mirror of
https://github.com/bpatrik/pigallery2.git
synced 2025-04-04 22:24:27 +02:00
285 lines
7.7 KiB
TypeScript
285 lines
7.7 KiB
TypeScript
import * as path from 'path';
|
|
import {constants as fsConstants, promises as fsp} from 'fs';
|
|
import * as os from 'os';
|
|
import * as crypto from 'crypto';
|
|
import {ProjectPath} from '../../ProjectPath';
|
|
import {Config} from '../../../common/config/private/Config';
|
|
import {PhotoWorker, RendererInput, ThumbnailSourceType,} from '../threading/PhotoWorker';
|
|
import {ITaskExecuter, TaskExecuter} from '../threading/TaskExecuter';
|
|
import {FaceRegion, PhotoDTO} from '../../../common/entities/PhotoDTO';
|
|
import {SupportedFormats} from '../../../common/SupportedFormats';
|
|
import {PersonEntry} from '../database/enitites/PersonEntry';
|
|
|
|
export class PhotoProcessing {
|
|
private static initDone = false;
|
|
private static taskQue: ITaskExecuter<RendererInput, void> = null;
|
|
private static readonly CONVERTED_EXTENSION = '.webp';
|
|
|
|
public static init(): void {
|
|
if (this.initDone === true) {
|
|
return;
|
|
}
|
|
|
|
if (Config.Server.Threading.enabled === true) {
|
|
if (Config.Server.Threading.thumbnailThreads > 0) {
|
|
Config.Media.Thumbnail.concurrentThumbnailGenerations =
|
|
Config.Server.Threading.thumbnailThreads;
|
|
} else {
|
|
Config.Media.Thumbnail.concurrentThumbnailGenerations = Math.max(
|
|
1,
|
|
os.cpus().length - 1
|
|
);
|
|
}
|
|
} else {
|
|
Config.Media.Thumbnail.concurrentThumbnailGenerations = 1;
|
|
}
|
|
|
|
this.taskQue = new TaskExecuter(
|
|
Config.Media.Thumbnail.concurrentThumbnailGenerations,
|
|
(input): Promise<void> => PhotoWorker.render(input)
|
|
);
|
|
|
|
this.initDone = true;
|
|
}
|
|
|
|
public static async generatePersonThumbnail(
|
|
person: PersonEntry
|
|
): Promise<string> {
|
|
// load parameters
|
|
const photo: PhotoDTO = person.sampleRegion.media;
|
|
const mediaPath = path.join(
|
|
ProjectPath.ImageFolder,
|
|
photo.directory.path,
|
|
photo.directory.name,
|
|
photo.name
|
|
);
|
|
const size: number = Config.Media.Thumbnail.personThumbnailSize;
|
|
const faceRegion = person.sampleRegion.media.metadata.faces.find(f => f.name === person.name);
|
|
// generate thumbnail path
|
|
const thPath = PhotoProcessing.generatePersonThumbnailPath(
|
|
mediaPath,
|
|
faceRegion,
|
|
size
|
|
);
|
|
|
|
// check if thumbnail already exist
|
|
try {
|
|
await fsp.access(thPath, fsConstants.R_OK);
|
|
return thPath;
|
|
} catch (e) {
|
|
// ignoring errors
|
|
}
|
|
|
|
const margin = {
|
|
x: Math.round(
|
|
faceRegion.box.width *
|
|
Config.Media.Thumbnail.personFaceMargin
|
|
),
|
|
y: Math.round(
|
|
faceRegion.box.height *
|
|
Config.Media.Thumbnail.personFaceMargin
|
|
),
|
|
};
|
|
|
|
// run on other thread
|
|
const input = {
|
|
type: ThumbnailSourceType.Photo,
|
|
mediaPath,
|
|
size,
|
|
outPath: thPath,
|
|
makeSquare: false,
|
|
cut: {
|
|
left: Math.round(
|
|
Math.max(0, faceRegion.box.left - margin.x / 2)
|
|
),
|
|
top: Math.round(
|
|
Math.max(0, faceRegion.box.top - margin.y / 2)
|
|
),
|
|
width: faceRegion.box.width + margin.x,
|
|
height: faceRegion.box.height + margin.y,
|
|
},
|
|
useLanczos3: Config.Media.Thumbnail.useLanczos3,
|
|
quality: Config.Media.Thumbnail.quality,
|
|
smartSubsample: Config.Media.Thumbnail.smartSubsample,
|
|
} as RendererInput;
|
|
input.cut.width = Math.min(
|
|
input.cut.width,
|
|
photo.metadata.size.width - input.cut.left
|
|
);
|
|
input.cut.height = Math.min(
|
|
input.cut.height,
|
|
photo.metadata.size.height - input.cut.top
|
|
);
|
|
|
|
await fsp.mkdir(ProjectPath.FacesFolder, {recursive: true});
|
|
await PhotoProcessing.taskQue.execute(input);
|
|
return thPath;
|
|
}
|
|
|
|
public static generateConvertedPath(mediaPath: string, size: number): string {
|
|
const file = path.basename(mediaPath);
|
|
return path.join(
|
|
ProjectPath.TranscodedFolder,
|
|
ProjectPath.getRelativePathToImages(path.dirname(mediaPath)),
|
|
file + '_' + size + 'q' + Config.Media.Thumbnail.quality + (Config.Media.Thumbnail.smartSubsample ? 'cs' : '') + PhotoProcessing.CONVERTED_EXTENSION
|
|
);
|
|
}
|
|
|
|
public static generatePersonThumbnailPath(
|
|
mediaPath: string,
|
|
faceRegion: FaceRegion,
|
|
size: number
|
|
): string {
|
|
return path.join(
|
|
ProjectPath.FacesFolder,
|
|
crypto
|
|
.createHash('md5')
|
|
.update(
|
|
mediaPath +
|
|
'_' +
|
|
faceRegion.name +
|
|
'_' +
|
|
faceRegion.box.left +
|
|
'_' +
|
|
faceRegion.box.top
|
|
)
|
|
.digest('hex') +
|
|
'_' +
|
|
size +
|
|
PhotoProcessing.CONVERTED_EXTENSION
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Tells if the path is valid with the current config
|
|
* @param convertedPath
|
|
*/
|
|
public static async isValidConvertedPath(
|
|
convertedPath: string
|
|
): Promise<boolean> {
|
|
const origFilePath = path.join(
|
|
ProjectPath.ImageFolder,
|
|
path.relative(
|
|
ProjectPath.TranscodedFolder,
|
|
convertedPath.substring(0, convertedPath.lastIndexOf('_'))
|
|
)
|
|
);
|
|
|
|
if (path.extname(convertedPath) !== PhotoProcessing.CONVERTED_EXTENSION) {
|
|
return false;
|
|
}
|
|
|
|
const sizeStr = convertedPath.substring(
|
|
convertedPath.lastIndexOf('_') + 1,
|
|
convertedPath.lastIndexOf('q')
|
|
);
|
|
|
|
const size = parseInt(sizeStr, 10);
|
|
|
|
if (
|
|
(size + '').length !== sizeStr.length ||
|
|
(Config.Media.Thumbnail.thumbnailSizes.indexOf(size) === -1 &&
|
|
Config.Media.Photo.Converting.resolution !== size)
|
|
) {
|
|
return false;
|
|
}
|
|
|
|
|
|
let qualityStr = convertedPath.substring(
|
|
convertedPath.lastIndexOf('q') + 1,
|
|
convertedPath.length - path.extname(convertedPath).length
|
|
);
|
|
|
|
|
|
if (Config.Media.Thumbnail.smartSubsample) {
|
|
if (!qualityStr.endsWith('cs')) { // remove chromatic subsampling flag if exists
|
|
return false;
|
|
}
|
|
qualityStr = qualityStr.slice(0, -2);
|
|
}
|
|
|
|
const quality = parseInt(qualityStr, 10);
|
|
|
|
if ((quality + '').length !== qualityStr.length ||
|
|
quality !== Config.Media.Thumbnail.quality) {
|
|
return false;
|
|
}
|
|
|
|
try {
|
|
await fsp.access(origFilePath, fsConstants.R_OK);
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
public static async convertPhoto(mediaPath: string): Promise<string> {
|
|
return this.generateThumbnail(
|
|
mediaPath,
|
|
Config.Media.Photo.Converting.resolution,
|
|
ThumbnailSourceType.Photo,
|
|
false
|
|
);
|
|
}
|
|
|
|
static async convertedPhotoExist(
|
|
mediaPath: string,
|
|
size: number
|
|
): Promise<boolean> {
|
|
// generate thumbnail path
|
|
const outPath = PhotoProcessing.generateConvertedPath(mediaPath, size);
|
|
|
|
// check if file already exist
|
|
try {
|
|
await fsp.access(outPath, fsConstants.R_OK);
|
|
return true;
|
|
} catch (e) {
|
|
// ignoring errors
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public static async generateThumbnail(
|
|
mediaPath: string,
|
|
size: number,
|
|
sourceType: ThumbnailSourceType,
|
|
makeSquare: boolean
|
|
): Promise<string> {
|
|
// generate thumbnail path
|
|
const outPath = PhotoProcessing.generateConvertedPath(mediaPath, size);
|
|
|
|
// check if file already exist
|
|
try {
|
|
await fsp.access(outPath, fsConstants.R_OK);
|
|
return outPath;
|
|
} catch (e) {
|
|
// ignoring errors
|
|
}
|
|
|
|
// run on other thread
|
|
const input = {
|
|
type: sourceType,
|
|
mediaPath,
|
|
size,
|
|
outPath,
|
|
makeSquare,
|
|
useLanczos3: Config.Media.Thumbnail.useLanczos3,
|
|
quality: Config.Media.Thumbnail.quality,
|
|
smartSubsample: Config.Media.Thumbnail.smartSubsample,
|
|
} as RendererInput;
|
|
|
|
const outDir = path.dirname(input.outPath);
|
|
|
|
await fsp.mkdir(outDir, {recursive: true});
|
|
await this.taskQue.execute(input);
|
|
return outPath;
|
|
}
|
|
|
|
public static isPhoto(fullPath: string): boolean {
|
|
const extension = path.extname(fullPath).toLowerCase();
|
|
return SupportedFormats.WithDots.Photos.indexOf(extension) !== -1;
|
|
}
|
|
}
|
|
|