backend performance improvement

This commit is contained in:
Patrik Braun 2017-07-23 22:37:29 +02:00
parent 1537efb5b5
commit 296fb8b422
8 changed files with 84 additions and 23 deletions

View File

@ -12,13 +12,13 @@ import {PhotoDTO} from "../../../common/entities/PhotoDTO";
import {Config} from "../../../common/config/private/Config"; import {Config} from "../../../common/config/private/Config";
import {ThumbnailProcessingLib} from "../../../common/config/private/IPrivateConfig"; import {ThumbnailProcessingLib} from "../../../common/config/private/IPrivateConfig";
import {ThumbnailTH} from "../../model/threading/ThreadPool"; import {ThumbnailTH} from "../../model/threading/ThreadPool";
import {RendererFactory, RendererInput} from "../../model/threading/ThumbnailWoker"; import {RendererInput} from "../../model/threading/ThumbnailWoker";
import {ITaskQue, TaskQue} from "../../model/threading/TaskQue";
export class ThumbnailGeneratorMWs { export class ThumbnailGeneratorMWs {
private static initDone = false; private static initDone = false;
private static ThumbnailFunction: (input: RendererInput) => Promise<void> = null; private static taskQue: ITaskQue = null;
private static threadPool: ThumbnailTH = null;
public static init() { public static init() {
if (this.initDone == true) { if (this.initDone == true) {
@ -33,11 +33,11 @@ export class ThumbnailGeneratorMWs {
Config.Client.concurrentThumbnailGenerations = 1; Config.Client.concurrentThumbnailGenerations = 1;
} }
this.ThumbnailFunction = RendererFactory.build(Config.Server.thumbnail.processingLibrary);
if (Config.Server.enableThreading == true && if (Config.Server.enableThreading == true &&
Config.Server.thumbnail.processingLibrary == ThumbnailProcessingLib.Jimp) { Config.Server.thumbnail.processingLibrary == ThumbnailProcessingLib.Jimp) {
this.threadPool = new ThumbnailTH(Config.Client.concurrentThumbnailGenerations); this.taskQue = new ThumbnailTH(Config.Client.concurrentThumbnailGenerations);
} else {
this.taskQue = new TaskQue(Config.Client.concurrentThumbnailGenerations);
} }
this.initDone = true; this.initDone = true;
@ -157,13 +157,8 @@ export class ThumbnailGeneratorMWs {
qualityPriority: Config.Server.thumbnail.qualityPriority qualityPriority: Config.Server.thumbnail.qualityPriority
}; };
try { try {
if (this.threadPool !== null) { await this.taskQue.execute(input);
await this.threadPool.execute(input);
return next(); return next();
} else {
await ThumbnailGeneratorMWs.ThumbnailFunction(input);
return next();
}
} catch (error) { } catch (error) {
return next(new ErrorDTO(ErrorCodes.THUMBNAIL_GENERATION_ERROR, "Error during generating thumbnail", error)); return next(new ErrorDTO(ErrorCodes.THUMBNAIL_GENERATION_ERROR, "Error during generating thumbnail", error));
} }

View File

@ -65,7 +65,6 @@ export class UserManager implements IUserManager {
users.splice(i, 1); users.splice(i, 1);
} }
} }
console.log(users);
return users; return users;
} }

View File

