package config import ( "context" "fmt" "net/http" "regexp" "strings" "time" "github.com/quic-go/quic-go" "golang.org/x/net/websocket" "m7s.live/engine/v4/log" "m7s.live/engine/v4/util" ) type PublishConfig interface { GetPublishConfig() Publish } type SubscribeConfig interface { GetSubscribeConfig() *Subscribe } type PullConfig interface { GetPullConfig() *Pull } type PushConfig interface { GetPushConfig() *Push } type Publish struct { PubAudio bool `default:"true" desc:"是否发布音频"` PubVideo bool `default:"true" desc:"是否发布视频"` InsertSEI bool `desc:"是否启用SEI插入"` // 是否启用SEI插入 KickExist bool `desc:"是否踢掉已经存在的发布者"` // 是否踢掉已经存在的发布者 PublishTimeout time.Duration `default:"10s" desc:"发布无数据超时"` // 发布无数据超时 WaitCloseTimeout time.Duration `desc:"延迟自动关闭(等待重连)"` // 延迟自动关闭(等待重连) DelayCloseTimeout time.Duration `desc:"延迟自动关闭(无订阅时)"` // 延迟自动关闭(无订阅时) IdleTimeout time.Duration `desc:"空闲(无订阅)超时"` // 空闲(无订阅)超时 PauseTimeout time.Duration `default:"30s" desc:"暂停超时时间"` // 暂停超时 BufferTime time.Duration `desc:"缓冲长度(单位:秒),0代表取最近关键帧"` // 缓冲长度(单位:秒),0代表取最近关键帧 SpeedLimit time.Duration `default:"500ms" desc:"速度限制最大等待时间,0则不等待"` //速度限制最大等待时间 Key string `desc:"发布鉴权key"` // 发布鉴权key SecretArgName string `default:"secret" desc:"发布鉴权参数名"` // 发布鉴权参数名 ExpireArgName string `default:"expire" desc:"发布鉴权失效时间参数名"` // 发布鉴权失效时间参数名 RingSize string `default:"256-1024" desc:"缓冲范围"` // 初始缓冲区大小 } func (c Publish) GetPublishConfig() Publish { return c } type Subscribe struct { SubAudio bool `default:"true" desc:"是否订阅音频"` SubVideo bool `default:"true" desc:"是否订阅视频"` SubVideoArgName string `default:"vts" desc:"定订阅的视频轨道参数名"` // 指定订阅的视频轨道参数名 SubAudioArgName string `default:"ats" desc:"指定订阅的音频轨道参数名"` // 指定订阅的音频轨道参数名 SubDataArgName string `default:"dts" desc:"指定订阅的数据轨道参数名"` // 指定订阅的数据轨道参数名 SubModeArgName string `desc:"指定订阅的模式参数名"` // 指定订阅的模式参数名 SubAudioTracks []string `desc:"指定订阅的音频轨道"` // 指定订阅的音频轨道 SubVideoTracks []string `desc:"指定订阅的视频轨道"` // 指定订阅的视频轨道 SubDataTracks []string `desc:"指定订阅的数据轨道"` // 指定订阅的数据轨道 SubMode int `desc:"订阅模式" enum:"0:实时模式,1:首屏后不进行追赶,2:从缓冲最大的关键帧开始播放"` // 0,实时模式:追赶发布者进度,在播放首屏后等待发布者的下一个关键帧,然后跳到该帧。1、首屏后不进行追赶。2、从缓冲最大的关键帧开始播放,也不追赶,需要发布者配置缓存长度 SyncMode int `desc:"同步模式" enum:"0:采用时间戳同步,1:采用写入时间同步"` // 0,采用时间戳同步,1,采用写入时间同步 IFrameOnly bool `desc:"只要关键帧"` // 只要关键帧 WaitTimeout time.Duration `default:"10s" desc:"等待流超时时间"` // 等待流超时 WriteBufferSize int `desc:"写缓冲大小"` // 写缓冲大小 Key string `desc:"订阅鉴权key"` // 订阅鉴权key SecretArgName string `default:"secret" desc:"订阅鉴权参数名"` // 订阅鉴权参数名 ExpireArgName string `default:"expire" desc:"订阅鉴权失效时间参数名"` // 订阅鉴权失效时间参数名 Internal bool `default:"false" desc:"是否内部订阅"` // 是否内部订阅 } func (c *Subscribe) GetSubscribeConfig() *Subscribe { return c } type Pull struct { RePull int // 断开后自动重拉,0 表示不自动重拉,-1 表示无限重拉,高于0 的数代表最大重拉次数 EnableRegexp bool // 是否启用正则表达式 PullOnStart map[string]string // 启动时拉流的列表 PullOnSub map[string]string // 订阅时自动拉流的列表 Proxy string // 代理地址 } func (p *Pull) GetPullConfig() *Pull { return p } func (p *Pull) CheckPullOnStart(streamPath string) string { if p.PullOnStart == nil { return "" } url, ok := p.PullOnStart[streamPath] if !ok && p.EnableRegexp { for k, url := range p.PullOnStart { if r, err := regexp.Compile(k); err != nil { if group := r.FindStringSubmatch(streamPath); group != nil { for i, value := range group { url = strings.Replace(url, fmt.Sprintf("$%d", i), value, -1) } return url } } return "" } } return url } func (p *Pull) CheckPullOnSub(streamPath string) string { if p.PullOnSub == nil { return "" } url, ok := p.PullOnSub[streamPath] if !ok && p.EnableRegexp { for k, url := range p.PullOnSub { if r, err := regexp.Compile(k); err == nil { if group := r.FindStringSubmatch(streamPath); group != nil { for i, value := range group { url = strings.Replace(url, fmt.Sprintf("$%d", i), value, -1) } return url } } return "" } } return url } func (p *Pull) AddPullOnStart(streamPath string, url string) { if p.PullOnStart == nil { p.PullOnStart = make(map[string]string) } p.PullOnStart[streamPath] = url } func (p *Pull) AddPullOnSub(streamPath string, url string) { if p.PullOnSub == nil { p.PullOnSub = make(map[string]string) } p.PullOnSub[streamPath] = url } type Push struct { EnableRegexp bool // 是否启用正则表达式 RePush int // 断开后自动重推,0 表示不自动重推,-1 表示无限重推,高于0 的数代表最大重推次数 PushList map[string]string // 自动推流列表 Proxy string // 代理地址 } func (p *Push) GetPushConfig() *Push { return p } func (p *Push) AddPush(url string, streamPath string) { if p.PushList == nil { p.PushList = make(map[string]string) } p.PushList[streamPath] = url } func (p *Push) CheckPush(streamPath string) string { url, ok := p.PushList[streamPath] if !ok && p.EnableRegexp { for k, url := range p.PushList { if r, err := regexp.Compile(k); err == nil { if group := r.FindStringSubmatch(streamPath); group != nil { for i, value := range group { url = strings.Replace(url, fmt.Sprintf("$%d", i), value, -1) } return url } } return "" } } return url } type Console struct { Server string `default:"console.monibuca.com:44944" desc:"远程控制台地址"` //远程控制台地址 Secret string `desc:"远程控制台密钥"` //远程控制台密钥 PublicAddr string `desc:"远程控制台公网地址"` //公网地址,提供远程控制台访问的地址,不配置的话使用自动识别的地址 PublicAddrTLS string `desc:"远程控制台公网TLS地址"` } type Engine struct { Publish Subscribe HTTP Console EnableAVCC bool `default:"true" desc:"启用AVCC格式,rtmp、http-flv协议使用"` //启用AVCC格式,rtmp、http-flv协议使用 EnableRTP bool `default:"true" desc:"启用RTP格式,rtsp、webrtc等协议使用"` //启用RTP格式,rtsp、webrtc等协议使用 EnableSubEvent bool `default:"true" desc:"启用订阅事件,禁用可以提高性能"` //启用订阅事件,禁用可以提高性能 EnableAuth bool `default:"true" desc:"启用鉴权"` //启用鉴权 LogLang string `default:"zh" desc:"日志语言" enum:"zh:中文,en:英文"` //日志语言 LogLevel string `default:"info" enum:"trace:跟踪,debug:调试,info:信息,warn:警告,error:错误"` //日志级别 EventBusSize int `default:"10" desc:"事件总线大小"` //事件总线大小 PulseInterval time.Duration `default:"5s" desc:"心跳事件间隔"` //心跳事件间隔 DisableAll bool `default:"false" desc:"禁用所有插件"` //禁用所有插件 RTPReorderBufferLen int `default:"50" desc:"RTP重排序缓冲区长度"` //RTP重排序缓冲区长度 PoolSize int `desc:"内存池大小"` //内存池大小 enableReport bool `default:"false"` //启用报告,用于统计和监控 reportStream quic.Stream // console server connection instanceId string // instance id 来自console } func (cfg *Engine) GetEnableReport() bool { return cfg.enableReport } func (cfg *Engine) GetInstanceId() string { return cfg.instanceId } var Global *Engine func (cfg *Engine) InitDefaultHttp() { Global = cfg cfg.HTTP.mux = http.NewServeMux() cfg.HTTP.ListenAddrTLS = ":8443" cfg.HTTP.ListenAddr = ":8080" } type myResponseWriter struct { } func (*myResponseWriter) Header() http.Header { return make(http.Header) } func (*myResponseWriter) WriteHeader(statusCode int) { } func (w *myResponseWriter) Flush() { } type myWsWriter struct { myResponseWriter *websocket.Conn } func (w *myWsWriter) Write(b []byte) (int, error) { return len(b), websocket.Message.Send(w.Conn, b) } func (cfg *Engine) WsRemote() { for { conn, err := websocket.Dial(cfg.Server, "", "https://console.monibuca.com") wr := &myWsWriter{Conn: conn} if err != nil { log.Error("connect to console server ", cfg.Server, " ", err) time.Sleep(time.Second * 5) continue } if err = websocket.Message.Send(conn, cfg.Secret); err != nil { time.Sleep(time.Second * 5) continue } var rMessage map[string]interface{} if err := websocket.JSON.Receive(conn, &rMessage); err == nil { if rMessage["code"].(float64) != 0 { log.Error("connect to console server ", cfg.Server, " ", rMessage["msg"]) return } else { log.Info("connect to console server ", cfg.Server, " success") } } for { var msg string err := websocket.Message.Receive(conn, &msg) if err != nil { log.Error("read console server error:", err) break } else { b, a, f := strings.Cut(msg, "\n") if f { if len(a) > 0 { req, err := http.NewRequest("POST", b, strings.NewReader(a)) if err != nil { log.Error("read console server error:", err) break } h, _ := cfg.mux.Handler(req) h.ServeHTTP(wr, req) } else { req, err := http.NewRequest("GET", b, nil) if err != nil { log.Error("read console server error:", err) break } h, _ := cfg.mux.Handler(req) h.ServeHTTP(wr, req) } } else { } } } } } func (cfg *Engine) OnEvent(event any) { switch v := event.(type) { case []byte: if cfg.reportStream != nil { cfg.reportStream.Write(v) cfg.reportStream.Write([]byte{0}) } case context.Context: util.RTPReorderBufferLen = uint16(cfg.RTPReorderBufferLen) if cfg.Secret != "" && cfg.Server != "" { if strings.HasPrefix(cfg.Console.Server, "wss") { go cfg.WsRemote() } else { go cfg.WtRemote(v) } } } }