2020-05-29 18:08:33 -07:00
package main
import (
"fmt"
2020-06-01 13:28:14 -07:00
"log"
2020-05-29 18:08:33 -07:00
"os"
"os/exec"
2020-06-01 16:53:31 -07:00
"path"
2020-06-09 01:52:15 -07:00
"strings"
2020-05-29 18:08:33 -07:00
)
2020-06-01 16:53:31 -07:00
func startFfmpeg ( configuration Config ) {
var outputDir = configuration . PublicHLSPath
2020-06-09 01:52:15 -07:00
var variantPlaylistPath = configuration . PublicHLSPath
2020-05-29 18:08:33 -07:00
2020-06-03 01:34:05 -07:00
if configuration . IPFS . Enabled || configuration . S3 . Enabled {
2020-06-01 16:53:31 -07:00
outputDir = configuration . PrivateHLSPath
2020-06-09 01:52:15 -07:00
variantPlaylistPath = configuration . PrivateHLSPath
2020-06-01 16:53:31 -07:00
}
2020-06-01 13:28:14 -07:00
2020-06-09 01:52:15 -07:00
outputDir = path . Join ( outputDir , "%v" )
// var masterPlaylistName = path.Join(configuration.PublicHLSPath, "%v", "stream.m3u8")
var variantPlaylistName = path . Join ( variantPlaylistPath , "%v" , "stream.m3u8" )
// var variantRootPath = configuration.PublicHLSPath
// variantRootPath = path.Join(variantRootPath, "%v")
// variantPlaylistName := path.Join("%v", "stream.m3u8")
log . Printf ( "Starting transcoder saving to /%s." , variantPlaylistName )
2020-06-01 17:42:36 -07:00
pipePath := getTempPipePath ( )
2020-06-01 13:28:14 -07:00
2020-06-09 01:52:15 -07:00
var videoMaps = make ( [ ] string , 0 )
var streamMaps = make ( [ ] string , 0 )
var audioMaps = make ( [ ] string , 0 )
2020-06-12 12:55:50 -07:00
var videoMapsString = ""
var audioMapsString = ""
var streamMappingString = ""
if configuration . VideoSettings . EnablePassthrough || len ( configuration . VideoSettings . StreamQualities ) == 0 {
fmt . Println ( "Enabling passthrough video" )
// videoMaps = append(videoMaps, fmt.Sprintf("-map 0:v -c:v copy"))
streamMaps = append ( streamMaps , fmt . Sprintf ( "v:%d,a:%d" , 0 , 0 ) )
} else {
for index , quality := range configuration . VideoSettings . StreamQualities {
videoMaps = append ( videoMaps , fmt . Sprintf ( "-map v:0 -c:v:%d libx264 -b:v:%d %s" , index , index , quality . Bitrate ) )
streamMaps = append ( streamMaps , fmt . Sprintf ( "v:%d,a:%d" , index , index ) )
videoMapsString = strings . Join ( videoMaps , " " )
audioMaps = append ( audioMaps , "-map a:0" )
audioMapsString = strings . Join ( audioMaps , " " ) + " -c:a copy" // Pass through audio for all the variants, don't reencode
}
2020-06-09 01:52:15 -07:00
}
2020-06-12 12:55:50 -07:00
streamMappingString = "-var_stream_map \"" + strings . Join ( streamMaps , " " ) + "\""
2020-06-09 01:52:15 -07:00
ffmpegFlags := [ ] string {
"-hide_banner" ,
2020-06-11 15:05:06 -07:00
"-re" ,
2020-06-09 01:52:15 -07:00
"-i pipe:" ,
2020-06-11 17:34:20 -07:00
// "-vf scale=900:-2", // Re-enable in the future with a config to togging resizing?
// "-sws_flags fast_bilinear",
2020-06-12 12:55:50 -07:00
videoMapsString , // All the different video variants
audioMapsString ,
2020-06-09 01:52:15 -07:00
"-master_pl_name stream.m3u8" ,
2020-06-11 17:34:20 -07:00
"-g 60" , "-keyint_min 60" , // create key frame (I-frame) every 48 frames (~2 seconds) - will later affect correct slicing of segments and alignment of renditions
"-framerate 30" ,
"-r 15" ,
2020-06-09 13:15:44 -07:00
"-preset " + configuration . VideoSettings . EncoderPreset ,
2020-06-11 17:34:20 -07:00
"-sc_threshold 0" , // don't create key frames on scene change - only according to -g
"-profile:v main" , // Main – for standard definition (SD) to 640× 480, High – for high definition (HD) to 1920× 1080
2020-06-09 01:52:15 -07:00
"-f hls" ,
"-hls_list_size 30" ,
"-hls_time 10" ,
"-strftime 1" ,
"-use_localtime 1" ,
"-hls_playlist_type event" ,
"-hls_segment_filename " + path . Join ( outputDir , "stream-%Y%m%d-%s.ts" ) ,
"-hls_flags delete_segments+program_date_time+temp_file" ,
"-segment_wrap 100" ,
2020-06-09 02:56:23 -07:00
"-tune zerolatency" ,
2020-06-12 12:55:50 -07:00
streamMappingString ,
2020-06-09 01:52:15 -07:00
variantPlaylistName ,
}
ffmpegFlagsString := strings . Join ( ffmpegFlags , " " )
ffmpegCmd := "cat " + pipePath + " | " + configuration . FFMpegPath + " " + ffmpegFlagsString
// fmt.Println(ffmpegCmd)
2020-06-01 13:28:14 -07:00
2020-06-09 18:28:07 -07:00
_ , err := exec . Command ( "sh" , "-c" , ffmpegCmd ) . Output ( )
2020-06-09 01:52:15 -07:00
fmt . Println ( err )
2020-05-29 18:08:33 -07:00
verifyError ( err )
}
func writePlaylist ( data string , filePath string ) {
f , err := os . Create ( filePath )
2020-06-01 12:15:07 -07:00
defer f . Close ( )
2020-05-29 18:08:33 -07:00
if err != nil {
fmt . Println ( err )
return
}
_ , err = f . WriteString ( data )
if err != nil {
fmt . Println ( err )
return
}
}