@ -35,7 +35,7 @@ export class DiskMangerWorker {
const absoluteDirectoryName = path.join(ProjectPath.ImageFolder, relativeDirectoryName); const absoluteDirectoryName = path.join(ProjectPath.ImageFolder, relativeDirectoryName);
const stat = fs.statSync(path.join(ProjectPath.ImageFolder, relativeDirectoryName)); const stat = fs.statSync(path.join(ProjectPath.ImageFolder, relativeDirectoryName));
let directory = <DirectoryDTO>{ const directory = <DirectoryDTO>{
name: directoryName, name: directoryName,
path: directoryParent, path: directoryParent,
lastModified: Math.max(stat.ctime.getTime(), stat.mtime.getTime()), lastModified: Math.max(stat.ctime.getTime(), stat.mtime.getTime()),
@ -51,8 +51,8 @@ export class DiskMangerWorker {
try { try {
for (let i = 0; i < list.length; i++) { for (let i = 0; i < list.length; i++) {
let file = list[i]; const file = list[i];
let fullFilePath = path.normalize(path.resolve(absoluteDirectoryName, file)); const fullFilePath = path.normalize(path.resolve(absoluteDirectoryName, file));
if (photosOnly == false && fs.statSync(fullFilePath).isDirectory()) { if (photosOnly == false && fs.statSync(fullFilePath).isDirectory()) {
const d = await DiskMangerWorker.scanDirectory(path.join(relativeDirectoryName, file), const d = await DiskMangerWorker.scanDirectory(path.join(relativeDirectoryName, file),
Config.Server.folderPreviewSize, true Config.Server.folderPreviewSize, true

View File

@ -0,0 +1,46 @@
import {RendererInput, ThumbnailWoker} from "./ThumbnailWoker";
import {Config} from "../../../common/config/private/Config";
interface QueTask {
data: RendererInput;
promise: { resolve: Function, reject: Function };
}
export interface ITaskQue {
execute(input: any): Promise<any>;
}
export class TaskQue implements ITaskQue {
private tasks: QueTask[] = [];
private taskInProgress: number = 0;
private run = async () => {
if (this.tasks.length == 0 || this.taskInProgress >= this.size) {
return;
}
this.taskInProgress++;
const task = this.tasks.shift();
try {
task.promise.resolve(await ThumbnailWoker.render(task.data, Config.Server.thumbnail.processingLibrary));
} catch (err) {
task.promise.reject(err);
}
this.taskInProgress--;
process.nextTick(this.run);
};
constructor(private size: number) {
}
execute(input: RendererInput): Promise<void> {
return new Promise((resolve: Function, reject: Function) => {
this.tasks.push({
data: input,
promise: {resolve: resolve, reject: reject}
});
this.run();
});
}
}

View File

@ -4,6 +4,7 @@ import {DiskManagerTask, ThumbnailTask, WorkerMessage, WorkerTask, WorkerTaskTyp
import {DirectoryDTO} from "../../../common/entities/DirectoryDTO"; import {DirectoryDTO} from "../../../common/entities/DirectoryDTO";
import {RendererInput} from "./ThumbnailWoker"; import {RendererInput} from "./ThumbnailWoker";
import {Config} from "../../../common/config/private/Config"; import {Config} from "../../../common/config/private/Config";
import {ITaskQue} from "./TaskQue";
interface PoolTask { interface PoolTask {
@ -26,7 +27,6 @@ export class ThreadPool {
Logger.silly("Creating thread pool with", size, "workers"); Logger.silly("Creating thread pool with", size, "workers");
for (let i = 0; i < size; i++) { for (let i = 0; i < size; i++) {
this.startWorker(); this.startWorker();
} }
} }
@ -83,14 +83,14 @@ export class ThreadPool {
return; return;
} }
const poolTask = this.tasks.pop(); const poolTask = this.tasks.shift();
worker.poolTask = poolTask; worker.poolTask = poolTask;
worker.worker.send(poolTask.task); worker.worker.send(poolTask.task);
}; };
} }
export class DiskManagerTH extends ThreadPool { export class DiskManagerTH extends ThreadPool implements ITaskQue {
execute(relativeDirectoryName: string): Promise<DirectoryDTO> { execute(relativeDirectoryName: string): Promise<DirectoryDTO> {
return super.executeTask(<DiskManagerTask>{ return super.executeTask(<DiskManagerTask>{
type: WorkerTaskTypes.diskManager, type: WorkerTaskTypes.diskManager,
@ -99,7 +99,7 @@ export class DiskManagerTH extends ThreadPool {
} }
} }
export class ThumbnailTH extends ThreadPool { export class ThumbnailTH extends ThreadPool implements ITaskQue {
execute(input: RendererInput): Promise<void> { execute(input: RendererInput): Promise<void> {
return super.executeTask(<ThumbnailTask>{ return super.executeTask(<ThumbnailTask>{
type: WorkerTaskTypes.thumbnail, type: WorkerTaskTypes.thumbnail,

View File

@ -2,6 +2,7 @@ import {DiskMangerWorker} from "./DiskMangerWorker";
import {Logger} from "../../Logger"; import {Logger} from "../../Logger";
import {RendererInput, ThumbnailWoker} from "./ThumbnailWoker"; import {RendererInput, ThumbnailWoker} from "./ThumbnailWoker";
import {ThumbnailProcessingLib} from "../../../common/config/private/IPrivateConfig"; import {ThumbnailProcessingLib} from "../../../common/config/private/IPrivateConfig";
export class Worker { export class Worker {
@ -13,6 +14,9 @@ export class Worker {
switch (task.type) { switch (task.type) {
case WorkerTaskTypes.diskManager: case WorkerTaskTypes.diskManager:
result = await DiskMangerWorker.scanDirectory((<DiskManagerTask>task).relativeDirectoryName); result = await DiskMangerWorker.scanDirectory((<DiskManagerTask>task).relativeDirectoryName);
if (global.gc) {
global.gc();
}
break; break;
case WorkerTaskTypes.thumbnail: case WorkerTaskTypes.thumbnail:
result = await ThumbnailWoker.render((<ThumbnailTask>task).input, (<ThumbnailTask>task).renderer); result = await ThumbnailWoker.render((<ThumbnailTask>task).input, (<ThumbnailTask>task).renderer);
@ -34,7 +38,7 @@ export class Worker {
} }
export enum WorkerTaskTypes{ export enum WorkerTaskTypes {
thumbnail, diskManager thumbnail, diskManager
} }

View File

@ -16,7 +16,8 @@ import {ThumbnailGeneratorMWs} from "./middlewares/thumbnail/ThumbnailGeneratorM
import {DiskManager} from "./model/DiskManger"; import {DiskManager} from "./model/DiskManger";
import {NotificationRouter} from "./routes/NotificationRouter"; import {NotificationRouter} from "./routes/NotificationRouter";
import {ConfigDiagnostics} from "./model/ConfigDiagnostics"; import {ConfigDiagnostics} from "./model/ConfigDiagnostics";
import _session = require('cookie-session');
const _session = require('cookie-session');
const LOG_TAG = "[server]"; const LOG_TAG = "[server]";

16
backend/tsconfig.json Normal file
View File

@ -0,0 +1,16 @@
{
"compileOnSave": true,
"compilerOptions": {
"module": "commonjs",
"skipLibCheck": true,
"sourceMap": false,
"declaration": false,
"moduleResolution": "node",
"emitDecoratorMetadata": true,
"experimentalDecorators": true,
"target": "es6",
"typeRoots": [
"../node_modules/@types"
]
}
}