2020-10-15 00:07:38 +03:00
package ffmpeg
import (
"bytes"
"io"
"os"
"path/filepath"
"strconv"
"strings"
"net/http"
"github.com/owncast/owncast/config"
"github.com/owncast/owncast/utils"
log "github.com/sirupsen/logrus"
)
2020-11-13 02:14:59 +03:00
// FileWriterReceiverServiceCallback are to be fired when transcoder responses are written to disk.
2020-10-15 00:07:38 +03:00
type FileWriterReceiverServiceCallback interface {
SegmentWritten ( localFilePath string )
VariantPlaylistWritten ( localFilePath string )
MasterPlaylistWritten ( localFilePath string )
}
2020-11-13 02:14:59 +03:00
// FileWriterReceiverService accepts transcoder responses via HTTP and fires the callbacks.
2020-10-15 00:07:38 +03:00
type FileWriterReceiverService struct {
callbacks FileWriterReceiverServiceCallback
}
2020-11-13 02:14:59 +03:00
// SetupFileWriterReceiverService will start listening for transcoder responses.
2020-10-15 00:07:38 +03:00
func ( s * FileWriterReceiverService ) SetupFileWriterReceiverService ( callbacks FileWriterReceiverServiceCallback ) {
s . callbacks = callbacks
httpServer := http . NewServeMux ( )
httpServer . HandleFunc ( "/" , s . uploadHandler )
localListenerAddress := "127.0.0.1:" + strconv . Itoa ( config . Config . GetPublicWebServerPort ( ) + 1 )
2020-11-12 23:42:18 +03:00
go func ( ) {
if err := http . ListenAndServe ( localListenerAddress , httpServer ) ; err != nil {
log . Fatal ( err )
}
} ( )
2020-10-15 00:07:38 +03:00
log . Traceln ( "Transcoder response listening on: " + localListenerAddress )
}
func ( s * FileWriterReceiverService ) uploadHandler ( w http . ResponseWriter , r * http . Request ) {
if r . Method != "PUT" {
http . Error ( w , http . StatusText ( http . StatusBadRequest ) , http . StatusBadRequest )
return
}
path := r . URL . Path
writePath := filepath . Join ( config . PrivateHLSStoragePath , path )
var buf bytes . Buffer
2020-11-13 01:42:01 +03:00
_ , _ = io . Copy ( & buf , r . Body )
2020-10-15 00:07:38 +03:00
data := buf . Bytes ( )
2020-11-13 01:42:01 +03:00
2020-10-15 00:07:38 +03:00
f , err := os . Create ( writePath )
if err != nil {
2020-11-15 05:39:53 +03:00
returnError ( err , w )
2020-10-15 00:07:38 +03:00
return
}
defer f . Close ( )
_ , err = f . Write ( data )
if err != nil {
2020-11-15 05:39:53 +03:00
returnError ( err , w )
2020-10-15 00:07:38 +03:00
return
}
s . fileWritten ( writePath )
w . WriteHeader ( http . StatusOK )
}
var _inWarningState = false
func ( s * FileWriterReceiverService ) fileWritten ( path string ) {
index := utils . GetIndexFromFilePath ( path )
if utils . GetRelativePathFromAbsolutePath ( path ) == "hls/stream.m3u8" {
s . callbacks . MasterPlaylistWritten ( path )
} else if strings . HasSuffix ( path , ".ts" ) {
performanceMonitorKey := "segmentWritten-" + index
averagePerformance := utils . GetAveragePerformance ( performanceMonitorKey )
utils . StartPerformanceMonitor ( performanceMonitorKey )
s . callbacks . SegmentWritten ( path )
2020-11-13 01:18:32 +03:00
if averagePerformance != 0 && averagePerformance > float64 ( config . Config . GetVideoSegmentSecondsLength ( ) ) {
2020-10-15 00:07:38 +03:00
if ! _inWarningState {
log . Warnln ( "slow encoding for variant" , index , "if this continues you may see buffering or errors. troubleshoot this issue by visiting https://owncast.online/docs/troubleshooting/" )
_inWarningState = true
}
} else {
_inWarningState = false
}
} else if strings . HasSuffix ( path , ".m3u8" ) {
s . callbacks . VariantPlaylistWritten ( path )
}
}
2020-11-15 05:39:53 +03:00
func returnError ( err error , w http . ResponseWriter ) {
2020-10-15 00:07:38 +03:00
log . Errorln ( err )
http . Error ( w , http . StatusText ( http . StatusInternalServerError ) + ": " + err . Error ( ) , http . StatusInternalServerError )
}