diff --git a/mobile/openapi/doc/AssetApi.md b/mobile/openapi/doc/AssetApi.md index 7ddea756c0..4b6171d61a 100644 Binary files a/mobile/openapi/doc/AssetApi.md and b/mobile/openapi/doc/AssetApi.md differ diff --git a/mobile/openapi/doc/GetAssetCountByTimeBucketDto.md b/mobile/openapi/doc/GetAssetCountByTimeBucketDto.md index e770c3f9bb..5ba4078539 100644 Binary files a/mobile/openapi/doc/GetAssetCountByTimeBucketDto.md and b/mobile/openapi/doc/GetAssetCountByTimeBucketDto.md differ diff --git a/mobile/openapi/lib/api/asset_api.dart b/mobile/openapi/lib/api/asset_api.dart index ff272d4c27..8b0eaa2be8 100644 Binary files a/mobile/openapi/lib/api/asset_api.dart and b/mobile/openapi/lib/api/asset_api.dart differ diff --git a/mobile/openapi/lib/model/get_asset_count_by_time_bucket_dto.dart b/mobile/openapi/lib/model/get_asset_count_by_time_bucket_dto.dart index 619c5fe868..57e2836319 100644 Binary files a/mobile/openapi/lib/model/get_asset_count_by_time_bucket_dto.dart and b/mobile/openapi/lib/model/get_asset_count_by_time_bucket_dto.dart differ diff --git a/mobile/openapi/test/asset_api_test.dart b/mobile/openapi/test/asset_api_test.dart index 79c74cc8e0..75d6e953b8 100644 Binary files a/mobile/openapi/test/asset_api_test.dart and b/mobile/openapi/test/asset_api_test.dart differ diff --git a/mobile/openapi/test/get_asset_count_by_time_bucket_dto_test.dart b/mobile/openapi/test/get_asset_count_by_time_bucket_dto_test.dart index 5fa7c11bec..ca0f586a6b 100644 Binary files a/mobile/openapi/test/get_asset_count_by_time_bucket_dto_test.dart and b/mobile/openapi/test/get_asset_count_by_time_bucket_dto_test.dart differ diff --git a/server/apps/immich/src/api-v1/asset/asset-repository.ts b/server/apps/immich/src/api-v1/asset/asset-repository.ts index de7a9df9e3..6f2bf49527 100644 --- a/server/apps/immich/src/api-v1/asset/asset-repository.ts +++ b/server/apps/immich/src/api-v1/asset/asset-repository.ts @@ -6,7 +6,7 @@ import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm/repository/Repository'; import { CuratedObjectsResponseDto } from './response-dto/curated-objects-response.dto'; import { AssetCountByTimeBucket } from './response-dto/asset-count-by-time-group-response.dto'; -import { TimeGroupEnum } from './dto/get-asset-count-by-time-bucket.dto'; +import { GetAssetCountByTimeBucketDto, TimeGroupEnum } from './dto/get-asset-count-by-time-bucket.dto'; import { GetAssetByTimeBucketDto } from './dto/get-asset-by-time-bucket.dto'; import { AssetCountByUserIdResponseDto } from './response-dto/asset-count-by-user-id-response.dto'; import { CheckExistingAssetsDto } from './dto/check-existing-assets.dto'; @@ -37,7 +37,7 @@ export interface IAssetRepository { getLocationsByUserId(userId: string): Promise; getDetectedObjectsByUserId(userId: string): Promise; getSearchPropertiesByUserId(userId: string): Promise; - getAssetCountByTimeBucket(userId: string, timeBucket: TimeGroupEnum): Promise; + getAssetCountByTimeBucket(userId: string, dto: GetAssetCountByTimeBucketDto): Promise; getAssetCountByUserId(userId: string): Promise; getArchivedAssetCountByUserId(userId: string): Promise; getAssetByTimeBucket(userId: string, getAssetByTimeBucketDto: GetAssetByTimeBucketDto): Promise; @@ -119,36 +119,35 @@ export class AssetRepository implements IAssetRepository { return builder.getMany(); } - async getAssetCountByTimeBucket(userId: string, timeBucket: TimeGroupEnum) { - let result: AssetCountByTimeBucket[] = []; + async getAssetCountByTimeBucket( + userId: string, + dto: GetAssetCountByTimeBucketDto, + ): Promise { + const builder = this.assetRepository + .createQueryBuilder('asset') + .select(`COUNT(asset.id)::int`, 'count') + .where('"ownerId" = :userId', { userId: userId }) + .andWhere('asset.isVisible = true') + .andWhere('asset.isArchived = false'); - if (timeBucket === TimeGroupEnum.Month) { - result = await this.assetRepository - .createQueryBuilder('asset') - .select(`COUNT(asset.id)::int`, 'count') + // Using a parameter for this doesn't work https://github.com/typeorm/typeorm/issues/7308 + if (dto.timeGroup === TimeGroupEnum.Month) { + builder .addSelect(`date_trunc('month', "fileCreatedAt")`, 'timeBucket') - .where('"ownerId" = :userId', { userId: userId }) - .andWhere('asset.resizePath is not NULL') - .andWhere('asset.isVisible = true') - .andWhere('asset.isArchived = false') .groupBy(`date_trunc('month', "fileCreatedAt")`) - .orderBy(`date_trunc('month', "fileCreatedAt")`, 'DESC') - .getRawMany(); - } else if (timeBucket === TimeGroupEnum.Day) { - result = await this.assetRepository - .createQueryBuilder('asset') - .select(`COUNT(asset.id)::int`, 'count') + .orderBy(`date_trunc('month', "fileCreatedAt")`, 'DESC'); + } else if (dto.timeGroup === TimeGroupEnum.Day) { + builder .addSelect(`date_trunc('day', "fileCreatedAt")`, 'timeBucket') - .where('"ownerId" = :userId', { userId: userId }) - .andWhere('asset.resizePath is not NULL') - .andWhere('asset.isVisible = true') - .andWhere('asset.isArchived = false') .groupBy(`date_trunc('day', "fileCreatedAt")`) - .orderBy(`date_trunc('day', "fileCreatedAt")`, 'DESC') - .getRawMany(); + .orderBy(`date_trunc('day', "fileCreatedAt")`, 'DESC'); } - return result; + if (!dto.withoutThumbs) { + builder.andWhere('asset.resizePath is not NULL'); + } + + return builder.getRawMany(); } async getSearchPropertiesByUserId(userId: string): Promise { @@ -231,7 +230,7 @@ export class AssetRepository implements IAssetRepository { return this.assetRepository.find({ where: { ownerId, - resizePath: Not(IsNull()), + resizePath: dto.withoutThumbs ? undefined : Not(IsNull()), isVisible: true, isFavorite: dto.isFavorite, isArchived: dto.isArchived, diff --git a/server/apps/immich/src/api-v1/asset/asset.service.ts b/server/apps/immich/src/api-v1/asset/asset.service.ts index dfc62f0c98..f0fe84d717 100644 --- a/server/apps/immich/src/api-v1/asset/asset.service.ts +++ b/server/apps/immich/src/api-v1/asset/asset.service.ts @@ -533,7 +533,7 @@ export class AssetService { const result = await this._assetRepository.getAssetCountByTimeBucket( getAssetCountByTimeBucketDto.userId || authUser.id, - getAssetCountByTimeBucketDto.timeGroup, + getAssetCountByTimeBucketDto, ); return mapAssetCountByTimeBucket(result); diff --git a/server/apps/immich/src/api-v1/asset/dto/asset-search.dto.ts b/server/apps/immich/src/api-v1/asset/dto/asset-search.dto.ts index 84bededd7b..8a908e746d 100644 --- a/server/apps/immich/src/api-v1/asset/dto/asset-search.dto.ts +++ b/server/apps/immich/src/api-v1/asset/dto/asset-search.dto.ts @@ -16,6 +16,14 @@ export class AssetSearchDto { @Transform(toBoolean) isArchived?: boolean; + /** + * Include assets without thumbnails + */ + @IsOptional() + @IsBoolean() + @Transform(toBoolean) + withoutThumbs?: boolean; + @IsOptional() @IsNumber() skip?: number; diff --git a/server/apps/immich/src/api-v1/asset/dto/get-asset-count-by-time-bucket.dto.ts b/server/apps/immich/src/api-v1/asset/dto/get-asset-count-by-time-bucket.dto.ts index 58104d5d68..d4e6c3c45a 100644 --- a/server/apps/immich/src/api-v1/asset/dto/get-asset-count-by-time-bucket.dto.ts +++ b/server/apps/immich/src/api-v1/asset/dto/get-asset-count-by-time-bucket.dto.ts @@ -1,5 +1,7 @@ import { ApiProperty } from '@nestjs/swagger'; -import { IsNotEmpty, IsOptional, IsUUID } from 'class-validator'; +import { Transform } from 'class-transformer'; +import { IsBoolean, IsNotEmpty, IsOptional, IsUUID } from 'class-validator'; +import { toBoolean } from '../../../utils/transform.util'; export enum TimeGroupEnum { Day = 'day', @@ -19,4 +21,12 @@ export class GetAssetCountByTimeBucketDto { @IsUUID('4') @ApiProperty({ format: 'uuid' }) userId?: string; + + /** + * Include assets without thumbnails + */ + @IsOptional() + @IsBoolean() + @Transform(toBoolean) + withoutThumbs?: boolean; } diff --git a/server/immich-openapi-specs.json b/server/immich-openapi-specs.json index 36a4dad1d8..41c4a4669b 100644 --- a/server/immich-openapi-specs.json +++ b/server/immich-openapi-specs.json @@ -3380,6 +3380,15 @@ "type": "boolean" } }, + { + "name": "withoutThumbs", + "required": false, + "in": "query", + "description": "Include assets without thumbnails", + "schema": { + "type": "boolean" + } + }, { "name": "skip", "required": false, @@ -6221,6 +6230,10 @@ "userId": { "type": "string", "format": "uuid" + }, + "withoutThumbs": { + "type": "boolean", + "description": "Include assets without thumbnails" } }, "required": [ diff --git a/web/src/api/open-api/api.ts b/web/src/api/open-api/api.ts index b777df42c1..6ec6ffcb37 100644 --- a/web/src/api/open-api/api.ts +++ b/web/src/api/open-api/api.ts @@ -1396,6 +1396,12 @@ export interface GetAssetCountByTimeBucketDto { * @memberof GetAssetCountByTimeBucketDto */ 'userId'?: string; + /** + * Include assets without thumbnails + * @type {boolean} + * @memberof GetAssetCountByTimeBucketDto + */ + 'withoutThumbs'?: boolean; } @@ -4999,12 +5005,13 @@ export const AssetApiAxiosParamCreator = function (configuration?: Configuration * @param {string} [userId] * @param {boolean} [isFavorite] * @param {boolean} [isArchived] + * @param {boolean} [withoutThumbs] Include assets without thumbnails * @param {number} [skip] * @param {string} [ifNoneMatch] ETag of data already cached on the client * @param {*} [options] Override http request option. * @throws {RequiredError} */ - getAllAssets: async (userId?: string, isFavorite?: boolean, isArchived?: boolean, skip?: number, ifNoneMatch?: string, options: AxiosRequestConfig = {}): Promise => { + getAllAssets: async (userId?: string, isFavorite?: boolean, isArchived?: boolean, withoutThumbs?: boolean, skip?: number, ifNoneMatch?: string, options: AxiosRequestConfig = {}): Promise => { const localVarPath = `/asset`; // use dummy base URL string because the URL constructor only accepts absolute URLs. const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL); @@ -5038,6 +5045,10 @@ export const AssetApiAxiosParamCreator = function (configuration?: Configuration localVarQueryParameter['isArchived'] = isArchived; } + if (withoutThumbs !== undefined) { + localVarQueryParameter['withoutThumbs'] = withoutThumbs; + } + if (skip !== undefined) { localVarQueryParameter['skip'] = skip; } @@ -5970,13 +5981,14 @@ export const AssetApiFp = function(configuration?: Configuration) { * @param {string} [userId] * @param {boolean} [isFavorite] * @param {boolean} [isArchived] + * @param {boolean} [withoutThumbs] Include assets without thumbnails * @param {number} [skip] * @param {string} [ifNoneMatch] ETag of data already cached on the client * @param {*} [options] Override http request option. * @throws {RequiredError} */ - async getAllAssets(userId?: string, isFavorite?: boolean, isArchived?: boolean, skip?: number, ifNoneMatch?: string, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise>> { - const localVarAxiosArgs = await localVarAxiosParamCreator.getAllAssets(userId, isFavorite, isArchived, skip, ifNoneMatch, options); + async getAllAssets(userId?: string, isFavorite?: boolean, isArchived?: boolean, withoutThumbs?: boolean, skip?: number, ifNoneMatch?: string, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise>> { + const localVarAxiosArgs = await localVarAxiosParamCreator.getAllAssets(userId, isFavorite, isArchived, withoutThumbs, skip, ifNoneMatch, options); return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration); }, /** @@ -6259,13 +6271,14 @@ export const AssetApiFactory = function (configuration?: Configuration, basePath * @param {string} [userId] * @param {boolean} [isFavorite] * @param {boolean} [isArchived] + * @param {boolean} [withoutThumbs] Include assets without thumbnails * @param {number} [skip] * @param {string} [ifNoneMatch] ETag of data already cached on the client * @param {*} [options] Override http request option. * @throws {RequiredError} */ - getAllAssets(userId?: string, isFavorite?: boolean, isArchived?: boolean, skip?: number, ifNoneMatch?: string, options?: any): AxiosPromise> { - return localVarFp.getAllAssets(userId, isFavorite, isArchived, skip, ifNoneMatch, options).then((request) => request(axios, basePath)); + getAllAssets(userId?: string, isFavorite?: boolean, isArchived?: boolean, withoutThumbs?: boolean, skip?: number, ifNoneMatch?: string, options?: any): AxiosPromise> { + return localVarFp.getAllAssets(userId, isFavorite, isArchived, withoutThumbs, skip, ifNoneMatch, options).then((request) => request(axios, basePath)); }, /** * @@ -6627,6 +6640,13 @@ export interface AssetApiGetAllAssetsRequest { */ readonly isArchived?: boolean + /** + * Include assets without thumbnails + * @type {boolean} + * @memberof AssetApiGetAllAssets + */ + readonly withoutThumbs?: boolean + /** * * @type {number} @@ -7071,7 +7091,7 @@ export class AssetApi extends BaseAPI { * @memberof AssetApi */ public getAllAssets(requestParameters: AssetApiGetAllAssetsRequest = {}, options?: AxiosRequestConfig) { - return AssetApiFp(this.configuration).getAllAssets(requestParameters.userId, requestParameters.isFavorite, requestParameters.isArchived, requestParameters.skip, requestParameters.ifNoneMatch, options).then((request) => request(this.axios, this.basePath)); + return AssetApiFp(this.configuration).getAllAssets(requestParameters.userId, requestParameters.isFavorite, requestParameters.isArchived, requestParameters.withoutThumbs, requestParameters.skip, requestParameters.ifNoneMatch, options).then((request) => request(this.axios, this.basePath)); } /** diff --git a/web/src/lib/components/photos-page/asset-grid.svelte b/web/src/lib/components/photos-page/asset-grid.svelte index 0adbe217aa..65b0d09d56 100644 --- a/web/src/lib/components/photos-page/asset-grid.svelte +++ b/web/src/lib/components/photos-page/asset-grid.svelte @@ -29,7 +29,8 @@ const { data: assetCountByTimebucket } = await api.assetApi.getAssetCountByTimeBucket({ getAssetCountByTimeBucketDto: { timeGroup: TimeGroupEnum.Month, - userId: user?.id + userId: user?.id, + withoutThumbs: true } }); bucketInfo = assetCountByTimebucket; diff --git a/web/src/lib/components/shared-components/side-bar/side-bar.svelte b/web/src/lib/components/shared-components/side-bar/side-bar.svelte index 92e8e47191..92db34f738 100644 --- a/web/src/lib/components/shared-components/side-bar/side-bar.svelte +++ b/web/src/lib/components/shared-components/side-bar/side-bar.svelte @@ -30,7 +30,10 @@ const getFavoriteCount = async () => { try { - const { data: assets } = await api.assetApi.getAllAssets({ isFavorite: true }); + const { data: assets } = await api.assetApi.getAllAssets({ + isFavorite: true, + withoutThumbs: true + }); return { favorites: assets.length diff --git a/web/src/routes/(user)/archive/+page.svelte b/web/src/routes/(user)/archive/+page.svelte index bcbb649297..2617d77bf6 100644 --- a/web/src/routes/(user)/archive/+page.svelte +++ b/web/src/routes/(user)/archive/+page.svelte @@ -26,7 +26,10 @@ onMount(async () => { try { - const { data: assets } = await api.assetApi.getAllAssets({ isArchived: true }); + const { data: assets } = await api.assetApi.getAllAssets({ + isArchived: true, + withoutThumbs: true + }); $archivedAsset = assets; } catch { handleError(Error, 'Unable to load archived assets'); diff --git a/web/src/routes/(user)/favorites/+page.svelte b/web/src/routes/(user)/favorites/+page.svelte index 7b2ad8054b..8efeb1b866 100644 --- a/web/src/routes/(user)/favorites/+page.svelte +++ b/web/src/routes/(user)/favorites/+page.svelte @@ -28,7 +28,10 @@ onMount(async () => { try { - const { data: assets } = await api.assetApi.getAllAssets({ isFavorite: true }); + const { data: assets } = await api.assetApi.getAllAssets({ + isFavorite: true, + withoutThumbs: true + }); favorites = assets; } catch { handleError(Error, 'Unable to load favorites');