2023-12-14 17:55:40 +01:00
|
|
|
import { BadRequestException, Inject, Injectable } from '@nestjs/common';
|
2024-03-17 20:16:02 +01:00
|
|
|
import { OnEvent } from '@nestjs/event-emitter';
|
2024-03-14 06:52:30 +01:00
|
|
|
import { Trie } from 'mnemonist';
|
2023-09-20 13:16:33 +02:00
|
|
|
import { R_OK } from 'node:constants';
|
2024-02-02 04:18:00 +01:00
|
|
|
import { EventEmitter } from 'node:events';
|
2023-09-20 13:16:33 +02:00
|
|
|
import { Stats } from 'node:fs';
|
2024-02-02 04:18:00 +01:00
|
|
|
import path, { basename, parse } from 'node:path';
|
2024-01-31 17:26:51 +01:00
|
|
|
import picomatch from 'picomatch';
|
2024-03-20 19:32:04 +01:00
|
|
|
import { mimeTypes } from 'src/domain/domain.constant';
|
|
|
|
import { JOBS_ASSET_PAGINATION_SIZE, JobName } from 'src/domain/job/job.constants';
|
|
|
|
import { IBaseJob, IEntityJob, ILibraryFileJob, ILibraryRefreshJob } from 'src/domain/job/job.interface';
|
2023-09-20 13:16:33 +02:00
|
|
|
import {
|
|
|
|
CreateLibraryDto,
|
|
|
|
LibraryResponseDto,
|
|
|
|
LibraryStatsResponseDto,
|
|
|
|
ScanLibraryDto,
|
2024-02-29 19:35:37 +01:00
|
|
|
SearchLibraryDto,
|
2023-09-20 13:16:33 +02:00
|
|
|
UpdateLibraryDto,
|
2024-02-20 16:53:12 +01:00
|
|
|
ValidateLibraryDto,
|
|
|
|
ValidateLibraryImportPathResponseDto,
|
|
|
|
ValidateLibraryResponseDto,
|
2023-10-09 05:52:12 +02:00
|
|
|
mapLibrary,
|
2024-03-20 19:32:04 +01:00
|
|
|
} from 'src/domain/library/library.dto';
|
|
|
|
import { IAssetRepository, WithProperty } from 'src/domain/repositories/asset.repository';
|
|
|
|
import { InternalEvent, InternalEventMap } from 'src/domain/repositories/communication.repository';
|
|
|
|
import { ICryptoRepository } from 'src/domain/repositories/crypto.repository';
|
|
|
|
import { DatabaseLock, IDatabaseRepository } from 'src/domain/repositories/database.repository';
|
|
|
|
import { IJobRepository, JobStatus } from 'src/domain/repositories/job.repository';
|
|
|
|
import { ILibraryRepository } from 'src/domain/repositories/library.repository';
|
|
|
|
import { IStorageRepository, StorageEventType } from 'src/domain/repositories/storage.repository';
|
|
|
|
import { ISystemConfigRepository } from 'src/domain/repositories/system-config.repository';
|
|
|
|
import { StorageCore } from 'src/domain/storage/storage.core';
|
|
|
|
import { SystemConfigCore } from 'src/domain/system-config/system-config.core';
|
|
|
|
import { AssetType } from 'src/infra/entities/asset.entity';
|
|
|
|
import { LibraryEntity, LibraryType } from 'src/infra/entities/library.entity';
|
|
|
|
import { ImmichLogger } from 'src/infra/logger';
|
2024-03-20 21:04:03 +01:00
|
|
|
import { handlePromiseError, usePagination } from 'src/utils';
|
|
|
|
import { validateCronExpression } from 'src/validation';
|
2023-09-20 13:16:33 +02:00
|
|
|
|
2024-03-14 06:52:30 +01:00
|
|
|
const LIBRARY_SCAN_BATCH_SIZE = 5000;
|
|
|
|
|
2023-09-20 13:16:33 +02:00
|
|
|
@Injectable()
|
2024-01-31 09:15:54 +01:00
|
|
|
export class LibraryService extends EventEmitter {
|
2023-12-14 17:55:40 +01:00
|
|
|
readonly logger = new ImmichLogger(LibraryService.name);
|
2023-10-31 21:19:12 +01:00
|
|
|
private configCore: SystemConfigCore;
|
2024-01-31 09:15:54 +01:00
|
|
|
private watchLibraries = false;
|
2024-03-07 18:36:53 +01:00
|
|
|
private watchLock = false;
|
2024-03-05 23:23:06 +01:00
|
|
|
private watchers: Record<string, () => Promise<void>> = {};
|
2024-01-31 09:15:54 +01:00
|
|
|
|
2023-09-20 13:16:33 +02:00
|
|
|
constructor(
|
|
|
|
@Inject(IAssetRepository) private assetRepository: IAssetRepository,
|
2023-10-31 21:19:12 +01:00
|
|
|
@Inject(ISystemConfigRepository) configRepository: ISystemConfigRepository,
|
2023-09-20 13:16:33 +02:00
|
|
|
@Inject(ICryptoRepository) private cryptoRepository: ICryptoRepository,
|
|
|
|
@Inject(IJobRepository) private jobRepository: IJobRepository,
|
|
|
|
@Inject(ILibraryRepository) private repository: ILibraryRepository,
|
|
|
|
@Inject(IStorageRepository) private storageRepository: IStorageRepository,
|
2024-03-07 18:36:53 +01:00
|
|
|
@Inject(IDatabaseRepository) private databaseRepository: IDatabaseRepository,
|
2023-09-20 13:16:33 +02:00
|
|
|
) {
|
2024-01-31 09:15:54 +01:00
|
|
|
super();
|
2023-10-31 21:19:12 +01:00
|
|
|
this.configCore = SystemConfigCore.create(configRepository);
|
|
|
|
}
|
|
|
|
|
|
|
|
async init() {
|
|
|
|
const config = await this.configCore.getConfig();
|
2024-03-07 18:36:53 +01:00
|
|
|
|
2024-01-31 17:26:51 +01:00
|
|
|
const { watch, scan } = config.library;
|
2024-03-07 18:36:53 +01:00
|
|
|
|
|
|
|
// This ensures that library watching only occurs in one microservice
|
|
|
|
// TODO: we could make the lock be per-library instead of global
|
|
|
|
this.watchLock = await this.databaseRepository.tryLock(DatabaseLock.LibraryWatch);
|
|
|
|
|
|
|
|
this.watchLibraries = this.watchLock && watch.enabled;
|
|
|
|
|
2023-10-31 21:19:12 +01:00
|
|
|
this.jobRepository.addCronJob(
|
|
|
|
'libraryScan',
|
2024-01-31 17:26:51 +01:00
|
|
|
scan.cronExpression,
|
2024-03-05 23:23:06 +01:00
|
|
|
() =>
|
|
|
|
handlePromiseError(
|
|
|
|
this.jobRepository.queue({ name: JobName.LIBRARY_QUEUE_SCAN_ALL, data: { force: false } }),
|
|
|
|
this.logger,
|
|
|
|
),
|
2024-01-31 17:26:51 +01:00
|
|
|
scan.enabled,
|
2023-10-31 21:19:12 +01:00
|
|
|
);
|
|
|
|
|
2024-01-31 09:15:54 +01:00
|
|
|
if (this.watchLibraries) {
|
|
|
|
await this.watchAll();
|
|
|
|
}
|
|
|
|
|
2024-03-05 23:23:06 +01:00
|
|
|
this.configCore.config$.subscribe(({ library }) => {
|
2024-01-31 17:26:51 +01:00
|
|
|
this.jobRepository.updateCronJob('libraryScan', library.scan.cronExpression, library.scan.enabled);
|
2024-01-31 09:15:54 +01:00
|
|
|
|
2024-01-31 17:26:51 +01:00
|
|
|
if (library.watch.enabled !== this.watchLibraries) {
|
2024-03-07 18:36:53 +01:00
|
|
|
// Watch configuration changed, update accordingly
|
2024-01-31 17:26:51 +01:00
|
|
|
this.watchLibraries = library.watch.enabled;
|
2024-03-05 23:23:06 +01:00
|
|
|
handlePromiseError(this.watchLibraries ? this.watchAll() : this.unwatchAll(), this.logger);
|
2024-01-31 09:15:54 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-03-17 20:16:02 +01:00
|
|
|
@OnEvent(InternalEvent.VALIDATE_CONFIG)
|
|
|
|
validateConfig({ newConfig }: InternalEventMap[InternalEvent.VALIDATE_CONFIG]) {
|
|
|
|
const { scan } = newConfig.library;
|
|
|
|
if (!validateCronExpression(scan.cronExpression)) {
|
|
|
|
throw new Error(`Invalid cron expression ${scan.cronExpression}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-31 09:15:54 +01:00
|
|
|
private async watch(id: string): Promise<boolean> {
|
|
|
|
if (!this.watchLibraries) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const library = await this.findOrFail(id);
|
|
|
|
|
|
|
|
if (library.type !== LibraryType.EXTERNAL) {
|
|
|
|
throw new BadRequestException('Can only watch external libraries');
|
|
|
|
} else if (library.importPaths.length === 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
await this.unwatch(id);
|
|
|
|
|
|
|
|
this.logger.log(`Starting to watch library ${library.id} with import path(s) ${library.importPaths}`);
|
|
|
|
|
|
|
|
const matcher = picomatch(`**/*{${mimeTypes.getSupportedFileExtensions().join(',')}}`, {
|
|
|
|
nocase: true,
|
|
|
|
ignore: library.exclusionPatterns,
|
|
|
|
});
|
|
|
|
|
2024-02-13 14:48:47 +01:00
|
|
|
let _resolve: () => void;
|
|
|
|
const ready$ = new Promise<void>((resolve) => (_resolve = resolve));
|
|
|
|
|
|
|
|
this.watchers[id] = this.storageRepository.watch(
|
|
|
|
library.importPaths,
|
|
|
|
{
|
2024-02-28 21:20:10 +01:00
|
|
|
usePolling: false,
|
2024-02-13 14:48:47 +01:00
|
|
|
ignoreInitial: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
onReady: () => _resolve(),
|
2024-03-05 23:23:06 +01:00
|
|
|
onAdd: (path) => {
|
|
|
|
const handler = async () => {
|
|
|
|
this.logger.debug(`File add event received for ${path} in library ${library.id}}`);
|
|
|
|
if (matcher(path)) {
|
|
|
|
await this.scanAssets(library.id, [path], library.ownerId, false);
|
|
|
|
}
|
2024-03-07 18:36:53 +01:00
|
|
|
this.emit(StorageEventType.ADD, path);
|
2024-03-05 23:23:06 +01:00
|
|
|
};
|
|
|
|
return handlePromiseError(handler(), this.logger);
|
2024-02-13 14:48:47 +01:00
|
|
|
},
|
2024-03-05 23:23:06 +01:00
|
|
|
onChange: (path) => {
|
|
|
|
const handler = async () => {
|
|
|
|
this.logger.debug(`Detected file change for ${path} in library ${library.id}`);
|
|
|
|
if (matcher(path)) {
|
|
|
|
// Note: if the changed file was not previously imported, it will be imported now.
|
|
|
|
await this.scanAssets(library.id, [path], library.ownerId, false);
|
|
|
|
}
|
2024-03-07 18:36:53 +01:00
|
|
|
this.emit(StorageEventType.CHANGE, path);
|
2024-03-05 23:23:06 +01:00
|
|
|
};
|
|
|
|
return handlePromiseError(handler(), this.logger);
|
2024-02-13 14:48:47 +01:00
|
|
|
},
|
2024-03-05 23:23:06 +01:00
|
|
|
onUnlink: (path) => {
|
|
|
|
const handler = async () => {
|
|
|
|
this.logger.debug(`Detected deleted file at ${path} in library ${library.id}`);
|
|
|
|
const asset = await this.assetRepository.getByLibraryIdAndOriginalPath(library.id, path);
|
|
|
|
if (asset && matcher(path)) {
|
2024-03-20 03:42:10 +01:00
|
|
|
await this.assetRepository.update({ id: asset.id, isOffline: true });
|
2024-03-05 23:23:06 +01:00
|
|
|
}
|
2024-03-07 18:36:53 +01:00
|
|
|
this.emit(StorageEventType.UNLINK, path);
|
2024-03-05 23:23:06 +01:00
|
|
|
};
|
|
|
|
return handlePromiseError(handler(), this.logger);
|
2024-02-13 14:48:47 +01:00
|
|
|
},
|
|
|
|
onError: (error) => {
|
|
|
|
this.logger.error(`Library watcher for library ${library.id} encountered error: ${error}`);
|
2024-03-07 18:36:53 +01:00
|
|
|
this.emit(StorageEventType.ERROR, error);
|
2024-02-13 14:48:47 +01:00
|
|
|
},
|
|
|
|
},
|
|
|
|
);
|
2024-01-31 09:15:54 +01:00
|
|
|
|
|
|
|
// Wait for the watcher to initialize before returning
|
2024-02-13 14:48:47 +01:00
|
|
|
await ready$;
|
2024-01-31 09:15:54 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
async unwatch(id: string) {
|
2024-02-13 14:48:47 +01:00
|
|
|
if (this.watchers[id]) {
|
2024-01-31 09:15:54 +01:00
|
|
|
await this.watchers[id]();
|
|
|
|
delete this.watchers[id];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-07 18:36:53 +01:00
|
|
|
async teardown() {
|
|
|
|
await this.unwatchAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
private async unwatchAll() {
|
|
|
|
if (!this.watchLock) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-01-31 09:15:54 +01:00
|
|
|
for (const id in this.watchers) {
|
|
|
|
await this.unwatch(id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async watchAll() {
|
2024-03-07 18:36:53 +01:00
|
|
|
if (!this.watchLock) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-01-31 09:15:54 +01:00
|
|
|
const libraries = await this.repository.getAll(false, LibraryType.EXTERNAL);
|
|
|
|
|
|
|
|
for (const library of libraries) {
|
|
|
|
await this.watch(library.id);
|
|
|
|
}
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-03-18 21:59:53 +01:00
|
|
|
async getStatistics(id: string): Promise<LibraryStatsResponseDto> {
|
|
|
|
await this.findOrFail(id);
|
2023-09-20 13:16:33 +02:00
|
|
|
return this.repository.getStatistics(id);
|
|
|
|
}
|
|
|
|
|
2024-03-18 21:59:53 +01:00
|
|
|
async get(id: string): Promise<LibraryResponseDto> {
|
2023-09-20 13:16:33 +02:00
|
|
|
const library = await this.findOrFail(id);
|
|
|
|
return mapLibrary(library);
|
|
|
|
}
|
|
|
|
|
2024-03-18 21:59:53 +01:00
|
|
|
async getAll(dto: SearchLibraryDto): Promise<LibraryResponseDto[]> {
|
2024-02-29 19:35:37 +01:00
|
|
|
const libraries = await this.repository.getAll(false, dto.type);
|
|
|
|
return libraries.map((library) => mapLibrary(library));
|
|
|
|
}
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
async handleQueueCleanup(): Promise<JobStatus> {
|
2023-09-20 13:16:33 +02:00
|
|
|
this.logger.debug('Cleaning up any pending library deletions');
|
|
|
|
const pendingDeletion = await this.repository.getAllDeleted();
|
2024-01-01 21:45:42 +01:00
|
|
|
await this.jobRepository.queueAll(
|
|
|
|
pendingDeletion.map((libraryToDelete) => ({ name: JobName.LIBRARY_DELETE, data: { id: libraryToDelete.id } })),
|
|
|
|
);
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.SUCCESS;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-03-18 21:59:53 +01:00
|
|
|
async create(dto: CreateLibraryDto): Promise<LibraryResponseDto> {
|
2023-09-20 13:16:33 +02:00
|
|
|
switch (dto.type) {
|
2024-02-02 04:18:00 +01:00
|
|
|
case LibraryType.EXTERNAL: {
|
2023-09-20 13:16:33 +02:00
|
|
|
if (!dto.name) {
|
|
|
|
dto.name = 'New External Library';
|
|
|
|
}
|
|
|
|
break;
|
2024-02-02 04:18:00 +01:00
|
|
|
}
|
|
|
|
case LibraryType.UPLOAD: {
|
2023-09-20 13:16:33 +02:00
|
|
|
if (!dto.name) {
|
|
|
|
dto.name = 'New Upload Library';
|
|
|
|
}
|
|
|
|
if (dto.importPaths && dto.importPaths.length > 0) {
|
|
|
|
throw new BadRequestException('Upload libraries cannot have import paths');
|
|
|
|
}
|
|
|
|
if (dto.exclusionPatterns && dto.exclusionPatterns.length > 0) {
|
|
|
|
throw new BadRequestException('Upload libraries cannot have exclusion patterns');
|
|
|
|
}
|
2024-01-31 09:15:54 +01:00
|
|
|
if (dto.isWatched) {
|
|
|
|
throw new BadRequestException('Upload libraries cannot be watched');
|
|
|
|
}
|
2023-09-20 13:16:33 +02:00
|
|
|
break;
|
2024-02-02 04:18:00 +01:00
|
|
|
}
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const library = await this.repository.create({
|
2024-03-18 21:59:53 +01:00
|
|
|
ownerId: dto.ownerId,
|
2023-09-20 13:16:33 +02:00
|
|
|
name: dto.name,
|
|
|
|
type: dto.type,
|
|
|
|
importPaths: dto.importPaths ?? [],
|
|
|
|
exclusionPatterns: dto.exclusionPatterns ?? [],
|
|
|
|
isVisible: dto.isVisible ?? true,
|
|
|
|
});
|
|
|
|
|
2024-03-18 21:59:53 +01:00
|
|
|
this.logger.log(`Creating ${dto.type} library for ${dto.ownerId}}`);
|
2024-01-31 09:15:54 +01:00
|
|
|
|
2024-03-07 18:36:53 +01:00
|
|
|
if (dto.type === LibraryType.EXTERNAL) {
|
2024-01-31 09:15:54 +01:00
|
|
|
await this.watch(library.id);
|
|
|
|
}
|
|
|
|
|
2023-09-20 13:16:33 +02:00
|
|
|
return mapLibrary(library);
|
|
|
|
}
|
|
|
|
|
2024-01-31 09:15:54 +01:00
|
|
|
private async scanAssets(libraryId: string, assetPaths: string[], ownerId: string, force = false) {
|
2024-03-14 14:43:05 +01:00
|
|
|
this.logger.verbose(`Queuing refresh of ${assetPaths.length} asset(s)`);
|
|
|
|
|
|
|
|
// We perform this in batches to save on memory when performing large refreshes (greater than 1M assets)
|
|
|
|
const batchSize = 5000;
|
|
|
|
for (let i = 0; i < assetPaths.length; i += batchSize) {
|
|
|
|
const batch = assetPaths.slice(i, i + batchSize);
|
|
|
|
await this.jobRepository.queueAll(
|
|
|
|
batch.map((assetPath) => ({
|
|
|
|
name: JobName.LIBRARY_SCAN_ASSET,
|
|
|
|
data: {
|
|
|
|
id: libraryId,
|
|
|
|
assetPath: assetPath,
|
|
|
|
ownerId,
|
|
|
|
force,
|
|
|
|
},
|
|
|
|
})),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.logger.debug('Asset refresh queue completed');
|
2024-01-31 09:15:54 +01:00
|
|
|
}
|
|
|
|
|
2024-02-20 16:53:12 +01:00
|
|
|
private async validateImportPath(importPath: string): Promise<ValidateLibraryImportPathResponseDto> {
|
|
|
|
const validation = new ValidateLibraryImportPathResponseDto();
|
|
|
|
validation.importPath = importPath;
|
|
|
|
|
2024-03-15 23:01:58 +01:00
|
|
|
if (StorageCore.isImmichPath(importPath)) {
|
|
|
|
validation.message = 'Cannot use media upload folder for external libraries';
|
|
|
|
return validation;
|
|
|
|
}
|
|
|
|
|
2024-02-20 16:53:12 +01:00
|
|
|
try {
|
|
|
|
const stat = await this.storageRepository.stat(importPath);
|
|
|
|
if (!stat.isDirectory()) {
|
|
|
|
validation.message = 'Not a directory';
|
|
|
|
return validation;
|
|
|
|
}
|
|
|
|
} catch (error: any) {
|
|
|
|
if (error.code === 'ENOENT') {
|
|
|
|
validation.message = 'Path does not exist (ENOENT)';
|
|
|
|
return validation;
|
|
|
|
}
|
|
|
|
validation.message = String(error);
|
|
|
|
return validation;
|
|
|
|
}
|
|
|
|
|
|
|
|
const access = await this.storageRepository.checkFileExists(importPath, R_OK);
|
|
|
|
|
|
|
|
if (!access) {
|
|
|
|
validation.message = 'Lacking read permission for folder';
|
|
|
|
return validation;
|
|
|
|
}
|
|
|
|
|
|
|
|
validation.isValid = true;
|
|
|
|
return validation;
|
|
|
|
}
|
|
|
|
|
2024-03-18 21:59:53 +01:00
|
|
|
async validate(id: string, dto: ValidateLibraryDto): Promise<ValidateLibraryResponseDto> {
|
|
|
|
const importPaths = await Promise.all(
|
|
|
|
(dto.importPaths || []).map((importPath) => this.validateImportPath(importPath)),
|
|
|
|
);
|
|
|
|
return { importPaths };
|
2024-02-20 16:53:12 +01:00
|
|
|
}
|
|
|
|
|
2024-03-18 21:59:53 +01:00
|
|
|
async update(id: string, dto: UpdateLibraryDto): Promise<LibraryResponseDto> {
|
|
|
|
await this.findOrFail(id);
|
2023-09-20 13:16:33 +02:00
|
|
|
const library = await this.repository.update({ id, ...dto });
|
2024-01-31 09:15:54 +01:00
|
|
|
|
2024-02-20 16:53:12 +01:00
|
|
|
if (dto.importPaths) {
|
2024-03-18 21:59:53 +01:00
|
|
|
const validation = await this.validate(id, { importPaths: dto.importPaths });
|
2024-02-20 16:53:12 +01:00
|
|
|
if (validation.importPaths) {
|
|
|
|
for (const path of validation.importPaths) {
|
|
|
|
if (!path.isValid) {
|
|
|
|
throw new BadRequestException(`Invalid import path: ${path.message}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-31 09:15:54 +01:00
|
|
|
if (dto.importPaths || dto.exclusionPatterns) {
|
|
|
|
// Re-watch library to use new paths and/or exclusion patterns
|
|
|
|
await this.watch(id);
|
|
|
|
}
|
|
|
|
|
2023-09-20 13:16:33 +02:00
|
|
|
return mapLibrary(library);
|
|
|
|
}
|
|
|
|
|
2024-03-18 21:59:53 +01:00
|
|
|
async delete(id: string) {
|
2023-09-20 13:16:33 +02:00
|
|
|
const library = await this.findOrFail(id);
|
2024-03-18 21:59:53 +01:00
|
|
|
const uploadCount = await this.repository.getUploadLibraryCount(library.ownerId);
|
2023-09-20 13:16:33 +02:00
|
|
|
if (library.type === LibraryType.UPLOAD && uploadCount <= 1) {
|
|
|
|
throw new BadRequestException('Cannot delete the last upload library');
|
|
|
|
}
|
|
|
|
|
2024-01-31 09:15:54 +01:00
|
|
|
if (this.watchLibraries) {
|
|
|
|
await this.unwatch(id);
|
|
|
|
}
|
|
|
|
|
2023-09-20 13:16:33 +02:00
|
|
|
await this.repository.softDelete(id);
|
|
|
|
await this.jobRepository.queue({ name: JobName.LIBRARY_DELETE, data: { id } });
|
|
|
|
}
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
async handleDeleteLibrary(job: IEntityJob): Promise<JobStatus> {
|
2023-09-20 13:16:33 +02:00
|
|
|
const library = await this.repository.get(job.id, true);
|
|
|
|
if (!library) {
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.FAILED;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO use pagination
|
2023-10-07 22:42:08 +02:00
|
|
|
const assetIds = await this.repository.getAssetIds(job.id, true);
|
2023-09-20 13:16:33 +02:00
|
|
|
this.logger.debug(`Will delete ${assetIds.length} asset(s) in library ${job.id}`);
|
2024-01-01 21:45:42 +01:00
|
|
|
await this.jobRepository.queueAll(
|
|
|
|
assetIds.map((assetId) => ({ name: JobName.ASSET_DELETION, data: { id: assetId, fromExternal: true } })),
|
|
|
|
);
|
2023-10-07 19:44:10 +02:00
|
|
|
|
2023-10-07 22:42:08 +02:00
|
|
|
if (assetIds.length === 0) {
|
|
|
|
this.logger.log(`Deleting library ${job.id}`);
|
|
|
|
await this.repository.delete(job.id);
|
|
|
|
}
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.SUCCESS;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
async handleAssetRefresh(job: ILibraryFileJob): Promise<JobStatus> {
|
2023-09-20 13:16:33 +02:00
|
|
|
const assetPath = path.normalize(job.assetPath);
|
|
|
|
|
|
|
|
const existingAssetEntity = await this.assetRepository.getByLibraryIdAndOriginalPath(job.id, assetPath);
|
|
|
|
|
|
|
|
let stats: Stats;
|
|
|
|
try {
|
|
|
|
stats = await this.storageRepository.stat(assetPath);
|
|
|
|
} catch (error: Error | any) {
|
|
|
|
// Can't access file, probably offline
|
|
|
|
if (existingAssetEntity) {
|
|
|
|
// Mark asset as offline
|
|
|
|
this.logger.debug(`Marking asset as offline: ${assetPath}`);
|
|
|
|
|
2024-03-20 03:42:10 +01:00
|
|
|
await this.assetRepository.update({ id: existingAssetEntity.id, isOffline: true });
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.SUCCESS;
|
2023-09-20 13:16:33 +02:00
|
|
|
} else {
|
|
|
|
// File can't be accessed and does not already exist in db
|
2024-02-29 19:35:37 +01:00
|
|
|
throw new BadRequestException('Cannot access file', { cause: error });
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let doImport = false;
|
|
|
|
let doRefresh = false;
|
|
|
|
|
2023-10-09 05:16:13 +02:00
|
|
|
if (job.force) {
|
2023-09-20 13:16:33 +02:00
|
|
|
doRefresh = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!existingAssetEntity) {
|
|
|
|
// This asset is new to us, read it from disk
|
|
|
|
this.logger.debug(`Importing new asset: ${assetPath}`);
|
|
|
|
doImport = true;
|
|
|
|
} else if (stats.mtime.toISOString() !== existingAssetEntity.fileModifiedAt.toISOString()) {
|
|
|
|
// File modification time has changed since last time we checked, re-read from disk
|
|
|
|
this.logger.debug(
|
|
|
|
`File modification time has changed, re-importing asset: ${assetPath}. Old mtime: ${existingAssetEntity.fileModifiedAt}. New mtime: ${stats.mtime}`,
|
|
|
|
);
|
|
|
|
doRefresh = true;
|
2023-10-09 05:16:13 +02:00
|
|
|
} else if (!job.force && stats && !existingAssetEntity.isOffline) {
|
2023-09-20 13:16:33 +02:00
|
|
|
// Asset exists on disk and in db and mtime has not changed. Also, we are not forcing refresn. Therefore, do nothing
|
|
|
|
this.logger.debug(`Asset already exists in database and on disk, will not import: ${assetPath}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (stats && existingAssetEntity?.isOffline) {
|
|
|
|
// File was previously offline but is now online
|
|
|
|
this.logger.debug(`Marking previously-offline asset as online: ${assetPath}`);
|
2024-03-20 03:42:10 +01:00
|
|
|
await this.assetRepository.update({ id: existingAssetEntity.id, isOffline: false });
|
2023-09-20 13:16:33 +02:00
|
|
|
doRefresh = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!doImport && !doRefresh) {
|
|
|
|
// If we don't import, exit here
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.SKIPPED;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
let assetType: AssetType;
|
|
|
|
|
|
|
|
if (mimeTypes.isImage(assetPath)) {
|
|
|
|
assetType = AssetType.IMAGE;
|
|
|
|
} else if (mimeTypes.isVideo(assetPath)) {
|
|
|
|
assetType = AssetType.VIDEO;
|
|
|
|
} else {
|
|
|
|
throw new BadRequestException(`Unsupported file type ${assetPath}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: doesn't xmp replace the file extension? Will need investigation
|
|
|
|
let sidecarPath: string | null = null;
|
|
|
|
if (await this.storageRepository.checkFileExists(`${assetPath}.xmp`, R_OK)) {
|
|
|
|
sidecarPath = `${assetPath}.xmp`;
|
|
|
|
}
|
|
|
|
|
2024-02-02 04:18:00 +01:00
|
|
|
const deviceAssetId = `${basename(assetPath)}`.replaceAll(/\s+/g, '');
|
2023-09-20 13:16:33 +02:00
|
|
|
|
|
|
|
let assetId;
|
|
|
|
if (doImport) {
|
|
|
|
const library = await this.repository.get(job.id, true);
|
|
|
|
if (library?.deletedAt) {
|
|
|
|
this.logger.error('Cannot import asset into deleted library');
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.FAILED;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-01-31 09:15:54 +01:00
|
|
|
const pathHash = this.cryptoRepository.hashSha1(`path:${assetPath}`);
|
|
|
|
|
2023-09-20 13:16:33 +02:00
|
|
|
// TODO: In wait of refactoring the domain asset service, this function is just manually written like this
|
|
|
|
const addedAsset = await this.assetRepository.create({
|
|
|
|
ownerId: job.ownerId,
|
|
|
|
libraryId: job.id,
|
|
|
|
checksum: pathHash,
|
|
|
|
originalPath: assetPath,
|
|
|
|
deviceAssetId: deviceAssetId,
|
|
|
|
deviceId: 'Library Import',
|
2023-09-24 15:14:25 +02:00
|
|
|
fileCreatedAt: stats.mtime,
|
2023-09-20 13:16:33 +02:00
|
|
|
fileModifiedAt: stats.mtime,
|
2023-10-05 00:11:11 +02:00
|
|
|
localDateTime: stats.mtime,
|
2023-09-20 13:16:33 +02:00
|
|
|
type: assetType,
|
2024-03-20 05:40:28 +01:00
|
|
|
originalFileName: parse(assetPath).base,
|
2023-09-20 13:16:33 +02:00
|
|
|
sidecarPath,
|
|
|
|
isReadOnly: true,
|
|
|
|
isExternal: true,
|
|
|
|
});
|
|
|
|
assetId = addedAsset.id;
|
|
|
|
} else if (doRefresh && existingAssetEntity) {
|
|
|
|
assetId = existingAssetEntity.id;
|
|
|
|
await this.assetRepository.updateAll([existingAssetEntity.id], {
|
2023-09-24 15:14:25 +02:00
|
|
|
fileCreatedAt: stats.mtime,
|
2023-09-20 13:16:33 +02:00
|
|
|
fileModifiedAt: stats.mtime,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
// Not importing and not refreshing, do nothing
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.SKIPPED;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
this.logger.debug(`Queuing metadata extraction for: ${assetPath}`);
|
|
|
|
|
|
|
|
await this.jobRepository.queue({ name: JobName.METADATA_EXTRACTION, data: { id: assetId, source: 'upload' } });
|
|
|
|
|
|
|
|
if (assetType === AssetType.VIDEO) {
|
|
|
|
await this.jobRepository.queue({ name: JobName.VIDEO_CONVERSION, data: { id: assetId } });
|
|
|
|
}
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.SUCCESS;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-03-18 21:59:53 +01:00
|
|
|
async queueScan(id: string, dto: ScanLibraryDto) {
|
|
|
|
const library = await this.findOrFail(id);
|
|
|
|
if (library.type !== LibraryType.EXTERNAL) {
|
2023-09-20 13:16:33 +02:00
|
|
|
throw new BadRequestException('Can only refresh external libraries');
|
|
|
|
}
|
|
|
|
|
|
|
|
await this.jobRepository.queue({
|
|
|
|
name: JobName.LIBRARY_SCAN,
|
|
|
|
data: {
|
|
|
|
id,
|
|
|
|
refreshModifiedFiles: dto.refreshModifiedFiles ?? false,
|
|
|
|
refreshAllFiles: dto.refreshAllFiles ?? false,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-03-18 21:59:53 +01:00
|
|
|
async queueRemoveOffline(id: string) {
|
2023-09-20 13:16:33 +02:00
|
|
|
this.logger.verbose(`Removing offline files from library: ${id}`);
|
2024-03-18 21:59:53 +01:00
|
|
|
await this.jobRepository.queue({ name: JobName.LIBRARY_REMOVE_OFFLINE, data: { id } });
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
async handleQueueAllScan(job: IBaseJob): Promise<JobStatus> {
|
2023-09-20 13:16:33 +02:00
|
|
|
this.logger.debug(`Refreshing all external libraries: force=${job.force}`);
|
|
|
|
|
|
|
|
// Queue cleanup
|
|
|
|
await this.jobRepository.queue({ name: JobName.LIBRARY_QUEUE_CLEANUP, data: {} });
|
|
|
|
|
|
|
|
// Queue all library refresh
|
|
|
|
const libraries = await this.repository.getAll(true, LibraryType.EXTERNAL);
|
2024-01-01 21:45:42 +01:00
|
|
|
await this.jobRepository.queueAll(
|
|
|
|
libraries.map((library) => ({
|
2023-09-20 13:16:33 +02:00
|
|
|
name: JobName.LIBRARY_SCAN,
|
|
|
|
data: {
|
|
|
|
id: library.id,
|
|
|
|
refreshModifiedFiles: !job.force,
|
|
|
|
refreshAllFiles: job.force ?? false,
|
|
|
|
},
|
2024-01-01 21:45:42 +01:00
|
|
|
})),
|
|
|
|
);
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.SUCCESS;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
async handleOfflineRemoval(job: IEntityJob): Promise<JobStatus> {
|
2023-10-07 19:44:10 +02:00
|
|
|
const assetPagination = usePagination(JOBS_ASSET_PAGINATION_SIZE, (pagination) =>
|
|
|
|
this.assetRepository.getWith(pagination, WithProperty.IS_OFFLINE, job.id),
|
|
|
|
);
|
2023-09-20 13:16:33 +02:00
|
|
|
|
|
|
|
for await (const assets of assetPagination) {
|
2023-10-07 19:44:10 +02:00
|
|
|
this.logger.debug(`Removing ${assets.length} offline assets`);
|
2024-01-01 21:45:42 +01:00
|
|
|
await this.jobRepository.queueAll(
|
|
|
|
assets.map((asset) => ({ name: JobName.ASSET_DELETION, data: { id: asset.id, fromExternal: true } })),
|
|
|
|
);
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.SUCCESS;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
async handleQueueAssetRefresh(job: ILibraryRefreshJob): Promise<JobStatus> {
|
2023-09-20 13:16:33 +02:00
|
|
|
const library = await this.repository.get(job.id);
|
|
|
|
if (!library || library.type !== LibraryType.EXTERNAL) {
|
|
|
|
this.logger.warn('Can only refresh external libraries');
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.FAILED;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-03-14 14:43:05 +01:00
|
|
|
this.logger.log(`Refreshing library: ${job.id}`);
|
2024-02-20 16:53:12 +01:00
|
|
|
|
2024-03-14 06:52:30 +01:00
|
|
|
const crawledAssetPaths = await this.getPathTrie(library);
|
2024-03-11 03:30:57 +01:00
|
|
|
this.logger.debug(`Found ${crawledAssetPaths.size} asset(s) when crawling import paths ${library.importPaths}`);
|
2023-09-20 13:16:33 +02:00
|
|
|
|
2024-03-11 03:30:57 +01:00
|
|
|
const assetIdsToMarkOffline = [];
|
|
|
|
const assetIdsToMarkOnline = [];
|
2024-03-14 06:52:30 +01:00
|
|
|
const pagination = usePagination(LIBRARY_SCAN_BATCH_SIZE, (pagination) =>
|
2024-03-11 03:30:57 +01:00
|
|
|
this.assetRepository.getLibraryAssetPaths(pagination, library.id),
|
|
|
|
);
|
|
|
|
|
2024-03-14 14:43:05 +01:00
|
|
|
this.logger.verbose(`Crawled asset paths paginated`);
|
|
|
|
|
2024-03-14 06:52:30 +01:00
|
|
|
const shouldScanAll = job.refreshAllFiles || job.refreshModifiedFiles;
|
2024-03-11 03:30:57 +01:00
|
|
|
for await (const page of pagination) {
|
|
|
|
for (const asset of page) {
|
|
|
|
const isOffline = !crawledAssetPaths.has(asset.originalPath);
|
|
|
|
if (isOffline && !asset.isOffline) {
|
|
|
|
assetIdsToMarkOffline.push(asset.id);
|
2024-03-14 14:43:05 +01:00
|
|
|
this.logger.verbose(`Added to mark-offline list: ${asset.originalPath}`);
|
2024-03-11 03:30:57 +01:00
|
|
|
}
|
2023-09-20 13:16:33 +02:00
|
|
|
|
2024-03-11 03:30:57 +01:00
|
|
|
if (!isOffline && asset.isOffline) {
|
|
|
|
assetIdsToMarkOnline.push(asset.id);
|
2024-03-14 14:43:05 +01:00
|
|
|
this.logger.verbose(`Added to mark-online list: ${asset.originalPath}`);
|
2024-03-11 03:30:57 +01:00
|
|
|
}
|
|
|
|
|
2024-03-14 06:52:30 +01:00
|
|
|
if (!shouldScanAll) {
|
|
|
|
crawledAssetPaths.delete(asset.originalPath);
|
|
|
|
}
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
2024-03-11 03:30:57 +01:00
|
|
|
}
|
|
|
|
|
2024-03-14 14:43:05 +01:00
|
|
|
this.logger.verbose(`Crawled assets have been checked for online/offline status`);
|
|
|
|
|
2024-03-11 03:30:57 +01:00
|
|
|
if (assetIdsToMarkOffline.length > 0) {
|
|
|
|
this.logger.debug(`Found ${assetIdsToMarkOffline.length} offline asset(s) previously marked as online`);
|
|
|
|
await this.assetRepository.updateAll(assetIdsToMarkOffline, { isOffline: true });
|
|
|
|
}
|
|
|
|
|
|
|
|
if (assetIdsToMarkOnline.length > 0) {
|
|
|
|
this.logger.debug(`Found ${assetIdsToMarkOnline.length} online asset(s) previously marked as offline`);
|
|
|
|
await this.assetRepository.updateAll(assetIdsToMarkOnline, { isOffline: false });
|
|
|
|
}
|
|
|
|
|
2024-03-14 06:52:30 +01:00
|
|
|
if (crawledAssetPaths.size > 0) {
|
|
|
|
if (!shouldScanAll) {
|
|
|
|
this.logger.debug(`Will import ${crawledAssetPaths.size} new asset(s)`);
|
|
|
|
}
|
|
|
|
|
2024-03-15 23:01:58 +01:00
|
|
|
let batch = [];
|
2024-03-14 06:52:30 +01:00
|
|
|
for (const assetPath of crawledAssetPaths) {
|
|
|
|
batch.push(assetPath);
|
2023-09-20 13:16:33 +02:00
|
|
|
|
2024-03-14 06:52:30 +01:00
|
|
|
if (batch.length >= LIBRARY_SCAN_BATCH_SIZE) {
|
|
|
|
await this.scanAssets(job.id, batch, library.ownerId, job.refreshAllFiles ?? false);
|
2024-03-15 23:01:58 +01:00
|
|
|
batch = [];
|
2024-03-14 06:52:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (batch.length > 0) {
|
|
|
|
await this.scanAssets(job.id, batch, library.ownerId, job.refreshAllFiles ?? false);
|
|
|
|
}
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
await this.repository.update({ id: job.id, refreshedAt: new Date() });
|
|
|
|
|
2024-03-15 14:16:54 +01:00
|
|
|
return JobStatus.SUCCESS;
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
|
2024-03-14 06:52:30 +01:00
|
|
|
private async getPathTrie(library: LibraryEntity): Promise<Trie<string>> {
|
|
|
|
const pathValidation = await Promise.all(
|
|
|
|
library.importPaths.map(async (importPath) => await this.validateImportPath(importPath)),
|
|
|
|
);
|
|
|
|
|
|
|
|
const validImportPaths = pathValidation
|
|
|
|
.map((validation) => {
|
|
|
|
if (!validation.isValid) {
|
|
|
|
this.logger.error(`Skipping invalid import path: ${validation.importPath}. Reason: ${validation.message}`);
|
|
|
|
}
|
|
|
|
return validation;
|
|
|
|
})
|
|
|
|
.filter((validation) => validation.isValid)
|
|
|
|
.map((validation) => validation.importPath);
|
|
|
|
|
|
|
|
const generator = this.storageRepository.walk({
|
|
|
|
pathsToCrawl: validImportPaths,
|
|
|
|
exclusionPatterns: library.exclusionPatterns,
|
|
|
|
});
|
|
|
|
|
|
|
|
const trie = new Trie<string>();
|
|
|
|
for await (const filePath of generator) {
|
|
|
|
trie.add(filePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
return trie;
|
|
|
|
}
|
|
|
|
|
2023-09-20 13:16:33 +02:00
|
|
|
private async findOrFail(id: string) {
|
|
|
|
const library = await this.repository.get(id);
|
|
|
|
if (!library) {
|
|
|
|
throw new BadRequestException('Library not found');
|
|
|
|
}
|
|
|
|
return library;
|
|
|
|
}
|
|
|
|
}
|