2024-04-16 23:30:31 +02:00
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
2024-04-19 17:50:13 +02:00
|
|
|
import { exiftool } from 'exiftool-vendored';
|
2024-03-20 19:32:04 +01:00
|
|
|
import ffmpeg, { FfprobeData } from 'fluent-ffmpeg';
|
2024-09-28 00:10:39 +02:00
|
|
|
import { Duration } from 'luxon';
|
2024-03-20 19:32:04 +01:00
|
|
|
import fs from 'node:fs/promises';
|
|
|
|
import { Writable } from 'node:stream';
|
|
|
|
import sharp from 'sharp';
|
2024-11-08 07:30:59 +01:00
|
|
|
import { ORIENTATION_TO_SHARP_ROTATION } from 'src/constants';
|
2024-09-28 00:10:39 +02:00
|
|
|
import { Colorspace, LogLevel } from 'src/enum';
|
2024-04-16 23:30:31 +02:00
|
|
|
import { ILoggerRepository } from 'src/interfaces/logger.interface';
|
2024-03-05 23:23:06 +01:00
|
|
|
import {
|
2024-09-28 19:47:24 +02:00
|
|
|
DecodeToBufferOptions,
|
|
|
|
GenerateThumbhashOptions,
|
|
|
|
GenerateThumbnailOptions,
|
2024-03-05 23:23:06 +01:00
|
|
|
IMediaRepository,
|
2024-04-19 17:50:13 +02:00
|
|
|
ImageDimensions,
|
2024-09-28 00:10:39 +02:00
|
|
|
ProbeOptions,
|
2024-05-27 21:20:07 +02:00
|
|
|
TranscodeCommand,
|
2024-03-05 23:23:06 +01:00
|
|
|
VideoInfo,
|
2024-03-21 12:59:49 +01:00
|
|
|
} from 'src/interfaces/media.interface';
|
2024-03-21 04:15:09 +01:00
|
|
|
import { handlePromiseError } from 'src/utils/misc';
|
2023-04-04 16:48:02 +02:00
|
|
|
|
2024-09-28 00:10:39 +02:00
|
|
|
const probe = (input: string, options: string[]): Promise<FfprobeData> =>
|
|
|
|
new Promise((resolve, reject) =>
|
|
|
|
ffmpeg.ffprobe(input, options, (error, data) => (error ? reject(error) : resolve(data))),
|
|
|
|
);
|
2023-08-02 03:56:10 +02:00
|
|
|
sharp.concurrency(0);
|
2024-01-21 05:10:14 +01:00
|
|
|
sharp.cache({ files: 0 });
|
2023-02-25 15:12:03 +01:00
|
|
|
|
2024-09-28 00:10:39 +02:00
|
|
|
type ProgressEvent = {
|
|
|
|
frames: number;
|
|
|
|
currentFps: number;
|
|
|
|
currentKbps: number;
|
|
|
|
targetSize: number;
|
|
|
|
timemark: string;
|
|
|
|
percent?: number;
|
|
|
|
};
|
|
|
|
|
2024-04-16 23:30:31 +02:00
|
|
|
@Injectable()
|
2023-02-25 15:12:03 +01:00
|
|
|
export class MediaRepository implements IMediaRepository {
|
2024-04-16 23:30:31 +02:00
|
|
|
constructor(@Inject(ILoggerRepository) private logger: ILoggerRepository) {
|
|
|
|
this.logger.setContext(MediaRepository.name);
|
|
|
|
}
|
2024-04-19 17:50:13 +02:00
|
|
|
|
|
|
|
async extract(input: string, output: string): Promise<boolean> {
|
|
|
|
try {
|
|
|
|
await exiftool.extractJpgFromRaw(input, output);
|
|
|
|
} catch (error: any) {
|
|
|
|
this.logger.debug('Could not extract JPEG from image, trying preview', error.message);
|
|
|
|
try {
|
|
|
|
await exiftool.extractPreview(input, output);
|
|
|
|
} catch (error: any) {
|
|
|
|
this.logger.debug('Could not extract preview from image', error.message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-09-28 19:47:24 +02:00
|
|
|
decodeImage(input: string, options: DecodeToBufferOptions) {
|
|
|
|
return this.getImageDecodingPipeline(input, options).raw().toBuffer({ resolveWithObject: true });
|
|
|
|
}
|
2024-05-08 15:09:34 +02:00
|
|
|
|
2024-09-28 19:47:24 +02:00
|
|
|
async generateThumbnail(input: string | Buffer, options: GenerateThumbnailOptions, output: string): Promise<void> {
|
|
|
|
await this.getImageDecodingPipeline(input, options)
|
2023-12-26 22:27:51 +01:00
|
|
|
.toFormat(options.format, {
|
|
|
|
quality: options.quality,
|
|
|
|
// this is default in libvips (except the threshold is 90), but we need to set it manually in sharp
|
|
|
|
chromaSubsampling: options.quality >= 80 ? '4:4:4' : '4:2:0',
|
|
|
|
})
|
2023-08-07 22:35:25 +02:00
|
|
|
.toFile(output);
|
2023-02-25 15:12:03 +01:00
|
|
|
}
|
2023-04-04 16:48:02 +02:00
|
|
|
|
2024-09-28 19:47:24 +02:00
|
|
|
private getImageDecodingPipeline(input: string | Buffer, options: DecodeToBufferOptions) {
|
|
|
|
let pipeline = sharp(input, {
|
|
|
|
// some invalid images can still be processed by sharp, but we want to fail on them by default to avoid crashes
|
|
|
|
failOn: options.processInvalidImages ? 'none' : 'error',
|
|
|
|
limitInputPixels: false,
|
|
|
|
raw: options.raw,
|
2024-10-02 22:53:17 +02:00
|
|
|
})
|
|
|
|
.pipelineColorspace(options.colorspace === Colorspace.SRGB ? 'srgb' : 'rgb16')
|
|
|
|
.withIccProfile(options.colorspace);
|
2024-09-28 19:47:24 +02:00
|
|
|
|
|
|
|
if (!options.raw) {
|
2024-11-08 07:30:59 +01:00
|
|
|
const { angle, flip, flop } = options.orientation ? ORIENTATION_TO_SHARP_ROTATION[options.orientation] : {};
|
|
|
|
pipeline = pipeline.rotate(angle);
|
|
|
|
if (flip) {
|
|
|
|
pipeline = pipeline.flip();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flop) {
|
|
|
|
pipeline = pipeline.flop();
|
|
|
|
}
|
2024-09-28 19:47:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (options.crop) {
|
|
|
|
pipeline = pipeline.extract(options.crop);
|
|
|
|
}
|
|
|
|
|
|
|
|
return pipeline.resize(options.size, options.size, { fit: 'outside', withoutEnlargement: true });
|
|
|
|
}
|
|
|
|
|
|
|
|
async generateThumbhash(input: string | Buffer, options: GenerateThumbhashOptions): Promise<Buffer> {
|
|
|
|
const [{ rgbaToThumbHash }, { data, info }] = await Promise.all([
|
|
|
|
import('thumbhash'),
|
|
|
|
sharp(input, options)
|
|
|
|
.resize(100, 100, { fit: 'inside', withoutEnlargement: true })
|
|
|
|
.raw()
|
|
|
|
.ensureAlpha()
|
|
|
|
.toBuffer({ resolveWithObject: true }),
|
|
|
|
]);
|
|
|
|
return Buffer.from(rgbaToThumbHash(info.width, info.height, data));
|
|
|
|
}
|
|
|
|
|
2024-09-28 00:10:39 +02:00
|
|
|
async probe(input: string, options?: ProbeOptions): Promise<VideoInfo> {
|
|
|
|
const results = await probe(input, options?.countFrames ? ['-count_packets'] : []); // gets frame count quickly: https://stackoverflow.com/a/28376817
|
2023-04-04 16:48:02 +02:00
|
|
|
return {
|
2023-04-06 05:32:59 +02:00
|
|
|
format: {
|
|
|
|
formatName: results.format.format_name,
|
|
|
|
formatLongName: results.format.format_long_name,
|
2024-10-29 15:21:30 +01:00
|
|
|
duration: this.parseFloat(results.format.duration),
|
|
|
|
bitrate: this.parseInt(results.format.bit_rate),
|
2023-04-06 05:32:59 +02:00
|
|
|
},
|
|
|
|
videoStreams: results.streams
|
|
|
|
.filter((stream) => stream.codec_type === 'video')
|
2024-08-08 00:36:37 +02:00
|
|
|
.filter((stream) => !stream.disposition?.attached_pic)
|
2023-04-06 05:32:59 +02:00
|
|
|
.map((stream) => ({
|
2023-08-29 11:01:42 +02:00
|
|
|
index: stream.index,
|
2024-10-29 15:21:30 +01:00
|
|
|
height: this.parseInt(stream.height),
|
|
|
|
width: this.parseInt(stream.width),
|
2023-08-02 03:56:10 +02:00
|
|
|
codecName: stream.codec_name === 'h265' ? 'hevc' : stream.codec_name,
|
2023-04-06 05:32:59 +02:00
|
|
|
codecType: stream.codec_type,
|
2024-09-28 00:10:39 +02:00
|
|
|
frameCount: this.parseInt(options?.countFrames ? stream.nb_read_packets : stream.nb_frames),
|
|
|
|
rotation: this.parseInt(stream.rotation),
|
2023-08-07 22:35:25 +02:00
|
|
|
isHDR: stream.color_transfer === 'smpte2084' || stream.color_transfer === 'arib-std-b67',
|
2024-09-28 00:10:39 +02:00
|
|
|
bitrate: this.parseInt(stream.bit_rate),
|
2024-11-01 01:48:23 +01:00
|
|
|
pixelFormat: stream.pix_fmt || 'yuv420p',
|
2023-04-06 05:32:59 +02:00
|
|
|
})),
|
|
|
|
audioStreams: results.streams
|
|
|
|
.filter((stream) => stream.codec_type === 'audio')
|
|
|
|
.map((stream) => ({
|
2023-08-29 11:01:42 +02:00
|
|
|
index: stream.index,
|
2023-04-06 05:32:59 +02:00
|
|
|
codecType: stream.codec_type,
|
|
|
|
codecName: stream.codec_name,
|
2024-09-28 00:10:39 +02:00
|
|
|
frameCount: this.parseInt(options?.countFrames ? stream.nb_read_packets : stream.nb_frames),
|
2023-04-06 05:32:59 +02:00
|
|
|
})),
|
2023-04-04 16:48:02 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-05-27 21:20:07 +02:00
|
|
|
transcode(input: string, output: string | Writable, options: TranscodeCommand): Promise<void> {
|
2023-05-22 20:07:43 +02:00
|
|
|
if (!options.twoPass) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2024-08-07 17:45:30 +02:00
|
|
|
this.configureFfmpegCall(input, output, options)
|
|
|
|
.on('error', reject)
|
|
|
|
.on('end', () => resolve())
|
|
|
|
.run();
|
2023-05-22 20:07:43 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-09-03 08:21:51 +02:00
|
|
|
if (typeof output !== 'string') {
|
2024-02-02 04:18:00 +01:00
|
|
|
throw new TypeError('Two-pass transcoding does not support writing to a stream');
|
2023-09-03 08:21:51 +02:00
|
|
|
}
|
|
|
|
|
2023-05-22 20:07:43 +02:00
|
|
|
// two-pass allows for precise control of bitrate at the cost of running twice
|
|
|
|
// recommended for vp9 for better quality and compression
|
2023-04-04 16:48:02 +02:00
|
|
|
return new Promise((resolve, reject) => {
|
2023-10-30 15:39:37 +01:00
|
|
|
// first pass output is not saved as only the .log file is needed
|
|
|
|
this.configureFfmpegCall(input, '/dev/null', options)
|
2023-05-22 20:07:43 +02:00
|
|
|
.addOptions('-pass', '1')
|
|
|
|
.addOptions('-passlogfile', output)
|
|
|
|
.addOptions('-f null')
|
2023-10-30 15:39:37 +01:00
|
|
|
.on('error', reject)
|
2023-05-22 20:07:43 +02:00
|
|
|
.on('end', () => {
|
|
|
|
// second pass
|
2023-10-30 15:39:37 +01:00
|
|
|
this.configureFfmpegCall(input, output, options)
|
2023-05-22 20:07:43 +02:00
|
|
|
.addOptions('-pass', '2')
|
|
|
|
.addOptions('-passlogfile', output)
|
2023-10-30 15:39:37 +01:00
|
|
|
.on('error', reject)
|
2024-03-05 23:23:06 +01:00
|
|
|
.on('end', () => handlePromiseError(fs.unlink(`${output}-0.log`), this.logger))
|
|
|
|
.on('end', () => handlePromiseError(fs.rm(`${output}-0.log.mbtree`, { force: true }), this.logger))
|
2024-08-07 17:45:30 +02:00
|
|
|
.on('end', () => resolve())
|
2023-05-22 20:07:43 +02:00
|
|
|
.run();
|
|
|
|
})
|
2023-04-04 16:48:02 +02:00
|
|
|
.run();
|
|
|
|
});
|
|
|
|
}
|
2023-06-18 05:22:31 +02:00
|
|
|
|
2024-04-19 17:50:13 +02:00
|
|
|
async getImageDimensions(input: string): Promise<ImageDimensions> {
|
|
|
|
const { width = 0, height = 0 } = await sharp(input).metadata();
|
|
|
|
return { width, height };
|
|
|
|
}
|
|
|
|
|
2024-05-27 21:20:07 +02:00
|
|
|
private configureFfmpegCall(input: string, output: string | Writable, options: TranscodeCommand) {
|
2024-09-28 00:10:39 +02:00
|
|
|
const ffmpegCall = ffmpeg(input, { niceness: 10 })
|
2024-03-12 06:19:12 +01:00
|
|
|
.inputOptions(options.inputOptions)
|
|
|
|
.outputOptions(options.outputOptions)
|
|
|
|
.output(output)
|
2024-09-28 00:10:39 +02:00
|
|
|
.on('start', (command: string) => this.logger.debug(command))
|
|
|
|
.on('error', (error, _, stderr) => this.logger.error(stderr || error));
|
|
|
|
|
|
|
|
const { frameCount, percentInterval } = options.progress;
|
|
|
|
const frameInterval = Math.ceil(frameCount / (100 / percentInterval));
|
|
|
|
if (this.logger.isLevelEnabled(LogLevel.DEBUG) && frameCount && frameInterval) {
|
|
|
|
let lastProgressFrame: number = 0;
|
|
|
|
ffmpegCall.on('progress', (progress: ProgressEvent) => {
|
|
|
|
if (progress.frames - lastProgressFrame < frameInterval) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
lastProgressFrame = progress.frames;
|
|
|
|
const percent = ((progress.frames / frameCount) * 100).toFixed(2);
|
2024-10-23 14:49:20 +02:00
|
|
|
const ms = progress.currentFps ? Math.floor((frameCount - progress.frames) / progress.currentFps) * 1000 : 0;
|
2024-09-28 00:10:39 +02:00
|
|
|
const duration = ms ? Duration.fromMillis(ms).rescale().toHuman({ unitDisplay: 'narrow' }) : '';
|
|
|
|
const outputText = output instanceof Writable ? 'stream' : output.split('/').pop();
|
|
|
|
this.logger.debug(
|
|
|
|
`Transcoding ${percent}% done${duration ? `, estimated ${duration} remaining` : ''} for output ${outputText}`,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return ffmpegCall;
|
|
|
|
}
|
|
|
|
|
|
|
|
private parseInt(value: string | number | undefined): number {
|
|
|
|
return Number.parseInt(value as string) || 0;
|
2024-03-12 06:19:12 +01:00
|
|
|
}
|
2024-10-29 15:21:30 +01:00
|
|
|
|
|
|
|
private parseFloat(value: string | number | undefined): number {
|
|
|
|
return Number.parseFloat(value as string) || 0;
|
|
|
|
}
|
2023-02-25 15:12:03 +01:00
|
|
|
}
|