* feat: initial watch support * allow offline files * chore: ignore query errors when resetting e2e db * revert db query * add savepoint * guard the user query * chore: openapi and db migration * wip * support multiple libraries * fix tests * wip * can now cleanup chokidar watchers * fix unit tests * add library watch queue * add missing init from merge * wip * can now filter file extensions * remove watch api from non job client * Fix e2e test * watch library with updated import path and exclusion pattern * add library watch frontend ui * case sensitive watching extensions * can auto watch libraries * move watcher e2e tests to separate file * don't watch libraries from a queue * use event emitters * shorten e2e test timeout * refactor chokidar code to filesystem provider * expose chokidar parameters to config file * fix storage mock * set default config for library watching * add fs provider mocks * cleanup * add more unit tests for watcher * chore: fix format + sql * add more tests * move unwatch feature back to library service * add file event unit tests * chore: formatting * add documentation * fix e2e tests * chore: fix e2e tests * fix library updating * test cleanup * fix typo * cleanup * fixing as per pr comments * reduce library watch config file * update storage config and mocks * move negative event tests to unit tests * fix library watcher e2e * make watch configuration global * remove the feature flag * refactor watcher teardown * fix microservices init * centralize asset scan job queue * improve docs * add more tests * chore: open api * initialize app service * fix docs * fix library watch feature flag * Update docs/docs/features/libraries.md Co-authored-by: Daniel Dietzler <36593685+danieldietzler@users.noreply.github.com> * fix: import right app service * don't be truthy * fix test speling * stricter library update tests * move fs watcher mock to external file * subscribe to config changes * docker does not need polling * make library watch() private * feat: add configuration ui --------- Co-authored-by: Daniel Dietzler <36593685+danieldietzler@users.noreply.github.com> Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
129 lines
2.6 KiB
TypeScript
129 lines
2.6 KiB
TypeScript
import { LibraryEntity, LibraryType } from '@app/infra/entities';
|
|
import { ApiProperty } from '@nestjs/swagger';
|
|
import { IsBoolean, IsEnum, IsNotEmpty, IsOptional, IsString } from 'class-validator';
|
|
import { ValidateUUID } from '../domain.util';
|
|
|
|
export class CreateLibraryDto {
|
|
@IsEnum(LibraryType)
|
|
@ApiProperty({ enumName: 'LibraryType', enum: LibraryType })
|
|
type!: LibraryType;
|
|
|
|
@IsString()
|
|
@IsOptional()
|
|
@IsNotEmpty()
|
|
name?: string;
|
|
|
|
@IsOptional()
|
|
@IsBoolean()
|
|
isVisible?: boolean;
|
|
|
|
@IsOptional()
|
|
@IsString({ each: true })
|
|
@IsNotEmpty({ each: true })
|
|
importPaths?: string[];
|
|
|
|
@IsOptional()
|
|
@IsString({ each: true })
|
|
@IsNotEmpty({ each: true })
|
|
exclusionPatterns?: string[];
|
|
|
|
@IsOptional()
|
|
@IsBoolean()
|
|
isWatched?: boolean;
|
|
}
|
|
|
|
export class UpdateLibraryDto {
|
|
@IsOptional()
|
|
@IsString()
|
|
@IsNotEmpty()
|
|
name?: string;
|
|
|
|
@IsOptional()
|
|
@IsBoolean()
|
|
isVisible?: boolean;
|
|
|
|
@IsOptional()
|
|
@IsString({ each: true })
|
|
@IsNotEmpty({ each: true })
|
|
importPaths?: string[];
|
|
|
|
@IsOptional()
|
|
@IsNotEmpty({ each: true })
|
|
@IsString({ each: true })
|
|
exclusionPatterns?: string[];
|
|
}
|
|
|
|
export class CrawlOptionsDto {
|
|
pathsToCrawl!: string[];
|
|
includeHidden? = false;
|
|
exclusionPatterns?: string[];
|
|
}
|
|
|
|
export class LibrarySearchDto {
|
|
@ValidateUUID({ optional: true })
|
|
userId?: string;
|
|
}
|
|
|
|
export class ScanLibraryDto {
|
|
@IsBoolean()
|
|
@IsOptional()
|
|
refreshModifiedFiles?: boolean;
|
|
|
|
@IsBoolean()
|
|
@IsOptional()
|
|
refreshAllFiles?: boolean = false;
|
|
}
|
|
|
|
export class LibraryResponseDto {
|
|
id!: string;
|
|
ownerId!: string;
|
|
name!: string;
|
|
|
|
@ApiProperty({ enumName: 'LibraryType', enum: LibraryType })
|
|
type!: LibraryType;
|
|
|
|
@ApiProperty({ type: 'integer' })
|
|
assetCount!: number;
|
|
|
|
importPaths!: string[];
|
|
|
|
exclusionPatterns!: string[];
|
|
|
|
createdAt!: Date;
|
|
updatedAt!: Date;
|
|
refreshedAt!: Date | null;
|
|
}
|
|
|
|
export class LibraryStatsResponseDto {
|
|
@ApiProperty({ type: 'integer' })
|
|
photos = 0;
|
|
|
|
@ApiProperty({ type: 'integer' })
|
|
videos = 0;
|
|
|
|
@ApiProperty({ type: 'integer' })
|
|
total = 0;
|
|
|
|
@ApiProperty({ type: 'integer', format: 'int64' })
|
|
usage = 0;
|
|
}
|
|
|
|
export function mapLibrary(entity: LibraryEntity): LibraryResponseDto {
|
|
let assetCount = 0;
|
|
if (entity.assets) {
|
|
assetCount = entity.assets.length;
|
|
}
|
|
return {
|
|
id: entity.id,
|
|
ownerId: entity.ownerId,
|
|
type: entity.type,
|
|
name: entity.name,
|
|
createdAt: entity.createdAt,
|
|
updatedAt: entity.updatedAt,
|
|
refreshedAt: entity.refreshedAt,
|
|
assetCount,
|
|
importPaths: entity.importPaths,
|
|
exclusionPatterns: entity.exclusionPatterns,
|
|
};
|
|
}
|