1
0
mirror of https://github.com/bpatrik/pigallery2.git synced 2025-01-10 04:07:35 +02:00

Making file job load files from DB in batches. #494

This commit is contained in:
Patrik J. Braun 2022-12-18 23:47:13 +01:00
parent 3012bc9908
commit 9fdd732ce5
3 changed files with 102 additions and 18 deletions

View File

@ -25,6 +25,12 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
public readonly ConfigTemplate: ConfigTemplateEntry[] = []; public readonly ConfigTemplate: ConfigTemplateEntry[] = [];
directoryQueue: string[] = []; directoryQueue: string[] = [];
fileQueue: string[] = []; fileQueue: string[] = [];
DBProcessing = {
mediaLoaded: 0,
hasMoreMedia: true,
initiated: false
};
protected constructor(private scanFilter: DirectoryScanSettings) { protected constructor(private scanFilter: DirectoryScanSettings) {
super(); super();
@ -43,6 +49,11 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
protected async init(): Promise<void> { protected async init(): Promise<void> {
this.directoryQueue = []; this.directoryQueue = [];
this.fileQueue = []; this.fileQueue = [];
this.DBProcessing = {
mediaLoaded: 0,
hasMoreMedia: true,
initiated: false
};
this.directoryQueue.push('/'); this.directoryQueue.push('/');
} }
@ -59,22 +70,17 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
protected abstract processFile(filePath: string): Promise<void>; protected abstract processFile(filePath: string): Promise<void>;
protected async step(): Promise<boolean> { protected async step(): Promise<boolean> {
if (this.directoryQueue.length === 0 && this.fileQueue.length === 0) { const DBBased = this.config.indexedOnly &&
Config.Server.Database.type !== DatabaseType.memory;
if (
this.fileQueue.length === 0 &&
((this.directoryQueue.length === 0 && !DBBased) ||
(DBBased &&
this.DBProcessing.hasMoreMedia === false))) {
return false; return false;
} }
if (this.directoryQueue.length > 0) { const processOneFile = async () => {
if (
this.config.indexedOnly === true &&
Config.Server.Database.type !== DatabaseType.memory
) {
await this.loadAllMediaFilesFromDB();
this.directoryQueue = [];
} else {
await this.loadADirectoryFromDisk();
}
} else if (this.fileQueue.length > 0) {
this.Progress.Left = this.fileQueue.length;
const filePath = this.fileQueue.shift(); const filePath = this.fileQueue.shift();
try { try {
if ((await this.shouldProcess(filePath)) === true) { if ((await this.shouldProcess(filePath)) === true) {
@ -95,7 +101,32 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
'Error during processing file:' + filePath + ', ' + e.toString() 'Error during processing file:' + filePath + ', ' + e.toString()
); );
} }
};
if (!DBBased) {
if (this.directoryQueue.length > 0) {
await this.loadADirectoryFromDisk();
} else if (this.fileQueue.length > 0) {
this.Progress.Left = this.fileQueue.length;
await processOneFile();
}
} else {
if (!this.DBProcessing.initiated) {
this.Progress.log('Counting files from db');
Logger.silly(LOG_TAG, 'Counting files from db');
this.Progress.All = await this.countMediaFromDB();
Logger.silly(LOG_TAG, 'Found:' + this.Progress.All);
this.DBProcessing.initiated = true;
return true;
}
if (this.fileQueue.length === 0) {
await this.loadMediaFilesFromDB();
} else {
this.Progress.Left = this.fileQueue.length;
await processOneFile();
}
} }
return true; return true;
} }
@ -138,16 +169,21 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
} }
} }
private async loadAllMediaFilesFromDB(): Promise<void> { private async loadMediaFilesFromDB(): Promise<void> {
if (this.scanFilter.noVideo === true && if (this.scanFilter.noVideo === true &&
this.scanFilter.noPhoto === true && this.scanFilter.noPhoto === true &&
this.scanFilter.noMetaFile === true) { this.scanFilter.noMetaFile === true) {
return; return;
} }
this.Progress.log('Loading files from db'); const logStr = `Loading next batch of files from db. Skipping: ${this.DBProcessing.mediaLoaded}, looking for more: ${Config.Server.Jobs.mediaProcessingBatchSize}`;
Logger.silly(LOG_TAG, 'Loading files from db'); this.Progress.log(logStr);
Logger.silly(LOG_TAG, logStr);
const hasMoreFile = {
media: false,
metafile: false
};
const connection = await SQLConnection.getConnection(); const connection = await SQLConnection.getConnection();
if (!this.scanFilter.noVideo || if (!this.scanFilter.noVideo ||
!this.scanFilter.noPhoto) { !this.scanFilter.noPhoto) {
@ -165,8 +201,12 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
.createQueryBuilder('media') .createQueryBuilder('media')
.select(['media.name', 'directory.name', 'directory.path']) .select(['media.name', 'directory.name', 'directory.path'])
.leftJoin('media.directory', 'directory') .leftJoin('media.directory', 'directory')
.offset(this.DBProcessing.mediaLoaded)
.limit(Config.Server.Jobs.mediaProcessingBatchSize)
.getMany(); .getMany();
hasMoreFile.media = result.length > 0;
this.DBProcessing.mediaLoaded += result.length;
const scannedAndFiltered = await this.filterMediaFiles(result); const scannedAndFiltered = await this.filterMediaFiles(result);
for (const item of scannedAndFiltered) { for (const item of scannedAndFiltered) {
this.fileQueue.push( this.fileQueue.push(
@ -186,9 +226,13 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
.createQueryBuilder('file') .createQueryBuilder('file')
.select(['file.name', 'directory.name', 'directory.path']) .select(['file.name', 'directory.name', 'directory.path'])
.leftJoin('file.directory', 'directory') .leftJoin('file.directory', 'directory')
.offset(this.DBProcessing.mediaLoaded)
.limit(Config.Server.Jobs.mediaProcessingBatchSize)
.getMany(); .getMany();
hasMoreFile.metafile = result.length > 0;
this.DBProcessing.mediaLoaded += result.length;
const scannedAndFiltered = await this.filterMetaFiles(result); const scannedAndFiltered = await this.filterMetaFiles(result);
for (const item of scannedAndFiltered) { for (const item of scannedAndFiltered) {
this.fileQueue.push( this.fileQueue.push(
@ -201,5 +245,42 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
); );
} }
} }
this.DBProcessing.hasMoreMedia = hasMoreFile.media || hasMoreFile.metafile;
}
private async countMediaFromDB(): Promise<number> {
if (this.scanFilter.noVideo === true &&
this.scanFilter.noPhoto === true &&
this.scanFilter.noMetaFile === true) {
return;
}
let count = 0;
const connection = await SQLConnection.getConnection();
if (!this.scanFilter.noVideo ||
!this.scanFilter.noPhoto) {
let usedEntity = MediaEntity;
if (this.scanFilter.noVideo === true) {
usedEntity = PhotoEntity;
} else if (this.scanFilter.noPhoto === true) {
usedEntity = VideoEntity;
}
count += await connection
.getRepository(usedEntity)
.createQueryBuilder('media')
.getCount();
if (!this.scanFilter.noMetaFile) {
count += await connection
.getRepository(FileEntity)
.createQueryBuilder('file')
.getCount();
}
return count;
}
} }
} }

View File

@ -3,6 +3,7 @@ import {
JobProgressLogDTO, JobProgressLogDTO,
JobProgressStates, JobProgressStates,
} from '../../../../common/entities/job/JobProgressDTO'; } from '../../../../common/entities/job/JobProgressDTO';
import {Config} from '../../../../common/config/private/Config';
export class JobProgress { export class JobProgress {
private steps = { private steps = {
@ -86,7 +87,7 @@ export class JobProgress {
}; };
log(log: string): void { log(log: string): void {
while (this.logs.length > 10) { while (this.logs.length > Config.Server.Jobs.maxSavedProgress) {
this.logs.shift(); this.logs.shift();
} }
this.logs.push({ this.logs.push({

View File

@ -316,7 +316,9 @@ export class JobScheduleConfig implements JobScheduleDTO {
@SubConfigClass() @SubConfigClass()
export class ServerJobConfig { export class ServerJobConfig {
@ConfigProperty({type: 'integer', description: 'Job history size'}) @ConfigProperty({type: 'integer', description: 'Job history size'})
maxSavedProgress: number = 10; maxSavedProgress: number = 20;
@ConfigProperty({type: 'integer', description: 'Job loads this many photos or videos form the DB for processing'})
mediaProcessingBatchSize: number = 1000;
@ConfigProperty({arrayType: JobScheduleConfig}) @ConfigProperty({arrayType: JobScheduleConfig})
scheduled: JobScheduleConfig[] = [ scheduled: JobScheduleConfig[] = [
new JobScheduleConfig( new JobScheduleConfig(