mirror of
https://gitlab.crans.org/nounous/ghostream.git
synced 2024-12-22 19:42:20 +00:00
Use only one instance of FFPMEG per SRT stream instead of one per additional cast
This commit is contained in:
parent
351870e92a
commit
9049935ee4
@ -13,8 +13,8 @@ type Options struct {
|
|||||||
|
|
||||||
var (
|
var (
|
||||||
options Options
|
options Options
|
||||||
ffmpegInstances = make(map[string][]*exec.Cmd)
|
ffmpegInstances = make(map[string]*exec.Cmd)
|
||||||
ffmpegInputStreams = make(map[string][]*io.WriteCloser)
|
ffmpegInputStreams = make(map[string]*io.WriteCloser)
|
||||||
)
|
)
|
||||||
|
|
||||||
// New Load configuration
|
// New Load configuration
|
||||||
@ -25,71 +25,70 @@ func New(cfg *Options) error {
|
|||||||
|
|
||||||
// RegisterStream Declare a new open stream and create ffmpeg instances
|
// RegisterStream Declare a new open stream and create ffmpeg instances
|
||||||
func RegisterStream(streamKey string) {
|
func RegisterStream(streamKey string) {
|
||||||
ffmpegInstances[streamKey] = []*exec.Cmd{}
|
if len(options.Outputs[streamKey]) == 0 {
|
||||||
ffmpegInputStreams[streamKey] = []*io.WriteCloser{}
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// TODO Export the list of multicasts
|
params := []string{"-re", "-i", "pipe:0"}
|
||||||
for _, stream := range options.Outputs[streamKey] {
|
for _, stream := range options.Outputs[streamKey] {
|
||||||
// Launch FFMPEG instance
|
|
||||||
// TODO Set optimal parameters
|
// TODO Set optimal parameters
|
||||||
ffmpeg := exec.Command("ffmpeg", "-re", "-i", "pipe:0", "-f", "flv", "-c:v", "libx264", "-preset",
|
params = append(params, "-f", "flv", "-c:v", "libx264", "-preset",
|
||||||
"veryfast", "-maxrate", "3000k", "-bufsize", "6000k", "-pix_fmt", "yuv420p", "-g", "50", "-c:a", "aac",
|
"veryfast", "-maxrate", "3000k", "-bufsize", "6000k", "-pix_fmt", "yuv420p", "-g", "50", "-c:a", "aac",
|
||||||
"-b:a", "160k", "-ac", "2", "-ar", "44100", stream)
|
"-b:a", "160k", "-ac", "2", "-ar", "44100", stream)
|
||||||
|
|
||||||
// Open pipes
|
|
||||||
input, err := ffmpeg.StdinPipe()
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
output, err := ffmpeg.StdoutPipe()
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
errOutput, err := ffmpeg.StderrPipe()
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
ffmpegInstances[streamKey] = append(ffmpegInstances[streamKey], ffmpeg)
|
|
||||||
ffmpegInputStreams[streamKey] = append(ffmpegInputStreams[streamKey], &input)
|
|
||||||
|
|
||||||
if err := ffmpeg.Start(); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Log ffmpeg output
|
|
||||||
go func() {
|
|
||||||
scanner := bufio.NewScanner(output)
|
|
||||||
for scanner.Scan() {
|
|
||||||
log.Println("[FFMPEG " + streamKey + "] " + scanner.Text())
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
// Log also error output
|
|
||||||
go func() {
|
|
||||||
scanner := bufio.NewScanner(errOutput)
|
|
||||||
for scanner.Scan() {
|
|
||||||
log.Println("[FFMPEG ERROR " + streamKey + "] " + scanner.Text())
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
}
|
}
|
||||||
|
// Launch FFMPEG instance
|
||||||
|
ffmpeg := exec.Command("ffmpeg", params...)
|
||||||
|
|
||||||
|
// Open pipes
|
||||||
|
input, err := ffmpeg.StdinPipe()
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
output, err := ffmpeg.StdoutPipe()
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
errOutput, err := ffmpeg.StderrPipe()
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
ffmpegInstances[streamKey] = ffmpeg
|
||||||
|
ffmpegInputStreams[streamKey] = &input
|
||||||
|
|
||||||
|
if err := ffmpeg.Start(); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Log ffmpeg output
|
||||||
|
go func() {
|
||||||
|
scanner := bufio.NewScanner(output)
|
||||||
|
for scanner.Scan() {
|
||||||
|
log.Println("[FFMPEG " + streamKey + "] " + scanner.Text())
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
// Log also error output
|
||||||
|
go func() {
|
||||||
|
scanner := bufio.NewScanner(errOutput)
|
||||||
|
for scanner.Scan() {
|
||||||
|
log.Println("[FFMPEG ERROR " + streamKey + "] " + scanner.Text())
|
||||||
|
}
|
||||||
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
// SendPacket When a SRT packet is received, transmit it to all FFMPEG instances related to the stream key
|
// SendPacket When a SRT packet is received, transmit it to all FFMPEG instances related to the stream key
|
||||||
func SendPacket(streamKey string, data []byte) {
|
func SendPacket(streamKey string, data []byte) {
|
||||||
for _, stdin := range ffmpegInputStreams[streamKey] {
|
stdin := ffmpegInputStreams[streamKey]
|
||||||
_, err := (*stdin).Write(data)
|
_, err := (*stdin).Write(data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error while sending a packet to external streaming server for key "+streamKey, err)
|
log.Println("Error while sending a packet to external streaming server for key "+streamKey, err)
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// CloseConnection When the stream is ended, close FFMPEG instances
|
// CloseConnection When the stream is ended, close FFMPEG instances
|
||||||
func CloseConnection(streamKey string) {
|
func CloseConnection(streamKey string) {
|
||||||
for _, ffmpeg := range ffmpegInstances[streamKey] {
|
ffmpeg := ffmpegInstances[streamKey]
|
||||||
if err := ffmpeg.Process.Kill(); err != nil {
|
if err := ffmpeg.Process.Kill(); err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
|
||||||
}
|
}
|
||||||
delete(ffmpegInstances, streamKey)
|
delete(ffmpegInstances, streamKey)
|
||||||
delete(ffmpegInputStreams, streamKey)
|
delete(ffmpegInputStreams, streamKey)
|
||||||
|
Loading…
Reference in New Issue
Block a user