86 lines
52 KiB
JavaScript
86 lines
52 KiB
JavaScript
|
|
#!/usr/bin/env node
|
|||
|
|
import{createRequire as NF}from"node:module";var WF=Object.create;var{getPrototypeOf:_F,defineProperty:iD,getOwnPropertyNames:QF}=Object;var HF=Object.prototype.hasOwnProperty;var FD=(u,D,C)=>{C=u!=null?WF(_F(u)):{};let F=D||!u||!u.__esModule?iD(C,"default",{value:u,enumerable:!0}):C;for(let E of QF(u))if(!HF.call(F,E))iD(F,E,{get:()=>u[E],enumerable:!0});return F};var x=(u,D)=>()=>(D||u((D={exports:{}}).exports,D),D.exports);var d=NF(import.meta.url);var _u=x((W3,Wu)=>{class ku{constructor(u,D,C){this.etaBufferLength=u||100,this.valueBuffer=[C],this.timeBuffer=[D],this.eta="0"}update(u,D,C){this.valueBuffer.push(D),this.timeBuffer.push(u),this.calculate(C-D)}getTime(){return this.eta}calculate(u){let D=this.valueBuffer.length,C=Math.min(this.etaBufferLength,D),F=this.valueBuffer[D-1]-this.valueBuffer[D-C],E=this.timeBuffer[D-1]-this.timeBuffer[D-C],B=F/E;this.valueBuffer=this.valueBuffer.slice(-this.etaBufferLength),this.timeBuffer=this.timeBuffer.slice(-this.etaBufferLength);let A=Math.ceil(u/B/1000);if(isNaN(A))this.eta="NULL";else if(!isFinite(A))this.eta="INF";else if(A>1e7)this.eta="INF";else if(A<0)this.eta=0;else this.eta=A}}Wu.exports=ku});var VD=x((_3,Hu)=>{var c=d("readline");class Qu{constructor(u){this.stream=u,this.linewrap=!0,this.dy=0}cursorSave(){if(!this.stream.isTTY)return;this.stream.write("\x1B7")}cursorRestore(){if(!this.stream.isTTY)return;this.stream.write("\x1B8")}cursor(u){if(!this.stream.isTTY)return;if(u)this.stream.write("\x1B[?25h");else this.stream.write("\x1B[?25l")}cursorTo(u=null,D=null){if(!this.stream.isTTY)return;c.cursorTo(this.stream,u,D)}cursorRelative(u=null,D=null){if(!this.stream.isTTY)return;this.dy=this.dy+D,c.moveCursor(this.stream,u,D)}cursorRelativeReset(){if(!this.stream.isTTY)return;c.moveCursor(this.stream,0,-this.dy),c.cursorTo(this.stream,0,null),this.dy=0}clearRight(){if(!this.stream.isTTY)return;c.clearLine(this.stream,1)}clearLine(){if(!this.stream.isTTY)return;c.clearLine(this.stream,0)}clearBottom(){if(!this.stream.isTTY)return;c.clearScreenDown(this.stream)}newline(){this.stream.write(`
|
|||
|
|
`),this.dy++}write(u,D=!1){if(this.linewrap===!0&&D===!1)this.stream.write(u.substr(0,this.getWidth()));else this.stream.write(u)}lineWrapping(u){if(!this.stream.isTTY)return;if(this.linewrap=u,u)this.stream.write("\x1B[?7h");else this.stream.write("\x1B[?7l")}isTTY(){return this.stream.isTTY===!0}getWidth(){return this.stream.columns||(this.stream.isTTY?80:200)}}Hu.exports=Qu});var zu=x((Q3,Nu)=>{Nu.exports=({onlyFirst:u=!1}={})=>{let D=["[\\u001B\\u009B][[\\]()#;?]*(?:(?:(?:(?:;[-a-zA-Z\\d\\/#&.:=?%@~_]+)*|[a-zA-Z\\d]+(?:;[-a-zA-Z\\d\\/#&.:=?%@~_]*)*)?\\u0007)","(?:(?:\\d{1,4}(?:;\\d{0,4})*)?[\\dA-PR-TZcf-ntqry=><~]))"].join("|");return new RegExp(D,u?void 0:"g")}});var Iu=x((H3,qu)=>{var cF=zu();qu.exports=(u)=>typeof u==="string"?u.replace(cF(),""):u});var Vu=x((N3,LD)=>{var xu=(u)=>{if(Number.isNaN(u))return!1;if(u>=4352&&(u<=4447||u===9001||u===9002||11904<=u&&u<=12871&&u!==12351||12880<=u&&u<=19903||19968<=u&&u<=42182||43360<=u&&u<=43388||44032<=u&&u<=55203||63744<=u&&u<=64255||65040<=u&&u<=65049||65072<=u&&u<=65131||65281<=u&&u<=65376||65504<=u&&u<=65510||110592<=u&&u<=110593||127488<=u&&u<=127569||131072<=u&&u<=262141))return!0;return!1};LD.exports=xu;LD.exports.default=xu});var Mu=x((z3,Lu)=>{Lu.exports=function(){return/\uD83C\uDFF4\uDB40\uDC67\uDB40\uDC62(?:\uDB40\uDC65\uDB40\uDC6E\uDB40\uDC67|\uDB40\uDC73\uDB40\uDC63\uDB40\uDC74|\uDB40\uDC77\uDB40\uDC6C\uDB40\uDC73)\uDB40\uDC7F|\uD83D\uDC68(?:\uD83C\uDFFC\u200D(?:\uD83E\uDD1D\u200D\uD83D\uDC68\uD83C\uDFFB|\uD83C[\uDF3E\uDF73\uDF93\uDFA4\uDFA8\uDFEB\uDFED]|\uD83D[\uDCBB\uDCBC\uDD27\uDD2C\uDE80\uDE92]|\uD83E[\uDDAF-\uDDB3\uDDBC\uDDBD])|\uD83C\uDFFF\u200D(?:\uD83E\uDD1D\u200D\uD83D\uDC68(?:\uD83C[\uDFFB-\uDFFE])|\uD83C[\uDF3E\uDF73\uDF93\uDFA4\uDFA8\uDFEB\uDFED]|\uD83D[\uDCBB\uDCBC\uDD27\uDD2C\uDE80\uDE92]|\uD83E[\uDDAF-\uDDB3\uDDBC\uDDBD])|\uD83C\uDFFE\u200D(?:\uD83E\uDD1D\u200D\uD83D\uDC68(?:\uD83C[\uDFFB-\uDFFD])|\uD83C[\uDF3E\uDF73\uDF93\uDFA4\uDFA8\uDFEB\uDFED]|\uD83D[\uDCBB\uDCBC\uDD27\uDD2C\uDE80\uDE92]|\uD83E[\uDDAF-\uDDB3\uDDBC\uDDBD])|\uD83C\uDFFD\u200D(?:\uD83E\uDD1D\u200D\uD83D\uDC68(?:\uD83C[\uDFFB\uDFFC])|\uD83C[\uDF3E\uDF73\uDF93\uDFA4\uDFA8\uDFEB\uDFED]|\uD83D[\uDCBB\uDCBC\uDD27\uDD2C\uDE80\uDE92]|\uD83E[\uDDAF-\uDDB3\uDDBC\uDDBD])|\u200D(?:\u2764\uFE0F\u200D(?:\uD83D\uDC8B\u200D)?\uD83D\uDC68|(?:\uD83D[\uDC68\uDC69])\u200D(?:\uD83D\uDC66\u200D\uD83D\uDC66|\uD83D\uDC67\u200D(?:\uD83D[\uDC66\uDC67]))|\uD83D\uDC66\u200D\uD83D\uDC66|\uD83D\uDC67\u200D(?:\uD83D[\uDC66\uDC67])|(?:\uD83D[\uDC68\uDC69])\u200D(?:\uD83D[\uDC66\uDC67])|[\u2695\u2696\u2708]\uFE0F|\uD83D[\uDC66\uDC67]|\uD83C[\uDF3E\uDF73\uDF93\uDFA4\uDFA8\uDFEB\uDFED]|\uD83D[\uDCBB\uDCBC\uDD27\uDD2C\uDE80\uDE92]|\uD83E[\uDDAF-\uDDB3\uDDBC\uDDBD])|(?:\uD83C\uDFFB\u200D[\u2695\u2696\u2708]|\uD83C\uDFFF\u200D[\u2695\u2696\u2708]|\uD83C\uDFFE\u200D[\u2695\u2696\u2708]|\uD83C\uDFFD\u200D[\u2695\u2696\u2708]|\uD83C\uDFFC\u200D[\u2695\u2696\u2708])\uFE0F|\uD83C\uDFFB\u200D(?:\uD83C[\uDF3E\uDF73\uDF93\uDFA4\uDFA8\uDFEB\uDFED]|\uD83D[\uDCBB\uDCBC\uDD27\uDD2C\uDE80\uDE92]|\uD83E[\uDDAF-\uDDB3\uDDBC\uDDBD])|\uD83C[\uDFFB-\uDFFF])|(?:\uD83E\uDDD1\uD83C\uDFFB\u200D\uD83E\uDD1D\u200D\uD83E\uDDD1|\uD83D\uDC69\uD83C\uDFFC\u200D\uD83E\uDD1D\u200D\uD83D\uDC69)\uD83C\uDFFB|\uD83E\uDDD1(?:\uD83C\uDFFF\u200D\uD83E\uDD1D\u200D\uD83E\uDDD1(?:\uD83C[\uDFFB-\uDFFF])|\u200D\uD83E\uDD1D\u200D\uD83E\uDDD1)|(?:\uD83E\uDDD1\uD83C\uDFFE\u200D\uD83E\uDD1D\u200D\uD83E\uDDD1|\uD83D\uDC69\uD83C\uDFFF\u200D\uD83E\uDD1D\u200D(?:\uD83D[\uDC68\uDC69]))(?:\uD83C[\uDFFB-\uDFFE])|(?:\uD83E\uDDD1\uD83C\uDFFC\u200D\uD83E\uDD1D\u200D\uD83E\uDDD1|\uD83D\uDC69\uD83C\uDFFD\u200D\uD83E\uDD1D\u200D\uD83D\uDC69)(?:\uD83C[\uDFFB\uDFFC])|\uD83D\uDC69(?:\uD83C\uDFFE\u200D(?:\uD83E\uDD1D\u200D\uD83D\uDC68(?:\uD83C[\uDFFB-\uDFFD\uDFFF])|\uD83C[\uDF3E\uDF73\uDF93\uDFA4\uDFA8\uDFEB\uDFED]|\uD83D[\uDCBB\uDCBC\uDD27\uDD2C\uDE80\uDE92]|\uD83E[\uDDAF-\uDDB3\uDDBC\uDDBD])|\uD83C\uDFFC\u200D(?:\uD83E\uDD1D\u200D\uD83D\uDC68(?:\uD83C[\uDFFB\uDFFD-\uDFFF])|\uD83C[\uDF3E\uDF73\uDF93\uDFA4\uDFA8\uDFEB\uDFED]|\uD83D[\uDCBB\uDCBC\uDD27\uDD2C\uDE80\uDE92]|\uD
|
|||
|
|
=== FFmpeg Command [${new Date().toISOString()}] ===
|
|||
|
|
ffmpeg ${u.join(" ")}
|
|||
|
|
`;return await P(E),new Promise((B,A)=>{let $=e("ffmpeg",u),J="";$.stderr.on("data",(G)=>{let X=G.toString();if(J+=X,D&&C){let Y=X.match(/time=(\d{2}):(\d{2}):(\d{2}\.\d{2})/);if(Y){let Z=parseInt(Y[1]),K=parseInt(Y[2]),_=parseFloat(Y[3]),U=Z*3600+K*60+_,q=Math.min(100,U/C*100);D(q)}}}),$.on("error",(G)=>{P(`ERROR: ${G.message}
|
|||
|
|
`),A(Error(`FFmpeg error: ${G.message}`))}),$.on("close",(G)=>{if(G===0){let Y=J.split(`
|
|||
|
|
`).filter((Z)=>Z.trim()).slice(-10).join(`
|
|||
|
|
`);P(`SUCCESS: Exit code ${G}
|
|||
|
|
--- Last 10 lines of output ---
|
|||
|
|
${Y}
|
|||
|
|
`),B()}else P(`FAILED: Exit code ${G}
|
|||
|
|
--- Full error output ---
|
|||
|
|
${J}
|
|||
|
|
`),A(Error(`FFmpeg failed with exit code ${G}
|
|||
|
|
${J}`))})})}async function QD(u){let C=`
|
|||
|
|
=== MP4Box Command [${new Date().toISOString()}] ===
|
|||
|
|
MP4Box ${u.join(" ")}
|
|||
|
|
`;return await P(C),new Promise((F,E)=>{let B=e("MP4Box",u),A="",$="";B.stdout.on("data",(J)=>{A+=J.toString()}),B.stderr.on("data",(J)=>{$+=J.toString()}),B.on("error",(J)=>{P(`ERROR: ${J.message}
|
|||
|
|
`),E(Error(`MP4Box error: ${J.message}`))}),B.on("close",(J)=>{if(J===0){let Y=(A||$).split(`
|
|||
|
|
`).filter((Z)=>Z.trim()).slice(-10).join(`
|
|||
|
|
`);P(`SUCCESS: Exit code ${J}
|
|||
|
|
--- Last 10 lines of output ---
|
|||
|
|
${Y}
|
|||
|
|
`),F()}else{let G=$||A;P(`FAILED: Exit code ${J}
|
|||
|
|
--- Full error output ---
|
|||
|
|
${G}
|
|||
|
|
`),E(Error(`MP4Box failed with exit code ${J}
|
|||
|
|
${G}`))}})})}import{spawn as qF}from"node:child_process";async function a(u){return new Promise((D,C)=>{let F=qF("ffprobe",["-v","error","-show_entries","stream=width,height,duration,r_frame_rate,codec_name,codec_type,bit_rate","-show_entries","format=duration","-of","json",u]),E="";F.stdout.on("data",(B)=>{E+=B.toString()}),F.on("error",(B)=>{C(Error(`ffprobe error: ${B.message}`))}),F.on("close",(B)=>{if(B!==0){C(Error(`ffprobe failed with exit code ${B}`));return}try{let A=JSON.parse(E),$=A.streams.find((U)=>U.codec_type==="video"),J=A.streams.find((U)=>U.codec_type==="audio"),G=A.format;if(!$){C(Error("No video stream found in input file"));return}let X=30;if($.r_frame_rate){let[U,q]=$.r_frame_rate.split("/").map(Number);if(U&&q&&q!==0)X=U/q}let Y=parseFloat($.duration||G.duration||"0"),Z=A.streams.find((U)=>U.codec_type==="audio"&&U.bit_rate),K=Z?.bit_rate?Math.round(parseInt(Z.bit_rate)/1000):void 0,_=$.bit_rate?Math.round(parseInt($.bit_rate)/1000):void 0;D({width:$.width,height:$.height,duration:Y,fps:X,codec:$.codec_name,hasAudio:Boolean(J),audioBitrate:K,videoBitrate:_})}catch(A){C(Error(`Failed to parse ffprobe output: ${A}`))}})})}function CD(u,D=256){if(!u)return`${D}k`;let C=Math.min(u,D);if(C<=64)return"64k";if(C<=96)return"96k";if(C<=128)return"128k";if(C<=192)return"192k";return"256k"}function ED(u){let D=Math.floor(u/3600),C=Math.floor(u%3600/60),F=u%60;return`${String(D).padStart(2,"0")}:${String(C).padStart(2,"0")}:${F.toFixed(3).padStart(6,"0")}`}import{mkdir as IF,access as xF,constants as VF}from"node:fs/promises";async function m(u){try{await xF(u,VF.F_OK)}catch{await IF(u,{recursive:!0})}}function OF(u,D){let C=u*D;if(C<=230400)return 0.08;if(C<=409920)return 0.075;if(C<=921600)return 0.07;if(C<=2073600)return 0.065;if(C<=3686400)return 0.06;return 0.055}function y(u,D,C=30,F){let E=OF(u,D),B=Math.round(u*D*C*E/1000);if(F&&B>F)B=F;return`${B}k`}var HD=[{name:"360p",width:640,height:360,videoBitrate:y(640,360,30),audioBitrate:"192k"},{name:"480p",width:854,height:480,videoBitrate:y(854,480,30),audioBitrate:"192k"},{name:"720p",width:1280,height:720,videoBitrate:y(1280,720,30),audioBitrate:"192k"},{name:"1080p",width:1920,height:1080,videoBitrate:y(1920,1080,30),audioBitrate:"256k"},{name:"1440p",width:2560,height:1440,videoBitrate:y(2560,1440,30),audioBitrate:"256k"},{name:"2160p",width:3840,height:2160,videoBitrate:y(3840,2160,30),audioBitrate:"256k"}];function DD(u,D,C=30,F){let E=[],B=HD.filter((A)=>{return A.width<=u&&A.height<=D});for(let A of B)E.push({...A,videoBitrate:y(A.width,A.height,30,F),fps:30});return E}function RF(u,D,C){return{...u,name:`${u.name}-${D}`,videoBitrate:y(u.width,u.height,D,C),fps:D}}function oD(u){let C=u.trim().match(/^(\d+)p?(?:[@-](\d+))?$/i);if(!C)return null;let F=C[1]+"p",E=C[2]?parseInt(C[2]):30;return{resolution:F,fps:E}}function tD(u,D=30,C){let F=HD.find((E)=>E.name===u);if(!F)return null;if(D===30)return{...F,videoBitrate:y(F.width,F.height,30,C),fps:30};return RF(F,D,C)}function jF(u,D,C,F){let E=oD(u);if(!E)return{error:`Invalid profile format: ${u}. Use format like: 360, 720@60, 1080-60`};let B=tD(E.resolution,E.fps);if(!B)return{error:`Unknown resolution: ${E.resolution}. Available: 360, 480, 720, 1080, 1440, 2160`};if(B.width>D||B.height>C)return{error:`Source resolution (${D}x${C}) is lower than ${u} (${B.width}x${B.height})`};let A=120,$=E.fps,J;if(E.fps>F)$=Math.min(F,A),J=`Requested ${E.fps} FPS in ${u}, but source is ${F} FPS. Using ${$} FPS instead`;else if(E.fps>A)$=A,J=`Requested ${E.fps} FPS in ${u} exceeds maximum ${A} FPS. Using ${$} FPS instead`;return J?{warning:J,adjustedFps:$}:{}}function BD(u,D,C,F,E){let B=[],A=[],$=[];for(let J of u){let G=jF(J,D,C,F);if(G.error){A.push(G.error);continue}if(G.warning)$.push(G.warning);let X=oD(J);if(!X)continue;let Y=G.adjustedFps!==void 0?G.adjustedFps:X.fps,Z=tD(X.resolution,Y,E);if(Z)B.push(Z)}return{profiles:B,errors:A,warnings:$}}import{join as h}from"node:path";import{readdir as wF,unlink as eD,rmdir as SF,writeFile as Du}from"node:fs/promises";async function uu(u,D,C="00:00:00"){let F=h(D,"po
|
|||
|
|
|
|||
|
|
`;for(let $=0;$<u;$++){let J=$*D,G=($+1)*D,X=Math.floor($/E),Z=$%E*C,K=X*F;A+=`${ED(J)} --> ${ED(G)}
|
|||
|
|
`,A+=`${B}#xywh=${Z},${K},${C},${F}
|
|||
|
|
|
|||
|
|
`}return A}import{join as bF}from"node:path";function yF(u,D,C){if(C)if(D==="h264")return 32;else return 42;else if(D==="h264"){if(u<=360)return 25;if(u<=480)return 24;if(u<=720)return 23;if(u<=1080)return 22;if(u<=1440)return 21;return 20}else{if(u<=360)return 40;if(u<=480)return 38;if(u<=720)return 35;if(u<=1080)return 32;if(u<=1440)return 30;return 28}}async function Cu(u,D,C,F,E,B,A,$,J,G,X,Y){let Z=bF(D,`video_${J}_${C.name}.mp4`),K=["-y","-i",u,"-c:v",F],_=F.includes("nvenc")||F.includes("qsv")||F.includes("amf")||F.includes("vaapi")||F.includes("videotoolbox")||F.includes("v4l2"),U;if(_&&G?.cq!==void 0)U=G.cq;else if(!_&&G?.crf!==void 0)U=G.crf;else U=yF(C.height,J,_);if(F==="h264_nvenc")K.push("-rc:v","vbr"),K.push("-cq",String(U)),K.push("-preset",E),K.push("-2pass","0");else if(F==="av1_nvenc")K.push("-rc:v","vbr"),K.push("-cq",String(U)),K.push("-preset",E),K.push("-2pass","0");else if(F==="av1_qsv")K.push("-preset",E),K.push("-global_quality",String(U));else if(F==="h264_qsv")K.push("-preset",E),K.push("-global_quality",String(U));else if(F==="av1_amf")K.push("-quality","balanced"),K.push("-rc","cqp"),K.push("-qp_i",String(U)),K.push("-qp_p",String(U));else if(F==="h264_amf")K.push("-quality","balanced"),K.push("-rc","cqp"),K.push("-qp_i",String(U)),K.push("-qp_p",String(U));else if(F==="libsvtav1")K.push("-crf",String(U)),K.push("-preset",E),K.push("-svtav1-params","tune=0:enable-overlays=1");else if(F==="libx264")K.push("-crf",String(U)),K.push("-preset",E);else K.push("-preset",E);let q=J==="av1"?0.6:1,k=Math.round(parseInt(C.videoBitrate)*q*1.5);K.push("-maxrate",`${k}k`),K.push("-bufsize",`${k*2}k`);let W=C.fps||30,N=Math.round(W*A);K.push("-g",String(N),"-keyint_min",String(N),"-sc_threshold","0");let V=[`scale=${C.width}:${C.height}`];if(X){if(X.deinterlace)V.push("yadif");if(X.denoise)V.push("hqdn3d");if(X.customFilters)V.push(...X.customFilters)}K.push("-vf",V.join(","));let b=parseInt(C.audioBitrate)||256,R=CD($,b);if(K.push("-c:a","aac","-b:a",R),X?.audioNormalize)K.push("-af","loudnorm");return K.push("-f","mp4",Z),await g(K,Y,B),Z}async function Eu(u,D,C,F,E,B,A,$,J,G,X,Y,Z,K){let _=new Map;if(J&&C.length>1)for(let U=0;U<C.length;U+=G){let q=C.slice(U,U+G),k=q.map((N)=>Cu(u,D,N,F,E,B,A,$,X,Y,Z,(V)=>{if(K)K(N.name,V)}));(await Promise.all(k)).forEach((N,V)=>{let b=q[V];_.set(b.name,N)})}else for(let U of C){let q=await Cu(u,D,U,F,E,B,A,$,X,Y,Z,(k)=>{if(K)K(U.name,k)});_.set(U.name,q)}return _}import{join as M}from"node:path";import{readdir as Ju,rename as e8,mkdir as D3,writeFile as ID}from"node:fs/promises";import{readFile as ND,writeFile as zD}from"node:fs/promises";async function Bu(u){let D=await ND(u,"utf-8");D=D.replace(/\/\/>/g,"/>"),D=D.replace(/\/\s+\/>/g,"/>"),D=D.replace(/(<Representation[^>]+)\s+\/>/g,"$1/>"),D=D.replace(/<Representation\s+([^>]+)\/>\s*<\/Representation>/g,"<Representation $1/>"),D=D.replace(/<Representation\s+([^>]+)\/>\s*(<AudioChannelConfiguration[^>]*\/>)/g,`<Representation $1>
|
|||
|
|
$2
|
|||
|
|
</Representation>`),D=D.replace(/<Representation\s+([^>]+)>\s*(?=<(?:Representation|\/AdaptationSet))/g,`<Representation $1/>
|
|||
|
|
`),await zD(u,D,"utf-8")}async function Au(u,D,C){let F=await ND(u,"utf-8");F=F.replace(/media="\$RepresentationID\$_\$Number\$\.m4s"/g,'media="$RepresentationID$/$RepresentationID$_$Number$.m4s"'),F=F.replace(/initialization="\$RepresentationID\$_\.mp4"/g,'initialization="$RepresentationID$/$RepresentationID$_.mp4"'),await zD(u,F,"utf-8")}async function Zu(u){let C=(await ND(u,"utf-8")).split(`
|
|||
|
|
`),F=[],E=0;while(E<C.length){let B=C[E];if(B.includes("<AdaptationSet")&&B.includes("maxWidth")){let A=E,$=[B],J=[],G=[],X=[],Y=!1;E++;while(E<C.length&&!C[E].includes("</AdaptationSet>")){let Z=C[E];if(Z.includes("<SegmentTemplate"))Y=!0;if(Y){if(J.push(Z),Z.includes("</SegmentTemplate>"))Y=!1}else if(Z.includes("<Representation")&&Z.includes("-h264"))G.push(Z);else if(Z.includes("<Representation")&&Z.includes("-av1"))X.push(Z);E++}if(G.length>0&&X.length>0)F.push(B),J.forEach((Z)=>F.push(Z)),G.forEach((Z)=>F.push(Z)),F.push(" </AdaptationSet>"),F.push(B),J.forEach((Z)=>F.push(Z)),X.forEach((Z)=>F.push(Z)),F.push(" </AdaptationSet>");else{F.push(B);for(let Z=A+1;Z<E;Z++)F.push(C[Z]);F.push(C[E])}E++}else F.push(B),E++}await zD(u,F.join(`
|
|||
|
|
`),"utf-8")}function qD(u,D,C){let F=`#EXTM3U
|
|||
|
|
`;F+=`#EXT-X-VERSION:6
|
|||
|
|
`,F+=`#EXT-X-TARGETDURATION:${Math.ceil(C)}
|
|||
|
|
`,F+=`#EXT-X-MEDIA-SEQUENCE:1
|
|||
|
|
`,F+=`#EXT-X-INDEPENDENT-SEGMENTS
|
|||
|
|
`,F+=`#EXT-X-MAP:URI="${D}"
|
|||
|
|
`;for(let E of u)F+=`#EXTINF:${C},
|
|||
|
|
`,F+=`${E}
|
|||
|
|
`;return F+=`#EXT-X-ENDLIST
|
|||
|
|
`,F}function $u(u,D){let C=`#EXTM3U
|
|||
|
|
`;if(C+=`#EXT-X-VERSION:6
|
|||
|
|
`,C+=`#EXT-X-INDEPENDENT-SEGMENTS
|
|||
|
|
|
|||
|
|
`,D)C+=`#EXT-X-MEDIA:TYPE=AUDIO,GROUP-ID="audio",NAME="audio",AUTOSELECT=YES,URI="audio/playlist.m3u8",CHANNELS="2"
|
|||
|
|
|
|||
|
|
`;for(let F of u){let E=D?"avc1.4D4020,mp4a.40.2":"avc1.4D4020";if(C+=`#EXT-X-STREAM-INF:BANDWIDTH=${F.bandwidth},CODECS="${E}",RESOLUTION=${F.resolution},FRAME-RATE=${F.fps}`,D)C+=',AUDIO="audio"';C+=`
|
|||
|
|
`,C+=`${F.path}
|
|||
|
|
|
|||
|
|
`}return C}async function vF(u,D,C,F,E,B){let A=M(D,"manifest.mpd"),$=["-dash",String(F*1000),"-frag",String(F*1000),"-rap","-segment-timeline","-segment-name","$RepresentationID$_$Number$","-out",A],J=!0;for(let[G,X]of u.entries())for(let Y of C){let Z=X.get(Y.name);if(!Z)throw Error(`MP4 file not found for profile: ${Y.name}, codec: ${G}`);let K=E==="dual"?`${Y.name}-${G}`:Y.name;if($.push(`${Z}#video:id=${K}`),J&&B)$.push(`${Z}#audio:id=audio`),J=!1}if(await QD($),await PF(D,C,E,B),await Au(A,C,E),E==="dual")await Zu(A);return await Bu(A),A}async function PF(u,D,C,F){let{readdir:E,rename:B,mkdir:A}=await import("node:fs/promises"),$=[];if(C==="h264"||C==="dual")$.push("h264");if(C==="av1"||C==="dual")$.push("av1");let J=[];for(let Y of $)for(let Z of D){let K=C==="dual"?`${Z.name}-${Y}`:Z.name;J.push(K);let _=M(u,K);await A(_,{recursive:!0})}let G=M(u,"audio");if(F)await A(G,{recursive:!0});let X=await E(u);for(let Y of X){if(Y==="manifest.mpd")continue;if(F&&(Y.startsWith("audio_")||Y==="audio_init.m4s")){let Z=M(u,Y),K=M(G,Y);await B(Z,K);continue}for(let Z of J)if(Y.startsWith(`${Z}_`)){let K=M(u,Y),_=M(u,Z,Y);await B(K,_);break}}}async function Ku(u,D,C,F,E,B,A){let $,J;if(B==="dash"||B==="both")$=await vF(u,D,C,F,E,A);if(B==="hls"||B==="both")J=await hF(D,C,F,E,A);return{manifestPath:$,hlsManifestPath:J}}async function hF(u,D,C,F,E){let B=M(u,"master.m3u8"),A=[];for(let X of D){let Y=F==="dual"?`${X.name}-h264`:X.name,Z=M(u,Y),K=await Ju(Z),_=K.filter((N)=>N.endsWith(".m4s")).sort((N,V)=>{let b=parseInt(N.match(/_(\d+)\.m4s$/)?.[1]||"0"),R=parseInt(V.match(/_(\d+)\.m4s$/)?.[1]||"0");return b-R}),U=K.find((N)=>N.endsWith("_.mp4"));if(!U||_.length===0)continue;let q=qD(_,U,C),k=M(Z,"playlist.m3u8");await ID(k,q,"utf-8");let W=parseInt(X.videoBitrate)*1000;A.push({path:`${Y}/playlist.m3u8`,bandwidth:W,resolution:`${X.width}x${X.height}`,fps:X.fps||30})}let $,J=[];if(E){let X=M(u,"audio"),Y=[];try{Y=await Ju(X)}catch{Y=[]}if(J=Y.filter((Z)=>Z.endsWith(".m4s")).sort((Z,K)=>{let _=parseInt(Z.match(/_(\d+)\.m4s$/)?.[1]||"0"),U=parseInt(K.match(/_(\d+)\.m4s$/)?.[1]||"0");return _-U}),$=Y.find((Z)=>Z.endsWith("_.mp4")),$&&J.length>0){let Z=qD(J,$,C);await ID(M(X,"playlist.m3u8"),Z,"utf-8")}}let G=$u(A,E&&$!==void 0&&J.length>0);return await ID(B,G,"utf-8"),B}async function xD(u){let{input:D,outputDir:C,segmentDuration:F=2,profiles:E,customProfiles:B,codec:A="dual",format:$="both",useNvenc:J,hardwareAccelerator:G,quality:X,generateThumbnails:Y=!0,thumbnailConfig:Z={},generatePoster:K=!0,posterTimecode:_="00:00:00",parallel:U=!0,onProgress:q}=u,k=ZD("/tmp",`dash-converter-${fF()}`);await m(k);let W=Yu(D,Gu(D)),N=ZD(C,W);await m(N);let V=ZD(N,"conversion.log");_D(V);let{writeFile:b}=await import("node:fs/promises"),R=`===========================================
|
|||
|
|
DASH Conversion Log
|
|||
|
|
Started: ${new Date().toISOString()}
|
|||
|
|
Input: ${D}
|
|||
|
|
Output: ${N}
|
|||
|
|
Codec: ${A}
|
|||
|
|
Format: ${$}
|
|||
|
|
===========================================
|
|||
|
|
`;await b(V,R,"utf-8");try{return await gF(D,C,k,F,E,B,A,$,J,G,X,Y,Z,K,_,U,q)}finally{let{appendFile:XD}=await import("node:fs/promises");try{await XD(V,`
|
|||
|
|
Completed: ${new Date().toISOString()}
|
|||
|
|
`,"utf-8")}catch(YD){}try{await Uu(k,{recursive:!0,force:!0})}catch(YD){console.warn(`Warning: Failed to cleanup temp directory: ${k}`)}}}async function gF(u,D,C,F,E,B,A,$,J,G,X,Y,Z,K,_,U,q){if(!await p())throw Error("FFmpeg is not installed or not in PATH");if(!await n())throw Error("MP4Box is not installed or not in PATH. Install gpac package.");let k=(H,I,UD,kD)=>{if(q)q({stage:H,percent:I,message:UD,currentProfile:kD})};k("analyzing",0,"Analyzing input video...");let W=await a(u),N=W.hasAudio,V=G&&G!=="auto"?G:J===!0?"nvenc":J===!1?"cpu":"auto",b=await s(),{selected:R,h264Encoder:XD,av1Encoder:YD,warnings:mD}=mF(b,V,A);if(mD.length>0)for(let H of mD)console.warn(`⚠️ ${H}`);let O;if(B&&B.length>0){let H=BD(B,W.width,W.height,W.fps,W.videoBitrate);if(H.errors.length>0){console.warn(`
|
|||
|
|
❌ Profile errors:`);for(let I of H.errors)console.warn(` - ${I}`);console.warn("")}if(H.warnings.length>0){console.warn(`
|
|||
|
|
⚠️ Profile warnings:`);for(let I of H.warnings)console.warn(` - ${I}`);console.warn("")}if(O=H.profiles,O.length===0)throw Error("No valid profiles found in custom list. Check errors above.")}else if(E)O=E;else O=DD(W.width,W.height,W.fps,W.videoBitrate);if(O.length===0)throw Error("No suitable profiles found for input video resolution");let AF=Yu(u,Gu(u)),o=ZD(D,AF);try{await Uu(o,{recursive:!0,force:!0})}catch(H){}await m(o);let v=[];if(A==="h264"||A==="dual"){let H=XD||"libx264",I=Xu(H,"h264");v.push({type:"h264",codec:H,preset:I})}if(A==="av1"||A==="dual"){let H=YD||"libsvtav1",I=Xu(H,"av1");v.push({type:"av1",codec:H,preset:I})}let ZF=v.map((H)=>H.type.toUpperCase()).join(" + "),$F=R==="cpu"?"CPU":R.toUpperCase();k("analyzing",20,`Using ${ZF} encoding (${$F})`,void 0);let JF=R==="cpu"?2:3,GD=new Map;for(let H=0;H<v.length;H++){let{type:I,codec:UD,preset:kD}=v[H],nD=H/v.length,YF=1/v.length;k("encoding",25+nD*40,`Stage 1: Encoding ${I.toUpperCase()} (${O.length} profiles)...`);let GF=I==="h264"?X?.h264:X?.av1,UF=await Eu(u,C,O,UD,kD,W.duration,F,W.audioBitrate,U,JF,I,GF,void 0,(t,sD)=>{let V8=O.findIndex((kF)=>kF.name===t),aD=25+nD*40,rD=sD/100*(40*YF/O.length);if(k("encoding",aD+rD,`Encoding ${I.toUpperCase()} ${t}...`,`${I}-${t}`),q)q({stage:"encoding",percent:aD+rD,currentProfile:`${I}-${t}`,profilePercent:sD,message:`Encoding ${I.toUpperCase()} ${t}...`})});GD.set(I,UF)}k("encoding",65,"Stage 1 complete: All codecs and profiles encoded"),k("encoding",70,"Stage 2: Creating segments and manifests...");let{manifestPath:KF,hlsManifestPath:XF}=await Ku(GD,o,O,F,A,$,N),cD=[];for(let H of GD.values())cD.push(...Array.from(H.values()));k("encoding",80,"Stage 2 complete: All formats packaged");let lD,dD;if(Y){k("thumbnails",80,"Generating thumbnail sprites...");let H={width:Z.width||160,height:Z.height||90,interval:Z.interval||1,columns:Z.columns||10},I=await Fu(u,o,W.duration,H);lD=I.spritePath,dD=I.vttPath,k("thumbnails",90,"Thumbnails generated")}let pD;if(K)k("thumbnails",92,"Generating poster image..."),pD=await uu(u,o,_),k("thumbnails",95,"Poster generated");return k("manifest",95,"Finalizing..."),k("complete",100,"Conversion complete!"),{manifestPath:KF,hlsManifestPath:XF,videoPaths:cD,thumbnailSpritePath:lD,thumbnailVttPath:dD,posterPath:pD,duration:W.duration,profiles:O,usedNvenc:v.some((H)=>H.codec.includes("nvenc")),selectedAccelerator:R,codecType:A,format:$}}var AD={nvenc:100,qsv:90,amf:80,vaapi:70,videotoolbox:65,v4l2:60,cpu:1};function mF(u,D,C){let F=C==="h264"||C==="dual",E=C==="av1"||C==="dual",B=[],A=new Set(["nvenc","qsv","amf"]),$=u.filter((k)=>F&&k.h264Encoder||E&&k.av1Encoder),J=$.filter((k)=>A.has(k.accelerator)),G=(k)=>$.find((W)=>W.accelerator===k),X;if(D!=="auto"){if(D==="cpu")X=void 0;else if(!A.has(D))B.push(`Ускоритель "${D}" пока не поддерживается, использую CPU`);else if(X=G(D),!X)throw Error(`Аппаратный ускоритель "${D}" недоступен в системе`)}else if(X=(J.length>0?J:[]).sort((W,N)=>(AD[N.accelerator]||0)-(AD[W.accelerator]||0))[0],!X&&$.length>0)B.push("Доступен аппаратный ускоритель, но он пока не поддерживается пайплайном, использую CPU");let Z=(J.length>0?J:[]).sort((k,W)=>(AD[W.accelerator]||0)-(AD[k.accelerator]||0)),K=(k)=>{let W=k==="h264"?X?.h264Encoder:X?.av1Encoder;if(W)return{encoder:W,accel:X?.accelerator};let N=Z.find((V)=>k==="h264"?V.h264Encoder:V.av1Encoder);if(N){if(D!=="auto"&&X)B.push(`Выбранный ускоритель "${X.accelerator}" не поддерживает ${k.toUpperCase()}, использую ${N.accelerator}`);return{encoder:k==="h264"?N.h264Encoder:N.av1Encoder,accel:N.accelerator}}if(D!=="auto"&&D!=="cpu")B.push(`Ускоритель "${D}" не поддерживает ${k.toUpperCase()}, использую CPU`);return{encoder:void 0,accel:"cpu"}},_=F?K("h264"):{encoder:void 0,accel:X?.accelerator},U=E?K("av1"):{encoder:void 0,accel:X?.accelerator};return{selected:X?.accelerator||_.accel||U.acc
|
|||
|
|
Options:`),console.error(" -r, --resolutions Video resolutions (e.g., 360,480,720 or 720@60,1080@60)"),console.error(" -c, --codec Video codec: av1, h264, or dual (default: dual)"),console.error(" -f, --format Streaming format: dash, hls, or both (default: both)"),console.error(" -p, --poster Poster timecode (e.g., 00:00:05 or 10)"),console.error(" --accel <type> Hardware accelerator: auto|nvenc|qsv|amf|cpu (default: auto)"),console.error(`
|
|||
|
|
Quality Options (override defaults):`),console.error(" --h264-cq <value> H.264 GPU CQ value (0-51, lower = better, default: auto)"),console.error(" --h264-crf <value> H.264 CPU CRF value (0-51, lower = better, default: auto)"),console.error(" --av1-cq <value> AV1 GPU CQ value (0-51, lower = better, default: auto)"),console.error(" --av1-crf <value> AV1 CPU CRF value (0-63, lower = better, default: auto)"),console.error(`
|
|||
|
|
Examples:`),console.error(" create-vod video.mp4"),console.error(" create-vod video.mp4 ./output"),console.error(" create-vod video.mp4 -r 360,480,720"),console.error(" create-vod video.mp4 -c av1 --av1-cq 40"),console.error(" create-vod video.mp4 -c dual --h264-cq 30 --av1-cq 39"),console.error(" create-vod video.mp4 -f hls"),console.error(" create-vod video.mp4 -c dual -f both"),console.error(" create-vod video.mp4 -r 720@60,1080@60,2160@60 -c av1 -f dash"),console.error(" create-vod video.mp4 -p 00:00:05"),console.error(" create-vod video.mp4 ./output -r 720,1080 -c dual -f both -p 10 --h264-cq 28 --av1-cq 37"),process.exit(1);console.log(`\uD83D\uDD0D Checking system...
|
|||
|
|
`);var EF=await p(),BF=await n(),gD=await s(),DF={nvenc:100,qsv:90,amf:80,vaapi:70,videotoolbox:65,v4l2:60},KD=gD.slice().sort((u,D)=>(DF[D.accelerator]||0)-(DF[u.accelerator]||0))[0];console.log(`FFmpeg: ${EF?"✅":"❌"}`);console.log(`MP4Box: ${BF?"✅":"❌"}`);var _8=Array.from(new Set(gD.map((u)=>u.accelerator.toUpperCase()))),vD=KD?KD.accelerator.toUpperCase():void 0,uF=_8.filter((u)=>u!==vD),Q8=vD?`✅ ${vD}${uF.length>0?` (${uF.join(", ")})`:""}`:"❌";console.log(`Hardware: ${Q8}`);console.log("");if(!EF)console.error("❌ FFmpeg not found. Please install FFmpeg first."),process.exit(1);if(!BF)console.error("❌ MP4Box not found. Please install: sudo pacman -S gpac"),process.exit(1);var H8=gD.some((u)=>u.av1Encoder);if((l==="av1"||l==="dual")&&!H8)console.error("⚠️ Warning: AV1 encoding requested but no hardware AV1 encoder found."),console.error(" CPU-based AV1 encoding (libsvtav1) will be VERY slow."),console.error(` Consider using --codec h264 for faster encoding.
|
|||
|
|
`);if((r==="hls"||r==="both")&&l==="av1")console.error("❌ Error: HLS format requires H.264 codec for Safari/iOS compatibility."),console.error(` Please use --codec h264 or --codec dual with --format hls
|
|||
|
|
`),process.exit(1);console.log(`\uD83D\uDCCA Analyzing video...
|
|||
|
|
`);var L=await a(i),N8=W8(i),z8=(N8.size/1048576).toFixed(2);console.log("\uD83D\uDCF9 Video Information:");console.log(` File: ${i}`);console.log(` Size: ${z8} MB`);console.log(` Resolution: ${L.width}x${L.height}`);console.log(` FPS: ${L.fps.toFixed(2)}`);console.log(` Duration: ${Math.floor(L.duration/60)}m ${Math.floor(L.duration%60)}s`);console.log(` Codec: ${L.codec}`);if(L.videoBitrate)console.log(` Video Bitrate: ${(L.videoBitrate/1000).toFixed(2)} Mbps`);if(L.audioBitrate)console.log(` Audio Bitrate: ${L.audioBitrate} kbps`);var PD=[];if(uD&&uD.length>0){let u=BD(uD,L.width,L.height,L.fps,L.videoBitrate);if(u.errors.length>0)console.error(`
|
|||
|
|
❌ Profile errors:`),u.errors.forEach((D)=>console.error(` - ${D}`)),process.exit(1);if(u.warnings.length>0)console.warn(`
|
|||
|
|
⚠️ Profile warnings:`),u.warnings.forEach((D)=>console.warn(` - ${D}`));PD=u.profiles.map((D)=>D.name)}else PD=DD(L.width,L.height,L.fps,L.videoBitrate).map((D)=>D.name);var q8=r==="both"?"DASH (manifest.mpd), HLS (master.m3u8)":r==="dash"?"DASH (manifest.mpd)":"HLS (master.m3u8)",I8=!0,x8=hD||"00:00:00";console.log(`
|
|||
|
|
\uD83D\uDCE6 Parameters:`);console.log(` Input: ${i}`);console.log(` Output: ${CF}`);console.log(` Codec: ${l}${l==="dual"?" (AV1 + H.264)":""}`);console.log(` Profiles: ${PD.join(", ")}`);console.log(` Manifests: ${q8}`);console.log(` Poster: ${x8} (will be generated)`);console.log(` Thumbnails: ${I8?"yes (with VTT)":"no"}`);console.log(` Accelerator: ${KD?KD.accelerator.toUpperCase():"CPU"}`);var f;if(j!==void 0||w!==void 0||S!==void 0||T!==void 0){if(f={},j!==void 0||w!==void 0){if(f.h264={},j!==void 0)f.h264.cq=j;if(w!==void 0)f.h264.crf=w;console.log(`\uD83C\uDF9A️ H.264 Quality: ${j!==void 0?`CQ ${j}`:""}${w!==void 0?` CRF ${w}`:""}`)}if(S!==void 0||T!==void 0){if(f.av1={},S!==void 0)f.av1.cq=S;if(T!==void 0)f.av1.crf=T;console.log(`\uD83C\uDF9A️ AV1 Quality: ${S!==void 0?`CQ ${S}`:""}${T!==void 0?` CRF ${T}`:""}`)}}console.log(`
|
|||
|
|
\uD83D\uDE80 Starting conversion...
|
|||
|
|
`);var JD=new yD.default.MultiBar({format:"{stage} | {bar} | {percentage}% | {name}",barCompleteChar:"█",barIncompleteChar:"░",hideCursor:!0,clearOnComplete:!1,stopOnComplete:!0},yD.default.Presets.shades_classic),TD={},bD=null;try{let u=await xD({input:i,outputDir:CF,customProfiles:uD,posterTimecode:hD,codec:l,format:r,segmentDuration:2,hardwareAccelerator:FF,quality:f,generateThumbnails:!0,generatePoster:!0,parallel:!0,onProgress:(D)=>{let C=D.stage==="encoding"?"Encoding":D.stage==="thumbnails"?"Thumbnails":D.stage==="manifest"?"Manifest":D.stage==="analyzing"?"Analyzing":"Complete";if(D.stage==="encoding"&&D.currentProfile){if(!TD[D.currentProfile])TD[D.currentProfile]=JD.create(100,0,{stage:"Encode",name:D.currentProfile});let F=D.profilePercent??D.percent;TD[D.currentProfile].update(F,{stage:"Encode",name:D.currentProfile})}if(!bD)bD=JD.create(100,0,{stage:C,name:"Overall"});bD.update(D.percent,{stage:C,name:D.message||"Overall"})}});JD.stop(),console.log(`
|
|||
|
|
✅ Conversion completed successfully!
|
|||
|
|
`)}catch(u){JD.stop(),console.error(`
|
|||
|
|
|
|||
|
|
❌ Error during conversion:`),console.error(u),process.exit(1)}
|