Merge pull request #423 from HaveAGitGat/transcode_plugin
Create Tdarr_Plugin_00td_action_transcode.jsmake-only-subtitle-default
commit
a7b4de1152
@ -0,0 +1,605 @@
|
|||||||
|
/* eslint no-plusplus: ["error", { "allowForLoopAfterthoughts": true }] */
|
||||||
|
const details = () => ({
|
||||||
|
id: 'Tdarr_Plugin_00td_action_transcode',
|
||||||
|
Stage: 'Pre-processing',
|
||||||
|
Name: 'Transcode a video file',
|
||||||
|
Type: 'Video',
|
||||||
|
Operation: 'Transcode',
|
||||||
|
Description: 'Transcode a video file using ffmpeg. GPU transcoding will be used if possible.',
|
||||||
|
Version: '3.1',
|
||||||
|
Tags: 'pre-processing,ffmpeg,video only,nvenc h265,configurable',
|
||||||
|
Inputs: [
|
||||||
|
{
|
||||||
|
name: 'target_codec',
|
||||||
|
type: 'string',
|
||||||
|
defaultValue: 'hevc',
|
||||||
|
inputUI: {
|
||||||
|
type: 'dropdown',
|
||||||
|
options: [
|
||||||
|
'hevc',
|
||||||
|
// 'vp9',
|
||||||
|
'h264',
|
||||||
|
// 'vp8',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
tooltip: 'Specify the codec to use',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'target_bitrate_multiplier',
|
||||||
|
type: 'number',
|
||||||
|
defaultValue: 0.5,
|
||||||
|
inputUI: {
|
||||||
|
type: 'text',
|
||||||
|
},
|
||||||
|
tooltip: `
|
||||||
|
Specify the multiplier to use to calculate the target bitrate.
|
||||||
|
Default of 0.5 will roughly half the size of the file.
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'try_use_gpu',
|
||||||
|
type: 'boolean',
|
||||||
|
defaultValue: true,
|
||||||
|
inputUI: {
|
||||||
|
type: 'dropdown',
|
||||||
|
options: [
|
||||||
|
'false',
|
||||||
|
'true',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
tooltip: 'If enabled then will use GPU if possible.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'container',
|
||||||
|
type: 'string',
|
||||||
|
defaultValue: 'mkv',
|
||||||
|
inputUI: {
|
||||||
|
type: 'dropdown',
|
||||||
|
options: [
|
||||||
|
'mkv',
|
||||||
|
'mp4',
|
||||||
|
'avi',
|
||||||
|
'ts',
|
||||||
|
'original',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
tooltip: `Specify output container of file. Use 'original' wihout quotes to keep original container.
|
||||||
|
\\n Ensure that all stream types you may have are supported by your chosen container.
|
||||||
|
\\n mkv is recommended`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'bitrate_cutoff',
|
||||||
|
type: 'number',
|
||||||
|
defaultValue: 0,
|
||||||
|
inputUI: {
|
||||||
|
type: 'text',
|
||||||
|
},
|
||||||
|
tooltip: `Specify bitrate cutoff, files with a current bitrate lower then this will not be transcoded.
|
||||||
|
\\n Rate is in kbps.
|
||||||
|
\\n Leave empty to disable.
|
||||||
|
\\nExample:\\n
|
||||||
|
6000
|
||||||
|
|
||||||
|
\\nExample:\\n
|
||||||
|
4000`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'enable_10bit',
|
||||||
|
type: 'boolean',
|
||||||
|
defaultValue: false,
|
||||||
|
inputUI: {
|
||||||
|
type: 'dropdown',
|
||||||
|
options: [
|
||||||
|
'false',
|
||||||
|
'true',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
tooltip: `Specify if output file should be 10bit. Default is false.
|
||||||
|
\\nExample:\\n
|
||||||
|
true
|
||||||
|
|
||||||
|
\\nExample:\\n
|
||||||
|
false`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'bframes_enabled',
|
||||||
|
type: 'boolean',
|
||||||
|
defaultValue: false,
|
||||||
|
inputUI: {
|
||||||
|
type: 'dropdown',
|
||||||
|
options: [
|
||||||
|
'false',
|
||||||
|
'true',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
tooltip: `Specify if b frames should be used.
|
||||||
|
\\n Using B frames should decrease file sizes but are only supported on newer GPUs.
|
||||||
|
\\n Default is false.
|
||||||
|
\\nExample:\\n
|
||||||
|
true
|
||||||
|
|
||||||
|
\\nExample:\\n
|
||||||
|
false`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'bframes_value',
|
||||||
|
type: 'number',
|
||||||
|
defaultValue: 5,
|
||||||
|
inputUI: {
|
||||||
|
type: 'text',
|
||||||
|
},
|
||||||
|
tooltip: 'Specify number of bframes to use.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'force_conform',
|
||||||
|
type: 'boolean',
|
||||||
|
defaultValue: false,
|
||||||
|
inputUI: {
|
||||||
|
type: 'dropdown',
|
||||||
|
options: [
|
||||||
|
'false',
|
||||||
|
'true',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
tooltip: `Make the file conform to output containers requirements.
|
||||||
|
\\n Drop hdmv_pgs_subtitle/eia_608/subrip/timed_id3 for MP4.
|
||||||
|
\\n Drop data streams/mov_text/eia_608/timed_id3 for MKV.
|
||||||
|
\\n Default is false.
|
||||||
|
\\nExample:\\n
|
||||||
|
true
|
||||||
|
|
||||||
|
\\nExample:\\n
|
||||||
|
false`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'exclude_gpus',
|
||||||
|
type: 'string',
|
||||||
|
defaultValue: '',
|
||||||
|
inputUI: {
|
||||||
|
type: 'text',
|
||||||
|
},
|
||||||
|
tooltip: `Specify the id(s) of any GPUs that needs to be excluded from assigning transcoding tasks.
|
||||||
|
\\n Seperate with a comma (,). Leave empty to disable.
|
||||||
|
\\n Get GPU numbers in the node by running 'nvidia-smi'
|
||||||
|
\\nExample:\\n
|
||||||
|
0,1,3,8
|
||||||
|
|
||||||
|
\\nExample:\\n
|
||||||
|
3
|
||||||
|
|
||||||
|
\\nExample:\\n
|
||||||
|
0`,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
const bframeSupport = [
|
||||||
|
'hevc_nvenc',
|
||||||
|
'h264_nvenc',
|
||||||
|
];
|
||||||
|
|
||||||
|
const hasEncoder = async ({
|
||||||
|
ffmpegPath,
|
||||||
|
encoder,
|
||||||
|
inputArgs,
|
||||||
|
}) => {
|
||||||
|
const { exec } = require('child_process');
|
||||||
|
let isEnabled = false;
|
||||||
|
try {
|
||||||
|
isEnabled = await new Promise((resolve) => {
|
||||||
|
const command = `${ffmpegPath} ${inputArgs || ''} -f lavfi -i color=c=black:s=256x256:d=1:r=30`
|
||||||
|
+ ` -c:v ${encoder} -f null /dev/null`;
|
||||||
|
exec(command, (
|
||||||
|
error,
|
||||||
|
// stdout,
|
||||||
|
// stderr,
|
||||||
|
) => {
|
||||||
|
if (error) {
|
||||||
|
resolve(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
resolve(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.log(err);
|
||||||
|
}
|
||||||
|
|
||||||
|
return isEnabled;
|
||||||
|
};
|
||||||
|
|
||||||
|
// credit to UNCode101 for this
|
||||||
|
const getBestNvencDevice = async ({
|
||||||
|
response,
|
||||||
|
inputs,
|
||||||
|
nvencDevice,
|
||||||
|
}) => {
|
||||||
|
const { execSync } = require('child_process');
|
||||||
|
let gpu_num = -1;
|
||||||
|
let lowest_gpu_util = 100000;
|
||||||
|
let result_util = 0;
|
||||||
|
let gpu_count = -1;
|
||||||
|
let gpu_names = '';
|
||||||
|
const gpus_to_exclude = inputs.exclude_gpus === '' ? [] : inputs.exclude_gpus.split(',').map(Number);
|
||||||
|
try {
|
||||||
|
gpu_names = execSync('nvidia-smi --query-gpu=name --format=csv,noheader');
|
||||||
|
gpu_names = gpu_names.toString().trim();
|
||||||
|
gpu_names = gpu_names.split(/\r?\n/);
|
||||||
|
/* When nvidia-smi returns an error it contains 'nvidia-smi' in the error
|
||||||
|
Example: Linux: nvidia-smi: command not found
|
||||||
|
Windows: 'nvidia-smi' is not recognized as an internal or external command,
|
||||||
|
operable program or batch file. */
|
||||||
|
if (!gpu_names[0].includes('nvidia-smi')) {
|
||||||
|
gpu_count = gpu_names.length;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
response.infoLog += 'Error in reading nvidia-smi output! \n';
|
||||||
|
// response.infoLog += error.message;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gpu_count > 0) {
|
||||||
|
for (let gpui = 0; gpui < gpu_count; gpui++) {
|
||||||
|
// Check if GPU # is in GPUs to exclude
|
||||||
|
if (gpus_to_exclude.includes(gpui)) {
|
||||||
|
response.infoLog += `GPU ${gpui}: ${gpu_names[gpui]} is in exclusion list, will not be used!\n`;
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
const cmd_gpu = `nvidia-smi --query-gpu=utilization.gpu --format=csv,noheader,nounits -i ${gpui}`;
|
||||||
|
result_util = parseInt(execSync(cmd_gpu), 10);
|
||||||
|
if (!Number.isNaN(result_util)) { // != "No devices were found") {
|
||||||
|
response.infoLog += `GPU ${gpui} : Utilization ${result_util}%\n`;
|
||||||
|
|
||||||
|
if (result_util < lowest_gpu_util) {
|
||||||
|
gpu_num = gpui;
|
||||||
|
lowest_gpu_util = result_util;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
response.infoLog += `Error in reading GPU ${gpui} Utilization\nError: ${error}\n`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (gpu_num >= 0) {
|
||||||
|
// eslint-disable-next-line no-param-reassign
|
||||||
|
nvencDevice.inputArgs = `-hwaccel_device ${gpu_num}`;
|
||||||
|
// eslint-disable-next-line no-param-reassign
|
||||||
|
nvencDevice.outputArgs = `-gpu ${gpu_num}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return nvencDevice;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getEncoder = async ({
|
||||||
|
response,
|
||||||
|
inputs,
|
||||||
|
otherArguments,
|
||||||
|
}) => {
|
||||||
|
if (
|
||||||
|
otherArguments.workerType
|
||||||
|
&& otherArguments.workerType.includes('gpu')
|
||||||
|
&& inputs.try_use_gpu && (inputs.target_codec === 'hevc' || inputs.target_codec === 'h264')) {
|
||||||
|
const gpuEncoders = [
|
||||||
|
{
|
||||||
|
encoder: 'hevc_nvenc',
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
encoder: 'hevc_amf',
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
encoder: 'hevc_qsv',
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
encoder: 'hevc_vaapi',
|
||||||
|
inputArgs: '-hwaccel vaapi -hwaccel_device /dev/dri/renderD128 -hwaccel_output_format vaapi',
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
encoder: 'hevc_videotoolbox',
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
encoder: 'h264_nvenc',
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
encoder: 'h264_amf',
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
encoder: 'h264_qsv',
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
encoder: 'h264_videotoolbox',
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const filteredGpuEncoders = gpuEncoders.filter((device) => device.encoder.includes(inputs.target_codec));
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-restricted-syntax
|
||||||
|
for (const gpuEncoder of filteredGpuEncoders) {
|
||||||
|
// eslint-disable-next-line no-await-in-loop
|
||||||
|
gpuEncoder.enabled = await hasEncoder({
|
||||||
|
ffmpegPath: otherArguments.ffmpegPath,
|
||||||
|
encoder: gpuEncoder.encoder,
|
||||||
|
inputArgs: gpuEncoder.inputArgs,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const enabledDevices = gpuEncoders.filter((device) => device.enabled === true);
|
||||||
|
|
||||||
|
if (enabledDevices.length > 0) {
|
||||||
|
if (enabledDevices[0].encoder.includes('nvenc')) {
|
||||||
|
return getBestNvencDevice({
|
||||||
|
response,
|
||||||
|
inputs,
|
||||||
|
nvencDevice: enabledDevices[0],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return enabledDevices[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (inputs.target_codec === 'hevc') {
|
||||||
|
return {
|
||||||
|
encoder: 'libx265',
|
||||||
|
inputArgs: '',
|
||||||
|
};
|
||||||
|
} if (inputs.target_codec === 'h264') {
|
||||||
|
return {
|
||||||
|
encoder: 'libx264',
|
||||||
|
inputArgs: '',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
encoder: '',
|
||||||
|
inputArgs: '',
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
|
const plugin = async (file, librarySettings, inputs, otherArguments) => {
|
||||||
|
const lib = require('../methods/lib')();
|
||||||
|
// eslint-disable-next-line no-unused-vars,no-param-reassign
|
||||||
|
inputs = lib.loadDefaultValues(inputs, details);
|
||||||
|
const response = {
|
||||||
|
processFile: false,
|
||||||
|
preset: '',
|
||||||
|
handBrakeMode: false,
|
||||||
|
FFmpegMode: true,
|
||||||
|
reQueueAfter: true,
|
||||||
|
infoLog: '',
|
||||||
|
};
|
||||||
|
|
||||||
|
const encoderProperties = await getEncoder({
|
||||||
|
response,
|
||||||
|
inputs,
|
||||||
|
otherArguments,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (inputs.container === 'original') {
|
||||||
|
response.container = `.${file.container}`;
|
||||||
|
} else {
|
||||||
|
response.container = `.${inputs.container}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if file is a video. If it isn't then exit plugin.
|
||||||
|
if (file.fileMedium !== 'video') {
|
||||||
|
response.infoLog += 'File is not a video. \n';
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
let duration = 0;
|
||||||
|
|
||||||
|
// Get duration in seconds
|
||||||
|
if (parseFloat(file.ffProbeData?.format?.duration) > 0) {
|
||||||
|
duration = parseFloat(file.ffProbeData?.format?.duration);
|
||||||
|
} else if (typeof file.meta.Duration !== 'undefined') {
|
||||||
|
duration = file.meta.Duration;
|
||||||
|
} else {
|
||||||
|
duration = file.ffProbeData.streams[0].duration;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set up required variables.
|
||||||
|
let videoIdx = 0;
|
||||||
|
let CPU10 = false;
|
||||||
|
let extraArguments = '';
|
||||||
|
let genpts = '';
|
||||||
|
let bitrateSettings = '';
|
||||||
|
|
||||||
|
// Used from here https://blog.frame.io/2017/03/06/calculate-video-bitrates/
|
||||||
|
|
||||||
|
const currentBitrate = (file.file_size * 1024 * 1024 * 8) / duration;
|
||||||
|
// Use the same calculation used for currentBitrate but divide it in half to get targetBitrate.
|
||||||
|
// Logic of h265 can be half the bitrate as h264 without losing quality.
|
||||||
|
|
||||||
|
const targetBitrate = currentBitrate * inputs.target_bitrate_multiplier;
|
||||||
|
// Allow some leeway under and over the targetBitrate.
|
||||||
|
|
||||||
|
const minimumBitrate = (targetBitrate * 0.7);
|
||||||
|
|
||||||
|
const maximumBitrate = (targetBitrate * 1.3);
|
||||||
|
|
||||||
|
// If Container .ts or .avi set genpts to fix unknown timestamp
|
||||||
|
if (inputs.container === 'ts' || inputs.container === 'avi') {
|
||||||
|
genpts = '-fflags +genpts';
|
||||||
|
}
|
||||||
|
|
||||||
|
// If targetBitrate comes out as 0 then something has gone wrong and bitrates could not be calculated.
|
||||||
|
// Cancel plugin completely.
|
||||||
|
if (targetBitrate === 0) {
|
||||||
|
response.infoLog += 'Target bitrate could not be calculated. Skipping this plugin. \n';
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if inputs.bitrate cutoff has something entered.
|
||||||
|
// (Entered means user actually wants something to happen, empty would disable this).
|
||||||
|
// Checks if currentBitrate is below inputs.bitrate_cutoff.
|
||||||
|
// If so then cancel plugin without touching original files.
|
||||||
|
if (currentBitrate <= inputs.bitrate_cutoff) {
|
||||||
|
response.infoLog += `Current bitrate is below set cutoff of ${inputs.bitrate_cutoff}. Cancelling plugin. \n`;
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if force_conform option is checked.
|
||||||
|
// If so then check streams and add any extra parameters required to make file conform with output format.
|
||||||
|
if (inputs.force_conform === true) {
|
||||||
|
if (inputs.container === 'mkv') {
|
||||||
|
extraArguments += '-map -0:d ';
|
||||||
|
for (let i = 0; i < file.ffProbeData.streams.length; i++) {
|
||||||
|
try {
|
||||||
|
if (
|
||||||
|
file.ffProbeData.streams[i].codec_name
|
||||||
|
.toLowerCase() === 'mov_text'
|
||||||
|
|| file.ffProbeData.streams[i].codec_name
|
||||||
|
.toLowerCase() === 'eia_608'
|
||||||
|
|| file.ffProbeData.streams[i].codec_name
|
||||||
|
.toLowerCase() === 'timed_id3'
|
||||||
|
) {
|
||||||
|
extraArguments += `-map -0:${i} `;
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
// Error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (inputs.container === 'mp4') {
|
||||||
|
for (let i = 0; i < file.ffProbeData.streams.length; i++) {
|
||||||
|
try {
|
||||||
|
if (
|
||||||
|
file.ffProbeData.streams[i].codec_name
|
||||||
|
.toLowerCase() === 'hdmv_pgs_subtitle'
|
||||||
|
|| file.ffProbeData.streams[i].codec_name
|
||||||
|
.toLowerCase() === 'eia_608'
|
||||||
|
|| file.ffProbeData.streams[i].codec_name
|
||||||
|
.toLowerCase() === 'subrip'
|
||||||
|
|| file.ffProbeData.streams[i].codec_name
|
||||||
|
.toLowerCase() === 'timed_id3'
|
||||||
|
) {
|
||||||
|
extraArguments += `-map -0:${i} `;
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
// Error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if 10bit variable is true.
|
||||||
|
if (inputs.enable_10bit === true) {
|
||||||
|
// If set to true then add 10bit argument
|
||||||
|
extraArguments += '-pix_fmt p010le ';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if b frame variable is true.
|
||||||
|
if (bframeSupport.includes(encoderProperties.encoder) && inputs.bframes_enabled === true) {
|
||||||
|
// If set to true then add b frames argument
|
||||||
|
extraArguments += `-bf ${inputs.bframes_value} `;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Go through each stream in the file.
|
||||||
|
for (let i = 0; i < file.ffProbeData.streams.length; i++) {
|
||||||
|
// Check if stream is a video.
|
||||||
|
let codec_type = '';
|
||||||
|
try {
|
||||||
|
codec_type = file.ffProbeData.streams[i].codec_type.toLowerCase();
|
||||||
|
} catch (err) {
|
||||||
|
// err
|
||||||
|
}
|
||||||
|
if (codec_type === 'video') {
|
||||||
|
// Check if codec of stream is mjpeg/png, if so then remove this "video" stream.
|
||||||
|
// mjpeg/png are usually embedded pictures that can cause havoc with plugins.
|
||||||
|
if (file.ffProbeData.streams[i].codec_name === 'mjpeg' || file.ffProbeData.streams[i].codec_name === 'png') {
|
||||||
|
extraArguments += `-map -v:${videoIdx} `;
|
||||||
|
}
|
||||||
|
// Check if codec of stream is hevc or vp9 AND check if file.container matches inputs.container.
|
||||||
|
// If so nothing for plugin to do.
|
||||||
|
if (
|
||||||
|
inputs.target_codec === file.ffProbeData.streams[i].codec_name
|
||||||
|
&& file.container === inputs.container
|
||||||
|
) {
|
||||||
|
response.infoLog += `File is already ${inputs.target_codec} and in ${inputs.container}. \n`;
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
// Check if codec of stream is hevc or vp9
|
||||||
|
// AND check if file.container does NOT match inputs.container.
|
||||||
|
// If so remux file.
|
||||||
|
if (
|
||||||
|
|
||||||
|
inputs.target_codec === file.ffProbeData.streams[i].codec_name
|
||||||
|
|
||||||
|
&& file.container !== inputs.container
|
||||||
|
) {
|
||||||
|
response.infoLog += `File is in ${inputs.target_codec} but `
|
||||||
|
+ `is not in ${inputs.container} container. Remuxing. \n`;
|
||||||
|
response.preset = `<io> -map 0 -c copy ${extraArguments}`;
|
||||||
|
response.processFile = true;
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if video stream is HDR or 10bit
|
||||||
|
if (
|
||||||
|
inputs.target_codec === 'hevc'
|
||||||
|
&& (file.ffProbeData.streams[i].profile === 'High 10'
|
||||||
|
|| file.ffProbeData.streams[i].bits_per_raw_sample === '10')
|
||||||
|
) {
|
||||||
|
CPU10 = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Increment videoIdx.
|
||||||
|
videoIdx += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set bitrateSettings variable using bitrate information calulcated earlier.
|
||||||
|
bitrateSettings = `-b:v ${targetBitrate} -minrate ${minimumBitrate} `
|
||||||
|
+ `-maxrate ${maximumBitrate} -bufsize ${currentBitrate}`;
|
||||||
|
// Print to infoLog information around file & bitrate settings.
|
||||||
|
response.infoLog += `Container for output selected as ${inputs.container}. \n`;
|
||||||
|
response.infoLog += `Current bitrate = ${currentBitrate} \n`;
|
||||||
|
response.infoLog += 'Bitrate settings: \n';
|
||||||
|
response.infoLog += `Target = ${targetBitrate} \n`;
|
||||||
|
response.infoLog += `Minimum = ${minimumBitrate} \n`;
|
||||||
|
response.infoLog += `Maximum = ${maximumBitrate} \n`;
|
||||||
|
|
||||||
|
if (encoderProperties.encoder.includes('nvenc')) {
|
||||||
|
if (file.video_codec_name === 'h263') {
|
||||||
|
response.preset = '-c:v h263_cuvid';
|
||||||
|
} else if (file.video_codec_name === 'h264' && CPU10 === false) {
|
||||||
|
response.preset = '-c:v h264_cuvid';
|
||||||
|
} else if (file.video_codec_name === 'mjpeg') {
|
||||||
|
response.preset = '-c:v mjpeg_cuvid';
|
||||||
|
} else if (file.video_codec_name === 'mpeg1') {
|
||||||
|
response.preset = '-c:v mpeg1_cuvid';
|
||||||
|
} else if (file.video_codec_name === 'mpeg2') {
|
||||||
|
response.preset = '-c:v mpeg2_cuvid';
|
||||||
|
} else if (file.video_codec_name === 'mpeg4') {
|
||||||
|
response.preset = '-c:v mpeg4_cuvid';
|
||||||
|
} else if (file.video_codec_name === 'vc1') {
|
||||||
|
response.preset = '-c:v vc1_cuvid';
|
||||||
|
} else if (file.video_codec_name === 'vp8') {
|
||||||
|
response.preset = '-c:v vp8_cuvid';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const vEncode = `-cq:v 19 ${bitrateSettings}`;
|
||||||
|
|
||||||
|
response.preset += ` ${encoderProperties.inputArgs ? encoderProperties.inputArgs : ''} ${genpts}<io>`
|
||||||
|
+ ` -map 0 -c copy -c:v ${encoderProperties.encoder}`
|
||||||
|
+ ` ${encoderProperties.outputArgs ? encoderProperties.outputArgs : ''}`
|
||||||
|
+ ` ${vEncode}`
|
||||||
|
+ ` -spatial_aq:v 1 -rc-lookahead:v 32 -max_muxing_queue_size 9999 ${extraArguments}`;
|
||||||
|
response.processFile = true;
|
||||||
|
response.infoLog += `File is not in ${inputs.target_codec}. Transcoding. \n`;
|
||||||
|
return response;
|
||||||
|
};
|
||||||
|
module.exports.details = details;
|
||||||
|
module.exports.plugin = plugin;
|
||||||
@ -0,0 +1,142 @@
|
|||||||
|
/* eslint max-len: 0 */
|
||||||
|
const _ = require('lodash');
|
||||||
|
const run = require('../helpers/run');
|
||||||
|
|
||||||
|
const tests = [
|
||||||
|
{
|
||||||
|
input: {
|
||||||
|
file: _.cloneDeep(require('../sampleData/media/sampleH264_1.json')),
|
||||||
|
librarySettings: {},
|
||||||
|
inputs: {},
|
||||||
|
otherArguments: {},
|
||||||
|
},
|
||||||
|
output: {
|
||||||
|
processFile: true,
|
||||||
|
preset: ' <io> -map 0 -c copy -c:v libx265 -cq:v 19 -b:v 795571.5361445782 -minrate 556900.0753012047 -maxrate 1034242.9969879518 -bufsize 1591143.0722891565 -spatial_aq:v 1 -rc-lookahead:v 32 -max_muxing_queue_size 9999 ',
|
||||||
|
handBrakeMode: false,
|
||||||
|
FFmpegMode: true,
|
||||||
|
reQueueAfter: true,
|
||||||
|
infoLog: 'Container for output selected as mkv. \n'
|
||||||
|
+ 'Current bitrate = 1591143.0722891565 \n'
|
||||||
|
+ 'Bitrate settings: \n'
|
||||||
|
+ 'Target = 795571.5361445782 \n'
|
||||||
|
+ 'Minimum = 556900.0753012047 \n'
|
||||||
|
+ 'Maximum = 1034242.9969879518 \n'
|
||||||
|
+ 'File is not in hevc. Transcoding. \n',
|
||||||
|
container: '.mkv',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: {
|
||||||
|
file: _.cloneDeep(require('../sampleData/media/sampleH265_1.json')),
|
||||||
|
librarySettings: {},
|
||||||
|
inputs: {},
|
||||||
|
otherArguments: {},
|
||||||
|
},
|
||||||
|
output: {
|
||||||
|
processFile: false,
|
||||||
|
preset: '',
|
||||||
|
handBrakeMode: false,
|
||||||
|
FFmpegMode: true,
|
||||||
|
reQueueAfter: true,
|
||||||
|
infoLog: 'File is already hevc and in mkv. \n',
|
||||||
|
container: '.mkv',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
input: {
|
||||||
|
file: _.cloneDeep(require('../sampleData/media/sampleH264_1.json')),
|
||||||
|
librarySettings: {},
|
||||||
|
inputs: {
|
||||||
|
target_codec: 'hevc',
|
||||||
|
target_bitrate_multiplier: 0.75,
|
||||||
|
try_use_gpu: true,
|
||||||
|
container: 'mkv',
|
||||||
|
bitrate_cutoff: 0,
|
||||||
|
|
||||||
|
enable_10bit: false,
|
||||||
|
bframes_enabled: false,
|
||||||
|
bframes_value: 5,
|
||||||
|
force_conform: false,
|
||||||
|
|
||||||
|
},
|
||||||
|
otherArguments: {},
|
||||||
|
},
|
||||||
|
output: {
|
||||||
|
processFile: true,
|
||||||
|
preset: ' <io> -map 0 -c copy -c:v libx265 -cq:v 19 -b:v 1193357.3042168673 -minrate 835350.1129518071 -maxrate 1551364.4954819276 -bufsize 1591143.0722891565 -spatial_aq:v 1 -rc-lookahead:v 32 -max_muxing_queue_size 9999 ',
|
||||||
|
handBrakeMode: false,
|
||||||
|
FFmpegMode: true,
|
||||||
|
reQueueAfter: true,
|
||||||
|
infoLog: 'Container for output selected as mkv. \n'
|
||||||
|
+ 'Current bitrate = 1591143.0722891565 \n'
|
||||||
|
+ 'Bitrate settings: \n'
|
||||||
|
+ 'Target = 1193357.3042168673 \n'
|
||||||
|
+ 'Minimum = 835350.1129518071 \n'
|
||||||
|
+ 'Maximum = 1551364.4954819276 \n'
|
||||||
|
+ 'File is not in hevc. Transcoding. \n',
|
||||||
|
container: '.mkv',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
input: {
|
||||||
|
file: _.cloneDeep(require('../sampleData/media/sampleH264_1.json')),
|
||||||
|
librarySettings: {},
|
||||||
|
inputs: {
|
||||||
|
target_codec: 'h264',
|
||||||
|
target_bitrate_multiplier: 0.75,
|
||||||
|
try_use_gpu: true,
|
||||||
|
container: 'mkv',
|
||||||
|
bitrate_cutoff: 0,
|
||||||
|
enable_10bit: false,
|
||||||
|
bframes_enabled: false,
|
||||||
|
bframes_value: 5,
|
||||||
|
force_conform: false,
|
||||||
|
},
|
||||||
|
otherArguments: {},
|
||||||
|
},
|
||||||
|
output: {
|
||||||
|
processFile: true,
|
||||||
|
preset: '<io> -map 0 -c copy ',
|
||||||
|
handBrakeMode: false,
|
||||||
|
FFmpegMode: true,
|
||||||
|
reQueueAfter: true,
|
||||||
|
infoLog: 'File is in h264 but is not in mkv container. Remuxing. \n',
|
||||||
|
container: '.mkv',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
input: {
|
||||||
|
file: _.cloneDeep(require('../sampleData/media/sampleH264_1.json')),
|
||||||
|
librarySettings: {},
|
||||||
|
inputs: {
|
||||||
|
target_codec: 'hevc',
|
||||||
|
target_bitrate_multiplier: 0.75,
|
||||||
|
try_use_gpu: true,
|
||||||
|
container: 'mkv',
|
||||||
|
bitrate_cutoff: 10000000,
|
||||||
|
|
||||||
|
enable_10bit: false,
|
||||||
|
bframes_enabled: false,
|
||||||
|
bframes_value: 5,
|
||||||
|
force_conform: false,
|
||||||
|
|
||||||
|
},
|
||||||
|
otherArguments: {},
|
||||||
|
},
|
||||||
|
output: {
|
||||||
|
processFile: false,
|
||||||
|
preset: '',
|
||||||
|
handBrakeMode: false,
|
||||||
|
FFmpegMode: true,
|
||||||
|
reQueueAfter: true,
|
||||||
|
infoLog: 'Current bitrate is below set cutoff of 10000000. Cancelling plugin. \n',
|
||||||
|
container: '.mkv',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
run(tests);
|
||||||
Loading…
Reference in new issue