You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Tdarr_Plugins/Community/Tdarr_Plugin_e5c3_CnT_Remov...

388 lines
15 KiB

const fs = require('fs');
const execSync = require('child_process').execSync;
function details() {
return {
id: "Tdarr_Plugin_e5c3_CnT_Remove_Letterbox",
Stage: "Pre-processing",
Name: "Remove letterbox",
Type: "Video",
Operation:"Transcode",
Description: `Uses iiDrakeii's filter, and crops video files when letterboxing is detected.\nThis uses the FFMPEG NVENC transcoding(hw).\nIf a file is 4K it will be scaled down to 1080p.\nNow with user definable bitrates!(since 1.104 beta)\nCreated by @control#0405`,
Version: "1.4",
Link: "https://github.com/HaveAGitGat/Tdarr_Plugins/blob/master/Community/Tdarr_Plugin_e5c3_CnT_Remove_Letterbox.js",
Inputs: [
{
name: 'bitrate',
tooltip: `Desired bitrate for a 1080p video, minimum transcode size is based of this too!\\n 720p will be half of 1080p, 480p will be half of 720p.\\nThe default is '3000', this value is based of movies.\\nI would suggest 1500-2000 for series.\\nExample:\\n3000`
},
{
name: 'container',
tooltip: `Enter the output container of the new file.\\n Default: .mkv\\nExample:\\n.mkv`
},
],
Tags:'pre-processing,video only,ffmpeg,configurable',
}
}
function plugin(file, librarySettings, inputs, otherArguments) {
if (inputs.bitrate == "" || inputs.special == 'undefined') {
var min_bitrate = 6600;
var avg_rate = 3000;
var max_rate = 6000;
} else {
var min_bitrate = inputs.bitrate*2.2;
var avg_rate = inputs.bitrate;
var max_rate = inputs.bitrate*2;
}
var source = file.meta.SourceFile; //source file
var stats = fs.statSync(source);
var size = stats["size"]/1000000000;
size = size.toFixed(2);
var decoder = decoder_string(file); //decoder, before the input
var encoder = encoder_string_full(file, highres(file), crop_decider(file, create_crop_values(file).crop_height).crop, avg_rate, max_rate); //encoder
var process = 0; //decides if it should be processed
//default values that will be returned
var response = {
processFile: false,
preset: '',
container: '.mkv',
handBrakeMode: false,
FFmpegMode: true,
reQueueAfter: true,
infoLog: ''
}
if (inputs.container !== undefined) {
response.container = inputs.container;
console.log(`Changed container to: ` + inputs.container);
}
var returns = {
create_crop: create_crop_values(file),
crop: crop_decider(file, create_crop_values(file).crop_height),
size: size_check(file, min_bitrate)
}
var log = {
size: returns.size.log,
hevc: ``,
resolution: ``,
crop: returns.crop.log,
createcrop: returns.create_crop.log
}
//filters
if (size_check(file, min_bitrate).size == 1) {
if (hevc(file) == 1) {
log.hevc = `☑ - Video is not HEVC \n`;
process = 1;
} else {
log.hevc += "☒ - File is already in HEVC \n"
}
if (highres(file) == 1) {
process = 1;
log.resolution += `☑ - Resolution > 1080p.\n File will be transcoded to 1080p \n`;
} else {
log.resolution += `☒ - Resolution <= 1080p \n`;
}
if (crop_decider(file, create_crop_values(file).crop_height).crop != "0") {
process = 1;
}
}
response.infoLog += log.createcrop +
log.crop +
log.resolution +
log.size +
log.hevc;
response.preset = `${decoder}, -map 0:v:0 -map 0:a -map 0:s? ${encoder}`
//change response
if (process == 1) {
response.processFile = true;
response.infoLog += `File will be processed\n`;
} else if (file.forceProcessing === true) {
response.processFile = true;
response.infoLog += `Force processing!\n`;
} else if (response.container !== `.` + file.container) {
response.infoLog += `Container is not correct\nMuxing to ${response.container}!\n`;
response.preset = `${decoder}, -c copy`;
response.processFile = true;
} else {
response.infoLog += `Processing not necessary\n`;
}
return response;
}
function highres(file) {
//if file is larger than 1080p it should be transcoded
if (file.meta.ImageWidth > 1920) {
return 1;
} else {
return 0;
}
}
function create_crop_values(file) {
var source = file.meta.SourceFile; //source file
var dir = file.meta.Directory; //source directory
var sourcename = file.meta.FileName.substring(0, file.meta.FileName.lastIndexOf(".")); //filename without extension
var cropfile = `${dir}/${sourcename}.txt`; //location and name of the crop file
var returns = {
crop_height: 0, //return value for this function, required for crop_decider
log: ``
}
//create crop value
if (!fs.existsSync(`${cropfile}`)) {
returns.log += `Creating crop values...\n`;
execSync(otherArguments.ffmpegPath + ` -ss 300 -i \"${source}\" -frames:v 240 -vf cropdetect -f null - 2>&1 | awk \'/crop/ { print $NF }\' | tail -240 > \"${cropfile}\"`);
execSync(otherArguments.ffmpegPath + ` -ss 1200 -i \"${source}\" -frames:v 240 -vf cropdetect -f null - 2>&1 | awk \'/crop/ { print $NF }\' | tail -240 >> \"${cropfile}\"`);
} else {
returns.log += `Crop values already exist\n`;
}
//get data from copvalue.txt
var data = fs.readFileSync(`${cropfile}`).toString().split("\n"); //full data from cropvalue.txt
//get height of the supposed cropped video
//var crop_height = parseInt(data[0].substring(10, 14));
for (var c = 0; c < data.length; c++) {
if (parseInt(data[c].substring(10, 14)) > returns.crop_height) {
returns.crop_height = parseInt(data[c].substring(10, 14));
returns.log += `New cropheight: ${parseInt(data[c].substring(10, 14))}\n`;
}
}
return returns;
}
function hevc(file) {
//check if the file is already hevc, it will not be transcoded if true
if (file.ffProbeData.streams[0].codec_name) {
if ("hevc".toLowerCase().includes(file.ffProbeData.streams[0].codec_name.toLowerCase())) {
return 0;
} else {
return 1;
}
}
}
function decoder_string(file) {
var decoder = ``; //decoder, before the input
//use the correct decoder
if (file.video_codec_name == 'h263') {
decoder = `-c:v h263_cuvid`
}
else if (file.video_codec_name == 'h264') {
if (file.ffProbeData.streams[0].profile != 'High 10') { //Remove HW Decoding for High 10 Profile
decoder = `-c:v h264_cuvid`
}
}
else if (file.video_codec_name == 'mjpeg') {
decoder = `c:v mjpeg_cuvid`
}
else if (file.video_codec_name == 'mpeg1') {
decoder = `-c:v mpeg1_cuvid`
}
else if (file.video_codec_name == 'mpeg2') {
decoder = `-c:v mpeg2_cuvid`
}
else if (file.video_codec_name == 'vc1') {
decoder = `-c:v vc1_cuvid`
}
else if (file.video_codec_name == 'vp8') {
decoder = `-c:v vp8_cuvid`
}
else if (file.video_codec_name == 'vp9') {
decoder = `-c:v vp9_cuvid`
}
return decoder;
}
function crop_decider(file, crop_height) {
var returns = {
crop: `0`, //sets the crop filter
log: ``,
}
for (var i = 0; i < file.ffProbeData.streams.length; i++) {
if (file.ffProbeData.streams[i].width !== undefined) {
var imageWidth = file.ffProbeData.streams[i].width;
var imageHeight = file.ffProbeData.streams[i].height;
break;
}
}
var min_crop = parseInt(imageHeight*.98); //if the crop value is larger than this the file won't be cropped
//tree for resolution : quality
if (imageWidth >= 1300) { //file will be encoded if the resolution is 1080p, or greater (it will be downscaled)
//crop only if it is a larger crop than 1%;
if (crop_height < min_crop) {
var crop_hdis = parseInt((imageHeight-crop_height)/2);
if (crop_height >= 790) {
returns.crop = `-filter:0 crop=${imageWidth}:${crop_height}:0:${crop_hdis}`;
returns.log += `☑ - crop is larger than 1%\n`;
}
} else {
returns.log += `☒ - Crop is not necessary\n`;
}
} else if(imageWidth < 1300 && file.meta.ImageWidth >= 770) {
//crop only if it is a larger crop than 1%;
if (crop_height < min_crop) {
var crop_hdis = parseInt((imageHeight-crop_height)/2);
if (crop_height >= 530) {
returns.crop = `-filter:0 crop=${imageWidth}:${crop_height}:0:${crop_hdis}`;
returns.log += `☑ - crop is larger than 1%\n`;
}
} else {
returns.log += `☒ - Crop is not necessary\n`;
}
} else if(imageWidth < 770) { //file won't be cropped at this resolution
returns.log += `No crop: Resolution < 720p\n`;
}
return returns;
}
function size_check(file, min_bitrate) {
var duration = file.meta.Duration; //duration of video in seconds
var source = file.meta.SourceFile; //source file
var stats = fs.statSync(source);
var size = stats["size"]/1000000000;
size = size.toFixed(2);
var returns = {
size: 0,
log: ``
}
//tree for resolution : quality
if (file.video_resolution === "1080p" || file.video_resolution === "4KUHD") { //file will be encoded if the resolution is 1080p, or greater (it will be downscaled)
var min_transcode_size = (min_bitrate * duration * 0.125)/1000000; //minimum size in GB for transcode
min_transcode_size = min_transcode_size.toFixed(2);
//check if file is large enough for transcode
if (size >= (min_bitrate * duration * 0.125)/1000000) {
returns.log += `☑ - ${size}GB > ${min_transcode_size}GB\n`;
returns.size = 1;
} else {
returns.log += `☒ - ${size}GB < ${min_transcode_size}GB\n`;
}
} else if(file.video_resolution === "720p") { //file will be encoded if the resolution is 720p
var min_transcode_size = ((min_bitrate/2) * duration * 0.125)/1000000; //minimum size in GB for transcode
min_transcode_size = min_transcode_size.toFixed(2);
//check if file is large enough for transcode
if (size >= ((min_bitrate/2) * duration * 0.125)/1000000) {
returns.log += `☑ - ${size}GB > ${min_transcode_size}GB\n`;
returns.size = 1;
} else {
returns.log += `☒ - ${size}GB < ${min_transcode_size}GB\n`;
}
} else if(file.video_resolution === "480p" || file.video_resolution === "576p") { //file will be encoded if the resolution is 480p or 576p
var min_transcode_size = ((min_bitrate/4) * duration * 0.125)/1000000; //minimum size in GB for transcode
min_transcode_size = min_transcode_size.toFixed(2);
//check if file is large enough for transcode
if (size >= ((min_bitrate/4) * duration * 0.125)/1000000) {
returns.log += `☑ - ${size}GB > ${min_transcode_size}GB\n`;
returns.size = 1;
} else {
returns.log += `☒ - ${size}GB < ${min_transcode_size}GB\n`;
}
}
return returns;
}
function error_fix(file) {
var fix = {
sub_codec: 0, //changes to 1 if unwanted codec is found
muxing: 0
}
for (var i = 0; i < file.ffProbeData.streams.length; i++) {
//these subtitle codecs don't fit in a mkv container
if (file.ffProbeData.streams[i].codec_name && file.ffProbeData.streams[i].codec_type) {
if (file.ffProbeData.streams[i].codec_name.toLowerCase() == "eia_608" || file.ffProbeData.streams[i].codec_name.toLowerCase() == "mov_text" && file.ffProbeData.streams[i].codec_type.toLowerCase.includes("sub") && response.container == '.mkv') {
fix.sub_codec = 1;
}
//mitigate TrueHD audio causing Too many packets error
if (file.ffProbeData.streams[i].codec_name.toLowerCase() == "truehd" || file.ffProbeData.streams[i].codec_name.toLowerCase() == "dts" && file.ffProbeData.streams[i].codec_type.toLowerCase() == "audio" ) {
fix.muxing = 1;
}
}
}
return fix;
}
function encoder_string(file, avg_rate, max_rate) {
var encoder = ``; //encoder
var fix = error_fix(file);
var sub = ``;
//tree for resolution : quality
if (file.video_resolution === "1080p" || file.video_resolution === "4KUHD") { //file will be encoded if the resolution is 1080p, or greater (it will be downscaled)
encoder += ` -pix_fmt p010le -rc:v vbr_hq -qmin 0 -cq:v 26 -b:v ${avg_rate}k -maxrate:v ${max_rate}k`; //-qp 26
} else if(file.video_resolution === "720p") { //file will be encoded if the resolution is 720p
encoder += ` -pix_fmt p010le -rc:v vbr_hq -qmin 0 -cq:v 26 -b:v ${avg_rate/2}k -maxrate:v ${max_rate/2}k`; //-qp 28
} else if(file.video_resolution === "480p" || file.video_resolution === "576p") { //file will be encoded if the resolution is 480p or 576p
encoder += ` -pix_fmt p010le -rc:v vbr_hq -qmin 0 -cq:v 26 -b:v ${avg_rate/4}k -maxrate:v ${max_rate/4}k`; //-qp 30
}
encoder += ` -c:v hevc_nvenc -preset slow -rc-lookahead 32 -spatial_aq:v 1 -aq-strength:v 8 -a53cc 0 -dn`;
if (fix.sub_codec == 1) {
for (var i = 0; i < file.ffProbeData.streams.length; i++) {
if (file.ffProbeData.streams[i].codec_name) {
if (file.ffProbeData.streams[i].codec_name.toLowerCase() == "eia_608" || file.ffProbeData.streams[i].codec_name.toLowerCase() == "mov_text" && file.ffProbeData.streams[i].codec_type.toLowerCase().includes("sub")) {
sub += ` -c:${i} ass`;
} else {
if (file.ffProbeData.streams[i].codec_type) {
if (file.ffProbeData.streams[i].codec_type.toLowerCase().includes("sub")) {
sub += ` -c:${i} copy`;
}
}
}
}
}
} else {
sub = ` -c:s copy`
}
if (fix.muxing == 1) {
encoder += ` -max_muxing_queue_size 2048`;
}
return encoder + ` -c:a copy` + sub;
}
function encoder_string_full(file, highres, crop, avg_rate, max_rate) {
var encoder = encoder_string(file, avg_rate, max_rate);
console.log(`crop filter: ` + crop)
if (highres == 1 && crop != "0") {
return crop + `,scale=-1:1920 ` + encoder;
} else if (highres == 1) {
return `-filter:0 scale=-1:1920 ` + encoder;
} else if (crop != "0") {
return crop + encoder;
} else {
return encoder;
}
}
module.exports.details = details;
module.exports.plugin = plugin;