mirror of
https://github.com/immich-app/immich.git
synced 2026-07-04 03:45:59 -07:00
200 lines
7.1 KiB
TypeScript
200 lines
7.1 KiB
TypeScript
import { Kysely } from 'kysely';
|
|
import { SearchSuggestionType } from 'src/dtos/search.dto';
|
|
import { AlbumUserRole, AssetVisibility } from 'src/enum';
|
|
import { AccessRepository } from 'src/repositories/access.repository';
|
|
import { AssetRepository } from 'src/repositories/asset.repository';
|
|
import { DatabaseRepository } from 'src/repositories/database.repository';
|
|
import { LoggingRepository } from 'src/repositories/logging.repository';
|
|
import { PartnerRepository } from 'src/repositories/partner.repository';
|
|
import { PersonRepository } from 'src/repositories/person.repository';
|
|
import { SearchRepository } from 'src/repositories/search.repository';
|
|
import { DB } from 'src/schema';
|
|
import { SearchService } from 'src/services/search.service';
|
|
import { newMediumService } from 'test/medium.factory';
|
|
import { factory } from 'test/small.factory';
|
|
import { getKyselyDB } from 'test/utils';
|
|
|
|
let defaultDatabase: Kysely<DB>;
|
|
|
|
const setup = (db?: Kysely<DB>) => {
|
|
return newMediumService(SearchService, {
|
|
database: db || defaultDatabase,
|
|
real: [
|
|
AccessRepository,
|
|
AssetRepository,
|
|
DatabaseRepository,
|
|
SearchRepository,
|
|
PartnerRepository,
|
|
PersonRepository,
|
|
],
|
|
mock: [LoggingRepository],
|
|
});
|
|
};
|
|
|
|
beforeAll(async () => {
|
|
defaultDatabase = await getKyselyDB();
|
|
});
|
|
|
|
describe(SearchService.name, () => {
|
|
it('should work', () => {
|
|
const { sut } = setup();
|
|
expect(sut).toBeDefined();
|
|
});
|
|
|
|
it('should return assets', async () => {
|
|
const { sut, ctx } = setup();
|
|
const { user } = await ctx.newUser();
|
|
|
|
const assets = [];
|
|
const sizes = [12_334, 599, 123_456];
|
|
|
|
for (let i = 0; i < sizes.length; i++) {
|
|
const { asset } = await ctx.newAsset({ ownerId: user.id });
|
|
await ctx.newExif({ assetId: asset.id, fileSizeInByte: sizes[i] });
|
|
assets.push(asset);
|
|
}
|
|
|
|
const auth = factory.auth({ user: { id: user.id } });
|
|
|
|
await expect(sut.searchLargeAssets(auth, {})).resolves.toEqual([
|
|
expect.objectContaining({ id: assets[2].id }),
|
|
expect.objectContaining({ id: assets[0].id }),
|
|
expect.objectContaining({ id: assets[1].id }),
|
|
]);
|
|
});
|
|
|
|
describe('searchStatistics', () => {
|
|
it('should return statistics when filtering by personIds', async () => {
|
|
const { sut, ctx } = setup();
|
|
const { user } = await ctx.newUser();
|
|
const { asset } = await ctx.newAsset({ ownerId: user.id });
|
|
const { person } = await ctx.newPerson({ ownerId: user.id });
|
|
await ctx.newAssetFace({ assetId: asset.id, personId: person.id });
|
|
|
|
const auth = factory.auth({ user: { id: user.id } });
|
|
|
|
const result = await sut.searchStatistics(auth, { personIds: [person.id] });
|
|
|
|
expect(result).toEqual({ total: 1 });
|
|
});
|
|
|
|
it('should return zero when no assets match the personIds filter', async () => {
|
|
const { sut, ctx } = setup();
|
|
const { user } = await ctx.newUser();
|
|
const { person } = await ctx.newPerson({ ownerId: user.id });
|
|
|
|
const auth = factory.auth({ user: { id: user.id } });
|
|
|
|
const result = await sut.searchStatistics(auth, { personIds: [person.id] });
|
|
|
|
expect(result).toEqual({ total: 0 });
|
|
});
|
|
});
|
|
|
|
describe('withStacked option', () => {
|
|
it('should exclude stacked assets when withStacked is false', async () => {
|
|
const { sut, ctx } = setup();
|
|
const { user } = await ctx.newUser();
|
|
|
|
const { asset: primaryAsset } = await ctx.newAsset({ ownerId: user.id });
|
|
const { asset: stackedAsset } = await ctx.newAsset({ ownerId: user.id });
|
|
const { asset: unstackedAsset } = await ctx.newAsset({ ownerId: user.id });
|
|
|
|
await ctx.newStack({ ownerId: user.id }, [primaryAsset.id, stackedAsset.id]);
|
|
|
|
const auth = factory.auth({ user: { id: user.id } });
|
|
|
|
const response = await sut.searchMetadata(auth, { withStacked: false });
|
|
|
|
expect(response.assets.items.length).toBe(1);
|
|
expect(response.assets.items[0].id).toBe(unstackedAsset.id);
|
|
});
|
|
|
|
describe('visibility', () => {
|
|
it('should filter out locked assets in a default session', async () => {
|
|
const { sut, ctx } = setup();
|
|
const { user } = await ctx.newUser();
|
|
|
|
await ctx.newAsset({ ownerId: user.id, visibility: AssetVisibility.Locked });
|
|
|
|
const auth = factory.auth({ user: { id: user.id } });
|
|
|
|
const response = await sut.searchMetadata(auth, { withStacked: false });
|
|
|
|
expect(response.assets.items.length).toBe(0);
|
|
});
|
|
|
|
it('should return locked assets in an elevated session', async () => {
|
|
const { sut, ctx } = setup();
|
|
const { user } = await ctx.newUser();
|
|
|
|
await ctx.newAsset({ ownerId: user.id, visibility: AssetVisibility.Locked });
|
|
|
|
const auth = factory.auth({ user: { id: user.id }, session: { hasElevatedPermission: true } });
|
|
|
|
const response = await sut.searchMetadata(auth, { withStacked: false });
|
|
|
|
expect(response.assets.items.length).toBe(1);
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('albumIds option', () => {
|
|
it('should return assets from shared album', async () => {
|
|
const { sut, ctx } = setup();
|
|
const { user } = await ctx.newUser();
|
|
const { user: otherUser } = await ctx.newUser();
|
|
|
|
const { asset } = await ctx.newAsset({ ownerId: otherUser.id });
|
|
const { album } = await ctx.newAlbum({ ownerId: otherUser.id });
|
|
await ctx.newAlbumAsset({ albumId: album.id, assetId: asset.id });
|
|
await ctx.newAlbumUser({ albumId: album.id, userId: user.id, role: AlbumUserRole.Editor });
|
|
|
|
const auth = factory.auth({ user: { id: user.id } });
|
|
|
|
const response = await sut.searchMetadata(auth, { albumIds: [album.id] });
|
|
|
|
expect(response.assets.items.length).toBe(1);
|
|
});
|
|
|
|
it('should not return assets for album, a user is not in, when partner sharing is enabled', async () => {
|
|
const { sut, ctx } = setup();
|
|
const { user } = await ctx.newUser();
|
|
const { user: otherUser } = await ctx.newUser();
|
|
|
|
await ctx.newPartner({ sharedById: otherUser.id, sharedWithId: user.id });
|
|
|
|
const { asset } = await ctx.newAsset({ ownerId: otherUser.id });
|
|
const { album } = await ctx.newAlbum({ ownerId: otherUser.id });
|
|
await ctx.newAlbumAsset({ albumId: album.id, assetId: asset.id });
|
|
|
|
const auth = factory.auth({ user: { id: user.id } });
|
|
|
|
await expect(sut.searchMetadata(auth, { albumIds: [album.id] })).rejects.toThrow(
|
|
'Not found or no album.read access',
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('getSearchSuggestions', () => {
|
|
it('should filter out empty search suggestions', async () => {
|
|
const { sut, ctx } = setup();
|
|
const { user } = await ctx.newUser();
|
|
|
|
const { asset } = await ctx.newAsset({ ownerId: user.id });
|
|
await ctx.newExif({ assetId: asset.id, make: 'Canon' });
|
|
|
|
const { asset: assetWithEmptyMake } = await ctx.newAsset({ ownerId: user.id });
|
|
await ctx.newExif({ assetId: assetWithEmptyMake.id, make: '' });
|
|
|
|
const auth = factory.auth({ user: { id: user.id } });
|
|
const suggestions = await sut.getSearchSuggestions(auth, {
|
|
type: SearchSuggestionType.CAMERA_MAKE,
|
|
includeNull: true,
|
|
});
|
|
|
|
expect(suggestions).toEqual(['Canon', null]);
|
|
});
|
|
});
|
|
});
|