mirror of
https://github.com/immich-app/immich.git
synced 2026-03-03 00:47:02 +00:00
feat: add isTransparent to db (#26413)
This commit is contained in:
@@ -352,6 +352,7 @@ export const columns = {
|
||||
'asset_file.type',
|
||||
'asset_file.isEdited',
|
||||
'asset_file.isProgressive',
|
||||
'asset_file.isTransparent',
|
||||
],
|
||||
authUser: ['user.id', 'user.name', 'user.email', 'user.isAdmin', 'user.quotaUsageInBytes', 'user.quotaSizeInBytes'],
|
||||
authApiKey: ['api_key.id', 'api_key.permissions'],
|
||||
|
||||
@@ -216,7 +216,8 @@ select
|
||||
"asset_file"."path",
|
||||
"asset_file"."type",
|
||||
"asset_file"."isEdited",
|
||||
"asset_file"."isProgressive"
|
||||
"asset_file"."isProgressive",
|
||||
"asset_file"."isTransparent"
|
||||
from
|
||||
"asset_file"
|
||||
where
|
||||
|
||||
@@ -903,7 +903,10 @@ export class AssetRepository {
|
||||
}
|
||||
|
||||
async upsertFile(
|
||||
file: Pick<Insertable<AssetFileTable>, 'assetId' | 'path' | 'type' | 'isEdited' | 'isProgressive'>,
|
||||
file: Pick<
|
||||
Insertable<AssetFileTable>,
|
||||
'assetId' | 'path' | 'type' | 'isEdited' | 'isProgressive' | 'isTransparent'
|
||||
>,
|
||||
): Promise<void> {
|
||||
await this.db
|
||||
.insertInto('asset_file')
|
||||
@@ -917,7 +920,10 @@ export class AssetRepository {
|
||||
}
|
||||
|
||||
async upsertFiles(
|
||||
files: Pick<Insertable<AssetFileTable>, 'assetId' | 'path' | 'type' | 'isEdited' | 'isProgressive'>[],
|
||||
files: Pick<
|
||||
Insertable<AssetFileTable>,
|
||||
'assetId' | 'path' | 'type' | 'isEdited' | 'isProgressive' | 'isTransparent'
|
||||
>[],
|
||||
): Promise<void> {
|
||||
if (files.length === 0) {
|
||||
return;
|
||||
@@ -930,6 +936,7 @@ export class AssetRepository {
|
||||
oc.columns(['assetId', 'type', 'isEdited']).doUpdateSet((eb) => ({
|
||||
path: eb.ref('excluded.path'),
|
||||
isProgressive: eb.ref('excluded.isProgressive'),
|
||||
isTransparent: eb.ref('excluded.isTransparent'),
|
||||
})),
|
||||
)
|
||||
.execute();
|
||||
|
||||
@@ -0,0 +1,9 @@
|
||||
import { Kysely, sql } from 'kysely';
|
||||
|
||||
export async function up(db: Kysely<any>): Promise<void> {
|
||||
await sql`ALTER TABLE "asset_file" ADD "isTransparent" boolean NOT NULL DEFAULT false;`.execute(db);
|
||||
}
|
||||
|
||||
export async function down(db: Kysely<any>): Promise<void> {
|
||||
await sql`ALTER TABLE "asset_file" DROP COLUMN "isTransparent";`.execute(db);
|
||||
}
|
||||
@@ -43,4 +43,7 @@ export class AssetFileTable {
|
||||
|
||||
@Column({ type: 'boolean', default: false })
|
||||
isProgressive!: Generated<boolean>;
|
||||
|
||||
@Column({ type: 'boolean', default: false })
|
||||
isTransparent!: Generated<boolean>;
|
||||
}
|
||||
|
||||
@@ -468,6 +468,7 @@ describe(MediaService.name, () => {
|
||||
path: expect.any(String),
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
assetId: asset.id,
|
||||
@@ -475,6 +476,7 @@ describe(MediaService.name, () => {
|
||||
path: expect.any(String),
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
expect(mocks.asset.update).toHaveBeenCalledWith({ id: asset.id, thumbhash: thumbhashBuffer });
|
||||
@@ -509,6 +511,7 @@ describe(MediaService.name, () => {
|
||||
path: expect.any(String),
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
assetId: asset.id,
|
||||
@@ -516,6 +519,7 @@ describe(MediaService.name, () => {
|
||||
path: expect.any(String),
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
});
|
||||
@@ -549,6 +553,7 @@ describe(MediaService.name, () => {
|
||||
path: expect.any(String),
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
assetId: asset.id,
|
||||
@@ -556,6 +561,7 @@ describe(MediaService.name, () => {
|
||||
path: expect.any(String),
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
});
|
||||
@@ -771,10 +777,12 @@ describe(MediaService.name, () => {
|
||||
expect.objectContaining({
|
||||
type: AssetFileType.Preview,
|
||||
isProgressive: true,
|
||||
isTransparent: false,
|
||||
}),
|
||||
expect.objectContaining({
|
||||
type: AssetFileType.Thumbnail,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
}),
|
||||
]);
|
||||
});
|
||||
@@ -808,10 +816,12 @@ describe(MediaService.name, () => {
|
||||
expect.objectContaining({
|
||||
type: AssetFileType.Preview,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
}),
|
||||
expect.objectContaining({
|
||||
type: AssetFileType.Thumbnail,
|
||||
isProgressive: true,
|
||||
isTransparent: false,
|
||||
}),
|
||||
]);
|
||||
});
|
||||
@@ -830,10 +840,12 @@ describe(MediaService.name, () => {
|
||||
expect.objectContaining({
|
||||
type: AssetFileType.Preview,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
}),
|
||||
expect.objectContaining({
|
||||
type: AssetFileType.Thumbnail,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
}),
|
||||
]);
|
||||
});
|
||||
@@ -3583,6 +3595,7 @@ describe(MediaService.name, () => {
|
||||
path: '/new/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
assetId: asset.id,
|
||||
@@ -3590,6 +3603,7 @@ describe(MediaService.name, () => {
|
||||
path: '/new/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
|
||||
@@ -3600,6 +3614,7 @@ describe(MediaService.name, () => {
|
||||
type: AssetFileType.Preview,
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
assetId: 'asset-id',
|
||||
@@ -3607,6 +3622,7 @@ describe(MediaService.name, () => {
|
||||
type: AssetFileType.Thumbnail,
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
expect(mocks.asset.deleteFiles).not.toHaveBeenCalled();
|
||||
@@ -3624,6 +3640,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
id: 'file-2',
|
||||
@@ -3632,6 +3649,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
],
|
||||
};
|
||||
@@ -3643,6 +3661,7 @@ describe(MediaService.name, () => {
|
||||
path: '/new/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
assetId: asset.id,
|
||||
@@ -3650,6 +3669,7 @@ describe(MediaService.name, () => {
|
||||
path: '/new/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
|
||||
@@ -3660,6 +3680,7 @@ describe(MediaService.name, () => {
|
||||
type: AssetFileType.Preview,
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
assetId: 'asset-id',
|
||||
@@ -3667,6 +3688,7 @@ describe(MediaService.name, () => {
|
||||
type: AssetFileType.Thumbnail,
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
expect(mocks.asset.deleteFiles).not.toHaveBeenCalled();
|
||||
@@ -3687,6 +3709,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
id: 'file-2',
|
||||
@@ -3695,6 +3718,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
],
|
||||
};
|
||||
@@ -3710,6 +3734,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
id: 'file-2',
|
||||
@@ -3718,6 +3743,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
expect(mocks.job.queue).toHaveBeenCalledWith({
|
||||
@@ -3737,6 +3763,7 @@ describe(MediaService.name, () => {
|
||||
path: '/same/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
id: 'file-2',
|
||||
@@ -3745,6 +3772,7 @@ describe(MediaService.name, () => {
|
||||
path: '/same/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
],
|
||||
};
|
||||
@@ -3756,6 +3784,7 @@ describe(MediaService.name, () => {
|
||||
path: '/same/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
assetId: asset.id,
|
||||
@@ -3763,6 +3792,7 @@ describe(MediaService.name, () => {
|
||||
path: '/same/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
|
||||
@@ -3782,6 +3812,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
id: 'file-2',
|
||||
@@ -3790,6 +3821,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
],
|
||||
};
|
||||
@@ -3801,6 +3833,7 @@ describe(MediaService.name, () => {
|
||||
path: '/new/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
}, // replace
|
||||
{
|
||||
assetId: asset.id,
|
||||
@@ -3808,6 +3841,7 @@ describe(MediaService.name, () => {
|
||||
path: '/new/fullsize.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
}, // new
|
||||
]);
|
||||
|
||||
@@ -3818,6 +3852,7 @@ describe(MediaService.name, () => {
|
||||
type: AssetFileType.Preview,
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
assetId: 'asset-id',
|
||||
@@ -3825,6 +3860,7 @@ describe(MediaService.name, () => {
|
||||
type: AssetFileType.FullSize,
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
expect(mocks.asset.deleteFiles).toHaveBeenCalledWith([
|
||||
@@ -3835,6 +3871,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
expect(mocks.job.queue).toHaveBeenCalledWith({
|
||||
@@ -3867,6 +3904,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
],
|
||||
};
|
||||
@@ -3882,6 +3920,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
expect(mocks.job.queue).toHaveBeenCalledWith({
|
||||
@@ -3901,6 +3940,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
id: 'file-2',
|
||||
@@ -3909,6 +3949,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
],
|
||||
};
|
||||
@@ -3920,6 +3961,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/preview.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: true,
|
||||
isTransparent: false,
|
||||
},
|
||||
{
|
||||
assetId: asset.id,
|
||||
@@ -3927,6 +3969,7 @@ describe(MediaService.name, () => {
|
||||
path: '/old/thumbnail.jpg',
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
|
||||
@@ -3937,6 +3980,7 @@ describe(MediaService.name, () => {
|
||||
type: AssetFileType.Preview,
|
||||
isEdited: false,
|
||||
isProgressive: true,
|
||||
isTransparent: false,
|
||||
},
|
||||
]);
|
||||
expect(mocks.asset.deleteFiles).not.toHaveBeenCalled();
|
||||
|
||||
@@ -52,6 +52,7 @@ interface UpsertFileOptions {
|
||||
path: string;
|
||||
isEdited: boolean;
|
||||
isProgressive: boolean;
|
||||
isTransparent: boolean;
|
||||
}
|
||||
|
||||
type ThumbnailAsset = NonNullable<Awaited<ReturnType<AssetJobRepository['getForGenerateThumbnailJob']>>>;
|
||||
@@ -321,12 +322,14 @@ export class MediaService extends BaseService {
|
||||
format: previewFormat,
|
||||
isEdited: useEdits,
|
||||
isProgressive: !!image.preview.progressive && previewFormat !== ImageFormat.Webp,
|
||||
isTransparent,
|
||||
});
|
||||
const thumbnailFile = this.getImageFile(asset, {
|
||||
fileType: AssetFileType.Thumbnail,
|
||||
format: thumbnailFormat,
|
||||
isEdited: useEdits,
|
||||
isProgressive: !!image.thumbnail.progressive && thumbnailFormat !== ImageFormat.Webp,
|
||||
isTransparent,
|
||||
});
|
||||
this.storageCore.ensureFolders(previewFile.path);
|
||||
|
||||
@@ -350,6 +353,7 @@ export class MediaService extends BaseService {
|
||||
format: fullsizeFormat,
|
||||
isEdited: useEdits,
|
||||
isProgressive: !!image.fullsize.progressive && fullsizeFormat !== ImageFormat.Webp,
|
||||
isTransparent,
|
||||
});
|
||||
const fullsizeOptions = {
|
||||
...baseOptions,
|
||||
@@ -364,6 +368,7 @@ export class MediaService extends BaseService {
|
||||
format: extracted.format,
|
||||
isEdited: false,
|
||||
isProgressive: !!image.fullsize.progressive && image.fullsize.format !== ImageFormat.Webp,
|
||||
isTransparent,
|
||||
});
|
||||
this.storageCore.ensureFolders(fullsizeFile.path);
|
||||
|
||||
@@ -510,12 +515,14 @@ export class MediaService extends BaseService {
|
||||
format: image.preview.format,
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
});
|
||||
const thumbnailFile = this.getImageFile(asset, {
|
||||
fileType: AssetFileType.Thumbnail,
|
||||
format: image.thumbnail.format,
|
||||
isEdited: false,
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
});
|
||||
this.storageCore.ensureFolders(previewFile.path);
|
||||
|
||||
@@ -802,7 +809,10 @@ export class MediaService extends BaseService {
|
||||
}
|
||||
}
|
||||
|
||||
private async syncFiles(oldFiles: (AssetFile & { isProgressive: boolean })[], newFiles: UpsertFileOptions[]) {
|
||||
private async syncFiles(
|
||||
oldFiles: (AssetFile & { isProgressive: boolean; isTransparent: boolean })[],
|
||||
newFiles: UpsertFileOptions[],
|
||||
) {
|
||||
const toUpsert: UpsertFileOptions[] = [];
|
||||
const pathsToDelete: string[] = [];
|
||||
const toDelete = new Set(oldFiles);
|
||||
@@ -814,7 +824,11 @@ export class MediaService extends BaseService {
|
||||
}
|
||||
|
||||
// upsert new file path
|
||||
if (existingFile?.path !== newFile.path || existingFile.isProgressive !== newFile.isProgressive) {
|
||||
if (
|
||||
existingFile?.path !== newFile.path ||
|
||||
existingFile.isProgressive !== newFile.isProgressive ||
|
||||
existingFile.isTransparent !== newFile.isTransparent
|
||||
) {
|
||||
toUpsert.push(newFile);
|
||||
|
||||
// delete old file from disk
|
||||
@@ -882,7 +896,10 @@ export class MediaService extends BaseService {
|
||||
}
|
||||
}
|
||||
|
||||
private getImageFile(asset: ThumbnailPathEntity, options: ImagePathOptions & { isProgressive: boolean }) {
|
||||
private getImageFile(
|
||||
asset: ThumbnailPathEntity,
|
||||
options: ImagePathOptions & { isProgressive: boolean; isTransparent: boolean },
|
||||
) {
|
||||
const path = StorageCore.getImagePath(asset, options);
|
||||
return {
|
||||
assetId: asset.id,
|
||||
@@ -890,6 +907,7 @@ export class MediaService extends BaseService {
|
||||
path,
|
||||
isEdited: options.isEdited,
|
||||
isProgressive: options.isProgressive,
|
||||
isTransparent: options.isTransparent,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
@@ -26,6 +26,7 @@ export class AssetFileFactory {
|
||||
path: `/data/12/34/thumbs/${id.slice(0, 2)}/${id.slice(2, 4)}/${id}${isEdited ? '_edited' : ''}.jpg`,
|
||||
updateId: newUuidV7(),
|
||||
isProgressive: false,
|
||||
isTransparent: false,
|
||||
isEdited,
|
||||
...dto,
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user