2023-10-11 04:14:44 +02:00
|
|
|
import { AssetEntity, AssetPathType, AssetType, SystemConfig } from '@app/infra/entities';
|
2023-12-14 17:55:40 +01:00
|
|
|
import { ImmichLogger } from '@app/infra/logger';
|
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
2023-07-01 06:43:24 +02:00
|
|
|
import handlebar from 'handlebars';
|
|
|
|
import * as luxon from 'luxon';
|
|
|
|
import path from 'node:path';
|
|
|
|
import sanitize from 'sanitize-filename';
|
2023-05-22 20:05:06 +02:00
|
|
|
import { getLivePhotoMotionFilename, usePagination } from '../domain.util';
|
2023-05-26 21:43:24 +02:00
|
|
|
import { IEntityJob, JOBS_ASSET_PAGINATION_SIZE } from '../job';
|
2023-10-11 04:14:44 +02:00
|
|
|
import {
|
2023-12-30 16:09:33 +01:00
|
|
|
DatabaseLock,
|
2023-10-23 20:00:31 +02:00
|
|
|
IAlbumRepository,
|
2023-10-11 04:14:44 +02:00
|
|
|
IAssetRepository,
|
2023-12-29 19:41:33 +01:00
|
|
|
ICryptoRepository,
|
2023-12-30 16:09:33 +01:00
|
|
|
IDatabaseRepository,
|
2023-10-11 04:14:44 +02:00
|
|
|
IMoveRepository,
|
|
|
|
IPersonRepository,
|
|
|
|
IStorageRepository,
|
|
|
|
ISystemConfigRepository,
|
|
|
|
IUserRepository,
|
|
|
|
} from '../repositories';
|
2023-10-09 16:25:03 +02:00
|
|
|
import { StorageCore, StorageFolder } from '../storage';
|
2023-07-01 06:43:24 +02:00
|
|
|
import {
|
|
|
|
INITIAL_SYSTEM_CONFIG,
|
|
|
|
supportedDayTokens,
|
|
|
|
supportedHourTokens,
|
|
|
|
supportedMinuteTokens,
|
|
|
|
supportedMonthTokens,
|
|
|
|
supportedSecondTokens,
|
2023-09-28 19:47:31 +02:00
|
|
|
supportedWeekTokens,
|
2023-07-01 06:43:24 +02:00
|
|
|
supportedYearTokens,
|
|
|
|
} from '../system-config';
|
|
|
|
import { SystemConfigCore } from '../system-config/system-config.core';
|
2023-02-25 15:12:03 +01:00
|
|
|
|
2023-05-22 05:18:10 +02:00
|
|
|
export interface MoveAssetMetadata {
|
|
|
|
storageLabel: string | null;
|
|
|
|
filename: string;
|
|
|
|
}
|
|
|
|
|
2023-10-23 20:00:31 +02:00
|
|
|
interface RenderMetadata {
|
|
|
|
asset: AssetEntity;
|
|
|
|
filename: string;
|
|
|
|
extension: string;
|
|
|
|
albumName: string | null;
|
|
|
|
}
|
|
|
|
|
2023-02-25 15:12:03 +01:00
|
|
|
@Injectable()
|
|
|
|
export class StorageTemplateService {
|
2023-12-14 17:55:40 +01:00
|
|
|
private logger = new ImmichLogger(StorageTemplateService.name);
|
2023-07-01 06:43:24 +02:00
|
|
|
private configCore: SystemConfigCore;
|
2023-09-25 17:07:21 +02:00
|
|
|
private storageCore: StorageCore;
|
2023-10-23 20:00:31 +02:00
|
|
|
private template: {
|
|
|
|
compiled: HandlebarsTemplateDelegate<any>;
|
|
|
|
raw: string;
|
|
|
|
needsAlbum: boolean;
|
|
|
|
};
|
2023-02-25 15:12:03 +01:00
|
|
|
|
|
|
|
constructor(
|
2023-10-23 20:00:31 +02:00
|
|
|
@Inject(IAlbumRepository) private albumRepository: IAlbumRepository,
|
2023-02-25 15:12:03 +01:00
|
|
|
@Inject(IAssetRepository) private assetRepository: IAssetRepository,
|
|
|
|
@Inject(ISystemConfigRepository) configRepository: ISystemConfigRepository,
|
|
|
|
@Inject(INITIAL_SYSTEM_CONFIG) config: SystemConfig,
|
2023-10-11 04:14:44 +02:00
|
|
|
@Inject(IMoveRepository) moveRepository: IMoveRepository,
|
|
|
|
@Inject(IPersonRepository) personRepository: IPersonRepository,
|
2023-02-25 15:12:03 +01:00
|
|
|
@Inject(IStorageRepository) private storageRepository: IStorageRepository,
|
2023-05-22 05:18:10 +02:00
|
|
|
@Inject(IUserRepository) private userRepository: IUserRepository,
|
2023-12-29 19:41:33 +01:00
|
|
|
@Inject(ICryptoRepository) private cryptoRepository: ICryptoRepository,
|
2023-12-30 16:09:33 +01:00
|
|
|
@Inject(IDatabaseRepository) private databaseRepository: IDatabaseRepository,
|
2023-02-25 15:12:03 +01:00
|
|
|
) {
|
2023-10-23 20:00:31 +02:00
|
|
|
this.template = this.compile(config.storageTemplate.template);
|
2023-10-09 02:51:03 +02:00
|
|
|
this.configCore = SystemConfigCore.create(configRepository);
|
2023-07-01 06:43:24 +02:00
|
|
|
this.configCore.addValidator((config) => this.validate(config));
|
2023-10-23 20:00:31 +02:00
|
|
|
this.configCore.config$.subscribe((config) => {
|
|
|
|
const template = config.storageTemplate.template;
|
|
|
|
this.logger.debug(`Received config, compiling storage template: ${template}`);
|
|
|
|
this.template = this.compile(template);
|
|
|
|
});
|
2023-12-29 19:41:33 +01:00
|
|
|
this.storageCore = StorageCore.create(
|
|
|
|
assetRepository,
|
|
|
|
moveRepository,
|
|
|
|
personRepository,
|
|
|
|
cryptoRepository,
|
|
|
|
configRepository,
|
|
|
|
storageRepository,
|
|
|
|
);
|
2023-02-25 15:12:03 +01:00
|
|
|
}
|
|
|
|
|
2023-05-26 21:43:24 +02:00
|
|
|
async handleMigrationSingle({ id }: IEntityJob) {
|
2023-12-29 19:41:33 +01:00
|
|
|
const storageTemplateEnabled = (await this.configCore.getConfig()).storageTemplate.enabled;
|
|
|
|
if (!storageTemplateEnabled) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-05-26 21:43:24 +02:00
|
|
|
const [asset] = await this.assetRepository.getByIds([id]);
|
2023-03-28 22:04:11 +02:00
|
|
|
|
2023-10-31 16:01:32 +01:00
|
|
|
const user = await this.userRepository.get(asset.ownerId, {});
|
2023-05-26 21:43:24 +02:00
|
|
|
const storageLabel = user?.storageLabel || null;
|
|
|
|
const filename = asset.originalFileName || asset.id;
|
|
|
|
await this.moveAsset(asset, { storageLabel, filename });
|
|
|
|
|
|
|
|
// move motion part of live photo
|
|
|
|
if (asset.livePhotoVideoId) {
|
|
|
|
const [livePhotoVideo] = await this.assetRepository.getByIds([asset.livePhotoVideoId]);
|
|
|
|
const motionFilename = getLivePhotoMotionFilename(filename, livePhotoVideo.originalPath);
|
|
|
|
await this.moveAsset(livePhotoVideo, { storageLabel, filename: motionFilename });
|
2023-03-28 22:04:11 +02:00
|
|
|
}
|
2023-05-26 21:43:24 +02:00
|
|
|
return true;
|
2023-03-28 22:04:11 +02:00
|
|
|
}
|
|
|
|
|
2023-05-26 14:52:52 +02:00
|
|
|
async handleMigration() {
|
2023-07-01 06:43:24 +02:00
|
|
|
this.logger.log('Starting storage template migration');
|
2023-12-29 19:41:33 +01:00
|
|
|
const storageTemplateEnabled = (await this.configCore.getConfig()).storageTemplate.enabled;
|
|
|
|
if (!storageTemplateEnabled) {
|
|
|
|
this.logger.log('Storage template migration disabled, skipping');
|
|
|
|
return true;
|
|
|
|
}
|
2023-07-01 06:43:24 +02:00
|
|
|
const assetPagination = usePagination(JOBS_ASSET_PAGINATION_SIZE, (pagination) =>
|
|
|
|
this.assetRepository.getAll(pagination),
|
|
|
|
);
|
|
|
|
const users = await this.userRepository.getList();
|
|
|
|
|
|
|
|
for await (const assets of assetPagination) {
|
|
|
|
for (const asset of assets) {
|
|
|
|
const user = users.find((user) => user.id === asset.ownerId);
|
|
|
|
const storageLabel = user?.storageLabel || null;
|
|
|
|
const filename = asset.originalFileName || asset.id;
|
|
|
|
await this.moveAsset(asset, { storageLabel, filename });
|
2023-02-25 15:12:03 +01:00
|
|
|
}
|
|
|
|
}
|
2023-05-26 21:43:24 +02:00
|
|
|
|
2023-07-01 06:43:24 +02:00
|
|
|
this.logger.debug('Cleaning up empty directories...');
|
2023-10-14 19:12:59 +02:00
|
|
|
const libraryFolder = StorageCore.getBaseFolder(StorageFolder.LIBRARY);
|
2023-07-01 06:43:24 +02:00
|
|
|
await this.storageRepository.removeEmptyDirs(libraryFolder);
|
|
|
|
|
|
|
|
this.logger.log('Finished storage template migration');
|
|
|
|
|
2023-05-26 21:43:24 +02:00
|
|
|
return true;
|
2023-02-25 15:12:03 +01:00
|
|
|
}
|
|
|
|
|
2023-05-22 05:18:10 +02:00
|
|
|
async moveAsset(asset: AssetEntity, metadata: MoveAssetMetadata) {
|
2023-10-23 17:52:21 +02:00
|
|
|
if (asset.isReadOnly || asset.isExternal || StorageCore.isAndroidMotionPath(asset.originalPath)) {
|
2023-09-20 13:16:33 +02:00
|
|
|
// External assets are not affected by storage template
|
|
|
|
// TODO: shouldn't this only apply to external assets?
|
feat(server): support for read-only assets and importing existing items in the filesystem (#2715)
* Added read-only flag for assets, endpoint to trigger file import vs upload
* updated fixtures with new property
* if upload is 'read-only', ensure there is no existing asset at the designated originalPath
* added test for file import as well as detecting existing image at read-only destination location
* Added storage service test for a case where it should not move read-only assets
* upload doesn't need the read-only flag available, just importing
* default isReadOnly on import endpoint to true
* formatting fixes
* create-asset dto needs isReadOnly, so set it to false by default on create, updated api generation
* updated code to reflect changes in MR
* fixed read stream promise return type
* new index for originalPath, check for existing path on import, reglardless of user, to prevent duplicates
* refactor: import asset
* chore: open api
* chore: tests
* Added externalPath support for individual users, updated UI to allow this to be set by admin
* added missing var for externalPath in ui
* chore: open api
* fix: compilation issues
* fix: server test
* built api, fixed user-response dto to include externalPath
* reverted accidental commit
* bad commit of duplicate externalPath in user response dto
* fixed tests to include externalPath on expected result
* fix: unit tests
* centralized supported filetypes, perform file type checking of asset and sidecar during file import process
* centralized supported filetype check method to keep regex DRY
* fixed typo
* combined migrations into one
* update api
* Removed externalPath from shared-link code, added column to admin user page whether external paths / import is enabled or not
* update mimetype
* Fixed detect correct mimetype
* revert asset-upload config
* reverted domain.constant
* refactor
* fix mime-type issue
* fix format
---------
Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
2023-06-22 04:33:20 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-12-30 16:09:33 +01:00
|
|
|
return this.databaseRepository.withLock(DatabaseLock.StorageTemplateMigration, async () => {
|
|
|
|
const { id, sidecarPath, originalPath, exifInfo } = asset;
|
|
|
|
const oldPath = originalPath;
|
|
|
|
const newPath = await this.getTemplatePath(asset, metadata);
|
2023-10-11 04:14:44 +02:00
|
|
|
|
2023-12-30 16:09:33 +01:00
|
|
|
if (!exifInfo || !exifInfo.fileSizeInByte) {
|
|
|
|
this.logger.error(`Asset ${id} missing exif info, skipping storage template migration`);
|
|
|
|
return;
|
|
|
|
}
|
2023-12-29 19:41:33 +01:00
|
|
|
|
2023-12-30 16:09:33 +01:00
|
|
|
try {
|
2023-10-11 04:14:44 +02:00
|
|
|
await this.storageCore.moveFile({
|
|
|
|
entityId: id,
|
2023-12-30 16:09:33 +01:00
|
|
|
pathType: AssetPathType.ORIGINAL,
|
|
|
|
oldPath,
|
|
|
|
newPath,
|
|
|
|
assetInfo: { sizeInBytes: exifInfo.fileSizeInByte, checksum: asset.checksum },
|
2023-10-11 04:14:44 +02:00
|
|
|
});
|
2023-12-30 16:09:33 +01:00
|
|
|
if (sidecarPath) {
|
|
|
|
await this.storageCore.moveFile({
|
|
|
|
entityId: id,
|
|
|
|
pathType: AssetPathType.SIDECAR,
|
|
|
|
oldPath: sidecarPath,
|
|
|
|
newPath: `${newPath}.xmp`,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} catch (error: any) {
|
|
|
|
this.logger.error(`Problem applying storage template`, error?.stack, { id: asset.id, oldPath, newPath });
|
2023-02-25 15:12:03 +01:00
|
|
|
}
|
2023-12-30 16:09:33 +01:00
|
|
|
});
|
2023-02-25 15:12:03 +01:00
|
|
|
}
|
2023-07-01 06:43:24 +02:00
|
|
|
|
|
|
|
private async getTemplatePath(asset: AssetEntity, metadata: MoveAssetMetadata): Promise<string> {
|
|
|
|
const { storageLabel, filename } = metadata;
|
|
|
|
|
|
|
|
try {
|
|
|
|
const source = asset.originalPath;
|
|
|
|
const ext = path.extname(source).split('.').pop() as string;
|
|
|
|
const sanitized = sanitize(path.basename(filename, `.${ext}`));
|
2023-10-23 17:52:21 +02:00
|
|
|
const rootPath = StorageCore.getLibraryFolder({ id: asset.ownerId, storageLabel });
|
2023-10-23 20:00:31 +02:00
|
|
|
|
|
|
|
let albumName = null;
|
|
|
|
if (this.template.needsAlbum) {
|
|
|
|
const albums = await this.albumRepository.getByAssetId(asset.ownerId, asset.id);
|
|
|
|
albumName = albums?.[0]?.albumName || null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const storagePath = this.render(this.template.compiled, {
|
|
|
|
asset,
|
|
|
|
filename: sanitized,
|
|
|
|
extension: ext,
|
|
|
|
albumName,
|
|
|
|
});
|
2023-07-01 06:43:24 +02:00
|
|
|
const fullPath = path.normalize(path.join(rootPath, storagePath));
|
|
|
|
let destination = `${fullPath}.${ext}`;
|
|
|
|
|
|
|
|
if (!fullPath.startsWith(rootPath)) {
|
|
|
|
this.logger.warn(`Skipped attempt to access an invalid path: ${fullPath}. Path should start with ${rootPath}`);
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (source === destination) {
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* In case of migrating duplicate filename to a new path, we need to check if it is already migrated
|
|
|
|
* Due to the mechanism of appending +1, +2, +3, etc to the filename
|
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
* Source = upload/abc/def/FullSizeRender+7.heic
|
|
|
|
* Expected Destination = upload/abc/def/FullSizeRender.heic
|
|
|
|
*
|
|
|
|
* The file is already at the correct location, but since there are other FullSizeRender.heic files in the
|
|
|
|
* destination, it was renamed to FullSizeRender+7.heic.
|
|
|
|
*
|
|
|
|
* The lines below will be used to check if the differences between the source and destination is only the
|
|
|
|
* +7 suffix, and if so, it will be considered as already migrated.
|
|
|
|
*/
|
|
|
|
if (source.startsWith(fullPath) && source.endsWith(`.${ext}`)) {
|
|
|
|
const diff = source.replace(fullPath, '').replace(`.${ext}`, '');
|
|
|
|
const hasDuplicationAnnotation = /^\+\d+$/.test(diff);
|
|
|
|
if (hasDuplicationAnnotation) {
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let duplicateCount = 0;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
const exists = await this.storageRepository.checkFileExists(destination);
|
|
|
|
if (!exists) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
duplicateCount++;
|
|
|
|
destination = `${fullPath}+${duplicateCount}.${ext}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
return destination;
|
|
|
|
} catch (error: any) {
|
|
|
|
this.logger.error(`Unable to get template path for ${filename}`, error);
|
|
|
|
return asset.originalPath;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private validate(config: SystemConfig) {
|
|
|
|
try {
|
2023-10-23 20:00:31 +02:00
|
|
|
const { compiled } = this.compile(config.storageTemplate.template);
|
|
|
|
this.render(compiled, {
|
|
|
|
asset: {
|
|
|
|
fileCreatedAt: new Date(),
|
|
|
|
originalPath: '/upload/test/IMG_123.jpg',
|
|
|
|
type: AssetType.IMAGE,
|
|
|
|
id: 'd587e44b-f8c0-4832-9ba3-43268bbf5d4e',
|
|
|
|
} as AssetEntity,
|
|
|
|
filename: 'IMG_123',
|
|
|
|
extension: 'jpg',
|
|
|
|
albumName: 'album',
|
|
|
|
});
|
2023-07-01 06:43:24 +02:00
|
|
|
} catch (e) {
|
|
|
|
this.logger.warn(`Storage template validation failed: ${JSON.stringify(e)}`);
|
|
|
|
throw new Error(`Invalid storage template: ${e}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private compile(template: string) {
|
2023-10-23 20:00:31 +02:00
|
|
|
return {
|
|
|
|
raw: template,
|
|
|
|
compiled: handlebar.compile(template, { knownHelpers: undefined, strict: true }),
|
|
|
|
needsAlbum: template.indexOf('{{album}}') !== -1,
|
|
|
|
};
|
2023-07-01 06:43:24 +02:00
|
|
|
}
|
|
|
|
|
2023-10-23 20:00:31 +02:00
|
|
|
private render(template: HandlebarsTemplateDelegate<any>, options: RenderMetadata) {
|
|
|
|
const { filename, extension, asset, albumName } = options;
|
2023-07-01 06:43:24 +02:00
|
|
|
const substitutions: Record<string, string> = {
|
|
|
|
filename,
|
2023-10-23 20:00:31 +02:00
|
|
|
ext: extension,
|
2023-07-01 06:43:24 +02:00
|
|
|
filetype: asset.type == AssetType.IMAGE ? 'IMG' : 'VID',
|
|
|
|
filetypefull: asset.type == AssetType.IMAGE ? 'IMAGE' : 'VIDEO',
|
2023-10-20 23:17:17 +02:00
|
|
|
assetId: asset.id,
|
2023-10-23 20:00:31 +02:00
|
|
|
//just throw into the root if it doesn't belong to an album
|
|
|
|
album: (albumName && sanitize(albumName.replace(/\.+/g, ''))) || '.',
|
2023-07-01 06:43:24 +02:00
|
|
|
};
|
|
|
|
|
2023-10-03 20:05:44 +02:00
|
|
|
const systemTimeZone = Intl.DateTimeFormat().resolvedOptions().timeZone;
|
|
|
|
const zone = asset.exifInfo?.timeZone || systemTimeZone;
|
|
|
|
const dt = luxon.DateTime.fromJSDate(asset.fileCreatedAt, { zone });
|
2023-07-01 06:43:24 +02:00
|
|
|
|
|
|
|
const dateTokens = [
|
|
|
|
...supportedYearTokens,
|
|
|
|
...supportedMonthTokens,
|
2023-09-28 19:47:31 +02:00
|
|
|
...supportedWeekTokens,
|
2023-07-01 06:43:24 +02:00
|
|
|
...supportedDayTokens,
|
|
|
|
...supportedHourTokens,
|
|
|
|
...supportedMinuteTokens,
|
|
|
|
...supportedSecondTokens,
|
|
|
|
];
|
|
|
|
|
|
|
|
for (const token of dateTokens) {
|
|
|
|
substitutions[token] = dt.toFormat(token);
|
|
|
|
}
|
|
|
|
|
|
|
|
return template(substitutions);
|
|
|
|
}
|
2023-02-25 15:12:03 +01:00
|
|
|
}
|