mirror of
https://github.com/bpatrik/pigallery2.git
synced 2025-01-12 04:23:09 +02:00
implementing job history saving
This commit is contained in:
parent
47b1aa7b86
commit
ddb734e64a
@ -5,6 +5,7 @@ import {DiskMangerWorker} from '../src/backend/model/threading/DiskMangerWorker'
|
|||||||
import {IndexingManager} from '../src/backend/model/database/sql/IndexingManager';
|
import {IndexingManager} from '../src/backend/model/database/sql/IndexingManager';
|
||||||
import {SearchManager} from '../src/backend/model/database/sql/SearchManager';
|
import {SearchManager} from '../src/backend/model/database/sql/SearchManager';
|
||||||
import * as fs from 'fs';
|
import * as fs from 'fs';
|
||||||
|
import * as path from 'path';
|
||||||
import {SearchTypes} from '../src/common/entities/AutoCompleteItem';
|
import {SearchTypes} from '../src/common/entities/AutoCompleteItem';
|
||||||
import {Utils} from '../src/common/Utils';
|
import {Utils} from '../src/common/Utils';
|
||||||
import {GalleryManager} from '../src/backend/model/database/sql/GalleryManager';
|
import {GalleryManager} from '../src/backend/model/database/sql/GalleryManager';
|
||||||
@ -123,7 +124,7 @@ export class Benchmarks {
|
|||||||
fs.unlinkSync(this.dbPath);
|
fs.unlinkSync(this.dbPath);
|
||||||
}
|
}
|
||||||
Config.Server.Database.type = ServerConfig.DatabaseType.sqlite;
|
Config.Server.Database.type = ServerConfig.DatabaseType.sqlite;
|
||||||
Config.Server.Database.sqlite.storage = this.dbPath;
|
Config.Server.Database.dbFolder = path.dirname(this.dbPath);
|
||||||
await ObjectManagers.InitSQLManagers();
|
await ObjectManagers.InitSQLManagers();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -16,8 +16,7 @@ ENTRYPOINT ["npm", "start", "--", \
|
|||||||
# after a extensive job (like video converting), pigallery calls gc, to clean up everthing as fast as possible
|
# after a extensive job (like video converting), pigallery calls gc, to clean up everthing as fast as possible
|
||||||
"--expose-gc", \
|
"--expose-gc", \
|
||||||
"--config-path=/app/data/config/config.json", \
|
"--config-path=/app/data/config/config.json", \
|
||||||
"--Server-Database-sqlite-storage=/app/data/db/sqlite.db", \
|
"--Server-Database-dbFolder=/app/data/db", \
|
||||||
"--Server-Database-memory-usersFile=/app/data/db/users.db", \
|
|
||||||
"--Server-Media-folder=/app/data/images", \
|
"--Server-Media-folder=/app/data/images", \
|
||||||
"--Server-Media-tempFolder=/app/data/tmp"]
|
"--Server-Media-tempFolder=/app/data/tmp"]
|
||||||
EXPOSE 80
|
EXPOSE 80
|
||||||
|
@ -15,8 +15,7 @@ ENTRYPOINT ["npm", "start", "--", \
|
|||||||
# after a extensive job (like video converting), pigallery calls gc, to clean up everthing as fast as possible
|
# after a extensive job (like video converting), pigallery calls gc, to clean up everthing as fast as possible
|
||||||
"--expose-gc", \
|
"--expose-gc", \
|
||||||
"--config-path=/app/data/config/config.json", \
|
"--config-path=/app/data/config/config.json", \
|
||||||
"--Server-Database-sqlite-storage=/app/data/db/sqlite.db", \
|
"--Server-Database-dbFolder=/app/data/db", \
|
||||||
"--Server-Database-memory-usersFile=/app/data/db/users.db", \
|
|
||||||
"--Server-Media-folder=/app/data/images", \
|
"--Server-Media-folder=/app/data/images", \
|
||||||
"--Server-Media-tempFolder=/app/data/tmp"]
|
"--Server-Media-tempFolder=/app/data/tmp"]
|
||||||
EXPOSE 80
|
EXPOSE 80
|
||||||
|
@ -18,8 +18,7 @@ ENTRYPOINT ["npm", "start", "--", \
|
|||||||
# after a extensive job (like video converting), pigallery calls gc, to clean up everthing as fast as possible
|
# after a extensive job (like video converting), pigallery calls gc, to clean up everthing as fast as possible
|
||||||
"--expose-gc", \
|
"--expose-gc", \
|
||||||
"--config-path=/app/data/config/config.json", \
|
"--config-path=/app/data/config/config.json", \
|
||||||
"--Server-Database-sqlite-storage=/app/data/db/sqlite.db", \
|
"--Server-Database-dbFolder=/app/data/db", \
|
||||||
"--Server-Database-memory-usersFile=/app/data/db/users.db", \
|
|
||||||
"--Server-Media-folder=/app/data/images", \
|
"--Server-Media-folder=/app/data/images", \
|
||||||
"--Server-Media-tempFolder=/app/data/tmp"]
|
"--Server-Media-tempFolder=/app/data/tmp"]
|
||||||
EXPOSE 80
|
EXPOSE 80
|
||||||
|
@ -15,7 +15,7 @@ export class PhotoConverterMWs {
|
|||||||
}
|
}
|
||||||
const fullMediaPath = req.resultPipe;
|
const fullMediaPath = req.resultPipe;
|
||||||
|
|
||||||
const convertedVideo = PhotoProcessing.generateConvertedFilePath(fullMediaPath);
|
const convertedVideo = PhotoProcessing.generateConvertedPath(fullMediaPath, Config.Server.Media.Photo.Converting.resolution);
|
||||||
|
|
||||||
// check if transcoded video exist
|
// check if transcoded video exist
|
||||||
if (fs.existsSync(convertedVideo) === true) {
|
if (fs.existsSync(convertedVideo) === true) {
|
||||||
@ -24,8 +24,7 @@ export class PhotoConverterMWs {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (Config.Server.Media.Photo.Converting.onTheFly === true) {
|
if (Config.Server.Media.Photo.Converting.onTheFly === true) {
|
||||||
req.resultPipe = await PhotoProcessing.convertPhoto(fullMediaPath,
|
req.resultPipe = await PhotoProcessing.convertPhoto(fullMediaPath);
|
||||||
Config.Server.Media.Photo.Converting.resolution);
|
|
||||||
return next();
|
return next();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -149,7 +149,7 @@ export class ThumbnailGeneratorMWs {
|
|||||||
const fullMediaPath = path.join(ProjectPath.ImageFolder, photos[i].directory.path, photos[i].directory.name, photos[i].name);
|
const fullMediaPath = path.join(ProjectPath.ImageFolder, photos[i].directory.path, photos[i].directory.name, photos[i].name);
|
||||||
for (let j = 0; j < Config.Client.Media.Thumbnail.thumbnailSizes.length; j++) {
|
for (let j = 0; j < Config.Client.Media.Thumbnail.thumbnailSizes.length; j++) {
|
||||||
const size = Config.Client.Media.Thumbnail.thumbnailSizes[j];
|
const size = Config.Client.Media.Thumbnail.thumbnailSizes[j];
|
||||||
const thPath = PhotoProcessing.generateThumbnailPath(fullMediaPath, size);
|
const thPath = PhotoProcessing.generateConvertedPath(fullMediaPath, size);
|
||||||
if (fs.existsSync(thPath) === true) {
|
if (fs.existsSync(thPath) === true) {
|
||||||
if (typeof photos[i].readyThumbnails === 'undefined') {
|
if (typeof photos[i].readyThumbnails === 'undefined') {
|
||||||
photos[i].readyThumbnails = [];
|
photos[i].readyThumbnails = [];
|
||||||
@ -157,7 +157,7 @@ export class ThumbnailGeneratorMWs {
|
|||||||
photos[i].readyThumbnails.push(size);
|
photos[i].readyThumbnails.push(size);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const iconPath = PhotoProcessing.generateThumbnailPath(fullMediaPath, Config.Client.Media.Thumbnail.iconSize);
|
const iconPath = PhotoProcessing.generateConvertedPath(fullMediaPath, Config.Client.Media.Thumbnail.iconSize);
|
||||||
if (fs.existsSync(iconPath) === true) {
|
if (fs.existsSync(iconPath) === true) {
|
||||||
photos[i].readyIcon = true;
|
photos[i].readyIcon = true;
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
import {JobProgressDTO} from '../../../../common/entities/job/JobProgressDTO';
|
import {JobProgressDTO} from '../../../../common/entities/job/JobProgressDTO';
|
||||||
import {JobDTO} from '../../../../common/entities/job/JobDTO';
|
import {JobDTO} from '../../../../common/entities/job/JobDTO';
|
||||||
import {JobLastRunDTO} from '../../../../common/entities/job/JobLastRunDTO';
|
|
||||||
|
|
||||||
export interface IJobManager {
|
export interface IJobManager {
|
||||||
|
|
||||||
@ -17,5 +16,5 @@ export interface IJobManager {
|
|||||||
|
|
||||||
runSchedules(): void;
|
runSchedules(): void;
|
||||||
|
|
||||||
getJobLastRuns(): { [key: string]: { [key: string]: JobLastRunDTO } };
|
getJobLastRuns(): { [key: string]: JobProgressDTO };
|
||||||
}
|
}
|
||||||
|
@ -3,6 +3,7 @@ import {IUserManager} from '../interfaces/IUserManager';
|
|||||||
import {ProjectPath} from '../../../ProjectPath';
|
import {ProjectPath} from '../../../ProjectPath';
|
||||||
import {Utils} from '../../../../common/Utils';
|
import {Utils} from '../../../../common/Utils';
|
||||||
import * as fs from 'fs';
|
import * as fs from 'fs';
|
||||||
|
import * as path from 'path';
|
||||||
import {PasswordHelper} from '../../PasswordHelper';
|
import {PasswordHelper} from '../../PasswordHelper';
|
||||||
import {Config} from '../../../../common/config/private/Config';
|
import {Config} from '../../../../common/config/private/Config';
|
||||||
|
|
||||||
@ -13,7 +14,7 @@ export class UserManager implements IUserManager {
|
|||||||
|
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
this.dbPath = ProjectPath.getAbsolutePath(Config.Server.Database.memory.usersFile);
|
this.dbPath = path.join(ProjectPath.getAbsolutePath(Config.Server.Database.dbFolder), 'users.db');
|
||||||
if (fs.existsSync(this.dbPath)) {
|
if (fs.existsSync(this.dbPath)) {
|
||||||
this.loadDB();
|
this.loadDB();
|
||||||
}
|
}
|
||||||
|
@ -17,6 +17,7 @@ import {FileEntity} from './enitites/FileEntity';
|
|||||||
import {FaceRegionEntry} from './enitites/FaceRegionEntry';
|
import {FaceRegionEntry} from './enitites/FaceRegionEntry';
|
||||||
import {PersonEntry} from './enitites/PersonEntry';
|
import {PersonEntry} from './enitites/PersonEntry';
|
||||||
import {Utils} from '../../../../common/Utils';
|
import {Utils} from '../../../../common/Utils';
|
||||||
|
import * as path from 'path';
|
||||||
import {ServerConfig} from '../../../../common/config/private/IPrivateConfig';
|
import {ServerConfig} from '../../../../common/config/private/IPrivateConfig';
|
||||||
|
|
||||||
|
|
||||||
@ -178,11 +179,14 @@ export class SQLConnection {
|
|||||||
} else if (config.type === ServerConfig.DatabaseType.sqlite) {
|
} else if (config.type === ServerConfig.DatabaseType.sqlite) {
|
||||||
driver = {
|
driver = {
|
||||||
type: 'sqlite',
|
type: 'sqlite',
|
||||||
database: ProjectPath.getAbsolutePath(config.sqlite.storage)
|
database: path.join(ProjectPath.getAbsolutePath(config.dbFolder), 'sqlite.db')
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
return driver;
|
return driver;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static getSQLiteDB(config: ServerConfig.DataBaseConfig){
|
||||||
|
return path.join(ProjectPath.getAbsolutePath(config.dbFolder), 'sqlite.db');
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -31,10 +31,9 @@ export class ConfigDiagnostics {
|
|||||||
}
|
}
|
||||||
if (databaseConfig.type !== ServerConfig.DatabaseType.sqlite) {
|
if (databaseConfig.type !== ServerConfig.DatabaseType.sqlite) {
|
||||||
try {
|
try {
|
||||||
await this.checkReadWritePermission(ProjectPath.getAbsolutePath(databaseConfig.sqlite.storage));
|
await this.checkReadWritePermission(SQLConnection.getSQLiteDB(databaseConfig));
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
throw new Error('Cannot read or write sqlite storage file: ' +
|
throw new Error('Cannot read or write sqlite storage file: ' + SQLConnection.getSQLiteDB(databaseConfig));
|
||||||
ProjectPath.getAbsolutePath(databaseConfig.sqlite.storage));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -98,7 +98,7 @@ export class PhotoProcessing {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public static generateThumbnailPath(mediaPath: string, size: number): string {
|
public static generateConvertedPath(mediaPath: string, size: number): string {
|
||||||
const file = path.basename(mediaPath);
|
const file = path.basename(mediaPath);
|
||||||
return path.join(ProjectPath.TranscodedFolder,
|
return path.join(ProjectPath.TranscodedFolder,
|
||||||
ProjectPath.getRelativePathToImages(path.dirname(mediaPath)),
|
ProjectPath.getRelativePathToImages(path.dirname(mediaPath)),
|
||||||
@ -111,9 +111,6 @@ export class PhotoProcessing {
|
|||||||
.digest('hex') + '_' + size + '.jpg');
|
.digest('hex') + '_' + size + '.jpg');
|
||||||
}
|
}
|
||||||
|
|
||||||
public static generateConvertedFilePath(photoPath: string): string {
|
|
||||||
return this.generateThumbnailPath(photoPath, Config.Server.Media.Photo.Converting.resolution);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static async isValidConvertedPath(convertedPath: string): Promise<boolean> {
|
public static async isValidConvertedPath(convertedPath: string): Promise<boolean> {
|
||||||
const origFilePath = path.join(ProjectPath.ImageFolder,
|
const origFilePath = path.join(ProjectPath.ImageFolder,
|
||||||
@ -142,42 +139,35 @@ export class PhotoProcessing {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public static async convertPhoto(mediaPath: string, size: number) {
|
public static async convertPhoto(mediaPath: string) {
|
||||||
|
return this.generateThumbnail(mediaPath,
|
||||||
|
Config.Server.Media.Photo.Converting.resolution,
|
||||||
|
ThumbnailSourceType.Photo,
|
||||||
|
false);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static async convertedPhotoExist(mediaPath: string, size: number) {
|
||||||
|
|
||||||
// generate thumbnail path
|
// generate thumbnail path
|
||||||
const outPath = PhotoProcessing.generateConvertedFilePath(mediaPath);
|
const outPath = PhotoProcessing.generateConvertedPath(mediaPath, size);
|
||||||
|
|
||||||
|
|
||||||
// check if file already exist
|
// check if file already exist
|
||||||
try {
|
try {
|
||||||
await fsp.access(outPath, fsConstants.R_OK);
|
await fsp.access(outPath, fsConstants.R_OK);
|
||||||
return outPath;
|
return true;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
}
|
}
|
||||||
|
return false;
|
||||||
|
|
||||||
// run on other thread
|
|
||||||
const input = <RendererInput>{
|
|
||||||
type: ThumbnailSourceType.Photo,
|
|
||||||
mediaPath: mediaPath,
|
|
||||||
size: size,
|
|
||||||
outPath: outPath,
|
|
||||||
makeSquare: false,
|
|
||||||
qualityPriority: Config.Server.Media.Thumbnail.qualityPriority
|
|
||||||
};
|
|
||||||
|
|
||||||
const outDir = path.dirname(input.outPath);
|
|
||||||
|
|
||||||
await fsp.mkdir(outDir, {recursive: true});
|
|
||||||
await this.taskQue.execute(input);
|
|
||||||
return outPath;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async generateThumbnail(mediaPath: string,
|
public static async generateThumbnail(mediaPath: string,
|
||||||
size: number,
|
size: number,
|
||||||
sourceType: ThumbnailSourceType,
|
sourceType: ThumbnailSourceType,
|
||||||
makeSquare: boolean) {
|
makeSquare: boolean): Promise<string> {
|
||||||
// generate thumbnail path
|
// generate thumbnail path
|
||||||
const outPath = PhotoProcessing.generateThumbnailPath(mediaPath, size);
|
const outPath = PhotoProcessing.generateConvertedPath(mediaPath, size);
|
||||||
|
|
||||||
|
|
||||||
// check if file already exist
|
// check if file already exist
|
||||||
@ -209,5 +199,6 @@ export class PhotoProcessing {
|
|||||||
const extension = path.extname(fullPath).toLowerCase();
|
const extension = path.extname(fullPath).toLowerCase();
|
||||||
return SupportedFormats.WithDots.Photos.indexOf(extension) !== -1;
|
return SupportedFormats.WithDots.Photos.indexOf(extension) !== -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -40,6 +40,19 @@ export class VideoProcessing {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static async convertedVideoExist(videoPath: string): Promise<boolean> {
|
||||||
|
const outPath = this.generateConvertedFilePath(videoPath);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await fsp.access(outPath, fsConstants.R_OK);
|
||||||
|
return true;
|
||||||
|
} catch (e) {
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
public static async convertVideo(videoPath: string): Promise<void> {
|
public static async convertVideo(videoPath: string): Promise<void> {
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,50 +1,40 @@
|
|||||||
import {IJobManager} from '../database/interfaces/IJobManager';
|
import {IJobManager} from '../database/interfaces/IJobManager';
|
||||||
import {JobProgressDTO} from '../../../common/entities/job/JobProgressDTO';
|
import {JobProgressDTO, JobProgressStates} from '../../../common/entities/job/JobProgressDTO';
|
||||||
import {IJob} from './jobs/IJob';
|
import {IJob} from './jobs/IJob';
|
||||||
import {JobRepository} from './JobRepository';
|
import {JobRepository} from './JobRepository';
|
||||||
import {Config} from '../../../common/config/private/Config';
|
import {Config} from '../../../common/config/private/Config';
|
||||||
import {AfterJobTrigger, JobScheduleDTO, JobTriggerType} from '../../../common/entities/job/JobScheduleDTO';
|
import {AfterJobTrigger, JobScheduleDTO, JobTriggerType} from '../../../common/entities/job/JobScheduleDTO';
|
||||||
import {Logger} from '../../Logger';
|
import {Logger} from '../../Logger';
|
||||||
import {NotificationManager} from '../NotifocationManager';
|
import {NotificationManager} from '../NotifocationManager';
|
||||||
import {JobLastRunDTO, JobLastRunState} from '../../../common/entities/job/JobLastRunDTO';
|
import {IJobListener} from './jobs/IJobListener';
|
||||||
|
import {JobProgress} from './jobs/JobProgress';
|
||||||
|
import {JobProgressManager} from './JobProgressManager';
|
||||||
|
|
||||||
declare var global: NodeJS.Global;
|
|
||||||
|
|
||||||
const LOG_TAG = '[JobManager]';
|
const LOG_TAG = '[JobManager]';
|
||||||
|
|
||||||
export class JobManager implements IJobManager {
|
export class JobManager implements IJobManager, IJobListener {
|
||||||
|
|
||||||
protected timers: { schedule: JobScheduleDTO, timer: NodeJS.Timeout }[] = [];
|
protected timers: { schedule: JobScheduleDTO, timer: NodeJS.Timeout }[] = [];
|
||||||
|
protected progressManager: JobProgressManager = null;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
|
this.progressManager = new JobProgressManager();
|
||||||
this.runSchedules();
|
this.runSchedules();
|
||||||
}
|
}
|
||||||
|
|
||||||
getProgresses(): { [id: string]: JobProgressDTO } {
|
getProgresses(): { [id: string]: JobProgressDTO } {
|
||||||
const m: { [id: string]: JobProgressDTO } = {};
|
return this.progressManager.Running;
|
||||||
JobRepository.Instance.getAvailableJobs()
|
|
||||||
.filter(t => t.Progress)
|
|
||||||
.forEach(t => {
|
|
||||||
t.Progress.time.current = Date.now();
|
|
||||||
m[t.Name] = t.Progress;
|
|
||||||
});
|
|
||||||
return m;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
getJobLastRuns(): { [key: string]: { [key: string]: JobLastRunDTO } } {
|
getJobLastRuns(): { [key: string]: JobProgressDTO } {
|
||||||
const m: { [id: string]: { [id: string]: JobLastRunDTO } } = {};
|
return this.progressManager.Finished;
|
||||||
JobRepository.Instance.getAvailableJobs().forEach(t => {
|
|
||||||
m[t.Name] = t.LastRuns;
|
|
||||||
});
|
|
||||||
return m;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async run<T>(jobName: string, config: T): Promise<void> {
|
async run<T>(jobName: string, config: T): Promise<void> {
|
||||||
const t = this.findJob(jobName);
|
const t = this.findJob(jobName);
|
||||||
if (t) {
|
if (t) {
|
||||||
await t.start(config, (status: JobLastRunState) => {
|
t.JobListener = this;
|
||||||
this.onJobFinished(t, status);
|
await t.start(config);
|
||||||
});
|
|
||||||
} else {
|
} else {
|
||||||
Logger.warn(LOG_TAG, 'cannot find job to start:' + jobName);
|
Logger.warn(LOG_TAG, 'cannot find job to start:' + jobName);
|
||||||
}
|
}
|
||||||
@ -53,14 +43,19 @@ export class JobManager implements IJobManager {
|
|||||||
stop(jobName: string): void {
|
stop(jobName: string): void {
|
||||||
const t = this.findJob(jobName);
|
const t = this.findJob(jobName);
|
||||||
if (t) {
|
if (t) {
|
||||||
t.stop();
|
t.cancel();
|
||||||
} else {
|
} else {
|
||||||
Logger.warn(LOG_TAG, 'cannot find job to stop:' + jobName);
|
Logger.warn(LOG_TAG, 'cannot find job to stop:' + jobName);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async onJobFinished(job: IJob<any>, status: JobLastRunState): Promise<void> {
|
onProgressUpdate = (progress: JobProgress): void => {
|
||||||
if (status === JobLastRunState.canceled) { // if it was cancelled do not start the next one
|
this.progressManager.onJobProgressUpdate(progress.toDTO());
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
onJobFinished = async (job: IJob<any>, state: JobProgressStates): Promise<void> => {
|
||||||
|
if (state !== JobProgressStates.finished) { // if it was not finished peacefully, do not start the next one
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const sch = Config.Server.Jobs.scheduled.find(s => s.jobName === job.Name);
|
const sch = Config.Server.Jobs.scheduled.find(s => s.jobName === job.Name);
|
||||||
@ -75,7 +70,7 @@ export class JobManager implements IJobManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
getAvailableJobs(): IJob<any>[] {
|
getAvailableJobs(): IJob<any>[] {
|
||||||
return JobRepository.Instance.getAvailableJobs();
|
return JobRepository.Instance.getAvailableJobs();
|
||||||
|
83
src/backend/model/jobs/JobProgressManager.ts
Normal file
83
src/backend/model/jobs/JobProgressManager.ts
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
import {promises as fsp} from 'fs';
|
||||||
|
import * as path from 'path';
|
||||||
|
import {ProjectPath} from '../../ProjectPath';
|
||||||
|
import {Config} from '../../../common/config/private/Config';
|
||||||
|
import {JobProgressDTO, JobProgressStates} from '../../../common/entities/job/JobProgressDTO';
|
||||||
|
|
||||||
|
export class JobProgressManager {
|
||||||
|
db: { [key: string]: { progress: JobProgressDTO, timestamp: number } } = {};
|
||||||
|
private readonly dbPath: string;
|
||||||
|
private timer: NodeJS.Timeout = null;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this.dbPath = path.join(ProjectPath.getAbsolutePath(Config.Server.Database.dbFolder), 'jobs.db');
|
||||||
|
this.loadDB().catch(console.error);
|
||||||
|
}
|
||||||
|
|
||||||
|
get Running(): { [key: string]: JobProgressDTO } {
|
||||||
|
const m: { [key: string]: JobProgressDTO } = {};
|
||||||
|
for (const key of Object.keys(this.db)) {
|
||||||
|
if (this.db[key].progress.state === JobProgressStates.running) {
|
||||||
|
m[key] = this.db[key].progress;
|
||||||
|
m[key].time.end = Date.now();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return m;
|
||||||
|
}
|
||||||
|
|
||||||
|
get Finished(): { [key: string]: JobProgressDTO } {
|
||||||
|
const m: { [key: string]: JobProgressDTO } = {};
|
||||||
|
for (const key of Object.keys(this.db)) {
|
||||||
|
if (this.db[key].progress.state !== JobProgressStates.running) {
|
||||||
|
m[key] = this.db[key].progress;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return m;
|
||||||
|
}
|
||||||
|
|
||||||
|
onJobProgressUpdate(progress: JobProgressDTO) {
|
||||||
|
this.db[progress.HashName] = {progress: progress, timestamp: Date.now()};
|
||||||
|
this.delayedSave();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async loadDB() {
|
||||||
|
try {
|
||||||
|
await fsp.access(this.dbPath);
|
||||||
|
} catch (e) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const data = await fsp.readFile(this.dbPath, 'utf8');
|
||||||
|
this.db = JSON.parse(data);
|
||||||
|
|
||||||
|
while (Object.keys(this.db).length > Config.Server.Jobs.maxSavedProgress) {
|
||||||
|
let min: string = null;
|
||||||
|
for (const key of Object.keys(this.db)) {
|
||||||
|
if (min === null || this.db[min].timestamp > this.db[key].timestamp) {
|
||||||
|
min = key;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delete this.db[min];
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const key of Object.keys(this.db)) {
|
||||||
|
if (this.db[key].progress.state === JobProgressStates.running) {
|
||||||
|
this.db[key].progress.state = JobProgressStates.interrupted;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async saveDB() {
|
||||||
|
await fsp.writeFile(this.dbPath, JSON.stringify(this.db));
|
||||||
|
}
|
||||||
|
|
||||||
|
private delayedSave() {
|
||||||
|
if (this.timer !== null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.timer = setTimeout(async () => {
|
||||||
|
this.saveDB().catch(console.error);
|
||||||
|
this.timer = null;
|
||||||
|
}, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,4 +1,3 @@
|
|||||||
import {JobProgressDTO} from '../../../../common/entities/job/JobProgressDTO';
|
|
||||||
import {ObjectManagers} from '../../ObjectManagers';
|
import {ObjectManagers} from '../../ObjectManagers';
|
||||||
import {Config} from '../../../../common/config/private/Config';
|
import {Config} from '../../../../common/config/private/Config';
|
||||||
import {ConfigTemplateEntry, DefaultsJobs} from '../../../../common/entities/job/JobDTO';
|
import {ConfigTemplateEntry, DefaultsJobs} from '../../../../common/entities/job/JobDTO';
|
||||||
@ -19,9 +18,11 @@ export class DBRestJob extends Job {
|
|||||||
protected async init() {
|
protected async init() {
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async step(): Promise<JobProgressDTO> {
|
protected async step(): Promise<boolean> {
|
||||||
|
this.Progress.Left = 1;
|
||||||
|
this.Progress.Processed++;
|
||||||
await ObjectManagers.getInstance().IndexingManager.resetDB();
|
await ObjectManagers.getInstance().IndexingManager.resetDB();
|
||||||
return null;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,4 +1,3 @@
|
|||||||
import {JobProgressDTO} from '../../../../common/entities/job/JobProgressDTO';
|
|
||||||
import {ConfigTemplateEntry} from '../../../../common/entities/job/JobDTO';
|
import {ConfigTemplateEntry} from '../../../../common/entities/job/JobDTO';
|
||||||
import {Job} from './Job';
|
import {Job} from './Job';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
@ -53,14 +52,14 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
|
|||||||
return files;
|
return files;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected abstract async shouldProcess(file: FileDTO): Promise<boolean>;
|
||||||
protected abstract async processFile(file: FileDTO): Promise<void>;
|
protected abstract async processFile(file: FileDTO): Promise<void>;
|
||||||
|
|
||||||
protected async step(): Promise<JobProgressDTO> {
|
protected async step(): Promise<boolean> {
|
||||||
if (this.directoryQueue.length === 0 && this.fileQueue.length === 0) {
|
if (this.directoryQueue.length === 0 && this.fileQueue.length === 0) {
|
||||||
return null;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.progress.time.current = Date.now();
|
|
||||||
if (this.directoryQueue.length > 0) {
|
if (this.directoryQueue.length > 0) {
|
||||||
|
|
||||||
if (this.config.indexedOnly === true &&
|
if (this.config.indexedOnly === true &&
|
||||||
@ -71,24 +70,30 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
|
|||||||
await this.loadADirectoryFromDisk();
|
await this.loadADirectoryFromDisk();
|
||||||
}
|
}
|
||||||
} else if (this.fileQueue.length > 0) {
|
} else if (this.fileQueue.length > 0) {
|
||||||
|
this.Progress.Left = this.fileQueue.length;
|
||||||
const file = this.fileQueue.shift();
|
const file = this.fileQueue.shift();
|
||||||
this.progress.left = this.fileQueue.length;
|
|
||||||
this.progress.progress++;
|
|
||||||
const filePath = path.join(file.directory.path, file.directory.name, file.name);
|
const filePath = path.join(file.directory.path, file.directory.name, file.name);
|
||||||
this.progress.comment = 'processing: ' + filePath;
|
|
||||||
try {
|
try {
|
||||||
await this.processFile(file);
|
if ((await this.shouldProcess(file)) === true) {
|
||||||
|
this.Progress.Processed++;
|
||||||
|
this.Progress.log('processing: ' + filePath);
|
||||||
|
await this.processFile(file);
|
||||||
|
} else {
|
||||||
|
this.Progress.log('skipping: ' + filePath);
|
||||||
|
this.Progress.Skipped++;
|
||||||
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error(e);
|
console.error(e);
|
||||||
Logger.error(LOG_TAG, 'Error during processing file:' + filePath + ', ' + e.toString());
|
Logger.error(LOG_TAG, 'Error during processing file:' + filePath + ', ' + e.toString());
|
||||||
|
this.Progress.log('Error during processing file:' + filePath + ', ' + e.toString());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return this.progress;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async loadADirectoryFromDisk() {
|
private async loadADirectoryFromDisk() {
|
||||||
const directory = this.directoryQueue.shift();
|
const directory = this.directoryQueue.shift();
|
||||||
this.progress.comment = 'scanning directory: ' + directory;
|
this.Progress.log('scanning directory: ' + directory);
|
||||||
const scanned = await DiskManager.scanDirectory(directory, this.scanFilter);
|
const scanned = await DiskManager.scanDirectory(directory, this.scanFilter);
|
||||||
for (let i = 0; i < scanned.directories.length; i++) {
|
for (let i = 0; i < scanned.directories.length; i++) {
|
||||||
this.directoryQueue.push(path.join(scanned.directories[i].path, scanned.directories[i].name));
|
this.directoryQueue.push(path.join(scanned.directories[i].path, scanned.directories[i].name));
|
||||||
@ -106,7 +111,7 @@ export abstract class FileJob<S extends { indexedOnly: boolean } = { indexedOnly
|
|||||||
if (this.scanFilter.noVideo === true && this.scanFilter.noPhoto === true) {
|
if (this.scanFilter.noVideo === true && this.scanFilter.noPhoto === true) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this.progress.comment = 'Loading files from db';
|
this.Progress.log('Loading files from db');
|
||||||
Logger.silly(LOG_TAG, 'Loading files from db');
|
Logger.silly(LOG_TAG, 'Loading files from db');
|
||||||
|
|
||||||
const connection = await SQLConnection.getConnection();
|
const connection = await SQLConnection.getConnection();
|
||||||
|
@ -1,16 +1,16 @@
|
|||||||
import {JobProgressDTO} from '../../../../common/entities/job/JobProgressDTO';
|
|
||||||
import {JobDTO} from '../../../../common/entities/job/JobDTO';
|
import {JobDTO} from '../../../../common/entities/job/JobDTO';
|
||||||
import {JobLastRunDTO, JobLastRunState} from '../../../../common/entities/job/JobLastRunDTO';
|
import {JobProgress} from './JobProgress';
|
||||||
|
import {IJobListener} from './IJobListener';
|
||||||
|
|
||||||
export interface IJob<T> extends JobDTO {
|
export interface IJob<T> extends JobDTO {
|
||||||
Name: string;
|
Name: string;
|
||||||
Supported: boolean;
|
Supported: boolean;
|
||||||
Progress: JobProgressDTO;
|
Progress: JobProgress;
|
||||||
LastRuns: { [key: string]: JobLastRunDTO };
|
JobListener: IJobListener;
|
||||||
|
|
||||||
start(config: T, OnFinishCB: (status: JobLastRunState) => void): Promise<void>;
|
start(config: T): Promise<void>;
|
||||||
|
|
||||||
stop(): void;
|
cancel(): void;
|
||||||
|
|
||||||
toJSON(): JobDTO;
|
toJSON(): JobDTO;
|
||||||
}
|
}
|
||||||
|
9
src/backend/model/jobs/jobs/IJobListener.ts
Normal file
9
src/backend/model/jobs/jobs/IJobListener.ts
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import {JobProgress} from './JobProgress';
|
||||||
|
import {IJob} from './IJob';
|
||||||
|
import {JobProgressStates} from '../../../../common/entities/job/JobProgressDTO';
|
||||||
|
|
||||||
|
export interface IJobListener {
|
||||||
|
onJobFinished(job: IJob<any>, state: JobProgressStates): void;
|
||||||
|
|
||||||
|
onProgressUpdate(progress: JobProgress): void;
|
||||||
|
}
|
@ -1,10 +1,10 @@
|
|||||||
import {JobProgressDTO, JobState} from '../../../../common/entities/job/JobProgressDTO';
|
|
||||||
import {ObjectManagers} from '../../ObjectManagers';
|
import {ObjectManagers} from '../../ObjectManagers';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import {Config} from '../../../../common/config/private/Config';
|
import {Config} from '../../../../common/config/private/Config';
|
||||||
import {Job} from './Job';
|
import {Job} from './Job';
|
||||||
import {ConfigTemplateEntry, DefaultsJobs} from '../../../../common/entities/job/JobDTO';
|
import {ConfigTemplateEntry, DefaultsJobs} from '../../../../common/entities/job/JobDTO';
|
||||||
import {ServerConfig} from '../../../../common/config/private/IPrivateConfig';
|
import {ServerConfig} from '../../../../common/config/private/IPrivateConfig';
|
||||||
|
import {JobProgressStates} from '../../../../common/entities/job/JobProgressDTO';
|
||||||
|
|
||||||
declare var global: NodeJS.Global;
|
declare var global: NodeJS.Global;
|
||||||
const LOG_TAG = '[IndexingJob]';
|
const LOG_TAG = '[IndexingJob]';
|
||||||
@ -23,23 +23,22 @@ export class IndexingJob extends Job {
|
|||||||
this.directoriesToIndex.push('/');
|
this.directoriesToIndex.push('/');
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async step(): Promise<JobProgressDTO> {
|
protected async step(): Promise<boolean> {
|
||||||
if (this.directoriesToIndex.length === 0) {
|
if (this.directoriesToIndex.length === 0) {
|
||||||
return null;
|
return false;
|
||||||
}
|
}
|
||||||
const directory = this.directoriesToIndex.shift();
|
const directory = this.directoriesToIndex.shift();
|
||||||
this.progress.comment = directory;
|
this.Progress.log(directory);
|
||||||
this.progress.left = this.directoriesToIndex.length;
|
this.Progress.Left = this.directoriesToIndex.length;
|
||||||
const scanned = await ObjectManagers.getInstance().IndexingManager.indexDirectory(directory);
|
const scanned = await ObjectManagers.getInstance().IndexingManager.indexDirectory(directory);
|
||||||
if (this.state !== JobState.running) {
|
if (this.Progress.State !== JobProgressStates.running) {
|
||||||
return null;
|
return false;
|
||||||
}
|
}
|
||||||
this.progress.progress++;
|
this.Progress.Processed++;
|
||||||
this.progress.time.current = Date.now();
|
|
||||||
for (let i = 0; i < scanned.directories.length; i++) {
|
for (let i = 0; i < scanned.directories.length; i++) {
|
||||||
this.directoriesToIndex.push(path.join(scanned.directories[i].path, scanned.directories[i].name));
|
this.directoriesToIndex.push(path.join(scanned.directories[i].path, scanned.directories[i].name));
|
||||||
}
|
}
|
||||||
return this.progress;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,8 +1,9 @@
|
|||||||
import {JobProgressDTO, JobState} from '../../../../common/entities/job/JobProgressDTO';
|
|
||||||
import {Logger} from '../../../Logger';
|
import {Logger} from '../../../Logger';
|
||||||
import {IJob} from './IJob';
|
import {IJob} from './IJob';
|
||||||
import {ConfigTemplateEntry, JobDTO} from '../../../../common/entities/job/JobDTO';
|
import {ConfigTemplateEntry, JobDTO} from '../../../../common/entities/job/JobDTO';
|
||||||
import {JobLastRunDTO, JobLastRunState} from '../../../../common/entities/job/JobLastRunDTO';
|
import {JobProgress} from './JobProgress';
|
||||||
|
import {IJobListener} from './IJobListener';
|
||||||
|
import {JobProgressStates} from '../../../../common/entities/job/JobProgressDTO';
|
||||||
|
|
||||||
declare const process: any;
|
declare const process: any;
|
||||||
declare const global: any;
|
declare const global: any;
|
||||||
@ -10,13 +11,15 @@ declare const global: any;
|
|||||||
const LOG_TAG = '[JOB]';
|
const LOG_TAG = '[JOB]';
|
||||||
|
|
||||||
export abstract class Job<T = void> implements IJob<T> {
|
export abstract class Job<T = void> implements IJob<T> {
|
||||||
|
protected progress: JobProgress = null;
|
||||||
protected progress: JobProgressDTO = null;
|
|
||||||
protected state = JobState.idle;
|
|
||||||
protected config: T;
|
protected config: T;
|
||||||
protected prResolve: () => void;
|
protected prResolve: () => void;
|
||||||
protected IsInstant = false;
|
protected IsInstant = false;
|
||||||
protected lastRuns: { [key: string]: JobLastRunDTO } = {};
|
private jobListener: IJobListener;
|
||||||
|
|
||||||
|
public set JobListener(value: IJobListener) {
|
||||||
|
this.jobListener = value;
|
||||||
|
}
|
||||||
|
|
||||||
public abstract get Supported(): boolean;
|
public abstract get Supported(): boolean;
|
||||||
|
|
||||||
@ -24,45 +27,25 @@ export abstract class Job<T = void> implements IJob<T> {
|
|||||||
|
|
||||||
public abstract get ConfigTemplate(): ConfigTemplateEntry[];
|
public abstract get ConfigTemplate(): ConfigTemplateEntry[];
|
||||||
|
|
||||||
public get LastRuns(): { [key: string]: JobLastRunDTO } {
|
|
||||||
return this.lastRuns;
|
|
||||||
}
|
|
||||||
|
|
||||||
public get Progress(): JobProgressDTO {
|
public get Progress(): JobProgress {
|
||||||
return this.progress;
|
return this.progress;
|
||||||
}
|
}
|
||||||
|
|
||||||
public start(config: T, onFinishCB: (status: JobLastRunState) => void): Promise<void> {
|
public get CanRun() {
|
||||||
this.OnFinishCB = onFinishCB;
|
return this.Progress == null && this.Supported;
|
||||||
if (this.state === JobState.idle && this.Supported) {
|
}
|
||||||
|
|
||||||
|
public start(config: T): Promise<void> {
|
||||||
|
if (this.CanRun) {
|
||||||
Logger.info(LOG_TAG, 'Running job: ' + this.Name);
|
Logger.info(LOG_TAG, 'Running job: ' + this.Name);
|
||||||
this.config = config;
|
this.config = config;
|
||||||
this.progress = {
|
this.progress = new JobProgress(JobDTO.getHashName(this.Name, this.config));
|
||||||
progress: 0,
|
this.progress.OnChange = this.jobListener.onProgressUpdate;
|
||||||
left: 0,
|
|
||||||
comment: '',
|
|
||||||
state: JobState.running,
|
|
||||||
time: {
|
|
||||||
start: Date.now(),
|
|
||||||
current: Date.now()
|
|
||||||
}
|
|
||||||
};
|
|
||||||
this.lastRuns[JSON.stringify(this.config)] = {
|
|
||||||
all: 0,
|
|
||||||
done: 0,
|
|
||||||
comment: '',
|
|
||||||
config: this.config,
|
|
||||||
state: JobLastRunState.finished,
|
|
||||||
time: {
|
|
||||||
start: Date.now(),
|
|
||||||
end: Date.now()
|
|
||||||
}
|
|
||||||
};
|
|
||||||
const pr = new Promise<void>((resolve) => {
|
const pr = new Promise<void>((resolve) => {
|
||||||
this.prResolve = resolve;
|
this.prResolve = resolve;
|
||||||
});
|
});
|
||||||
this.init().catch(console.error);
|
this.init().catch(console.error);
|
||||||
this.state = JobState.running;
|
|
||||||
this.run();
|
this.run();
|
||||||
if (!this.IsInstant) { // if instant, wait for execution, otherwise, return right away
|
if (!this.IsInstant) { // if instant, wait for execution, otherwise, return right away
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
@ -74,11 +57,9 @@ export abstract class Job<T = void> implements IJob<T> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public stop(): void {
|
public cancel(): void {
|
||||||
Logger.info(LOG_TAG, 'Stopping job: ' + this.Name);
|
Logger.info(LOG_TAG, 'Stopping job: ' + this.Name);
|
||||||
this.state = JobState.stopping;
|
this.Progress.State = JobProgressStates.cancelling;
|
||||||
this.progress.state = JobState.stopping;
|
|
||||||
this.lastRuns[JSON.stringify(this.config)].state = JobLastRunState.canceled;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public toJSON(): JobDTO {
|
public toJSON(): JobDTO {
|
||||||
@ -88,18 +69,19 @@ export abstract class Job<T = void> implements IJob<T> {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
protected OnFinishCB = (status: JobLastRunState) => {
|
|
||||||
};
|
|
||||||
|
|
||||||
protected abstract async step(): Promise<JobProgressDTO>;
|
protected abstract async step(): Promise<boolean>;
|
||||||
|
|
||||||
protected abstract async init(): Promise<void>;
|
protected abstract async init(): Promise<void>;
|
||||||
|
|
||||||
private onFinish(): void {
|
private onFinish(): void {
|
||||||
this.lastRuns[JSON.stringify(this.config)].all = this.progress.left + this.progress.progress;
|
if (this.Progress.State === JobProgressStates.running) {
|
||||||
this.lastRuns[JSON.stringify(this.config)].done = this.progress.progress;
|
this.Progress.State = JobProgressStates.finished;
|
||||||
this.lastRuns[JSON.stringify(this.config)].time.end = Date.now();
|
}
|
||||||
|
if (this.Progress.State === JobProgressStates.cancelling) {
|
||||||
|
this.Progress.State = JobProgressStates.canceled;
|
||||||
|
}
|
||||||
|
const finishState = this.Progress.State;
|
||||||
this.progress = null;
|
this.progress = null;
|
||||||
if (global.gc) {
|
if (global.gc) {
|
||||||
global.gc();
|
global.gc();
|
||||||
@ -108,25 +90,19 @@ export abstract class Job<T = void> implements IJob<T> {
|
|||||||
if (this.IsInstant) {
|
if (this.IsInstant) {
|
||||||
this.prResolve();
|
this.prResolve();
|
||||||
}
|
}
|
||||||
this.OnFinishCB(this.lastRuns[JSON.stringify(this.config)].state);
|
this.jobListener.onJobFinished(this, finishState);
|
||||||
}
|
}
|
||||||
|
|
||||||
private run() {
|
private run() {
|
||||||
process.nextTick(async () => {
|
process.nextTick(async () => {
|
||||||
try {
|
try {
|
||||||
if (this.state === JobState.idle) {
|
if (this.Progress == null || this.Progress.State !== JobProgressStates.running) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let prg = null;
|
if (await this.step() === false) { // finished
|
||||||
if (this.state === JobState.running) {
|
|
||||||
prg = await this.step();
|
|
||||||
}
|
|
||||||
if (prg == null) { // finished
|
|
||||||
this.state = JobState.idle;
|
|
||||||
this.onFinish();
|
this.onFinish();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this.progress = prg;
|
|
||||||
this.run();
|
this.run();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
Logger.error(LOG_TAG, e);
|
Logger.error(LOG_TAG, e);
|
||||||
|
102
src/backend/model/jobs/jobs/JobProgress.ts
Normal file
102
src/backend/model/jobs/jobs/JobProgress.ts
Normal file
@ -0,0 +1,102 @@
|
|||||||
|
import {JobProgressDTO, JobProgressStates} from '../../../../common/entities/job/JobProgressDTO';
|
||||||
|
|
||||||
|
|
||||||
|
export class JobProgress {
|
||||||
|
private steps = {
|
||||||
|
all: 0,
|
||||||
|
processed: 0,
|
||||||
|
skipped: 0,
|
||||||
|
};
|
||||||
|
private state = JobProgressStates.running;
|
||||||
|
private time = {
|
||||||
|
start: <number>Date.now(),
|
||||||
|
end: <number>null,
|
||||||
|
};
|
||||||
|
private logs: string[] = [];
|
||||||
|
|
||||||
|
|
||||||
|
constructor(public readonly HashName: string) {
|
||||||
|
}
|
||||||
|
|
||||||
|
set OnChange(val: (progress: JobProgress) => void) {
|
||||||
|
this.onChange = val;
|
||||||
|
}
|
||||||
|
|
||||||
|
get Skipped(): number {
|
||||||
|
return this.steps.skipped;
|
||||||
|
}
|
||||||
|
|
||||||
|
set Skipped(value: number) {
|
||||||
|
this.steps.skipped = value;
|
||||||
|
this.time.end = Date.now();
|
||||||
|
this.onChange(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
get Processed(): number {
|
||||||
|
return this.steps.processed;
|
||||||
|
}
|
||||||
|
|
||||||
|
set Processed(value: number) {
|
||||||
|
this.steps.processed = value;
|
||||||
|
this.time.end = Date.now();
|
||||||
|
this.onChange(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
get Left(): number {
|
||||||
|
return this.steps.all - this.steps.processed - this.steps.skipped;
|
||||||
|
}
|
||||||
|
|
||||||
|
set Left(value: number) {
|
||||||
|
this.steps.all = value + this.steps.skipped + this.steps.processed;
|
||||||
|
this.time.end = Date.now();
|
||||||
|
this.onChange(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
get All(): number {
|
||||||
|
return this.steps.all;
|
||||||
|
}
|
||||||
|
|
||||||
|
set All(value: number) {
|
||||||
|
this.steps.all = value;
|
||||||
|
this.time.end = Date.now();
|
||||||
|
this.onChange(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
get State(): JobProgressStates {
|
||||||
|
return this.state;
|
||||||
|
}
|
||||||
|
|
||||||
|
set State(value: JobProgressStates) {
|
||||||
|
this.state = value;
|
||||||
|
this.time.end = Date.now();
|
||||||
|
this.onChange(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
get Logs(): string[] {
|
||||||
|
return this.logs;
|
||||||
|
}
|
||||||
|
|
||||||
|
onChange = (progress: JobProgress) => {
|
||||||
|
};
|
||||||
|
|
||||||
|
log(log: string) {
|
||||||
|
while (this.logs.length > 10) {
|
||||||
|
this.logs.shift();
|
||||||
|
}
|
||||||
|
this.logs.push(log);
|
||||||
|
this.onChange(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
toDTO(): JobProgressDTO {
|
||||||
|
return {
|
||||||
|
HashName: this.HashName,
|
||||||
|
state: this.state,
|
||||||
|
time: {
|
||||||
|
start: this.time.start,
|
||||||
|
end: this.time.end
|
||||||
|
},
|
||||||
|
logs: this.logs,
|
||||||
|
steps: this.steps
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
@ -21,11 +21,15 @@ export class PhotoConvertingJob extends FileJob {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
protected async shouldProcess(file: FileDTO): Promise<boolean> {
|
||||||
|
const mPath = path.join(ProjectPath.ImageFolder, file.directory.path, file.directory.name, file.name);
|
||||||
|
return !(await PhotoProcessing.convertedPhotoExist(mPath, Config.Server.Media.Photo.Converting.resolution));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
protected async processFile(file: FileDTO): Promise<void> {
|
protected async processFile(file: FileDTO): Promise<void> {
|
||||||
await PhotoProcessing.convertPhoto(path.join(ProjectPath.ImageFolder,
|
const mPath = path.join(ProjectPath.ImageFolder, file.directory.path, file.directory.name, file.name);
|
||||||
file.directory.path,
|
await PhotoProcessing.convertPhoto(mPath);
|
||||||
file.directory.name,
|
|
||||||
file.name), Config.Server.Media.Photo.Converting.resolution);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -3,7 +3,6 @@ import * as path from 'path';
|
|||||||
import * as util from 'util';
|
import * as util from 'util';
|
||||||
import {promises as fsp} from 'fs';
|
import {promises as fsp} from 'fs';
|
||||||
import {Job} from './Job';
|
import {Job} from './Job';
|
||||||
import {JobProgressDTO} from '../../../../common/entities/job/JobProgressDTO';
|
|
||||||
import {ProjectPath} from '../../../ProjectPath';
|
import {ProjectPath} from '../../../ProjectPath';
|
||||||
import {PhotoProcessing} from '../../fileprocessing/PhotoProcessing';
|
import {PhotoProcessing} from '../../fileprocessing/PhotoProcessing';
|
||||||
import {VideoProcessing} from '../../fileprocessing/VideoProcessing';
|
import {VideoProcessing} from '../../fileprocessing/VideoProcessing';
|
||||||
@ -61,39 +60,38 @@ export class TempFolderCleaningJob extends Job {
|
|||||||
const validFiles = [ProjectPath.TranscodedFolder, ProjectPath.FacesFolder];
|
const validFiles = [ProjectPath.TranscodedFolder, ProjectPath.FacesFolder];
|
||||||
for (let i = 0; i < files.length; ++i) {
|
for (let i = 0; i < files.length; ++i) {
|
||||||
if (validFiles.indexOf(files[i]) === -1) {
|
if (validFiles.indexOf(files[i]) === -1) {
|
||||||
|
this.Progress.Processed++;
|
||||||
if ((await fsp.stat(files[i])).isDirectory()) {
|
if ((await fsp.stat(files[i])).isDirectory()) {
|
||||||
await rimrafPR(files[i]);
|
await rimrafPR(files[i]);
|
||||||
} else {
|
} else {
|
||||||
await fsp.unlink(files[i]);
|
await fsp.unlink(files[i]);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
this.Progress.Skipped++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.progress.time.current = Date.now();
|
|
||||||
|
|
||||||
this.progress.comment = 'processing: ' + ProjectPath.TempFolder;
|
this.Progress.log('processing: ' + ProjectPath.TempFolder);
|
||||||
|
|
||||||
return this.progress;
|
return true;
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async stepConvertedDirectory() {
|
protected async stepConvertedDirectory() {
|
||||||
|
|
||||||
|
|
||||||
this.progress.time.current = Date.now();
|
|
||||||
|
|
||||||
|
|
||||||
const filePath = this.directoryQueue.shift();
|
const filePath = this.directoryQueue.shift();
|
||||||
const stat = await fsp.stat(filePath);
|
const stat = await fsp.stat(filePath);
|
||||||
|
|
||||||
this.progress.left = this.directoryQueue.length;
|
this.Progress.Left = this.directoryQueue.length;
|
||||||
this.progress.progress++;
|
this.Progress.log('processing: ' + filePath);
|
||||||
this.progress.comment = 'processing: ' + filePath;
|
|
||||||
if (stat.isDirectory()) {
|
if (stat.isDirectory()) {
|
||||||
if (await this.isValidDirectory(filePath) === false) {
|
if (await this.isValidDirectory(filePath) === false) {
|
||||||
|
this.Progress.Processed++;
|
||||||
await rimrafPR(filePath);
|
await rimrafPR(filePath);
|
||||||
} else {
|
} else {
|
||||||
|
this.Progress.Skipped++;
|
||||||
this.directoryQueue = this.directoryQueue.concat(await this.readDir(filePath));
|
this.directoryQueue = this.directoryQueue.concat(await this.readDir(filePath));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -101,12 +99,12 @@ export class TempFolderCleaningJob extends Job {
|
|||||||
await fsp.unlink(filePath);
|
await fsp.unlink(filePath);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return this.progress;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async step(): Promise<JobProgressDTO> {
|
protected async step(): Promise<boolean> {
|
||||||
if (this.directoryQueue.length === 0) {
|
if (this.directoryQueue.length === 0) {
|
||||||
return null;
|
return false;
|
||||||
}
|
}
|
||||||
if (this.tempRootCleaned === false) {
|
if (this.tempRootCleaned === false) {
|
||||||
this.tempRootCleaned = true;
|
this.tempRootCleaned = true;
|
||||||
|
@ -7,12 +7,12 @@ import {PhotoProcessing} from '../../fileprocessing/PhotoProcessing';
|
|||||||
import {ThumbnailSourceType} from '../../threading/PhotoWorker';
|
import {ThumbnailSourceType} from '../../threading/PhotoWorker';
|
||||||
import {MediaDTO} from '../../../../common/entities/MediaDTO';
|
import {MediaDTO} from '../../../../common/entities/MediaDTO';
|
||||||
import {FileDTO} from '../../../../common/entities/FileDTO';
|
import {FileDTO} from '../../../../common/entities/FileDTO';
|
||||||
import {JobLastRunState} from '../../../../common/entities/job/JobLastRunDTO';
|
|
||||||
|
|
||||||
const LOG_TAG = '[ThumbnailGenerationJob]';
|
const LOG_TAG = '[ThumbnailGenerationJob]';
|
||||||
|
|
||||||
|
|
||||||
export class ThumbnailGenerationJob extends FileJob<{ sizes: number[], indexedOnly: boolean }> {
|
export class ThumbnailGenerationJob extends FileJob<{ sizes: number[], indexedOnly: boolean }> {
|
||||||
|
|
||||||
public readonly Name = DefaultsJobs[DefaultsJobs['Thumbnail Generation']];
|
public readonly Name = DefaultsJobs[DefaultsJobs['Thumbnail Generation']];
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
@ -29,14 +29,14 @@ export class ThumbnailGenerationJob extends FileJob<{ sizes: number[], indexedOn
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
start(config: { sizes: number[], indexedOnly: boolean }, OnFinishCB: (status: JobLastRunState) => void): Promise<void> {
|
start(config: { sizes: number[], indexedOnly: boolean }): Promise<void> {
|
||||||
for (let i = 0; i < config.sizes.length; ++i) {
|
for (let i = 0; i < config.sizes.length; ++i) {
|
||||||
if (Config.Client.Media.Thumbnail.thumbnailSizes.indexOf(config.sizes[i]) === -1) {
|
if (Config.Client.Media.Thumbnail.thumbnailSizes.indexOf(config.sizes[i]) === -1) {
|
||||||
throw new Error('unknown thumbnails size: ' + config.sizes[i] + '. Add it to the possible thumbnail sizes.');
|
throw new Error('unknown thumbnails size: ' + config.sizes[i] + '. Add it to the possible thumbnail sizes.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return super.start(config, OnFinishCB);
|
return super.start(config);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async filterMediaFiles(files: FileDTO[]): Promise<FileDTO[]> {
|
protected async filterMediaFiles(files: FileDTO[]): Promise<FileDTO[]> {
|
||||||
@ -47,13 +47,22 @@ export class ThumbnailGenerationJob extends FileJob<{ sizes: number[], indexedOn
|
|||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async processFile(media: FileDTO): Promise<void> {
|
protected async shouldProcess(file: FileDTO): Promise<boolean> {
|
||||||
|
const mPath = path.join(ProjectPath.ImageFolder, file.directory.path, file.directory.name, file.name);
|
||||||
|
for (let i = 0; i < this.config.sizes.length; ++i) {
|
||||||
|
if (!(await PhotoProcessing.convertedPhotoExist(mPath, this.config.sizes[i]))) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const mPath = path.join(ProjectPath.ImageFolder, media.directory.path, media.directory.name, media.name);
|
protected async processFile(file: FileDTO): Promise<void> {
|
||||||
|
|
||||||
|
const mPath = path.join(ProjectPath.ImageFolder, file.directory.path, file.directory.name, file.name);
|
||||||
for (let i = 0; i < this.config.sizes.length; ++i) {
|
for (let i = 0; i < this.config.sizes.length; ++i) {
|
||||||
await PhotoProcessing.generateThumbnail(mPath,
|
await PhotoProcessing.generateThumbnail(mPath,
|
||||||
this.config.sizes[i],
|
this.config.sizes[i],
|
||||||
MediaDTO.isVideo(media) ? ThumbnailSourceType.Video : ThumbnailSourceType.Photo,
|
MediaDTO.isVideo(file) ? ThumbnailSourceType.Video : ThumbnailSourceType.Photo,
|
||||||
false);
|
false);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -20,12 +20,14 @@ export class VideoConvertingJob extends FileJob {
|
|||||||
return Config.Client.Media.Video.enabled === true;
|
return Config.Client.Media.Video.enabled === true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected async shouldProcess(file: FileDTO): Promise<boolean> {
|
||||||
|
const mPath = path.join(ProjectPath.ImageFolder, file.directory.path, file.directory.name, file.name);
|
||||||
|
return !(await VideoProcessing.convertedVideoExist(mPath));
|
||||||
|
}
|
||||||
|
|
||||||
protected async processFile(file: FileDTO): Promise<void> {
|
protected async processFile(file: FileDTO): Promise<void> {
|
||||||
await VideoProcessing.convertVideo(path.join(ProjectPath.ImageFolder,
|
const mPath = path.join(ProjectPath.ImageFolder, file.directory.path, file.directory.name, file.name);
|
||||||
file.directory.path,
|
await VideoProcessing.convertVideo(mPath);
|
||||||
file.directory.name,
|
|
||||||
file.name));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -27,19 +27,19 @@ export module ServerConfig {
|
|||||||
password: string;
|
password: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface SQLiteConfig {
|
/*
|
||||||
storage: string;
|
export interface SQLiteConfig {
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface MemoryConfig {
|
export interface MemoryConfig {
|
||||||
usersFile: string;
|
}*/
|
||||||
}
|
|
||||||
|
|
||||||
export interface DataBaseConfig {
|
export interface DataBaseConfig {
|
||||||
type: DatabaseType;
|
type: DatabaseType;
|
||||||
|
dbFolder: string;
|
||||||
mysql?: MySQLConfig;
|
mysql?: MySQLConfig;
|
||||||
sqlite?: SQLiteConfig;
|
// sqlite?: SQLiteConfig;
|
||||||
memory?: MemoryConfig;
|
// memory?: MemoryConfig;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ThumbnailConfig {
|
export interface ThumbnailConfig {
|
||||||
@ -78,6 +78,7 @@ export module ServerConfig {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface JobConfig {
|
export interface JobConfig {
|
||||||
|
maxSavedProgress: number;
|
||||||
scheduled: JobScheduleDTO[];
|
scheduled: JobScheduleDTO[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -43,18 +43,13 @@ export class PrivateConfigDefaultsClass extends PublicConfigClass implements IPr
|
|||||||
photoMetadataSize: 512 * 1024,
|
photoMetadataSize: 512 * 1024,
|
||||||
Database: {
|
Database: {
|
||||||
type: ServerConfig.DatabaseType.sqlite,
|
type: ServerConfig.DatabaseType.sqlite,
|
||||||
|
dbFolder: 'db',
|
||||||
mysql: {
|
mysql: {
|
||||||
host: '',
|
host: '',
|
||||||
username: '',
|
username: '',
|
||||||
password: '',
|
password: '',
|
||||||
database: ''
|
database: ''
|
||||||
|
|
||||||
},
|
|
||||||
sqlite: {
|
|
||||||
storage: 'sqlite.db'
|
|
||||||
},
|
|
||||||
memory: {
|
|
||||||
usersFile: 'user.db'
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
Sharing: {
|
Sharing: {
|
||||||
@ -75,6 +70,7 @@ export class PrivateConfigDefaultsClass extends PublicConfigClass implements IPr
|
|||||||
listingLimit: 1000
|
listingLimit: 1000
|
||||||
},
|
},
|
||||||
Jobs: {
|
Jobs: {
|
||||||
|
maxSavedProgress: 10,
|
||||||
scheduled: [
|
scheduled: [
|
||||||
{
|
{
|
||||||
name: DefaultsJobs[DefaultsJobs.Indexing],
|
name: DefaultsJobs[DefaultsJobs.Indexing],
|
||||||
|
@ -21,3 +21,9 @@ export interface JobDTO {
|
|||||||
Name: string;
|
Name: string;
|
||||||
ConfigTemplate: ConfigTemplateEntry[];
|
ConfigTemplate: ConfigTemplateEntry[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export module JobDTO {
|
||||||
|
export const getHashName = (jobName: string, config: any = {}) => {
|
||||||
|
return jobName + '-' + JSON.stringify(config);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
@ -1,15 +0,0 @@
|
|||||||
export enum JobLastRunState {
|
|
||||||
finished = 1, canceled = 2
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface JobLastRunDTO {
|
|
||||||
config: any;
|
|
||||||
done: number;
|
|
||||||
all: number;
|
|
||||||
state: JobLastRunState;
|
|
||||||
comment: string;
|
|
||||||
time: {
|
|
||||||
start: number,
|
|
||||||
end: number
|
|
||||||
};
|
|
||||||
}
|
|
@ -1,15 +1,19 @@
|
|||||||
export enum JobState {
|
export enum JobProgressStates {
|
||||||
idle = 1, running = 2, stopping = 3
|
running = 1, cancelling = 2, interrupted = 3, canceled = 4, finished = 5
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
export interface JobProgressDTO {
|
export interface JobProgressDTO {
|
||||||
progress: number;
|
HashName: string;
|
||||||
left: number;
|
steps: {
|
||||||
state: JobState;
|
all: number,
|
||||||
comment: string;
|
processed: number,
|
||||||
|
skipped: number,
|
||||||
|
};
|
||||||
|
state: JobProgressStates;
|
||||||
|
logs: string[];
|
||||||
time: {
|
time: {
|
||||||
start: number,
|
start: number,
|
||||||
current: number
|
end: number
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -21,6 +21,17 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group row">
|
||||||
|
<label class="col-md-2 control-label" for="dbFolder" i18n>Database folder</label>
|
||||||
|
<div class="col-md-10">
|
||||||
|
<input type="text" class="form-control" placeholder="db"
|
||||||
|
[(ngModel)]="settings.dbFolder" id="dbFolder" name="dbFolder" required>
|
||||||
|
</div>
|
||||||
|
<small class="form-text text-muted" i18n>
|
||||||
|
All file-based data will be stored here (sqlite database, user database in case of memory db, job history data)
|
||||||
|
</small>
|
||||||
|
</div>
|
||||||
|
|
||||||
<ng-container *ngIf="settings.type == DatabaseType.mysql">
|
<ng-container *ngIf="settings.type == DatabaseType.mysql">
|
||||||
|
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
@ -52,24 +63,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
<ng-container *ngIf="settings.type == DatabaseType.sqlite">
|
|
||||||
<div class="form-group row">
|
|
||||||
<label class="col-md-2 control-label" for="storage" i18n>Storage file</label>
|
|
||||||
<div class="col-md-10">
|
|
||||||
<input type="text" class="form-control" placeholder="sqlite.db"
|
|
||||||
[(ngModel)]="settings.sqlite.storage" id="storage" name="storage" required>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</ng-container>
|
|
||||||
<ng-container *ngIf="settings.type == DatabaseType.memory">
|
|
||||||
<div class="form-group row">
|
|
||||||
<label class="col-md-2 control-label" for="usersFile" i18n>User's file</label>
|
|
||||||
<div class="col-md-10">
|
|
||||||
<input type="text" class="form-control" placeholder="users.db"
|
|
||||||
[(ngModel)]="settings.memory.usersFile" id="usersFile" name="usersFile" required>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</ng-container>
|
|
||||||
|
|
||||||
<button class="btn btn-success float-right"
|
<button class="btn btn-success float-right"
|
||||||
[disabled]="!settingsForm.form.valid || !changed || inProgress"
|
[disabled]="!settingsForm.form.valid || !changed || inProgress"
|
||||||
|
@ -123,7 +123,7 @@
|
|||||||
</button>
|
</button>
|
||||||
<button class="btn btn-secondary ml-0"
|
<button class="btn btn-secondary ml-0"
|
||||||
*ngIf="Progress != null"
|
*ngIf="Progress != null"
|
||||||
[disabled]="inProgress || Progress.state !== JobState.running"
|
[disabled]="inProgress || Progress.state !== JobProgressStates.running"
|
||||||
(click)="cancelIndexing()" i18n>Cancel converting
|
(click)="cancelIndexing()" i18n>Cancel converting
|
||||||
</button>
|
</button>
|
||||||
<button class="btn btn-danger ml-2"
|
<button class="btn btn-danger ml-2"
|
||||||
|
@ -8,9 +8,9 @@ import {SettingsComponent} from '../_abstract/abstract.settings.component';
|
|||||||
import {Utils} from '../../../../../common/Utils';
|
import {Utils} from '../../../../../common/Utils';
|
||||||
import {I18n} from '@ngx-translate/i18n-polyfill';
|
import {I18n} from '@ngx-translate/i18n-polyfill';
|
||||||
import {ScheduledJobsService} from '../scheduled-jobs.service';
|
import {ScheduledJobsService} from '../scheduled-jobs.service';
|
||||||
import {DefaultsJobs} from '../../../../../common/entities/job/JobDTO';
|
import {DefaultsJobs, JobDTO} from '../../../../../common/entities/job/JobDTO';
|
||||||
import {ServerConfig} from '../../../../../common/config/private/IPrivateConfig';
|
import {ServerConfig} from '../../../../../common/config/private/IPrivateConfig';
|
||||||
import {JobState} from '../../../../../common/entities/job/JobProgressDTO';
|
import {JobProgressStates} from '../../../../../common/entities/job/JobProgressDTO';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-settings-indexing',
|
selector: 'app-settings-indexing',
|
||||||
@ -24,7 +24,7 @@ export class IndexingSettingsComponent extends SettingsComponent<ServerConfig.In
|
|||||||
|
|
||||||
|
|
||||||
types: { key: number; value: string }[] = [];
|
types: { key: number; value: string }[] = [];
|
||||||
JobState = JobState;
|
JobProgressStates = JobProgressStates;
|
||||||
|
|
||||||
constructor(_authService: AuthenticationService,
|
constructor(_authService: AuthenticationService,
|
||||||
_navigation: NavigationService,
|
_navigation: NavigationService,
|
||||||
@ -44,7 +44,7 @@ export class IndexingSettingsComponent extends SettingsComponent<ServerConfig.In
|
|||||||
}
|
}
|
||||||
|
|
||||||
get Progress() {
|
get Progress() {
|
||||||
return this.jobsService.progress.value[DefaultsJobs[DefaultsJobs.Indexing]];
|
return this.jobsService.progress.value[JobDTO.getHashName(DefaultsJobs[DefaultsJobs.Indexing])];
|
||||||
}
|
}
|
||||||
|
|
||||||
get excludeFolderList(): string {
|
get excludeFolderList(): string {
|
||||||
|
@ -36,7 +36,7 @@
|
|||||||
</button>
|
</button>
|
||||||
<button class="btn btn-secondary job-control-button"
|
<button class="btn btn-secondary job-control-button"
|
||||||
*ngIf="jobsService.progress.value[schedule.jobName]"
|
*ngIf="jobsService.progress.value[schedule.jobName]"
|
||||||
[disabled]="disableButtons || jobsService.progress.value[schedule.jobName].state !== JobState.running"
|
[disabled]="disableButtons || jobsService.progress.value[schedule.jobName].state !== JobProgressStates.running"
|
||||||
(click)="stop(schedule); $event.stopPropagation();"><span class="oi oi-media-stop"></span>
|
(click)="stop(schedule); $event.stopPropagation();"><span class="oi oi-media-stop"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
@ -129,7 +129,7 @@
|
|||||||
</button>
|
</button>
|
||||||
<button class="btn btn-secondary float-right"
|
<button class="btn btn-secondary float-right"
|
||||||
*ngIf="jobsService.progress.value[schedule.jobName]"
|
*ngIf="jobsService.progress.value[schedule.jobName]"
|
||||||
[disabled]="disableButtons || jobsService.progress.value[schedule.jobName].state !== JobState.running"
|
[disabled]="disableButtons || jobsService.progress.value[schedule.jobName].state !== JobProgressStates.running"
|
||||||
(click)="stop(schedule)" i18n>Stop
|
(click)="stop(schedule)" i18n>Stop
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
@ -17,10 +17,10 @@ import {
|
|||||||
} from '../../../../../common/entities/job/JobScheduleDTO';
|
} from '../../../../../common/entities/job/JobScheduleDTO';
|
||||||
import {Utils} from '../../../../../common/Utils';
|
import {Utils} from '../../../../../common/Utils';
|
||||||
import {ServerConfig} from '../../../../../common/config/private/IPrivateConfig';
|
import {ServerConfig} from '../../../../../common/config/private/IPrivateConfig';
|
||||||
import {ConfigTemplateEntry} from '../../../../../common/entities/job/JobDTO';
|
import {ConfigTemplateEntry, JobDTO} from '../../../../../common/entities/job/JobDTO';
|
||||||
import {JobState} from '../../../../../common/entities/job/JobProgressDTO';
|
|
||||||
import {Job} from '../../../../../backend/model/jobs/jobs/Job';
|
import {Job} from '../../../../../backend/model/jobs/jobs/Job';
|
||||||
import {ModalDirective} from 'ngx-bootstrap/modal';
|
import {ModalDirective} from 'ngx-bootstrap/modal';
|
||||||
|
import {JobProgressStates} from '../../../../../common/entities/job/JobProgressDTO';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-settings-jobs',
|
selector: 'app-settings-jobs',
|
||||||
@ -38,7 +38,7 @@ export class JobsSettingsComponent extends SettingsComponent<ServerConfig.JobCon
|
|||||||
JobTriggerType = JobTriggerType;
|
JobTriggerType = JobTriggerType;
|
||||||
periods: string[] = [];
|
periods: string[] = [];
|
||||||
showDetails: boolean[] = [];
|
showDetails: boolean[] = [];
|
||||||
JobState = JobState;
|
JobProgressStates = JobProgressStates;
|
||||||
newSchedule: JobScheduleDTO = {
|
newSchedule: JobScheduleDTO = {
|
||||||
name: '',
|
name: '',
|
||||||
config: null,
|
config: null,
|
||||||
@ -216,11 +216,11 @@ export class JobsSettingsComponent extends SettingsComponent<ServerConfig.JobCon
|
|||||||
}
|
}
|
||||||
|
|
||||||
getProgress(schedule: JobScheduleDTO) {
|
getProgress(schedule: JobScheduleDTO) {
|
||||||
return this.jobsService.progress.value[schedule.jobName];
|
return this.jobsService.progress.value[JobDTO.getHashName(schedule.jobName, schedule.config)];
|
||||||
}
|
}
|
||||||
|
|
||||||
getLastRun(schedule: JobScheduleDTO) {
|
getLastRun(schedule: JobScheduleDTO) {
|
||||||
return (this.jobsService.lastRuns.value[schedule.jobName] || {})[this.getConfigHash(schedule)];
|
return this.jobsService.lastRuns.value[JobDTO.getHashName(schedule.jobName, schedule.config)];
|
||||||
}
|
}
|
||||||
|
|
||||||
private getNextRunningDate(sch: JobScheduleDTO, list: JobScheduleDTO[], depth: number = 0): number {
|
private getNextRunningDate(sch: JobScheduleDTO, list: JobScheduleDTO[], depth: number = 0): number {
|
||||||
|
@ -2,26 +2,26 @@
|
|||||||
<div class="col-md-2 col-12" i18n>
|
<div class="col-md-2 col-12" i18n>
|
||||||
Last run:
|
Last run:
|
||||||
</div>
|
</div>
|
||||||
<div class="col-md-4 col-12">
|
<div class="col-md-4 col-12" title="Run between" i18n-title >
|
||||||
<span class="oi oi-clock" title="Run between" i18n-title aria-hidden="true"></span>
|
<span class="oi oi-clock" aria-hidden="true"></span>
|
||||||
{{lastRun.time.start | date:'medium'}} - {{lastRun.time.end | date:'mediumTime'}}
|
{{lastRun.time.start | date:'medium'}} - {{lastRun.time.end | date:'mediumTime'}}
|
||||||
</div>
|
</div>
|
||||||
<div class="col-md-3 col-6">
|
<div class="col-md-3 col-6" title="processed+skipped/all" i18n-title>
|
||||||
<span class="oi oi-check" title="done/all" i18n-title aria-hidden="true"></span>
|
<span class="oi oi-check" aria-hidden="true"></span>
|
||||||
{{lastRun.done}}/{{lastRun.all}}
|
{{lastRun.steps.processed}}+{{lastRun.steps.skipped}}/{{lastRun.steps.all}}
|
||||||
</div>
|
</div>
|
||||||
<div class="col-md-3 col-6">
|
<div class="col-md-3 col-6" title="Status" i18n-title >
|
||||||
<span class="oi oi-pulse" title="Status" i18n-title aria-hidden="true"></span>
|
<span class="oi oi-pulse" aria-hidden="true"></span>
|
||||||
{{JobLastRunState[lastRun.state]}}
|
{{JobProgressStates[lastRun.state]}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div *ngIf="progress">
|
<div *ngIf="progress">
|
||||||
|
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<div class="col-md-12">
|
<div class="col-md-12">
|
||||||
<input *ngIf="progress.state === JobState.running" type="text" class="form-control" disabled
|
<input *ngIf="progress.state === JobProgressStates.running" type="text" class="form-control" disabled
|
||||||
[ngModel]="progress.comment" name="details">
|
[ngModel]="progress.logs[progress.logs.length-1]" name="details">
|
||||||
<input *ngIf="progress.state === JobState.stopping" type="text" class="form-control" disabled value="Stopping"
|
<input *ngIf="progress.state === JobProgressStates.cancelling" type="text" class="form-control" disabled value="Cancelling: {{progress.logs[progress.logs.length-1]}}"
|
||||||
i18n-value name="details">
|
i18n-value name="details">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -30,22 +30,22 @@
|
|||||||
<div class="form-group row progress-row ">
|
<div class="form-group row progress-row ">
|
||||||
<div class="col-6 col-md-2 col-lg-1 text-md-right order-md-0" title="time elapsed" i18n-title>{{TimeElapsed| duration:':'}}</div>
|
<div class="col-6 col-md-2 col-lg-1 text-md-right order-md-0" title="time elapsed" i18n-title>{{TimeElapsed| duration:':'}}</div>
|
||||||
<div class="col-6 col-md-2 col-lg-1 order-md-2 text-right text-md-left" title="time left" i18n-title>{{TimeAll| duration:':'}}</div>
|
<div class="col-6 col-md-2 col-lg-1 order-md-2 text-right text-md-left" title="time left" i18n-title>{{TimeAll| duration:':'}}</div>
|
||||||
<div class="progress col-md-8 col-lg-10 order-md-1">
|
<div class="progress col-md-8 col-lg-10 order-md-1"
|
||||||
|
title="processed:{{progress.steps.processed}}+ skipped:{{progress.steps.skipped}} / all:{{progress.steps.all}}">
|
||||||
<div
|
<div
|
||||||
*ngIf="progress.progress + progress.left >0"
|
*ngIf="progress.steps.all >0"
|
||||||
class="progress-bar d-inline-block progress-bar-success {{progress.state === JobState.stopping ? 'bg-secondary' : ''}}"
|
class="progress-bar d-inline-block progress-bar-success {{progress.state === JobProgressStates.cancelling ? 'bg-secondary' : ''}}"
|
||||||
role="progressbar"
|
role="progressbar"
|
||||||
aria-valuenow="2"
|
aria-valuenow="2"
|
||||||
aria-valuemin="0"
|
aria-valuemin="0"
|
||||||
aria-valuemax="100"
|
aria-valuemax="100"
|
||||||
style="min-width: 2em;"
|
style="min-width: 2em;"
|
||||||
title="{{progress.progress}}/{{progress.progress + progress.left}}"
|
[style.width.%]="(progress.steps.processed+progress.steps.skipped/(progress.steps.all))*100">
|
||||||
[style.width.%]="(progress.progress/(progress.left+progress.progress))*100">
|
{{progress.steps.processed}}+{{progress.steps.skipped}}/{{progress.steps.all}}
|
||||||
{{progress.progress}}/{{progress.progress + progress.left}}
|
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
*ngIf="progress.progress + progress.left === 0"
|
*ngIf="progress.steps.all === 0"
|
||||||
class="progress-bar d-inline-block progress-bar-success progress-bar-striped progress-bar-animated {{progress.state === JobState.stopping ? 'bg-secondary' : ''}}"
|
class="progress-bar d-inline-block progress-bar-success progress-bar-striped progress-bar-animated {{progress.state === JobProgressStates.cancelling ? 'bg-secondary' : ''}}"
|
||||||
role="progressbar" aria-valuenow="100"
|
role="progressbar" aria-valuenow="100"
|
||||||
aria-valuemin="0" aria-valuemax="100" style="width: 100%">
|
aria-valuemin="0" aria-valuemax="100" style="width: 100%">
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import {Component, Input, OnChanges, OnDestroy} from '@angular/core';
|
import {Component, Input, OnChanges, OnDestroy} from '@angular/core';
|
||||||
import {JobProgressDTO, JobState} from '../../../../../../common/entities/job/JobProgressDTO';
|
import {JobProgressDTO, JobProgressStates} from '../../../../../../common/entities/job/JobProgressDTO';
|
||||||
import {Subscription, timer} from 'rxjs';
|
import {Subscription, timer} from 'rxjs';
|
||||||
import {JobLastRunDTO, JobLastRunState} from '../../../../../../common/entities/job/JobLastRunDTO';
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-settings-job-progress',
|
selector: 'app-settings-job-progress',
|
||||||
@ -11,10 +10,9 @@ import {JobLastRunDTO, JobLastRunState} from '../../../../../../common/entities/
|
|||||||
export class JobProgressComponent implements OnDestroy, OnChanges {
|
export class JobProgressComponent implements OnDestroy, OnChanges {
|
||||||
|
|
||||||
@Input() progress: JobProgressDTO;
|
@Input() progress: JobProgressDTO;
|
||||||
@Input() lastRun: JobLastRunDTO;
|
@Input() lastRun: JobProgressDTO;
|
||||||
JobState = JobState;
|
JobProgressStates = JobProgressStates;
|
||||||
timeCurrentCopy: number;
|
timeCurrentCopy: number;
|
||||||
JobLastRunState = JobLastRunState;
|
|
||||||
private timerSub: Subscription;
|
private timerSub: Subscription;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
@ -24,15 +22,15 @@ export class JobProgressComponent implements OnDestroy, OnChanges {
|
|||||||
if (!this.progress) {
|
if (!this.progress) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
return (this.progress.time.current - this.progress.time.start) /
|
return (this.progress.time.end - this.progress.time.start) /
|
||||||
this.progress.progress * (this.progress.left + this.progress.progress);
|
(this.progress.steps.processed + this.progress.steps.skipped) * this.progress.steps.all;
|
||||||
}
|
}
|
||||||
|
|
||||||
get TimeLeft(): number {
|
get TimeLeft(): number {
|
||||||
if (!this.progress) {
|
if (!this.progress) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
return (this.progress.time.current - this.progress.time.start) / this.progress.progress * this.progress.left;
|
return (this.progress.time.end - this.progress.time.start) / this.progress.steps.all;
|
||||||
}
|
}
|
||||||
|
|
||||||
get TimeElapsed() {
|
get TimeElapsed() {
|
||||||
@ -46,7 +44,7 @@ export class JobProgressComponent implements OnDestroy, OnChanges {
|
|||||||
if (!this.progress) {
|
if (!this.progress) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this.timeCurrentCopy = this.progress.time.current;
|
this.timeCurrentCopy = this.progress.time.end;
|
||||||
if (!this.timerSub) {
|
if (!this.timerSub) {
|
||||||
this.timerSub = timer(0, 1000).subscribe(() => {
|
this.timerSub = timer(0, 1000).subscribe(() => {
|
||||||
if (this.progress) {
|
if (this.progress) {
|
||||||
|
@ -111,7 +111,7 @@
|
|||||||
</button>
|
</button>
|
||||||
<button class="btn btn-secondary float-left ml-0"
|
<button class="btn btn-secondary float-left ml-0"
|
||||||
*ngIf="Progress != null"
|
*ngIf="Progress != null"
|
||||||
[disabled]="inProgress || Progress.state !== JobState.running"
|
[disabled]="inProgress || Progress.state !== JobProgressStates.running"
|
||||||
(click)="cancelPhotoConverting()" i18n>Cancel converting
|
(click)="cancelPhotoConverting()" i18n>Cancel converting
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
|
@ -9,9 +9,9 @@ import {I18n} from '@ngx-translate/i18n-polyfill';
|
|||||||
import {ScheduledJobsService} from '../scheduled-jobs.service';
|
import {ScheduledJobsService} from '../scheduled-jobs.service';
|
||||||
import {ServerConfig} from '../../../../../common/config/private/IPrivateConfig';
|
import {ServerConfig} from '../../../../../common/config/private/IPrivateConfig';
|
||||||
import {Utils} from '../../../../../common/Utils';
|
import {Utils} from '../../../../../common/Utils';
|
||||||
import {DefaultsJobs} from '../../../../../common/entities/job/JobDTO';
|
import {DefaultsJobs, JobDTO} from '../../../../../common/entities/job/JobDTO';
|
||||||
import {ErrorDTO} from '../../../../../common/entities/Error';
|
import {ErrorDTO} from '../../../../../common/entities/Error';
|
||||||
import {JobState} from '../../../../../common/entities/job/JobProgressDTO';
|
import {JobProgressStates} from '../../../../../common/entities/job/JobProgressDTO';
|
||||||
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
@ -28,7 +28,7 @@ export class PhotoSettingsComponent extends SettingsComponent<{
|
|||||||
}> {
|
}> {
|
||||||
resolutions = [720, 1080, 1440, 2160, 4320];
|
resolutions = [720, 1080, 1440, 2160, 4320];
|
||||||
PhotoProcessingLib = ServerConfig.PhotoProcessingLib;
|
PhotoProcessingLib = ServerConfig.PhotoProcessingLib;
|
||||||
JobState = JobState;
|
JobProgressStates = JobProgressStates;
|
||||||
|
|
||||||
libTypes = Utils
|
libTypes = Utils
|
||||||
.enumToArray(ServerConfig.PhotoProcessingLib).map((v) => {
|
.enumToArray(ServerConfig.PhotoProcessingLib).map((v) => {
|
||||||
@ -59,7 +59,7 @@ export class PhotoSettingsComponent extends SettingsComponent<{
|
|||||||
|
|
||||||
|
|
||||||
get Progress() {
|
get Progress() {
|
||||||
return this.jobsService.progress.value[DefaultsJobs[DefaultsJobs['Photo Converting']]];
|
return this.jobsService.progress.value[JobDTO.getHashName(DefaultsJobs[DefaultsJobs['Photo Converting']])];
|
||||||
}
|
}
|
||||||
|
|
||||||
async convertPhoto() {
|
async convertPhoto() {
|
||||||
|
@ -1,15 +1,14 @@
|
|||||||
import {EventEmitter, Injectable} from '@angular/core';
|
import {EventEmitter, Injectable} from '@angular/core';
|
||||||
import {BehaviorSubject} from 'rxjs';
|
import {BehaviorSubject} from 'rxjs';
|
||||||
import {JobProgressDTO} from '../../../../common/entities/job/JobProgressDTO';
|
import {JobProgressDTO, JobProgressStates} from '../../../../common/entities/job/JobProgressDTO';
|
||||||
import {NetworkService} from '../../model/network/network.service';
|
import {NetworkService} from '../../model/network/network.service';
|
||||||
import {JobLastRunDTO} from '../../../../common/entities/job/JobLastRunDTO';
|
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ScheduledJobsService {
|
export class ScheduledJobsService {
|
||||||
|
|
||||||
|
|
||||||
public progress: BehaviorSubject<{ [key: string]: JobProgressDTO }>;
|
public progress: BehaviorSubject<{ [key: string]: JobProgressDTO }>;
|
||||||
public lastRuns: BehaviorSubject<{ [key: string]: { [key: string]: JobLastRunDTO } }>;
|
public lastRuns: BehaviorSubject<{ [key: string]: { [key: string]: JobProgressStates } }>;
|
||||||
public onJobFinish: EventEmitter<string> = new EventEmitter<string>();
|
public onJobFinish: EventEmitter<string> = new EventEmitter<string>();
|
||||||
timer: number = null;
|
timer: number = null;
|
||||||
private subscribers = 0;
|
private subscribers = 0;
|
||||||
@ -19,17 +18,6 @@ export class ScheduledJobsService {
|
|||||||
this.lastRuns = new BehaviorSubject({});
|
this.lastRuns = new BehaviorSubject({});
|
||||||
}
|
}
|
||||||
|
|
||||||
public calcTimeElapsed(progress: JobProgressDTO) {
|
|
||||||
if (progress) {
|
|
||||||
return (progress.time.current - progress.time.start);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public calcTimeLeft(progress: JobProgressDTO) {
|
|
||||||
if (progress) {
|
|
||||||
return (progress.time.current - progress.time.start) / progress.progress * progress.left;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
subscribeToProgress(): void {
|
subscribeToProgress(): void {
|
||||||
this.incSubscribers();
|
this.incSubscribers();
|
||||||
@ -56,7 +44,7 @@ export class ScheduledJobsService {
|
|||||||
protected async getProgress(): Promise<void> {
|
protected async getProgress(): Promise<void> {
|
||||||
const prevPrg = this.progress.value;
|
const prevPrg = this.progress.value;
|
||||||
this.progress.next(await this._networkService.getJson<{ [key: string]: JobProgressDTO }>('/admin/jobs/scheduled/progress'));
|
this.progress.next(await this._networkService.getJson<{ [key: string]: JobProgressDTO }>('/admin/jobs/scheduled/progress'));
|
||||||
this.lastRuns.next(await this._networkService.getJson<{ [key: string]: { [key: string]: JobLastRunDTO } }>('/admin/jobs/scheduled/lastRun'));
|
this.lastRuns.next(await this._networkService.getJson<{ [key: string]: { [key: string]: JobProgressStates } }>('/admin/jobs/scheduled/lastRun'));
|
||||||
for (const prg in prevPrg) {
|
for (const prg in prevPrg) {
|
||||||
if (!this.progress.value.hasOwnProperty(prg)) {
|
if (!this.progress.value.hasOwnProperty(prg)) {
|
||||||
this.onJobFinish.emit(prg);
|
this.onJobFinish.emit(prg);
|
||||||
@ -64,6 +52,7 @@ export class ScheduledJobsService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected getProgressPeriodically() {
|
protected getProgressPeriodically() {
|
||||||
if (this.timer != null || this.subscribers === 0) {
|
if (this.timer != null || this.subscribers === 0) {
|
||||||
return;
|
return;
|
||||||
|
@ -85,7 +85,7 @@
|
|||||||
</button>
|
</button>
|
||||||
<button class="btn btn-secondary float-left ml-0"
|
<button class="btn btn-secondary float-left ml-0"
|
||||||
*ngIf="Progress != null"
|
*ngIf="Progress != null"
|
||||||
[disabled]="inProgress || Progress.state !== JobState.running"
|
[disabled]="inProgress || Progress.state !== JobProgressStates.running"
|
||||||
(click)="cancelJob()" i18n>Cancel thumbnail generation
|
(click)="cancelJob()" i18n>Cancel thumbnail generation
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
|
@ -7,10 +7,10 @@ import {ClientConfig} from '../../../../../common/config/public/ConfigClass';
|
|||||||
import {ThumbnailSettingsService} from './thumbnail.settings.service';
|
import {ThumbnailSettingsService} from './thumbnail.settings.service';
|
||||||
import {I18n} from '@ngx-translate/i18n-polyfill';
|
import {I18n} from '@ngx-translate/i18n-polyfill';
|
||||||
import {ServerConfig} from '../../../../../common/config/private/IPrivateConfig';
|
import {ServerConfig} from '../../../../../common/config/private/IPrivateConfig';
|
||||||
import {DefaultsJobs} from '../../../../../common/entities/job/JobDTO';
|
import {DefaultsJobs, JobDTO} from '../../../../../common/entities/job/JobDTO';
|
||||||
import {ErrorDTO} from '../../../../../common/entities/Error';
|
import {ErrorDTO} from '../../../../../common/entities/Error';
|
||||||
import {ScheduledJobsService} from '../scheduled-jobs.service';
|
import {ScheduledJobsService} from '../scheduled-jobs.service';
|
||||||
import {JobState} from '../../../../../common/entities/job/JobProgressDTO';
|
import {JobProgressStates} from '../../../../../common/entities/job/JobProgressDTO';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-settings-thumbnail',
|
selector: 'app-settings-thumbnail',
|
||||||
@ -22,7 +22,7 @@ import {JobState} from '../../../../../common/entities/job/JobProgressDTO';
|
|||||||
export class ThumbnailSettingsComponent
|
export class ThumbnailSettingsComponent
|
||||||
extends SettingsComponent<{ server: ServerConfig.ThumbnailConfig, client: ClientConfig.ThumbnailConfig }>
|
extends SettingsComponent<{ server: ServerConfig.ThumbnailConfig, client: ClientConfig.ThumbnailConfig }>
|
||||||
implements OnInit {
|
implements OnInit {
|
||||||
JobState = JobState;
|
JobProgressStates = JobProgressStates;
|
||||||
|
|
||||||
constructor(_authService: AuthenticationService,
|
constructor(_authService: AuthenticationService,
|
||||||
_navigation: NavigationService,
|
_navigation: NavigationService,
|
||||||
@ -49,7 +49,7 @@ export class ThumbnailSettingsComponent
|
|||||||
}
|
}
|
||||||
|
|
||||||
get Progress() {
|
get Progress() {
|
||||||
return this.jobsService.progress.value[DefaultsJobs[DefaultsJobs['Thumbnail Generation']]];
|
return this.jobsService.progress.value[JobDTO.getHashName(DefaultsJobs[DefaultsJobs['Thumbnail Generation']])];
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
|
@ -136,7 +136,7 @@
|
|||||||
</button>
|
</button>
|
||||||
<button class="btn btn-secondary float-left ml-0"
|
<button class="btn btn-secondary float-left ml-0"
|
||||||
*ngIf="Progress != null"
|
*ngIf="Progress != null"
|
||||||
[disabled]="inProgress || Progress.state !== JobState.running"
|
[disabled]="inProgress || Progress.state !== JobProgressStates.running"
|
||||||
(click)="cancelTranscoding()" i18n>Cancel transcoding
|
(click)="cancelTranscoding()" i18n>Cancel transcoding
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
|
@ -7,10 +7,10 @@ import {NotificationService} from '../../../model/notification.service';
|
|||||||
import {ClientConfig} from '../../../../../common/config/public/ConfigClass';
|
import {ClientConfig} from '../../../../../common/config/public/ConfigClass';
|
||||||
import {I18n} from '@ngx-translate/i18n-polyfill';
|
import {I18n} from '@ngx-translate/i18n-polyfill';
|
||||||
import {ScheduledJobsService} from '../scheduled-jobs.service';
|
import {ScheduledJobsService} from '../scheduled-jobs.service';
|
||||||
import {DefaultsJobs} from '../../../../../common/entities/job/JobDTO';
|
import {DefaultsJobs, JobDTO} from '../../../../../common/entities/job/JobDTO';
|
||||||
import {ErrorDTO} from '../../../../../common/entities/Error';
|
import {ErrorDTO} from '../../../../../common/entities/Error';
|
||||||
import {ServerConfig} from '../../../../../common/config/private/IPrivateConfig';
|
import {ServerConfig} from '../../../../../common/config/private/IPrivateConfig';
|
||||||
import { JobState } from '../../../../../common/entities/job/JobProgressDTO';
|
import {JobProgressStates} from '../../../../../common/entities/job/JobProgressDTO';
|
||||||
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
@ -27,7 +27,7 @@ export class VideoSettingsComponent extends SettingsComponent<{ server: ServerCo
|
|||||||
formats: ServerConfig.formatType[] = ['mp4', 'webm'];
|
formats: ServerConfig.formatType[] = ['mp4', 'webm'];
|
||||||
fps = [24, 25, 30, 48, 50, 60];
|
fps = [24, 25, 30, 48, 50, 60];
|
||||||
|
|
||||||
JobState = JobState;
|
JobProgressStates = JobProgressStates;
|
||||||
|
|
||||||
constructor(_authService: AuthenticationService,
|
constructor(_authService: AuthenticationService,
|
||||||
_navigation: NavigationService,
|
_navigation: NavigationService,
|
||||||
@ -48,7 +48,7 @@ export class VideoSettingsComponent extends SettingsComponent<{ server: ServerCo
|
|||||||
|
|
||||||
|
|
||||||
get Progress() {
|
get Progress() {
|
||||||
return this.jobsService.progress.value[DefaultsJobs[DefaultsJobs['Video Converting']]];
|
return this.jobsService.progress.value[JobDTO.getHashName(DefaultsJobs[DefaultsJobs['Video Converting']])];
|
||||||
}
|
}
|
||||||
|
|
||||||
get bitRate(): number {
|
get bitRate(): number {
|
||||||
|
@ -63,7 +63,7 @@ export class SQLTestHelper {
|
|||||||
await this.resetSQLite();
|
await this.resetSQLite();
|
||||||
|
|
||||||
Config.Server.Database.type = ServerConfig.DatabaseType.sqlite;
|
Config.Server.Database.type = ServerConfig.DatabaseType.sqlite;
|
||||||
Config.Server.Database.sqlite.storage = this.dbPath;
|
Config.Server.Database.dbFolder = path.dirname(this.dbPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async initMySQL() {
|
private async initMySQL() {
|
||||||
|
@ -32,7 +32,7 @@ describe('Typeorm integration', () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Config.Server.Database.type = ServerConfig.DatabaseType.sqlite;
|
Config.Server.Database.type = ServerConfig.DatabaseType.sqlite;
|
||||||
Config.Server.Database.sqlite.storage = dbPath;
|
Config.Server.Database.dbFolder = path.dirname(dbPath);
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -16,15 +16,18 @@ describe('PhotoProcessing', () => {
|
|||||||
const photoPath = path.join(ProjectPath.ImageFolder, 'test_png.png');
|
const photoPath = path.join(ProjectPath.ImageFolder, 'test_png.png');
|
||||||
|
|
||||||
expect(await PhotoProcessing
|
expect(await PhotoProcessing
|
||||||
.isValidConvertedPath(PhotoProcessing.generateConvertedFilePath(photoPath)))
|
.isValidConvertedPath(PhotoProcessing.generateConvertedPath(photoPath,
|
||||||
|
Config.Server.Media.Photo.Converting.resolution)))
|
||||||
.to.be.true;
|
.to.be.true;
|
||||||
|
|
||||||
expect(await PhotoProcessing
|
expect(await PhotoProcessing
|
||||||
.isValidConvertedPath(PhotoProcessing.generateConvertedFilePath(photoPath + 'noPath')))
|
.isValidConvertedPath(PhotoProcessing.generateConvertedPath(photoPath + 'noPath',
|
||||||
|
Config.Server.Media.Photo.Converting.resolution)))
|
||||||
.to.be.false;
|
.to.be.false;
|
||||||
|
|
||||||
{
|
{
|
||||||
const convertedPath = PhotoProcessing.generateConvertedFilePath(photoPath);
|
const convertedPath = PhotoProcessing.generateConvertedPath(photoPath,
|
||||||
|
Config.Server.Media.Photo.Converting.resolution);
|
||||||
Config.Server.Media.Photo.Converting.resolution = <any>1;
|
Config.Server.Media.Photo.Converting.resolution = <any>1;
|
||||||
expect(await PhotoProcessing.isValidConvertedPath(convertedPath)).to.be.false;
|
expect(await PhotoProcessing.isValidConvertedPath(convertedPath)).to.be.false;
|
||||||
}
|
}
|
||||||
@ -42,18 +45,18 @@ describe('PhotoProcessing', () => {
|
|||||||
for (let i = 0; i < Config.Client.Media.Thumbnail.thumbnailSizes.length; ++i) {
|
for (let i = 0; i < Config.Client.Media.Thumbnail.thumbnailSizes.length; ++i) {
|
||||||
const thSize = Config.Client.Media.Thumbnail.thumbnailSizes[i];
|
const thSize = Config.Client.Media.Thumbnail.thumbnailSizes[i];
|
||||||
expect(await PhotoProcessing
|
expect(await PhotoProcessing
|
||||||
.isValidConvertedPath(PhotoProcessing.generateThumbnailPath(photoPath, thSize)))
|
.isValidConvertedPath(PhotoProcessing.generateConvertedPath(photoPath, thSize)))
|
||||||
.to.be.true;
|
.to.be.true;
|
||||||
|
|
||||||
|
|
||||||
expect(await PhotoProcessing
|
expect(await PhotoProcessing
|
||||||
.isValidConvertedPath(PhotoProcessing.generateThumbnailPath(photoPath + 'noPath', thSize)))
|
.isValidConvertedPath(PhotoProcessing.generateConvertedPath(photoPath + 'noPath', thSize)))
|
||||||
.to.be.false;
|
.to.be.false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
expect(await PhotoProcessing
|
expect(await PhotoProcessing
|
||||||
.isValidConvertedPath(PhotoProcessing.generateThumbnailPath(photoPath, 30)))
|
.isValidConvertedPath(PhotoProcessing.generateConvertedPath(photoPath, 30)))
|
||||||
.to.be.false;
|
.to.be.false;
|
||||||
|
|
||||||
});
|
});
|
||||||
|
Loading…
Reference in New Issue
Block a user