2023-03-30 21:38:55 +02:00
|
|
|
import { AssetEntity, SystemConfig } from '@app/infra/entities';
|
2023-02-25 15:12:03 +01:00
|
|
|
import { Inject, Injectable, Logger } from '@nestjs/common';
|
|
|
|
import { IAssetRepository } from '../asset/asset.repository';
|
2023-03-25 15:50:57 +01:00
|
|
|
import { APP_MEDIA_LOCATION } from '../domain.constant';
|
2023-03-28 22:04:11 +02:00
|
|
|
import { getLivePhotoMotionFilename } from '../domain.util';
|
|
|
|
import { IAssetJob } from '../job';
|
2023-02-25 15:12:03 +01:00
|
|
|
import { IStorageRepository } from '../storage/storage.repository';
|
|
|
|
import { INITIAL_SYSTEM_CONFIG, ISystemConfigRepository } from '../system-config';
|
2023-05-22 05:18:10 +02:00
|
|
|
import { IUserRepository } from '../user/user.repository';
|
2023-02-25 15:12:03 +01:00
|
|
|
import { StorageTemplateCore } from './storage-template.core';
|
|
|
|
|
2023-05-22 05:18:10 +02:00
|
|
|
export interface MoveAssetMetadata {
|
|
|
|
storageLabel: string | null;
|
|
|
|
filename: string;
|
|
|
|
}
|
|
|
|
|
2023-02-25 15:12:03 +01:00
|
|
|
@Injectable()
|
|
|
|
export class StorageTemplateService {
|
|
|
|
private logger = new Logger(StorageTemplateService.name);
|
|
|
|
private core: StorageTemplateCore;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
@Inject(IAssetRepository) private assetRepository: IAssetRepository,
|
|
|
|
@Inject(ISystemConfigRepository) configRepository: ISystemConfigRepository,
|
|
|
|
@Inject(INITIAL_SYSTEM_CONFIG) config: SystemConfig,
|
|
|
|
@Inject(IStorageRepository) private storageRepository: IStorageRepository,
|
2023-05-22 05:18:10 +02:00
|
|
|
@Inject(IUserRepository) private userRepository: IUserRepository,
|
2023-02-25 15:12:03 +01:00
|
|
|
) {
|
|
|
|
this.core = new StorageTemplateCore(configRepository, config, storageRepository);
|
|
|
|
}
|
|
|
|
|
2023-03-28 22:04:11 +02:00
|
|
|
async handleTemplateMigrationSingle(data: IAssetJob) {
|
|
|
|
const { asset } = data;
|
|
|
|
|
|
|
|
try {
|
2023-05-22 05:18:10 +02:00
|
|
|
const user = await this.userRepository.get(asset.ownerId);
|
|
|
|
const storageLabel = user?.storageLabel || null;
|
2023-04-11 12:23:39 +02:00
|
|
|
const filename = asset.originalFileName || asset.id;
|
2023-05-22 05:18:10 +02:00
|
|
|
await this.moveAsset(asset, { storageLabel, filename });
|
2023-03-28 22:04:11 +02:00
|
|
|
|
|
|
|
// move motion part of live photo
|
|
|
|
if (asset.livePhotoVideoId) {
|
|
|
|
const [livePhotoVideo] = await this.assetRepository.getByIds([asset.livePhotoVideoId]);
|
|
|
|
const motionFilename = getLivePhotoMotionFilename(filename, livePhotoVideo.originalPath);
|
2023-05-22 05:18:10 +02:00
|
|
|
await this.moveAsset(livePhotoVideo, { storageLabel, filename: motionFilename });
|
2023-03-28 22:04:11 +02:00
|
|
|
}
|
|
|
|
} catch (error: any) {
|
|
|
|
this.logger.error('Error running single template migration', error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-25 15:12:03 +01:00
|
|
|
async handleTemplateMigration() {
|
|
|
|
try {
|
|
|
|
console.time('migrating-time');
|
|
|
|
const assets = await this.assetRepository.getAll();
|
2023-05-22 05:18:10 +02:00
|
|
|
const users = await this.userRepository.getList();
|
2023-02-25 15:12:03 +01:00
|
|
|
|
|
|
|
const livePhotoMap: Record<string, AssetEntity> = {};
|
|
|
|
|
|
|
|
for (const asset of assets) {
|
|
|
|
if (asset.livePhotoVideoId) {
|
|
|
|
livePhotoMap[asset.livePhotoVideoId] = asset;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const asset of assets) {
|
|
|
|
const livePhotoParentAsset = livePhotoMap[asset.id];
|
|
|
|
// TODO: remove livePhoto specific stuff once upload is fixed
|
2023-05-22 05:18:10 +02:00
|
|
|
const user = users.find((user) => user.id === asset.ownerId);
|
|
|
|
const storageLabel = user?.storageLabel || null;
|
2023-04-11 12:23:39 +02:00
|
|
|
const filename = asset.originalFileName || livePhotoParentAsset?.originalFileName || asset.id;
|
2023-05-22 05:18:10 +02:00
|
|
|
await this.moveAsset(asset, { storageLabel, filename });
|
2023-02-25 15:12:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
this.logger.debug('Cleaning up empty directories...');
|
2023-03-25 15:50:57 +01:00
|
|
|
await this.storageRepository.removeEmptyDirs(APP_MEDIA_LOCATION);
|
2023-02-25 15:12:03 +01:00
|
|
|
} catch (error: any) {
|
|
|
|
this.logger.error('Error running template migration', error);
|
|
|
|
} finally {
|
|
|
|
console.timeEnd('migrating-time');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: use asset core (once in domain)
|
2023-05-22 05:18:10 +02:00
|
|
|
async moveAsset(asset: AssetEntity, metadata: MoveAssetMetadata) {
|
|
|
|
const destination = await this.core.getTemplatePath(asset, metadata);
|
2023-02-25 15:12:03 +01:00
|
|
|
if (asset.originalPath !== destination) {
|
|
|
|
const source = asset.originalPath;
|
|
|
|
|
|
|
|
try {
|
|
|
|
await this.storageRepository.moveFile(asset.originalPath, destination);
|
|
|
|
try {
|
|
|
|
await this.assetRepository.save({ id: asset.id, originalPath: destination });
|
|
|
|
asset.originalPath = destination;
|
|
|
|
} catch (error: any) {
|
|
|
|
this.logger.warn('Unable to save new originalPath to database, undoing move', error?.stack);
|
|
|
|
await this.storageRepository.moveFile(destination, source);
|
|
|
|
}
|
|
|
|
} catch (error: any) {
|
|
|
|
this.logger.error(`Problem applying storage template`, error?.stack, { id: asset.id, source, destination });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return asset;
|
|
|
|
}
|
|
|
|
}
|