mirror of
https://github.com/immich-app/immich.git
synced 2025-01-04 02:46:47 +01:00
fix(server): search and explore issues (#2029)
* fix: send assets to typesense in batches * fix: run classs transformer on search endpoint * chore: log typesense filters
This commit is contained in:
parent
deb1e7f41f
commit
73a2063d96
9 changed files with 283 additions and 13 deletions
BIN
mobile/openapi/doc/SearchApi.md
generated
BIN
mobile/openapi/doc/SearchApi.md
generated
Binary file not shown.
BIN
mobile/openapi/lib/api/search_api.dart
generated
BIN
mobile/openapi/lib/api/search_api.dart
generated
Binary file not shown.
BIN
mobile/openapi/test/search_api_test.dart
generated
BIN
mobile/openapi/test/search_api_test.dart
generated
Binary file not shown.
|
@ -20,7 +20,7 @@ export class SearchController {
|
||||||
@Get()
|
@Get()
|
||||||
async search(
|
async search(
|
||||||
@GetAuthUser() authUser: AuthUserDto,
|
@GetAuthUser() authUser: AuthUserDto,
|
||||||
@Query(new ValidationPipe({ transform: true })) dto: SearchDto | any,
|
@Query(new ValidationPipe({ transform: true })) dto: SearchDto,
|
||||||
): Promise<SearchResponseDto> {
|
): Promise<SearchResponseDto> {
|
||||||
return this.searchService.search(authUser, dto);
|
return this.searchService.search(authUser, dto);
|
||||||
}
|
}
|
||||||
|
|
|
@ -620,7 +620,132 @@
|
||||||
"get": {
|
"get": {
|
||||||
"operationId": "search",
|
"operationId": "search",
|
||||||
"description": "",
|
"description": "",
|
||||||
"parameters": [],
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "q",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "query",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "clip",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "boolean"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "type",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"enum": [
|
||||||
|
"IMAGE",
|
||||||
|
"VIDEO",
|
||||||
|
"AUDIO",
|
||||||
|
"OTHER"
|
||||||
|
],
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "isFavorite",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "boolean"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "exifInfo.city",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "exifInfo.state",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "exifInfo.country",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "exifInfo.make",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "exifInfo.model",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "smartInfo.objects",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "smartInfo.tags",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "recent",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "boolean"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "motion",
|
||||||
|
"required": false,
|
||||||
|
"in": "query",
|
||||||
|
"schema": {
|
||||||
|
"type": "boolean"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
"responses": {
|
"responses": {
|
||||||
"200": {
|
"200": {
|
||||||
"description": "",
|
"description": "",
|
||||||
|
|
|
@ -145,7 +145,15 @@ export class SearchService {
|
||||||
// TODO: do this in batches based on searchIndexVersion
|
// TODO: do this in batches based on searchIndexVersion
|
||||||
const assets = this.patchAssets(await this.assetRepository.getAll({ isVisible: true }));
|
const assets = this.patchAssets(await this.assetRepository.getAll({ isVisible: true }));
|
||||||
this.logger.log(`Indexing ${assets.length} assets`);
|
this.logger.log(`Indexing ${assets.length} assets`);
|
||||||
await this.searchRepository.importAssets(assets, true);
|
|
||||||
|
const chunkSize = 1000;
|
||||||
|
for (let i = 0; i < assets.length; i += chunkSize) {
|
||||||
|
const end = i + chunkSize;
|
||||||
|
const chunk = assets.slice(i, end);
|
||||||
|
const done = end >= assets.length - 1;
|
||||||
|
await this.searchRepository.importAssets(chunk, done);
|
||||||
|
}
|
||||||
|
|
||||||
this.logger.debug('Finished re-indexing all assets');
|
this.logger.debug('Finished re-indexing all assets');
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
this.logger.error(`Unable to index all assets`, error?.stack);
|
this.logger.error(`Unable to index all assets`, error?.stack);
|
||||||
|
|
|
@ -365,7 +365,11 @@ export class TypesenseRepository implements ISearchRepository {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return _filters.join(' && ');
|
const result = _filters.join(' && ');
|
||||||
|
|
||||||
|
this.logger.debug(`Album filters are: ${result}`);
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
private getAssetFilters(filters: SearchFilter) {
|
private getAssetFilters(filters: SearchFilter) {
|
||||||
|
@ -382,6 +386,11 @@ export class TypesenseRepository implements ISearchRepository {
|
||||||
_filters.push(`${item.name}:${value}`);
|
_filters.push(`${item.name}:${value}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return _filters.join(' && ');
|
|
||||||
|
const result = _filters.join(' && ');
|
||||||
|
|
||||||
|
this.logger.debug(`Asset filters are: ${result}`);
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
126
web/src/api/open-api/api.ts
generated
126
web/src/api/open-api/api.ts
generated
|
@ -6761,10 +6761,24 @@ export const SearchApiAxiosParamCreator = function (configuration?: Configuratio
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
* @param {string} [q]
|
||||||
|
* @param {string} [query]
|
||||||
|
* @param {boolean} [clip]
|
||||||
|
* @param {'IMAGE' | 'VIDEO' | 'AUDIO' | 'OTHER'} [type]
|
||||||
|
* @param {boolean} [isFavorite]
|
||||||
|
* @param {string} [exifInfoCity]
|
||||||
|
* @param {string} [exifInfoState]
|
||||||
|
* @param {string} [exifInfoCountry]
|
||||||
|
* @param {string} [exifInfoMake]
|
||||||
|
* @param {string} [exifInfoModel]
|
||||||
|
* @param {Array<string>} [smartInfoObjects]
|
||||||
|
* @param {Array<string>} [smartInfoTags]
|
||||||
|
* @param {boolean} [recent]
|
||||||
|
* @param {boolean} [motion]
|
||||||
* @param {*} [options] Override http request option.
|
* @param {*} [options] Override http request option.
|
||||||
* @throws {RequiredError}
|
* @throws {RequiredError}
|
||||||
*/
|
*/
|
||||||
search: async (options: AxiosRequestConfig = {}): Promise<RequestArgs> => {
|
search: async (q?: string, query?: string, clip?: boolean, type?: 'IMAGE' | 'VIDEO' | 'AUDIO' | 'OTHER', isFavorite?: boolean, exifInfoCity?: string, exifInfoState?: string, exifInfoCountry?: string, exifInfoMake?: string, exifInfoModel?: string, smartInfoObjects?: Array<string>, smartInfoTags?: Array<string>, recent?: boolean, motion?: boolean, options: AxiosRequestConfig = {}): Promise<RequestArgs> => {
|
||||||
const localVarPath = `/search`;
|
const localVarPath = `/search`;
|
||||||
// use dummy base URL string because the URL constructor only accepts absolute URLs.
|
// use dummy base URL string because the URL constructor only accepts absolute URLs.
|
||||||
const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL);
|
const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL);
|
||||||
|
@ -6783,6 +6797,62 @@ export const SearchApiAxiosParamCreator = function (configuration?: Configuratio
|
||||||
|
|
||||||
// authentication cookie required
|
// authentication cookie required
|
||||||
|
|
||||||
|
if (q !== undefined) {
|
||||||
|
localVarQueryParameter['q'] = q;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (query !== undefined) {
|
||||||
|
localVarQueryParameter['query'] = query;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (clip !== undefined) {
|
||||||
|
localVarQueryParameter['clip'] = clip;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type !== undefined) {
|
||||||
|
localVarQueryParameter['type'] = type;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isFavorite !== undefined) {
|
||||||
|
localVarQueryParameter['isFavorite'] = isFavorite;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exifInfoCity !== undefined) {
|
||||||
|
localVarQueryParameter['exifInfo.city'] = exifInfoCity;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exifInfoState !== undefined) {
|
||||||
|
localVarQueryParameter['exifInfo.state'] = exifInfoState;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exifInfoCountry !== undefined) {
|
||||||
|
localVarQueryParameter['exifInfo.country'] = exifInfoCountry;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exifInfoMake !== undefined) {
|
||||||
|
localVarQueryParameter['exifInfo.make'] = exifInfoMake;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exifInfoModel !== undefined) {
|
||||||
|
localVarQueryParameter['exifInfo.model'] = exifInfoModel;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (smartInfoObjects) {
|
||||||
|
localVarQueryParameter['smartInfo.objects'] = smartInfoObjects;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (smartInfoTags) {
|
||||||
|
localVarQueryParameter['smartInfo.tags'] = smartInfoTags;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (recent !== undefined) {
|
||||||
|
localVarQueryParameter['recent'] = recent;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (motion !== undefined) {
|
||||||
|
localVarQueryParameter['motion'] = motion;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
setSearchParams(localVarUrlObj, localVarQueryParameter);
|
setSearchParams(localVarUrlObj, localVarQueryParameter);
|
||||||
|
@ -6824,11 +6894,25 @@ export const SearchApiFp = function(configuration?: Configuration) {
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
* @param {string} [q]
|
||||||
|
* @param {string} [query]
|
||||||
|
* @param {boolean} [clip]
|
||||||
|
* @param {'IMAGE' | 'VIDEO' | 'AUDIO' | 'OTHER'} [type]
|
||||||
|
* @param {boolean} [isFavorite]
|
||||||
|
* @param {string} [exifInfoCity]
|
||||||
|
* @param {string} [exifInfoState]
|
||||||
|
* @param {string} [exifInfoCountry]
|
||||||
|
* @param {string} [exifInfoMake]
|
||||||
|
* @param {string} [exifInfoModel]
|
||||||
|
* @param {Array<string>} [smartInfoObjects]
|
||||||
|
* @param {Array<string>} [smartInfoTags]
|
||||||
|
* @param {boolean} [recent]
|
||||||
|
* @param {boolean} [motion]
|
||||||
* @param {*} [options] Override http request option.
|
* @param {*} [options] Override http request option.
|
||||||
* @throws {RequiredError}
|
* @throws {RequiredError}
|
||||||
*/
|
*/
|
||||||
async search(options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<SearchResponseDto>> {
|
async search(q?: string, query?: string, clip?: boolean, type?: 'IMAGE' | 'VIDEO' | 'AUDIO' | 'OTHER', isFavorite?: boolean, exifInfoCity?: string, exifInfoState?: string, exifInfoCountry?: string, exifInfoMake?: string, exifInfoModel?: string, smartInfoObjects?: Array<string>, smartInfoTags?: Array<string>, recent?: boolean, motion?: boolean, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<SearchResponseDto>> {
|
||||||
const localVarAxiosArgs = await localVarAxiosParamCreator.search(options);
|
const localVarAxiosArgs = await localVarAxiosParamCreator.search(q, query, clip, type, isFavorite, exifInfoCity, exifInfoState, exifInfoCountry, exifInfoMake, exifInfoModel, smartInfoObjects, smartInfoTags, recent, motion, options);
|
||||||
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
|
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -6859,11 +6943,25 @@ export const SearchApiFactory = function (configuration?: Configuration, basePat
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
* @param {string} [q]
|
||||||
|
* @param {string} [query]
|
||||||
|
* @param {boolean} [clip]
|
||||||
|
* @param {'IMAGE' | 'VIDEO' | 'AUDIO' | 'OTHER'} [type]
|
||||||
|
* @param {boolean} [isFavorite]
|
||||||
|
* @param {string} [exifInfoCity]
|
||||||
|
* @param {string} [exifInfoState]
|
||||||
|
* @param {string} [exifInfoCountry]
|
||||||
|
* @param {string} [exifInfoMake]
|
||||||
|
* @param {string} [exifInfoModel]
|
||||||
|
* @param {Array<string>} [smartInfoObjects]
|
||||||
|
* @param {Array<string>} [smartInfoTags]
|
||||||
|
* @param {boolean} [recent]
|
||||||
|
* @param {boolean} [motion]
|
||||||
* @param {*} [options] Override http request option.
|
* @param {*} [options] Override http request option.
|
||||||
* @throws {RequiredError}
|
* @throws {RequiredError}
|
||||||
*/
|
*/
|
||||||
search(options?: any): AxiosPromise<SearchResponseDto> {
|
search(q?: string, query?: string, clip?: boolean, type?: 'IMAGE' | 'VIDEO' | 'AUDIO' | 'OTHER', isFavorite?: boolean, exifInfoCity?: string, exifInfoState?: string, exifInfoCountry?: string, exifInfoMake?: string, exifInfoModel?: string, smartInfoObjects?: Array<string>, smartInfoTags?: Array<string>, recent?: boolean, motion?: boolean, options?: any): AxiosPromise<SearchResponseDto> {
|
||||||
return localVarFp.search(options).then((request) => request(axios, basePath));
|
return localVarFp.search(q, query, clip, type, isFavorite, exifInfoCity, exifInfoState, exifInfoCountry, exifInfoMake, exifInfoModel, smartInfoObjects, smartInfoTags, recent, motion, options).then((request) => request(axios, basePath));
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -6897,12 +6995,26 @@ export class SearchApi extends BaseAPI {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
* @param {string} [q]
|
||||||
|
* @param {string} [query]
|
||||||
|
* @param {boolean} [clip]
|
||||||
|
* @param {'IMAGE' | 'VIDEO' | 'AUDIO' | 'OTHER'} [type]
|
||||||
|
* @param {boolean} [isFavorite]
|
||||||
|
* @param {string} [exifInfoCity]
|
||||||
|
* @param {string} [exifInfoState]
|
||||||
|
* @param {string} [exifInfoCountry]
|
||||||
|
* @param {string} [exifInfoMake]
|
||||||
|
* @param {string} [exifInfoModel]
|
||||||
|
* @param {Array<string>} [smartInfoObjects]
|
||||||
|
* @param {Array<string>} [smartInfoTags]
|
||||||
|
* @param {boolean} [recent]
|
||||||
|
* @param {boolean} [motion]
|
||||||
* @param {*} [options] Override http request option.
|
* @param {*} [options] Override http request option.
|
||||||
* @throws {RequiredError}
|
* @throws {RequiredError}
|
||||||
* @memberof SearchApi
|
* @memberof SearchApi
|
||||||
*/
|
*/
|
||||||
public search(options?: AxiosRequestConfig) {
|
public search(q?: string, query?: string, clip?: boolean, type?: 'IMAGE' | 'VIDEO' | 'AUDIO' | 'OTHER', isFavorite?: boolean, exifInfoCity?: string, exifInfoState?: string, exifInfoCountry?: string, exifInfoMake?: string, exifInfoModel?: string, smartInfoObjects?: Array<string>, smartInfoTags?: Array<string>, recent?: boolean, motion?: boolean, options?: AxiosRequestConfig) {
|
||||||
return SearchApiFp(this.configuration).search(options).then((request) => request(this.axios, this.basePath));
|
return SearchApiFp(this.configuration).search(q, query, clip, type, isFavorite, exifInfoCity, exifInfoState, exifInfoCountry, exifInfoMake, exifInfoModel, smartInfoObjects, smartInfoTags, recent, motion, options).then((request) => request(this.axios, this.basePath));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -9,7 +9,23 @@ export const load = (async ({ locals, parent, url }) => {
|
||||||
|
|
||||||
const term = url.searchParams.get('q') || url.searchParams.get('query') || undefined;
|
const term = url.searchParams.get('q') || url.searchParams.get('query') || undefined;
|
||||||
|
|
||||||
const { data: results } = await locals.api.searchApi.search({ params: url.searchParams });
|
const { data: results } = await locals.api.searchApi.search(
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
undefined,
|
||||||
|
{ params: url.searchParams }
|
||||||
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
user,
|
user,
|
||||||
|
|
Loading…
Reference in a new issue