-async function canDoQuickTranscode (path: string): Promise<boolean> {
- // NOTE: This could be optimized by running ffprobe only once (but it runs fast anyway)
- const videoStream = await getVideoStreamFromFile(path)
- const parsedAudio = await audio.get(path)
- const fps = await getVideoFileFPS(path)
- const bitRate = await getVideoFileBitrate(path)
- const resolution = await getVideoFileResolution(path)
-
- // check video params
- if (videoStream[ 'codec_name' ] !== 'h264') return false
- if (fps < VIDEO_TRANSCODING_FPS.MIN || fps > VIDEO_TRANSCODING_FPS.MAX) return false
- if (bitRate > getMaxBitrate(resolution.videoFileResolution, fps, VIDEO_TRANSCODING_FPS)) return false
-
- // check audio params (if audio stream exists)
- if (parsedAudio.audioStream) {
- if (parsedAudio.audioStream[ 'codec_name' ] !== 'aac') return false
-
- const maxAudioBitrate = audio.bitrate[ 'aac' ](parsedAudio.audioStream[ 'bit_rate' ])
- if (maxAudioBitrate !== -1 && parsedAudio.audioStream[ 'bit_rate' ] > maxAudioBitrate) return false
+function runLiveTranscoding (rtmpUrl: string, outPath: string, resolutions: number[], fps, deleteSegments: boolean) {
+ const command = getFFmpeg(rtmpUrl)
+ command.inputOption('-fflags nobuffer')
+
+ const varStreamMap: string[] = []
+
+ command.complexFilter([
+ {
+ inputs: '[v:0]',
+ filter: 'split',
+ options: resolutions.length,
+ outputs: resolutions.map(r => `vtemp${r}`)
+ },
+
+ ...resolutions.map(r => ({
+ inputs: `vtemp${r}`,
+ filter: 'scale',
+ options: `w=-2:h=${r}`,
+ outputs: `vout${r}`
+ }))
+ ])
+
+ command.outputOption('-b_strategy 1')
+ command.outputOption('-bf 16')
+ command.outputOption('-preset superfast')
+ command.outputOption('-level 3.1')
+ command.outputOption('-map_metadata -1')
+ command.outputOption('-pix_fmt yuv420p')
+ command.outputOption('-max_muxing_queue_size 1024')
+ command.outputOption('-g ' + (fps * 2))
+
+ for (let i = 0; i < resolutions.length; i++) {
+ const resolution = resolutions[i]
+
+ command.outputOption(`-map [vout${resolution}]`)
+ command.outputOption(`-c:v:${i} libx264`)
+ command.outputOption(`-b:v:${i} ${getTargetBitrate(resolution, fps, VIDEO_TRANSCODING_FPS)}`)
+
+ command.outputOption(`-map a:0`)
+ command.outputOption(`-c:a:${i} aac`)
+
+ varStreamMap.push(`v:${i},a:${i}`)
+ }
+
+ addDefaultLiveHLSParams(command, outPath, deleteSegments)
+
+ command.outputOption('-var_stream_map', varStreamMap.join(' '))
+
+ command.run()
+
+ return command
+}
+
+function runLiveMuxing (rtmpUrl: string, outPath: string, deleteSegments: boolean) {
+ const command = getFFmpeg(rtmpUrl)
+ command.inputOption('-fflags nobuffer')
+
+ command.outputOption('-c:v copy')
+ command.outputOption('-c:a copy')
+ command.outputOption('-map 0:a?')
+ command.outputOption('-map 0:v?')
+
+ addDefaultLiveHLSParams(command, outPath, deleteSegments)
+
+ command.run()
+
+ return command
+}
+
+async function hlsPlaylistToFragmentedMP4 (hlsDirectory: string, segmentFiles: string[], outputPath: string) {
+ const concatFilePath = join(hlsDirectory, 'concat.txt')
+
+ function cleaner () {
+ remove(concatFilePath)
+ .catch(err => logger.error('Cannot remove concat file in %s.', hlsDirectory, { err }))