mirror of
https://github.com/immich-app/immich.git
synced 2025-10-30 13:17:40 +09:00
feat: view similar photos (#21108)
* Enable filteing by example * Drop `@GenerateSql` for `getEmbedding`? * Improve error message * PR Feedback * Sort en.json * Add SQL * Fix lint * Drop test that is no longer valid * Fix i18n file sorting * Fix TS error * Add a `requireAccess` before pulling the embedding * Fix decorators * Run `make open-api` --------- Co-authored-by: Alex <alex.tran1502@gmail.com>
This commit is contained in:
@@ -1557,6 +1557,7 @@
|
|||||||
"purchase_server_description_2": "Supporter status",
|
"purchase_server_description_2": "Supporter status",
|
||||||
"purchase_server_title": "Server",
|
"purchase_server_title": "Server",
|
||||||
"purchase_settings_server_activated": "The server product key is managed by the admin",
|
"purchase_settings_server_activated": "The server product key is managed by the admin",
|
||||||
|
"query_asset_id": "Query Asset ID",
|
||||||
"queue_status": "Queuing {count}/{total}",
|
"queue_status": "Queuing {count}/{total}",
|
||||||
"rating": "Star rating",
|
"rating": "Star rating",
|
||||||
"rating_clear": "Clear rating",
|
"rating_clear": "Clear rating",
|
||||||
@@ -2077,6 +2078,7 @@
|
|||||||
"view_next_asset": "View next asset",
|
"view_next_asset": "View next asset",
|
||||||
"view_previous_asset": "View previous asset",
|
"view_previous_asset": "View previous asset",
|
||||||
"view_qr_code": "View QR code",
|
"view_qr_code": "View QR code",
|
||||||
|
"view_similar_photos": "View similar photos",
|
||||||
"view_stack": "View Stack",
|
"view_stack": "View Stack",
|
||||||
"view_user": "View User",
|
"view_user": "View User",
|
||||||
"viewer_remove_from_stack": "Remove from Stack",
|
"viewer_remove_from_stack": "Remove from Stack",
|
||||||
|
|||||||
38
mobile/openapi/lib/model/smart_search_dto.dart
generated
38
mobile/openapi/lib/model/smart_search_dto.dart
generated
@@ -31,7 +31,8 @@ class SmartSearchDto {
|
|||||||
this.model,
|
this.model,
|
||||||
this.page,
|
this.page,
|
||||||
this.personIds = const [],
|
this.personIds = const [],
|
||||||
required this.query,
|
this.query,
|
||||||
|
this.queryAssetId,
|
||||||
this.rating,
|
this.rating,
|
||||||
this.size,
|
this.size,
|
||||||
this.state,
|
this.state,
|
||||||
@@ -151,7 +152,21 @@ class SmartSearchDto {
|
|||||||
|
|
||||||
List<String> personIds;
|
List<String> personIds;
|
||||||
|
|
||||||
String query;
|
///
|
||||||
|
/// Please note: This property should have been non-nullable! Since the specification file
|
||||||
|
/// does not include a default value (using the "default:" property), however, the generated
|
||||||
|
/// source code must fall back to having a nullable type.
|
||||||
|
/// Consider adding a "default:" property in the specification file to hide this note.
|
||||||
|
///
|
||||||
|
String? query;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// Please note: This property should have been non-nullable! Since the specification file
|
||||||
|
/// does not include a default value (using the "default:" property), however, the generated
|
||||||
|
/// source code must fall back to having a nullable type.
|
||||||
|
/// Consider adding a "default:" property in the specification file to hide this note.
|
||||||
|
///
|
||||||
|
String? queryAssetId;
|
||||||
|
|
||||||
/// Minimum value: -1
|
/// Minimum value: -1
|
||||||
/// Maximum value: 5
|
/// Maximum value: 5
|
||||||
@@ -278,6 +293,7 @@ class SmartSearchDto {
|
|||||||
other.page == page &&
|
other.page == page &&
|
||||||
_deepEquality.equals(other.personIds, personIds) &&
|
_deepEquality.equals(other.personIds, personIds) &&
|
||||||
other.query == query &&
|
other.query == query &&
|
||||||
|
other.queryAssetId == queryAssetId &&
|
||||||
other.rating == rating &&
|
other.rating == rating &&
|
||||||
other.size == size &&
|
other.size == size &&
|
||||||
other.state == state &&
|
other.state == state &&
|
||||||
@@ -314,7 +330,8 @@ class SmartSearchDto {
|
|||||||
(model == null ? 0 : model!.hashCode) +
|
(model == null ? 0 : model!.hashCode) +
|
||||||
(page == null ? 0 : page!.hashCode) +
|
(page == null ? 0 : page!.hashCode) +
|
||||||
(personIds.hashCode) +
|
(personIds.hashCode) +
|
||||||
(query.hashCode) +
|
(query == null ? 0 : query!.hashCode) +
|
||||||
|
(queryAssetId == null ? 0 : queryAssetId!.hashCode) +
|
||||||
(rating == null ? 0 : rating!.hashCode) +
|
(rating == null ? 0 : rating!.hashCode) +
|
||||||
(size == null ? 0 : size!.hashCode) +
|
(size == null ? 0 : size!.hashCode) +
|
||||||
(state == null ? 0 : state!.hashCode) +
|
(state == null ? 0 : state!.hashCode) +
|
||||||
@@ -331,7 +348,7 @@ class SmartSearchDto {
|
|||||||
(withExif == null ? 0 : withExif!.hashCode);
|
(withExif == null ? 0 : withExif!.hashCode);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String toString() => 'SmartSearchDto[albumIds=$albumIds, city=$city, country=$country, createdAfter=$createdAfter, createdBefore=$createdBefore, deviceId=$deviceId, isEncoded=$isEncoded, isFavorite=$isFavorite, isMotion=$isMotion, isNotInAlbum=$isNotInAlbum, isOffline=$isOffline, language=$language, lensModel=$lensModel, libraryId=$libraryId, make=$make, model=$model, page=$page, personIds=$personIds, query=$query, rating=$rating, size=$size, state=$state, tagIds=$tagIds, takenAfter=$takenAfter, takenBefore=$takenBefore, trashedAfter=$trashedAfter, trashedBefore=$trashedBefore, type=$type, updatedAfter=$updatedAfter, updatedBefore=$updatedBefore, visibility=$visibility, withDeleted=$withDeleted, withExif=$withExif]';
|
String toString() => 'SmartSearchDto[albumIds=$albumIds, city=$city, country=$country, createdAfter=$createdAfter, createdBefore=$createdBefore, deviceId=$deviceId, isEncoded=$isEncoded, isFavorite=$isFavorite, isMotion=$isMotion, isNotInAlbum=$isNotInAlbum, isOffline=$isOffline, language=$language, lensModel=$lensModel, libraryId=$libraryId, make=$make, model=$model, page=$page, personIds=$personIds, query=$query, queryAssetId=$queryAssetId, rating=$rating, size=$size, state=$state, tagIds=$tagIds, takenAfter=$takenAfter, takenBefore=$takenBefore, trashedAfter=$trashedAfter, trashedBefore=$trashedBefore, type=$type, updatedAfter=$updatedAfter, updatedBefore=$updatedBefore, visibility=$visibility, withDeleted=$withDeleted, withExif=$withExif]';
|
||||||
|
|
||||||
Map<String, dynamic> toJson() {
|
Map<String, dynamic> toJson() {
|
||||||
final json = <String, dynamic>{};
|
final json = <String, dynamic>{};
|
||||||
@@ -417,7 +434,16 @@ class SmartSearchDto {
|
|||||||
// json[r'page'] = null;
|
// json[r'page'] = null;
|
||||||
}
|
}
|
||||||
json[r'personIds'] = this.personIds;
|
json[r'personIds'] = this.personIds;
|
||||||
|
if (this.query != null) {
|
||||||
json[r'query'] = this.query;
|
json[r'query'] = this.query;
|
||||||
|
} else {
|
||||||
|
// json[r'query'] = null;
|
||||||
|
}
|
||||||
|
if (this.queryAssetId != null) {
|
||||||
|
json[r'queryAssetId'] = this.queryAssetId;
|
||||||
|
} else {
|
||||||
|
// json[r'queryAssetId'] = null;
|
||||||
|
}
|
||||||
if (this.rating != null) {
|
if (this.rating != null) {
|
||||||
json[r'rating'] = this.rating;
|
json[r'rating'] = this.rating;
|
||||||
} else {
|
} else {
|
||||||
@@ -522,7 +548,8 @@ class SmartSearchDto {
|
|||||||
personIds: json[r'personIds'] is Iterable
|
personIds: json[r'personIds'] is Iterable
|
||||||
? (json[r'personIds'] as Iterable).cast<String>().toList(growable: false)
|
? (json[r'personIds'] as Iterable).cast<String>().toList(growable: false)
|
||||||
: const [],
|
: const [],
|
||||||
query: mapValueOfType<String>(json, r'query')!,
|
query: mapValueOfType<String>(json, r'query'),
|
||||||
|
queryAssetId: mapValueOfType<String>(json, r'queryAssetId'),
|
||||||
rating: num.parse('${json[r'rating']}'),
|
rating: num.parse('${json[r'rating']}'),
|
||||||
size: num.parse('${json[r'size']}'),
|
size: num.parse('${json[r'size']}'),
|
||||||
state: mapValueOfType<String>(json, r'state'),
|
state: mapValueOfType<String>(json, r'state'),
|
||||||
@@ -586,7 +613,6 @@ class SmartSearchDto {
|
|||||||
|
|
||||||
/// The list of required keys that must be present in a JSON.
|
/// The list of required keys that must be present in a JSON.
|
||||||
static const requiredKeys = <String>{
|
static const requiredKeys = <String>{
|
||||||
'query',
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -14571,6 +14571,10 @@
|
|||||||
"query": {
|
"query": {
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
|
"queryAssetId": {
|
||||||
|
"format": "uuid",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
"rating": {
|
"rating": {
|
||||||
"maximum": 5,
|
"maximum": 5,
|
||||||
"minimum": -1,
|
"minimum": -1,
|
||||||
@@ -14638,9 +14642,6 @@
|
|||||||
"type": "boolean"
|
"type": "boolean"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"required": [
|
|
||||||
"query"
|
|
||||||
],
|
|
||||||
"type": "object"
|
"type": "object"
|
||||||
},
|
},
|
||||||
"SourceType": {
|
"SourceType": {
|
||||||
|
|||||||
@@ -1014,7 +1014,8 @@ export type SmartSearchDto = {
|
|||||||
model?: string | null;
|
model?: string | null;
|
||||||
page?: number;
|
page?: number;
|
||||||
personIds?: string[];
|
personIds?: string[];
|
||||||
query: string;
|
query?: string;
|
||||||
|
queryAssetId?: string;
|
||||||
rating?: number;
|
rating?: number;
|
||||||
size?: number;
|
size?: number;
|
||||||
state?: string | null;
|
state?: string | null;
|
||||||
|
|||||||
@@ -128,12 +128,6 @@ describe(SearchController.name, () => {
|
|||||||
await request(ctx.getHttpServer()).post('/search/smart');
|
await request(ctx.getHttpServer()).post('/search/smart');
|
||||||
expect(ctx.authenticate).toHaveBeenCalled();
|
expect(ctx.authenticate).toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should require a query', async () => {
|
|
||||||
const { status, body } = await request(ctx.getHttpServer()).post('/search/smart').send({});
|
|
||||||
expect(status).toBe(400);
|
|
||||||
expect(body).toEqual(errorDto.badRequest(['query should not be empty', 'query must be a string']));
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('GET /search/explore', () => {
|
describe('GET /search/explore', () => {
|
||||||
|
|||||||
@@ -199,7 +199,12 @@ export class StatisticsSearchDto extends BaseSearchDto {
|
|||||||
export class SmartSearchDto extends BaseSearchWithResultsDto {
|
export class SmartSearchDto extends BaseSearchWithResultsDto {
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
query!: string;
|
@Optional()
|
||||||
|
query?: string;
|
||||||
|
|
||||||
|
@ValidateUUID({ optional: true })
|
||||||
|
@Optional()
|
||||||
|
queryAssetId?: string;
|
||||||
|
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
|
|||||||
@@ -123,6 +123,14 @@ offset
|
|||||||
$8
|
$8
|
||||||
commit
|
commit
|
||||||
|
|
||||||
|
-- SearchRepository.getEmbedding
|
||||||
|
select
|
||||||
|
*
|
||||||
|
from
|
||||||
|
"smart_search"
|
||||||
|
where
|
||||||
|
"assetId" = $1
|
||||||
|
|
||||||
-- SearchRepository.searchFaces
|
-- SearchRepository.searchFaces
|
||||||
begin
|
begin
|
||||||
set
|
set
|
||||||
|
|||||||
@@ -293,6 +293,13 @@ export class SearchRepository {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@GenerateSql({
|
||||||
|
params: [DummyValue.UUID],
|
||||||
|
})
|
||||||
|
async getEmbedding(assetId: string) {
|
||||||
|
return this.db.selectFrom('smart_search').selectAll().where('assetId', '=', assetId).executeTakeFirst();
|
||||||
|
}
|
||||||
|
|
||||||
@GenerateSql({
|
@GenerateSql({
|
||||||
params: [
|
params: [
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ import {
|
|||||||
SmartSearchDto,
|
SmartSearchDto,
|
||||||
StatisticsSearchDto,
|
StatisticsSearchDto,
|
||||||
} from 'src/dtos/search.dto';
|
} from 'src/dtos/search.dto';
|
||||||
import { AssetOrder, AssetVisibility } from 'src/enum';
|
import { AssetOrder, AssetVisibility, Permission } from 'src/enum';
|
||||||
import { BaseService } from 'src/services/base.service';
|
import { BaseService } from 'src/services/base.service';
|
||||||
import { requireElevatedPermission } from 'src/utils/access';
|
import { requireElevatedPermission } from 'src/utils/access';
|
||||||
import { getMyPartnerIds } from 'src/utils/asset.util';
|
import { getMyPartnerIds } from 'src/utils/asset.util';
|
||||||
@@ -113,14 +113,27 @@ export class SearchService extends BaseService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const userIds = this.getUserIdsToSearch(auth);
|
const userIds = this.getUserIdsToSearch(auth);
|
||||||
const key = machineLearning.clip.modelName + dto.query + dto.language;
|
let embedding;
|
||||||
let embedding = this.embeddingCache.get(key);
|
if (dto.query) {
|
||||||
if (!embedding) {
|
const key = machineLearning.clip.modelName + dto.query + dto.language;
|
||||||
embedding = await this.machineLearningRepository.encodeText(machineLearning.urls, dto.query, {
|
embedding = this.embeddingCache.get(key);
|
||||||
modelName: machineLearning.clip.modelName,
|
if (!embedding) {
|
||||||
language: dto.language,
|
embedding = await this.machineLearningRepository.encodeText(machineLearning.urls, dto.query, {
|
||||||
});
|
modelName: machineLearning.clip.modelName,
|
||||||
this.embeddingCache.set(key, embedding);
|
language: dto.language,
|
||||||
|
});
|
||||||
|
this.embeddingCache.set(key, embedding);
|
||||||
|
}
|
||||||
|
} else if (dto.queryAssetId) {
|
||||||
|
await this.requireAccess({ auth, permission: Permission.AssetRead, ids: [dto.queryAssetId] });
|
||||||
|
const getEmbeddingResponse = await this.searchRepository.getEmbedding(dto.queryAssetId);
|
||||||
|
const assetEmbedding = getEmbeddingResponse?.embedding;
|
||||||
|
if (!assetEmbedding) {
|
||||||
|
throw new BadRequestException(`Asset ${dto.queryAssetId} has no embedding`);
|
||||||
|
}
|
||||||
|
embedding = assetEmbedding;
|
||||||
|
} else {
|
||||||
|
throw new BadRequestException('Either `query` or `queryAssetId` must be set');
|
||||||
}
|
}
|
||||||
const page = dto.page ?? 1;
|
const page = dto.page ?? 1;
|
||||||
const size = dto.size || 100;
|
const size = dto.size || 100;
|
||||||
|
|||||||
@@ -22,6 +22,7 @@
|
|||||||
import ButtonContextMenu from '$lib/components/shared-components/context-menu/button-context-menu.svelte';
|
import ButtonContextMenu from '$lib/components/shared-components/context-menu/button-context-menu.svelte';
|
||||||
import MenuOption from '$lib/components/shared-components/context-menu/menu-option.svelte';
|
import MenuOption from '$lib/components/shared-components/context-menu/menu-option.svelte';
|
||||||
import { AppRoute } from '$lib/constants';
|
import { AppRoute } from '$lib/constants';
|
||||||
|
import { featureFlags } from '$lib/stores/server-config.store';
|
||||||
import { user } from '$lib/stores/user.store';
|
import { user } from '$lib/stores/user.store';
|
||||||
import { photoZoomState } from '$lib/stores/zoom-image.store';
|
import { photoZoomState } from '$lib/stores/zoom-image.store';
|
||||||
import { getAssetJobName, getSharedLink } from '$lib/utils';
|
import { getAssetJobName, getSharedLink } from '$lib/utils';
|
||||||
@@ -41,6 +42,7 @@
|
|||||||
import {
|
import {
|
||||||
mdiAlertOutline,
|
mdiAlertOutline,
|
||||||
mdiCogRefreshOutline,
|
mdiCogRefreshOutline,
|
||||||
|
mdiCompare,
|
||||||
mdiContentCopy,
|
mdiContentCopy,
|
||||||
mdiDatabaseRefreshOutline,
|
mdiDatabaseRefreshOutline,
|
||||||
mdiDotsVertical,
|
mdiDotsVertical,
|
||||||
@@ -98,6 +100,7 @@
|
|||||||
let isOwner = $derived($user && asset.ownerId === $user?.id);
|
let isOwner = $derived($user && asset.ownerId === $user?.id);
|
||||||
let showDownloadButton = $derived(sharedLink ? sharedLink.allowDownload : !asset.isOffline);
|
let showDownloadButton = $derived(sharedLink ? sharedLink.allowDownload : !asset.isOffline);
|
||||||
let isLocked = $derived(asset.visibility === AssetVisibility.Locked);
|
let isLocked = $derived(asset.visibility === AssetVisibility.Locked);
|
||||||
|
let smartSearchEnabled = $derived($featureFlags.loaded && $featureFlags.smartSearch);
|
||||||
|
|
||||||
// $: showEditorButton =
|
// $: showEditorButton =
|
||||||
// isOwner &&
|
// isOwner &&
|
||||||
@@ -225,6 +228,13 @@
|
|||||||
text={$t('view_in_timeline')}
|
text={$t('view_in_timeline')}
|
||||||
/>
|
/>
|
||||||
{/if}
|
{/if}
|
||||||
|
{#if !asset.isArchived && !asset.isTrashed && smartSearchEnabled}
|
||||||
|
<MenuOption
|
||||||
|
icon={mdiCompare}
|
||||||
|
onClick={() => goto(`${AppRoute.SEARCH}?query={"queryAssetId":"${stack?.primaryAssetId ?? asset.id}"}`)}
|
||||||
|
text={$t('view_similar_photos')}
|
||||||
|
/>
|
||||||
|
{/if}
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
{#if !asset.isTrashed}
|
{#if !asset.isTrashed}
|
||||||
|
|||||||
@@ -64,8 +64,16 @@
|
|||||||
return validQueryTypes.has(storedQueryType) ? storedQueryType : QueryType.SMART;
|
return validQueryTypes.has(storedQueryType) ? storedQueryType : QueryType.SMART;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let query = '';
|
||||||
|
if ('query' in searchQuery && searchQuery.query) {
|
||||||
|
query = searchQuery.query;
|
||||||
|
}
|
||||||
|
if ('originalFileName' in searchQuery && searchQuery.originalFileName) {
|
||||||
|
query = searchQuery.originalFileName;
|
||||||
|
}
|
||||||
|
|
||||||
let filter: SearchFilter = $state({
|
let filter: SearchFilter = $state({
|
||||||
query: 'query' in searchQuery ? searchQuery.query : searchQuery.originalFileName || '',
|
query,
|
||||||
queryType: defaultQueryType(),
|
queryType: defaultQueryType(),
|
||||||
personIds: new SvelteSet('personIds' in searchQuery ? searchQuery.personIds : []),
|
personIds: new SvelteSet('personIds' in searchQuery ? searchQuery.personIds : []),
|
||||||
tagIds:
|
tagIds:
|
||||||
|
|||||||
@@ -68,7 +68,7 @@
|
|||||||
|
|
||||||
const assetInteraction = new AssetInteraction();
|
const assetInteraction = new AssetInteraction();
|
||||||
|
|
||||||
type SearchTerms = MetadataSearchDto & Pick<SmartSearchDto, 'query'>;
|
type SearchTerms = MetadataSearchDto & Pick<SmartSearchDto, 'query' | 'queryAssetId'>;
|
||||||
let searchQuery = $derived(page.url.searchParams.get(QueryParameter.QUERY));
|
let searchQuery = $derived(page.url.searchParams.get(QueryParameter.QUERY));
|
||||||
let smartSearchEnabled = $derived($featureFlags.loaded && $featureFlags.smartSearch);
|
let smartSearchEnabled = $derived($featureFlags.loaded && $featureFlags.smartSearch);
|
||||||
let terms = $derived(searchQuery ? JSON.parse(searchQuery) : {});
|
let terms = $derived(searchQuery ? JSON.parse(searchQuery) : {});
|
||||||
@@ -164,7 +164,7 @@
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const { albums, assets } =
|
const { albums, assets } =
|
||||||
'query' in searchDto && smartSearchEnabled
|
('query' in searchDto || 'queryAssetId' in searchDto) && smartSearchEnabled
|
||||||
? await searchSmart({ smartSearchDto: searchDto })
|
? await searchSmart({ smartSearchDto: searchDto })
|
||||||
: await searchAssets({ metadataSearchDto: searchDto });
|
: await searchAssets({ metadataSearchDto: searchDto });
|
||||||
|
|
||||||
@@ -210,6 +210,7 @@
|
|||||||
tagIds: $t('tags'),
|
tagIds: $t('tags'),
|
||||||
originalFileName: $t('file_name'),
|
originalFileName: $t('file_name'),
|
||||||
description: $t('description'),
|
description: $t('description'),
|
||||||
|
queryAssetId: $t('query_asset_id'),
|
||||||
};
|
};
|
||||||
return keyMap[key] || key;
|
return keyMap[key] || key;
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user