mirror of
https://github.com/immich-app/immich.git
synced 2025-01-27 22:22:45 +01:00
fix(server): search suggestions include partner assets (#12269)
search suggestions now include partner assets Co-authored-by: Alex <alex.tran1502@gmail.com>
This commit is contained in:
parent
259bc8a6b0
commit
27e283e724
5 changed files with 26 additions and 25 deletions
server
src
interfaces
repositories
services
test/repositories
|
@ -53,9 +53,9 @@ export interface IMetadataRepository {
|
||||||
readTags(path: string): Promise<ImmichTags | null>;
|
readTags(path: string): Promise<ImmichTags | null>;
|
||||||
writeTags(path: string, tags: Partial<Tags>): Promise<void>;
|
writeTags(path: string, tags: Partial<Tags>): Promise<void>;
|
||||||
extractBinaryTag(tagName: string, path: string): Promise<Buffer>;
|
extractBinaryTag(tagName: string, path: string): Promise<Buffer>;
|
||||||
getCountries(userId: string): Promise<Array<string | null>>;
|
getCountries(userIds: string[]): Promise<Array<string | null>>;
|
||||||
getStates(userId: string, country?: string): Promise<Array<string | null>>;
|
getStates(userIds: string[], country?: string): Promise<Array<string | null>>;
|
||||||
getCities(userId: string, country?: string, state?: string): Promise<Array<string | null>>;
|
getCities(userIds: string[], country?: string, state?: string): Promise<Array<string | null>>;
|
||||||
getCameraMakes(userId: string, model?: string): Promise<Array<string | null>>;
|
getCameraMakes(userIds: string[], model?: string): Promise<Array<string | null>>;
|
||||||
getCameraModels(userId: string, make?: string): Promise<Array<string | null>>;
|
getCameraModels(userIds: string[], make?: string): Promise<Array<string | null>>;
|
||||||
}
|
}
|
||||||
|
|
|
@ -56,11 +56,11 @@ export class MetadataRepository implements IMetadataRepository {
|
||||||
}
|
}
|
||||||
|
|
||||||
@GenerateSql({ params: [DummyValue.UUID] })
|
@GenerateSql({ params: [DummyValue.UUID] })
|
||||||
async getCountries(userId: string): Promise<string[]> {
|
async getCountries(userIds: string[]): Promise<string[]> {
|
||||||
const results = await this.exifRepository
|
const results = await this.exifRepository
|
||||||
.createQueryBuilder('exif')
|
.createQueryBuilder('exif')
|
||||||
.leftJoin('exif.asset', 'asset')
|
.leftJoin('exif.asset', 'asset')
|
||||||
.where('asset.ownerId = :userId', { userId })
|
.where('asset.ownerId IN (:...userIds )', { userIds })
|
||||||
.select('exif.country', 'country')
|
.select('exif.country', 'country')
|
||||||
.distinctOn(['exif.country'])
|
.distinctOn(['exif.country'])
|
||||||
.getRawMany<{ country: string }>();
|
.getRawMany<{ country: string }>();
|
||||||
|
@ -69,11 +69,11 @@ export class MetadataRepository implements IMetadataRepository {
|
||||||
}
|
}
|
||||||
|
|
||||||
@GenerateSql({ params: [DummyValue.UUID, DummyValue.STRING] })
|
@GenerateSql({ params: [DummyValue.UUID, DummyValue.STRING] })
|
||||||
async getStates(userId: string, country: string | undefined): Promise<string[]> {
|
async getStates(userIds: string[], country: string | undefined): Promise<string[]> {
|
||||||
const query = this.exifRepository
|
const query = this.exifRepository
|
||||||
.createQueryBuilder('exif')
|
.createQueryBuilder('exif')
|
||||||
.leftJoin('exif.asset', 'asset')
|
.leftJoin('exif.asset', 'asset')
|
||||||
.where('asset.ownerId = :userId', { userId })
|
.where('asset.ownerId IN (:...userIds )', { userIds })
|
||||||
.select('exif.state', 'state')
|
.select('exif.state', 'state')
|
||||||
.distinctOn(['exif.state']);
|
.distinctOn(['exif.state']);
|
||||||
|
|
||||||
|
@ -87,11 +87,11 @@ export class MetadataRepository implements IMetadataRepository {
|
||||||
}
|
}
|
||||||
|
|
||||||
@GenerateSql({ params: [DummyValue.UUID, DummyValue.STRING, DummyValue.STRING] })
|
@GenerateSql({ params: [DummyValue.UUID, DummyValue.STRING, DummyValue.STRING] })
|
||||||
async getCities(userId: string, country: string | undefined, state: string | undefined): Promise<string[]> {
|
async getCities(userIds: string[], country: string | undefined, state: string | undefined): Promise<string[]> {
|
||||||
const query = this.exifRepository
|
const query = this.exifRepository
|
||||||
.createQueryBuilder('exif')
|
.createQueryBuilder('exif')
|
||||||
.leftJoin('exif.asset', 'asset')
|
.leftJoin('exif.asset', 'asset')
|
||||||
.where('asset.ownerId = :userId', { userId })
|
.where('asset.ownerId IN (:...userIds )', { userIds })
|
||||||
.select('exif.city', 'city')
|
.select('exif.city', 'city')
|
||||||
.distinctOn(['exif.city']);
|
.distinctOn(['exif.city']);
|
||||||
|
|
||||||
|
@ -109,11 +109,11 @@ export class MetadataRepository implements IMetadataRepository {
|
||||||
}
|
}
|
||||||
|
|
||||||
@GenerateSql({ params: [DummyValue.UUID, DummyValue.STRING] })
|
@GenerateSql({ params: [DummyValue.UUID, DummyValue.STRING] })
|
||||||
async getCameraMakes(userId: string, model: string | undefined): Promise<string[]> {
|
async getCameraMakes(userIds: string[], model: string | undefined): Promise<string[]> {
|
||||||
const query = this.exifRepository
|
const query = this.exifRepository
|
||||||
.createQueryBuilder('exif')
|
.createQueryBuilder('exif')
|
||||||
.leftJoin('exif.asset', 'asset')
|
.leftJoin('exif.asset', 'asset')
|
||||||
.where('asset.ownerId = :userId', { userId })
|
.where('asset.ownerId IN (:...userIds )', { userIds })
|
||||||
.select('exif.make', 'make')
|
.select('exif.make', 'make')
|
||||||
.distinctOn(['exif.make']);
|
.distinctOn(['exif.make']);
|
||||||
|
|
||||||
|
@ -126,11 +126,11 @@ export class MetadataRepository implements IMetadataRepository {
|
||||||
}
|
}
|
||||||
|
|
||||||
@GenerateSql({ params: [DummyValue.UUID, DummyValue.STRING] })
|
@GenerateSql({ params: [DummyValue.UUID, DummyValue.STRING] })
|
||||||
async getCameraModels(userId: string, make: string | undefined): Promise<string[]> {
|
async getCameraModels(userIds: string[], make: string | undefined): Promise<string[]> {
|
||||||
const query = this.exifRepository
|
const query = this.exifRepository
|
||||||
.createQueryBuilder('exif')
|
.createQueryBuilder('exif')
|
||||||
.leftJoin('exif.asset', 'asset')
|
.leftJoin('exif.asset', 'asset')
|
||||||
.where('asset.ownerId = :userId', { userId })
|
.where('asset.ownerId IN (:...userIds )', { userIds })
|
||||||
.select('exif.model', 'model')
|
.select('exif.model', 'model')
|
||||||
.distinctOn(['exif.model']);
|
.distinctOn(['exif.model']);
|
||||||
|
|
||||||
|
|
|
@ -103,7 +103,7 @@ describe(SearchService.name, () => {
|
||||||
await expect(
|
await expect(
|
||||||
sut.getSearchSuggestions(authStub.user1, { includeNull: true, type: SearchSuggestionType.COUNTRY }),
|
sut.getSearchSuggestions(authStub.user1, { includeNull: true, type: SearchSuggestionType.COUNTRY }),
|
||||||
).resolves.toEqual(['USA', null]);
|
).resolves.toEqual(['USA', null]);
|
||||||
expect(metadataMock.getCountries).toHaveBeenCalledWith(authStub.user1.user.id);
|
expect(metadataMock.getCountries).toHaveBeenCalledWith([authStub.user1.user.id]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return search suggestions (without null)', async () => {
|
it('should return search suggestions (without null)', async () => {
|
||||||
|
@ -111,7 +111,7 @@ describe(SearchService.name, () => {
|
||||||
await expect(
|
await expect(
|
||||||
sut.getSearchSuggestions(authStub.user1, { includeNull: false, type: SearchSuggestionType.COUNTRY }),
|
sut.getSearchSuggestions(authStub.user1, { includeNull: false, type: SearchSuggestionType.COUNTRY }),
|
||||||
).resolves.toEqual(['USA']);
|
).resolves.toEqual(['USA']);
|
||||||
expect(metadataMock.getCountries).toHaveBeenCalledWith(authStub.user1.user.id);
|
expect(metadataMock.getCountries).toHaveBeenCalledWith([authStub.user1.user.id]);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -121,26 +121,27 @@ export class SearchService {
|
||||||
}
|
}
|
||||||
|
|
||||||
async getSearchSuggestions(auth: AuthDto, dto: SearchSuggestionRequestDto) {
|
async getSearchSuggestions(auth: AuthDto, dto: SearchSuggestionRequestDto) {
|
||||||
const results = await this.getSuggestions(auth.user.id, dto);
|
const userIds = await this.getUserIdsToSearch(auth);
|
||||||
|
const results = await this.getSuggestions(userIds, dto);
|
||||||
return results.filter((result) => (dto.includeNull ? true : result !== null));
|
return results.filter((result) => (dto.includeNull ? true : result !== null));
|
||||||
}
|
}
|
||||||
|
|
||||||
private getSuggestions(userId: string, dto: SearchSuggestionRequestDto) {
|
private getSuggestions(userIds: string[], dto: SearchSuggestionRequestDto) {
|
||||||
switch (dto.type) {
|
switch (dto.type) {
|
||||||
case SearchSuggestionType.COUNTRY: {
|
case SearchSuggestionType.COUNTRY: {
|
||||||
return this.metadataRepository.getCountries(userId);
|
return this.metadataRepository.getCountries(userIds);
|
||||||
}
|
}
|
||||||
case SearchSuggestionType.STATE: {
|
case SearchSuggestionType.STATE: {
|
||||||
return this.metadataRepository.getStates(userId, dto.country);
|
return this.metadataRepository.getStates(userIds, dto.country);
|
||||||
}
|
}
|
||||||
case SearchSuggestionType.CITY: {
|
case SearchSuggestionType.CITY: {
|
||||||
return this.metadataRepository.getCities(userId, dto.country, dto.state);
|
return this.metadataRepository.getCities(userIds, dto.country, dto.state);
|
||||||
}
|
}
|
||||||
case SearchSuggestionType.CAMERA_MAKE: {
|
case SearchSuggestionType.CAMERA_MAKE: {
|
||||||
return this.metadataRepository.getCameraMakes(userId, dto.model);
|
return this.metadataRepository.getCameraMakes(userIds, dto.model);
|
||||||
}
|
}
|
||||||
case SearchSuggestionType.CAMERA_MODEL: {
|
case SearchSuggestionType.CAMERA_MODEL: {
|
||||||
return this.metadataRepository.getCameraModels(userId, dto.make);
|
return this.metadataRepository.getCameraModels(userIds, dto.make);
|
||||||
}
|
}
|
||||||
default: {
|
default: {
|
||||||
return [];
|
return [];
|
||||||
|
|
|
@ -5,7 +5,7 @@ export const newPartnerRepositoryMock = (): Mocked<IPartnerRepository> => {
|
||||||
return {
|
return {
|
||||||
create: vitest.fn(),
|
create: vitest.fn(),
|
||||||
remove: vitest.fn(),
|
remove: vitest.fn(),
|
||||||
getAll: vitest.fn(),
|
getAll: vitest.fn().mockResolvedValue([]),
|
||||||
get: vitest.fn(),
|
get: vitest.fn(),
|
||||||
update: vitest.fn(),
|
update: vitest.fn(),
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in a new issue