mirror of
				https://github.com/immich-app/immich.git
				synced 2025-11-04 03:27:09 -05: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
					
				@ -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…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user