import {Config} from '../src/common/config/private/Config'; import {ObjectManagers} from '../src/backend/model/ObjectManagers'; import {DiskMangerWorker} from '../src/backend/model/threading/DiskMangerWorker'; import {IndexingManager} from '../src/backend/model/database/sql/IndexingManager'; import * as path from 'path'; import * as fs from 'fs'; import {Utils} from '../src/common/Utils'; import {DirectoryDTO} from '../src/common/entities/DirectoryDTO'; import {DatabaseType, ReIndexingSensitivity} from '../src/common/config/private/PrivateConfig'; import {ProjectPath} from '../src/backend/ProjectPath'; import {Benchmark} from './Benchmark'; import {IndexingJob} from '../src/backend/model/jobs/jobs/IndexingJob'; import {IJob} from '../src/backend/model/jobs/jobs/IJob'; import {JobProgressStates} from '../src/common/entities/job/JobProgressDTO'; import {JobProgress} from '../src/backend/model/jobs/jobs/JobProgress'; import {ContentWrapper} from '../src/common/entities/ConentWrapper'; import {GalleryManager} from '../src/backend/model/database/sql/GalleryManager'; import {PersonManager} from '../src/backend/model/database/sql/PersonManager'; import {GalleryRouter} from '../src/backend/routes/GalleryRouter'; import {Express} from 'express'; import {PersonRouter} from '../src/backend/routes/PersonRouter'; import { ANDSearchQuery, ORSearchQuery, SearchQueryDTO, SearchQueryTypes, SomeOfSearchQuery, TextSearch, TextSearchQueryMatchTypes, TextSearchQueryTypes } from '../src/common/entities/SearchQueryDTO'; import {QueryKeywords, SearchQueryParser} from '../src/common/SearchQueryParser'; export interface BenchmarkResult { name: string; experiment?: string; duration: number; contentWrapper?: ContentWrapper; items?: number; subBenchmarks?: BenchmarkResult[]; } export class BMIndexingManager extends IndexingManager { public async saveToDB(scannedDirectory: DirectoryDTO): Promise { return super.saveToDB(scannedDirectory); } } class BMGalleryRouter extends GalleryRouter { public static addDirectoryList(app: Express): void { GalleryRouter.addDirectoryList(app); } public static addSearch(app: Express): void { GalleryRouter.addSearch(app); } public static addAutoComplete(app: Express): void { GalleryRouter.addAutoComplete(app); } } class BMPersonRouter extends PersonRouter { public static addGetPersons(app: Express): void { PersonRouter.addGetPersons(app); } } export class BenchmarkRunner { inited = false; private biggestDirPath: string = null; private readonly requestTemplate: any = { requestPipe: null, params: {}, query: {}, session: {} }; constructor(public RUNS: number) { Config.Client.authenticationRequired = false; } async bmSaveDirectory(): Promise { await this.init(); await this.resetDB(); const dir = await DiskMangerWorker.scanDirectory(this.biggestDirPath); const bm = new Benchmark('Saving directory to DB', null, (): Promise => this.resetDB()); bm.addAStep({ name: 'Saving directory to DB', fn: (): Promise => { const im = new BMIndexingManager(); return im.saveToDB(dir); } }); return await bm.run(this.RUNS); } async bmScanDirectory(): Promise { await this.init(); const bm = new Benchmark('Scanning directory'); bm.addAStep({ name: 'Scanning directory', fn: async (): Promise => new ContentWrapper(await DiskMangerWorker.scanDirectory(this.biggestDirPath)) }); return await bm.run(this.RUNS); } async bmListDirectory(): Promise { Config.Server.Indexing.reIndexingSensitivity = ReIndexingSensitivity.low; await this.init(); await this.setupDB(); const req = Utils.clone(this.requestTemplate); req.params.directory = this.biggestDirPath; const bm = new Benchmark('List directory', req, async (): Promise => { await ObjectManagers.reset(); await ObjectManagers.InitSQLManagers(); }); BMGalleryRouter.addDirectoryList(bm.BmExpressApp); return await bm.run(this.RUNS); } async bmListPersons(): Promise { await this.setupDB(); Config.Server.Indexing.reIndexingSensitivity = ReIndexingSensitivity.low; const bm = new Benchmark('Listing Faces', Utils.clone(this.requestTemplate), async (): Promise => { await ObjectManagers.reset(); await ObjectManagers.InitSQLManagers(); }); BMPersonRouter.addGetPersons(bm.BmExpressApp); return await bm.run(this.RUNS); } async bmAllSearch(): Promise<{ result: BenchmarkResult[], searchQuery: SearchQueryDTO }[]> { await this.setupDB(); const queryKeywords: QueryKeywords = { NSomeOf: '-of', and: 'and', caption: 'caption', directory: 'directory', file_name: 'file-name', from: 'from', keyword: 'keyword', landscape: 'landscape', maxRating: 'max-rating', maxResolution: 'max-resolution', minRating: 'min-rating', minResolution: 'min-resolution', or: 'or', orientation: 'orientation', any_text: 'any-text', person: 'person', portrait: 'portrait', position: 'position', someOf: 'some-of', to: 'to', kmFrom: 'km-from' }; const queryParser = new SearchQueryParser(queryKeywords); const names = (await ObjectManagers.getInstance().PersonManager.getAll()).sort((a, b) => b.count - a.count); const queries: { query: SearchQueryDTO, description: string }[] = TextSearchQueryTypes.map(t => { const q = { type: t, text: 'a' }; return { query: q, description: queryParser.stringify(q) }; }); // searching for everything queries.push({ query: { type: SearchQueryTypes.any_text, text: '.' } as TextSearch, description: queryParser.stringify({ type: SearchQueryTypes.any_text, text: '.' } as TextSearch) }); if (names.length > 0) { queries.push({ query: { type: SearchQueryTypes.person, text: names[0].name, matchType: TextSearchQueryMatchTypes.exact_match } as TextSearch, description: '' }); } if (names.length > 1) { queries.push({ query: { type: SearchQueryTypes.AND, list: [ { type: SearchQueryTypes.person, text: names[0].name, matchType: TextSearchQueryMatchTypes.exact_match } as TextSearch, { type: SearchQueryTypes.person, text: names[1].name, matchType: TextSearchQueryMatchTypes.exact_match } as TextSearch ] } as ANDSearchQuery, description: '' }); queries.push({ query: { type: SearchQueryTypes.OR, list: [ { type: SearchQueryTypes.person, text: names[0].name, matchType: TextSearchQueryMatchTypes.exact_match } as TextSearch, { type: SearchQueryTypes.person, text: names[1].name, matchType: TextSearchQueryMatchTypes.exact_match } as TextSearch ] } as ORSearchQuery, description: '' }); queries.push({ query: { type: SearchQueryTypes.SOME_OF, min: 2, list: names.map(n => ({ type: SearchQueryTypes.person, text: n.name, matchType: TextSearchQueryMatchTypes.exact_match } as TextSearch)) } as SomeOfSearchQuery, description: '' }); } const results: { result: BenchmarkResult[], searchQuery: SearchQueryDTO }[] = []; for (const entry of queries) { const req = Utils.clone(this.requestTemplate); req.params.searchQueryDTO = JSON.stringify(entry.query); const bm = new Benchmark('Searching for `' + entry.description + '`', req); BMGalleryRouter.addSearch(bm.BmExpressApp); results.push({result: await bm.run(this.RUNS), searchQuery: entry.query}); } return results; } async bmAutocomplete(text: string): Promise { await this.setupDB(); const req = Utils.clone(this.requestTemplate); req.params.text = text; const bm = new Benchmark('Auto complete for `' + text + '`', req); BMGalleryRouter.addAutoComplete(bm.BmExpressApp); return await bm.run(this.RUNS); } async getStatistic(): Promise { await this.setupDB(); const gm = new GalleryManager(); const pm = new PersonManager(); return 'directories: ' + await gm.countDirectories() + ', photos: ' + await gm.countPhotos() + ', videos: ' + await gm.countVideos() + ', diskUsage : ' + Utils.renderDataSize(await gm.countMediaSize()) + ', persons : ' + await pm.countFaces() + ', unique persons (faces): ' + (await pm.getAll()).length; } private async init(): Promise { if (this.inited === false) { await this.setupDB(); const gm = new GalleryManager(); let biggest = 0; let biggestPath = '/'; const queue = ['/']; while (queue.length > 0) { const dirPath = queue.shift(); const dir = await gm.listDirectory(dirPath); dir.directories.forEach((d): number => queue.push(path.join(d.path + d.name))); if (biggest < dir.media.length) { biggestPath = path.join(dir.path + dir.name); biggest = dir.media.length; } } this.biggestDirPath = biggestPath; console.log('updating path of biggest dir to: ' + this.biggestDirPath); this.inited = true; } return this.biggestDirPath; } private resetDB = async (): Promise => { Config.Server.Threading.enabled = false; await ObjectManagers.reset(); await fs.promises.rmdir(ProjectPath.DBFolder, {recursive: true}); Config.Server.Database.type = DatabaseType.sqlite; Config.Server.Jobs.scheduled = []; await ObjectManagers.InitSQLManagers(); }; private async setupDB(): Promise { Config.Server.Threading.enabled = false; await this.resetDB(); await new Promise((resolve, reject): void => { try { const indexingJob = new IndexingJob(); indexingJob.JobListener = { onJobFinished: (job: IJob, state: JobProgressStates, soloRun: boolean): void => { resolve(); }, onProgressUpdate: (progress: JobProgress): void => { } }; indexingJob.start({indexChangesOnly: false}).catch(console.error); } catch (e) { console.error(e); reject(e); } }); } }