mirror of
https://github.com/AlexxIT/go2rtc.git
synced 2025-09-27 04:36:12 +08:00
161 lines
4.1 KiB
Go
161 lines
4.1 KiB
Go
package ffmpeg
|
|
|
|
import (
|
|
"github.com/AlexxIT/go2rtc/cmd/app"
|
|
"github.com/AlexxIT/go2rtc/cmd/exec"
|
|
"github.com/AlexxIT/go2rtc/cmd/ffmpeg/device"
|
|
"github.com/AlexxIT/go2rtc/cmd/streams"
|
|
"github.com/AlexxIT/go2rtc/pkg/streamer"
|
|
"net/url"
|
|
"strings"
|
|
)
|
|
|
|
func Init() {
|
|
var cfg struct {
|
|
Mod map[string]string `yaml:"ffmpeg"`
|
|
}
|
|
|
|
// defaults
|
|
|
|
cfg.Mod = map[string]string{
|
|
"bin": "ffmpeg",
|
|
|
|
// inputs
|
|
"file": "-re -stream_loop -1 -i {input}",
|
|
"http": "-fflags nobuffer -flags low_delay -i {input}",
|
|
"rtsp": "-fflags nobuffer -flags low_delay -rtsp_transport tcp -timeout 5000000 -i {input}",
|
|
|
|
// output
|
|
"output": "-rtsp_transport tcp -f rtsp {output}",
|
|
|
|
// `-g 30` - group of picture, GOP, keyframe interval
|
|
// `-preset superfast` - we can't use ultrafast because it doesn't support `-profile main -level 4.1`
|
|
// `-tune zerolatency` - for minimal latency
|
|
// `-profile main -level 4.1` - most used streaming profile
|
|
// `-pix_fmt yuv420p` - if input pix format 4:2:2
|
|
"h264": "-codec:v libx264 -g 30 -preset superfast -tune zerolatency -profile main -level 4.1 -pix_fmt yuv420p",
|
|
"h264/ultra": "-codec:v libx264 -g 30 -preset ultrafast -tune zerolatency",
|
|
"h264/high": "-codec:v libx264 -g 30 -preset superfast -tune zerolatency",
|
|
"h265": "-codec:v libx265 -g 30 -preset ultrafast -tune zerolatency",
|
|
"mjpeg": "-codec:v mjpeg -force_duplicated_matrix 1 -huffman 0 -pix_fmt yuvj420p",
|
|
"opus": "-codec:a libopus -ar 48000 -ac 2",
|
|
"pcmu": "-codec:a pcm_mulaw -ar 8000 -ac 1",
|
|
"pcmu/16000": "-codec:a pcm_mulaw -ar 16000 -ac 1",
|
|
"pcmu/48000": "-codec:a pcm_mulaw -ar 48000 -ac 1",
|
|
"pcma": "-codec:a pcm_alaw -ar 8000 -ac 1",
|
|
"pcma/16000": "-codec:a pcm_alaw -ar 16000 -ac 1",
|
|
"pcma/48000": "-codec:a pcm_alaw -ar 48000 -ac 1",
|
|
"aac/16000": "-codec:a aac -ar 16000 -ac 1",
|
|
}
|
|
|
|
app.LoadConfig(&cfg)
|
|
|
|
tpl := cfg.Mod
|
|
|
|
streams.HandleFunc("ffmpeg", func(s string) (streamer.Producer, error) {
|
|
s = s[7:] // remove `ffmpeg:`
|
|
|
|
var query url.Values
|
|
var queryVideo, queryAudio bool
|
|
if i := strings.IndexByte(s, '#'); i > 0 {
|
|
query = parseQuery(s[i+1:])
|
|
queryVideo = query["video"] != nil
|
|
queryAudio = query["audio"] != nil
|
|
s = s[:i]
|
|
} else {
|
|
// by default query both video and audio
|
|
queryVideo = true
|
|
queryAudio = true
|
|
}
|
|
|
|
var input string
|
|
if i := strings.IndexByte(s, ':'); i > 0 {
|
|
switch s[:i] {
|
|
case "http", "https", "rtmp":
|
|
input = strings.Replace(tpl["http"], "{input}", s, 1)
|
|
case "rtsp", "rtsps":
|
|
// https://ffmpeg.org/ffmpeg-protocols.html#rtsp
|
|
// skip unnecessary input tracks
|
|
switch {
|
|
case queryVideo && queryAudio:
|
|
input = "-allowed_media_types video+audio "
|
|
case queryVideo:
|
|
input = "-allowed_media_types video "
|
|
case queryAudio:
|
|
input = "-allowed_media_types audio "
|
|
}
|
|
|
|
input += strings.Replace(tpl["rtsp"], "{input}", s, 1)
|
|
}
|
|
}
|
|
|
|
if input == "" {
|
|
if strings.HasPrefix(s, "device?") {
|
|
var err error
|
|
input, err = device.GetInput(s)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
} else {
|
|
input = strings.Replace(tpl["file"], "{input}", s, 1)
|
|
}
|
|
}
|
|
|
|
s = "exec:" + tpl["bin"] + " -hide_banner " + input
|
|
|
|
if query != nil {
|
|
for _, raw := range query["raw"] {
|
|
s += " " + raw
|
|
}
|
|
|
|
// TODO: multiple codecs via -map
|
|
// s += fmt.Sprintf(" -map 0:v:0 -c:v:%d copy", i)
|
|
|
|
for _, video := range query["video"] {
|
|
if video == "copy" {
|
|
s += " -codec:v copy"
|
|
} else {
|
|
s += " " + tpl[video]
|
|
}
|
|
}
|
|
|
|
for _, audio := range query["audio"] {
|
|
if audio == "copy" {
|
|
s += " -codec:a copy"
|
|
} else {
|
|
s += " " + tpl[audio]
|
|
}
|
|
}
|
|
|
|
switch {
|
|
case queryVideo && !queryAudio:
|
|
s += " -an"
|
|
case queryAudio && !queryVideo:
|
|
s += " -vn"
|
|
}
|
|
} else {
|
|
s += " -c copy"
|
|
}
|
|
|
|
s += " " + tpl["output"]
|
|
|
|
return exec.Handle(s)
|
|
})
|
|
|
|
device.Bin = cfg.Mod["bin"]
|
|
device.Init()
|
|
}
|
|
|
|
func parseQuery(s string) map[string][]string {
|
|
query := map[string][]string{}
|
|
for _, key := range strings.Split(s, "#") {
|
|
var value string
|
|
i := strings.IndexByte(key, '=')
|
|
if i > 0 {
|
|
key, value = key[:i], key[i+1:]
|
|
}
|
|
query[key] = append(query[key], value)
|
|
}
|
|
return query
|
|
}
|