mirror of
https://github.com/immich-app/immich
synced 2025-10-17 18:19:27 +00:00
fix: remove assets from shared link (#22935)
* fix remove assets from shared link * rename var * test: should remove individually shared asset * test: should share individually assets * fix failing tests
This commit is contained in:
parent
2919ee4c65
commit
a23dfff6cf
8 changed files with 98 additions and 3 deletions
|
|
@ -31,6 +31,7 @@ import { ProcessRepository } from 'src/repositories/process.repository';
|
||||||
import { SearchRepository } from 'src/repositories/search.repository';
|
import { SearchRepository } from 'src/repositories/search.repository';
|
||||||
import { ServerInfoRepository } from 'src/repositories/server-info.repository';
|
import { ServerInfoRepository } from 'src/repositories/server-info.repository';
|
||||||
import { SessionRepository } from 'src/repositories/session.repository';
|
import { SessionRepository } from 'src/repositories/session.repository';
|
||||||
|
import { SharedLinkAssetRepository } from 'src/repositories/shared-link-asset.repository';
|
||||||
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
||||||
import { StackRepository } from 'src/repositories/stack.repository';
|
import { StackRepository } from 'src/repositories/stack.repository';
|
||||||
import { StorageRepository } from 'src/repositories/storage.repository';
|
import { StorageRepository } from 'src/repositories/storage.repository';
|
||||||
|
|
@ -79,6 +80,7 @@ export const repositories = [
|
||||||
SessionRepository,
|
SessionRepository,
|
||||||
ServerInfoRepository,
|
ServerInfoRepository,
|
||||||
SharedLinkRepository,
|
SharedLinkRepository,
|
||||||
|
SharedLinkAssetRepository,
|
||||||
StackRepository,
|
StackRepository,
|
||||||
StorageRepository,
|
StorageRepository,
|
||||||
SyncRepository,
|
SyncRepository,
|
||||||
|
|
|
||||||
18
server/src/repositories/shared-link-asset.repository.ts
Normal file
18
server/src/repositories/shared-link-asset.repository.ts
Normal file
|
|
@ -0,0 +1,18 @@
|
||||||
|
import { Kysely } from 'kysely';
|
||||||
|
import { InjectKysely } from 'nestjs-kysely';
|
||||||
|
import { DB } from 'src/schema';
|
||||||
|
|
||||||
|
export class SharedLinkAssetRepository {
|
||||||
|
constructor(@InjectKysely() private db: Kysely<DB>) {}
|
||||||
|
|
||||||
|
async remove(sharedLinkId: string, assetsId: string[]) {
|
||||||
|
const deleted = await this.db
|
||||||
|
.deleteFrom('shared_link_asset')
|
||||||
|
.where('shared_link_asset.sharedLinksId', '=', sharedLinkId)
|
||||||
|
.where('shared_link_asset.assetsId', 'in', assetsId)
|
||||||
|
.returning('assetsId')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
return deleted.map((row) => row.assetsId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -38,6 +38,7 @@ import { ProcessRepository } from 'src/repositories/process.repository';
|
||||||
import { SearchRepository } from 'src/repositories/search.repository';
|
import { SearchRepository } from 'src/repositories/search.repository';
|
||||||
import { ServerInfoRepository } from 'src/repositories/server-info.repository';
|
import { ServerInfoRepository } from 'src/repositories/server-info.repository';
|
||||||
import { SessionRepository } from 'src/repositories/session.repository';
|
import { SessionRepository } from 'src/repositories/session.repository';
|
||||||
|
import { SharedLinkAssetRepository } from 'src/repositories/shared-link-asset.repository';
|
||||||
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
||||||
import { StackRepository } from 'src/repositories/stack.repository';
|
import { StackRepository } from 'src/repositories/stack.repository';
|
||||||
import { StorageRepository } from 'src/repositories/storage.repository';
|
import { StorageRepository } from 'src/repositories/storage.repository';
|
||||||
|
|
@ -89,6 +90,7 @@ export const BASE_SERVICE_DEPENDENCIES = [
|
||||||
ServerInfoRepository,
|
ServerInfoRepository,
|
||||||
SessionRepository,
|
SessionRepository,
|
||||||
SharedLinkRepository,
|
SharedLinkRepository,
|
||||||
|
SharedLinkAssetRepository,
|
||||||
StackRepository,
|
StackRepository,
|
||||||
StorageRepository,
|
StorageRepository,
|
||||||
SyncRepository,
|
SyncRepository,
|
||||||
|
|
@ -141,6 +143,7 @@ export class BaseService {
|
||||||
protected serverInfoRepository: ServerInfoRepository,
|
protected serverInfoRepository: ServerInfoRepository,
|
||||||
protected sessionRepository: SessionRepository,
|
protected sessionRepository: SessionRepository,
|
||||||
protected sharedLinkRepository: SharedLinkRepository,
|
protected sharedLinkRepository: SharedLinkRepository,
|
||||||
|
protected sharedLinkAssetRepository: SharedLinkAssetRepository,
|
||||||
protected stackRepository: StackRepository,
|
protected stackRepository: StackRepository,
|
||||||
protected storageRepository: StorageRepository,
|
protected storageRepository: StorageRepository,
|
||||||
protected syncRepository: SyncRepository,
|
protected syncRepository: SyncRepository,
|
||||||
|
|
|
||||||
|
|
@ -300,6 +300,7 @@ describe(SharedLinkService.name, () => {
|
||||||
mocks.sharedLink.get.mockResolvedValue(_.cloneDeep(sharedLinkStub.individual));
|
mocks.sharedLink.get.mockResolvedValue(_.cloneDeep(sharedLinkStub.individual));
|
||||||
mocks.sharedLink.create.mockResolvedValue(sharedLinkStub.individual);
|
mocks.sharedLink.create.mockResolvedValue(sharedLinkStub.individual);
|
||||||
mocks.sharedLink.update.mockResolvedValue(sharedLinkStub.individual);
|
mocks.sharedLink.update.mockResolvedValue(sharedLinkStub.individual);
|
||||||
|
mocks.sharedLinkAsset.remove.mockResolvedValue([assetStub.image.id]);
|
||||||
|
|
||||||
await expect(
|
await expect(
|
||||||
sut.removeAssets(authStub.admin, 'link-1', { assetIds: [assetStub.image.id, 'asset-2'] }),
|
sut.removeAssets(authStub.admin, 'link-1', { assetIds: [assetStub.image.id, 'asset-2'] }),
|
||||||
|
|
@ -308,6 +309,7 @@ describe(SharedLinkService.name, () => {
|
||||||
{ assetId: 'asset-2', success: false, error: AssetIdErrorReason.NOT_FOUND },
|
{ assetId: 'asset-2', success: false, error: AssetIdErrorReason.NOT_FOUND },
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
expect(mocks.sharedLinkAsset.remove).toHaveBeenCalledWith('link-1', [assetStub.image.id, 'asset-2']);
|
||||||
expect(mocks.sharedLink.update).toHaveBeenCalledWith({ ...sharedLinkStub.individual, assets: [] });
|
expect(mocks.sharedLink.update).toHaveBeenCalledWith({ ...sharedLinkStub.individual, assets: [] });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
||||||
|
|
@ -175,10 +175,12 @@ export class SharedLinkService extends BaseService {
|
||||||
throw new BadRequestException('Invalid shared link type');
|
throw new BadRequestException('Invalid shared link type');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const removedAssetIds = await this.sharedLinkAssetRepository.remove(id, dto.assetIds);
|
||||||
|
|
||||||
const results: AssetIdsResponseDto[] = [];
|
const results: AssetIdsResponseDto[] = [];
|
||||||
for (const assetId of dto.assetIds) {
|
for (const assetId of dto.assetIds) {
|
||||||
const hasAsset = sharedLink.assets.find((asset) => asset.id === assetId);
|
const wasRemoved = removedAssetIds.find((id) => id === assetId);
|
||||||
if (!hasAsset) {
|
if (!wasRemoved) {
|
||||||
results.push({ assetId, success: false, error: AssetIdErrorReason.NOT_FOUND });
|
results.push({ assetId, success: false, error: AssetIdErrorReason.NOT_FOUND });
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -33,6 +33,7 @@ import { PartnerRepository } from 'src/repositories/partner.repository';
|
||||||
import { PersonRepository } from 'src/repositories/person.repository';
|
import { PersonRepository } from 'src/repositories/person.repository';
|
||||||
import { SearchRepository } from 'src/repositories/search.repository';
|
import { SearchRepository } from 'src/repositories/search.repository';
|
||||||
import { SessionRepository } from 'src/repositories/session.repository';
|
import { SessionRepository } from 'src/repositories/session.repository';
|
||||||
|
import { SharedLinkAssetRepository } from 'src/repositories/shared-link-asset.repository';
|
||||||
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
||||||
import { StackRepository } from 'src/repositories/stack.repository';
|
import { StackRepository } from 'src/repositories/stack.repository';
|
||||||
import { StorageRepository } from 'src/repositories/storage.repository';
|
import { StorageRepository } from 'src/repositories/storage.repository';
|
||||||
|
|
@ -311,6 +312,7 @@ const newRealRepository = <T>(key: ClassConstructor<T>, db: Kysely<DB>): T => {
|
||||||
case SearchRepository:
|
case SearchRepository:
|
||||||
case SessionRepository:
|
case SessionRepository:
|
||||||
case SharedLinkRepository:
|
case SharedLinkRepository:
|
||||||
|
case SharedLinkAssetRepository:
|
||||||
case StackRepository:
|
case StackRepository:
|
||||||
case SyncRepository:
|
case SyncRepository:
|
||||||
case SyncCheckpointRepository:
|
case SyncCheckpointRepository:
|
||||||
|
|
|
||||||
|
|
@ -4,6 +4,7 @@ import { SharedLinkType } from 'src/enum';
|
||||||
import { AccessRepository } from 'src/repositories/access.repository';
|
import { AccessRepository } from 'src/repositories/access.repository';
|
||||||
import { DatabaseRepository } from 'src/repositories/database.repository';
|
import { DatabaseRepository } from 'src/repositories/database.repository';
|
||||||
import { LoggingRepository } from 'src/repositories/logging.repository';
|
import { LoggingRepository } from 'src/repositories/logging.repository';
|
||||||
|
import { SharedLinkAssetRepository } from 'src/repositories/shared-link-asset.repository';
|
||||||
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
||||||
import { StorageRepository } from 'src/repositories/storage.repository';
|
import { StorageRepository } from 'src/repositories/storage.repository';
|
||||||
import { DB } from 'src/schema';
|
import { DB } from 'src/schema';
|
||||||
|
|
@ -17,7 +18,7 @@ let defaultDatabase: Kysely<DB>;
|
||||||
const setup = (db?: Kysely<DB>) => {
|
const setup = (db?: Kysely<DB>) => {
|
||||||
return newMediumService(SharedLinkService, {
|
return newMediumService(SharedLinkService, {
|
||||||
database: db || defaultDatabase,
|
database: db || defaultDatabase,
|
||||||
real: [AccessRepository, DatabaseRepository, SharedLinkRepository],
|
real: [AccessRepository, DatabaseRepository, SharedLinkRepository, SharedLinkAssetRepository],
|
||||||
mock: [LoggingRepository, StorageRepository],
|
mock: [LoggingRepository, StorageRepository],
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
@ -62,4 +63,65 @@ describe(SharedLinkService.name, () => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should share individually assets', async () => {
|
||||||
|
const { sut, ctx } = setup();
|
||||||
|
|
||||||
|
const { user } = await ctx.newUser();
|
||||||
|
|
||||||
|
const assets = await Promise.all([
|
||||||
|
ctx.newAsset({ ownerId: user.id }),
|
||||||
|
ctx.newAsset({ ownerId: user.id }),
|
||||||
|
ctx.newAsset({ ownerId: user.id }),
|
||||||
|
]);
|
||||||
|
|
||||||
|
for (const { asset } of assets) {
|
||||||
|
await ctx.newExif({ assetId: asset.id, make: 'Canon' });
|
||||||
|
}
|
||||||
|
|
||||||
|
const sharedLinkRepo = ctx.get(SharedLinkRepository);
|
||||||
|
|
||||||
|
const sharedLink = await sharedLinkRepo.create({
|
||||||
|
key: randomBytes(16),
|
||||||
|
id: factory.uuid(),
|
||||||
|
userId: user.id,
|
||||||
|
allowUpload: false,
|
||||||
|
type: SharedLinkType.Individual,
|
||||||
|
assetIds: assets.map(({ asset }) => asset.id),
|
||||||
|
});
|
||||||
|
|
||||||
|
await expect(sut.getMine({ user, sharedLink }, {})).resolves.toMatchObject({
|
||||||
|
assets: assets.map(({ asset }) => expect.objectContaining({ id: asset.id })),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should remove individually shared asset', async () => {
|
||||||
|
const { sut, ctx } = setup();
|
||||||
|
|
||||||
|
const { user } = await ctx.newUser();
|
||||||
|
const auth = factory.auth({ user });
|
||||||
|
const { asset } = await ctx.newAsset({ ownerId: user.id });
|
||||||
|
await ctx.newExif({ assetId: asset.id, make: 'Canon' });
|
||||||
|
|
||||||
|
const sharedLinkRepo = ctx.get(SharedLinkRepository);
|
||||||
|
|
||||||
|
const sharedLink = await sharedLinkRepo.create({
|
||||||
|
key: randomBytes(16),
|
||||||
|
id: factory.uuid(),
|
||||||
|
userId: user.id,
|
||||||
|
allowUpload: false,
|
||||||
|
type: SharedLinkType.Individual,
|
||||||
|
assetIds: [asset.id],
|
||||||
|
});
|
||||||
|
|
||||||
|
await expect(sut.getMine({ user, sharedLink }, {})).resolves.toMatchObject({
|
||||||
|
assets: [expect.objectContaining({ id: asset.id })],
|
||||||
|
});
|
||||||
|
|
||||||
|
await sut.removeAssets(auth, sharedLink.id, {
|
||||||
|
assetIds: [asset.id],
|
||||||
|
});
|
||||||
|
|
||||||
|
await expect(sut.getMine({ user, sharedLink }, {})).resolves.toHaveProperty('assets', []);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
||||||
|
|
@ -47,6 +47,7 @@ import { ProcessRepository } from 'src/repositories/process.repository';
|
||||||
import { SearchRepository } from 'src/repositories/search.repository';
|
import { SearchRepository } from 'src/repositories/search.repository';
|
||||||
import { ServerInfoRepository } from 'src/repositories/server-info.repository';
|
import { ServerInfoRepository } from 'src/repositories/server-info.repository';
|
||||||
import { SessionRepository } from 'src/repositories/session.repository';
|
import { SessionRepository } from 'src/repositories/session.repository';
|
||||||
|
import { SharedLinkAssetRepository } from 'src/repositories/shared-link-asset.repository';
|
||||||
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
||||||
import { StackRepository } from 'src/repositories/stack.repository';
|
import { StackRepository } from 'src/repositories/stack.repository';
|
||||||
import { StorageRepository } from 'src/repositories/storage.repository';
|
import { StorageRepository } from 'src/repositories/storage.repository';
|
||||||
|
|
@ -236,6 +237,7 @@ export type ServiceOverrides = {
|
||||||
serverInfo: ServerInfoRepository;
|
serverInfo: ServerInfoRepository;
|
||||||
session: SessionRepository;
|
session: SessionRepository;
|
||||||
sharedLink: SharedLinkRepository;
|
sharedLink: SharedLinkRepository;
|
||||||
|
sharedLinkAsset: SharedLinkAssetRepository;
|
||||||
stack: StackRepository;
|
stack: StackRepository;
|
||||||
storage: StorageRepository;
|
storage: StorageRepository;
|
||||||
sync: SyncRepository;
|
sync: SyncRepository;
|
||||||
|
|
@ -307,6 +309,7 @@ export const newTestService = <T extends BaseService>(
|
||||||
serverInfo: automock(ServerInfoRepository, { args: [, loggerMock], strict: false }),
|
serverInfo: automock(ServerInfoRepository, { args: [, loggerMock], strict: false }),
|
||||||
session: automock(SessionRepository),
|
session: automock(SessionRepository),
|
||||||
sharedLink: automock(SharedLinkRepository),
|
sharedLink: automock(SharedLinkRepository),
|
||||||
|
sharedLinkAsset: automock(SharedLinkAssetRepository),
|
||||||
stack: automock(StackRepository),
|
stack: automock(StackRepository),
|
||||||
storage: newStorageRepositoryMock(),
|
storage: newStorageRepositoryMock(),
|
||||||
sync: automock(SyncRepository),
|
sync: automock(SyncRepository),
|
||||||
|
|
@ -357,6 +360,7 @@ export const newTestService = <T extends BaseService>(
|
||||||
overrides.serverInfo || (mocks.serverInfo as As<ServerInfoRepository>),
|
overrides.serverInfo || (mocks.serverInfo as As<ServerInfoRepository>),
|
||||||
overrides.session || (mocks.session as As<SessionRepository>),
|
overrides.session || (mocks.session as As<SessionRepository>),
|
||||||
overrides.sharedLink || (mocks.sharedLink as As<SharedLinkRepository>),
|
overrides.sharedLink || (mocks.sharedLink as As<SharedLinkRepository>),
|
||||||
|
overrides.sharedLinkAsset || (mocks.sharedLinkAsset as As<SharedLinkAssetRepository>),
|
||||||
overrides.stack || (mocks.stack as As<StackRepository>),
|
overrides.stack || (mocks.stack as As<StackRepository>),
|
||||||
overrides.storage || (mocks.storage as As<StorageRepository>),
|
overrides.storage || (mocks.storage as As<StorageRepository>),
|
||||||
overrides.sync || (mocks.sync as As<SyncRepository>),
|
overrides.sync || (mocks.sync as As<SyncRepository>),
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue