package core import ( "bufio" "context" "errors" "fmt" "net" "strings" "sync" "time" "github.com/bluenviron/gortsplib/v4/pkg/description" "github.com/bluenviron/gortsplib/v4/pkg/format" "github.com/bluenviron/mediacommon/pkg/codecs/h264" "github.com/bluenviron/mediacommon/pkg/codecs/h265" "github.com/bluenviron/mediacommon/pkg/formats/mpegts" "github.com/datarhei/gosrt" "github.com/google/uuid" "github.com/bluenviron/mediamtx/internal/asyncwriter" "github.com/bluenviron/mediamtx/internal/conf" "github.com/bluenviron/mediamtx/internal/externalcmd" "github.com/bluenviron/mediamtx/internal/logger" "github.com/bluenviron/mediamtx/internal/stream" "github.com/bluenviron/mediamtx/internal/unit" ) func durationGoToMPEGTS(v time.Duration) int64 { return int64(v.Seconds() * 90000) } type srtConnState int const ( srtConnStateRead srtConnState = iota + 1 srtConnStatePublish ) type srtConnPathManager interface { addReader(req pathAddReaderReq) pathAddReaderRes addPublisher(req pathAddPublisherReq) pathAddPublisherRes } type srtConnParent interface { logger.Writer closeConn(*srtConn) } type srtConn struct { *conn rtspAddress string readTimeout conf.StringDuration writeTimeout conf.StringDuration writeQueueSize int udpMaxPayloadSize int connReq srt.ConnRequest wg *sync.WaitGroup externalCmdPool *externalcmd.Pool pathManager srtConnPathManager parent srtConnParent ctx context.Context ctxCancel func() created time.Time uuid uuid.UUID mutex sync.RWMutex state srtConnState pathName string sconn srt.Conn chNew chan srtNewConnReq chSetConn chan srt.Conn } func newSRTConn( parentCtx context.Context, rtspAddress string, readTimeout conf.StringDuration, writeTimeout conf.StringDuration, writeQueueSize int, udpMaxPayloadSize int, connReq srt.ConnRequest, runOnConnect string, runOnConnectRestart bool, runOnDisconnect string, wg *sync.WaitGroup, externalCmdPool *externalcmd.Pool, pathManager srtConnPathManager, parent srtConnParent, ) *srtConn { ctx, ctxCancel := context.WithCancel(parentCtx) c := &srtConn{ rtspAddress: rtspAddress, readTimeout: readTimeout, writeTimeout: writeTimeout, writeQueueSize: writeQueueSize, udpMaxPayloadSize: udpMaxPayloadSize, connReq: connReq, wg: wg, externalCmdPool: externalCmdPool, pathManager: pathManager, parent: parent, ctx: ctx, ctxCancel: ctxCancel, created: time.Now(), uuid: uuid.New(), chNew: make(chan srtNewConnReq), chSetConn: make(chan srt.Conn), } c.conn = newConn( rtspAddress, runOnConnect, runOnConnectRestart, runOnDisconnect, externalCmdPool, c, ) c.Log(logger.Info, "opened") c.wg.Add(1) go c.run() return c } func (c *srtConn) close() { c.ctxCancel() } func (c *srtConn) Log(level logger.Level, format string, args ...interface{}) { c.parent.Log(level, "[conn %v] "+format, append([]interface{}{c.connReq.RemoteAddr()}, args...)...) } func (c *srtConn) ip() net.IP { return c.connReq.RemoteAddr().(*net.UDPAddr).IP } func (c *srtConn) run() { //nolint:dupl defer c.wg.Done() desc := c.apiReaderDescribe() c.conn.open(desc) defer c.conn.close(desc) err := c.runInner() c.ctxCancel() c.parent.closeConn(c) c.Log(logger.Info, "closed: %v", err) } func (c *srtConn) runInner() error { var req srtNewConnReq select { case req = <-c.chNew: case <-c.ctx.Done(): return errors.New("terminated") } answerSent, err := c.runInner2(req) if !answerSent { req.res <- nil } return err } func (c *srtConn) runInner2(req srtNewConnReq) (bool, error) { parts := strings.Split(req.connReq.StreamId(), ":") if (len(parts) != 2 && len(parts) != 4) || (parts[0] != "read" && parts[0] != "publish") { return false, fmt.Errorf("invalid streamid '%s':"+ " it must be 'action:pathname' or 'action:pathname:user:pass', "+ "where action is either read or publish, pathname is the path name, user and pass are the credentials", req.connReq.StreamId()) } pathName := parts[1] user := "" pass := "" if len(parts) == 4 { user, pass = parts[2], parts[3] } if parts[0] == "publish" { return c.runPublish(req, pathName, user, pass) } return c.runRead(req, pathName, user, pass) } func (c *srtConn) runPublish(req srtNewConnReq, pathName string, user string, pass string) (bool, error) { res := c.pathManager.addPublisher(pathAddPublisherReq{ author: c, pathName: pathName, credentials: authCredentials{ ip: c.ip(), user: user, pass: pass, proto: authProtocolSRT, id: &c.uuid, }, }) if res.err != nil { if terr, ok := res.err.(*errAuthentication); ok { // TODO: re-enable. Currently this freezes the listener. // wait some seconds to stop brute force attacks // <-time.After(srtPauseAfterAuthError) return false, terr } return false, res.err } defer res.path.removePublisher(pathRemovePublisherReq{author: c}) sconn, err := c.exchangeRequestWithConn(req) if err != nil { return true, err } c.mutex.Lock() c.state = srtConnStatePublish c.pathName = pathName c.sconn = sconn c.mutex.Unlock() readerErr := make(chan error) go func() { readerErr <- c.runPublishReader(sconn, res.path) }() select { case err := <-readerErr: sconn.Close() return true, err case <-c.ctx.Done(): sconn.Close() <-readerErr return true, errors.New("terminated") } } func (c *srtConn) runPublishReader(sconn srt.Conn, path *path) error { sconn.SetReadDeadline(time.Now().Add(time.Duration(c.readTimeout))) r, err := mpegts.NewReader(mpegts.NewBufferedReader(sconn)) if err != nil { return err } decodeErrLogger := logger.NewLimitedLogger(c) r.OnDecodeError(func(err error) { decodeErrLogger.Log(logger.Warn, err.Error()) }) var medias []*description.Media //nolint:prealloc var stream *stream.Stream var td *mpegts.TimeDecoder decodeTime := func(t int64) time.Duration { if td == nil { td = mpegts.NewTimeDecoder(t) } return td.Decode(t) } for _, track := range r.Tracks() { //nolint:dupl var medi *description.Media switch tcodec := track.Codec.(type) { case *mpegts.CodecH265: medi = &description.Media{ Type: description.MediaTypeVideo, Formats: []format.Format{&format.H265{ PayloadTyp: 96, }}, } r.OnDataH26x(track, func(pts int64, _ int64, au [][]byte) error { stream.WriteUnit(medi, medi.Formats[0], &unit.H265{ Base: unit.Base{ NTP: time.Now(), PTS: decodeTime(pts), }, AU: au, }) return nil }) case *mpegts.CodecH264: medi = &description.Media{ Type: description.MediaTypeVideo, Formats: []format.Format{&format.H264{ PayloadTyp: 96, PacketizationMode: 1, }}, } r.OnDataH26x(track, func(pts int64, _ int64, au [][]byte) error { stream.WriteUnit(medi, medi.Formats[0], &unit.H264{ Base: unit.Base{ NTP: time.Now(), PTS: decodeTime(pts), }, AU: au, }) return nil }) case *mpegts.CodecMPEG4Video: medi = &description.Media{ Type: description.MediaTypeVideo, Formats: []format.Format{&format.MPEG4Video{ PayloadTyp: 96, }}, } r.OnDataMPEGxVideo(track, func(pts int64, frame []byte) error { stream.WriteUnit(medi, medi.Formats[0], &unit.MPEG4Video{ Base: unit.Base{ NTP: time.Now(), PTS: decodeTime(pts), }, Frame: frame, }) return nil }) case *mpegts.CodecMPEG1Video: medi = &description.Media{ Type: description.MediaTypeVideo, Formats: []format.Format{&format.MPEG1Video{}}, } r.OnDataMPEGxVideo(track, func(pts int64, frame []byte) error { stream.WriteUnit(medi, medi.Formats[0], &unit.MPEG1Video{ Base: unit.Base{ NTP: time.Now(), PTS: decodeTime(pts), }, Frame: frame, }) return nil }) case *mpegts.CodecOpus: medi = &description.Media{ Type: description.MediaTypeAudio, Formats: []format.Format{&format.Opus{ PayloadTyp: 96, IsStereo: (tcodec.ChannelCount == 2), }}, } r.OnDataOpus(track, func(pts int64, packets [][]byte) error { stream.WriteUnit(medi, medi.Formats[0], &unit.Opus{ Base: unit.Base{ NTP: time.Now(), PTS: decodeTime(pts), }, Packets: packets, }) return nil }) case *mpegts.CodecMPEG4Audio: medi = &description.Media{ Type: description.MediaTypeAudio, Formats: []format.Format{&format.MPEG4Audio{ PayloadTyp: 96, SizeLength: 13, IndexLength: 3, IndexDeltaLength: 3, Config: &tcodec.Config, }}, } r.OnDataMPEG4Audio(track, func(pts int64, aus [][]byte) error { stream.WriteUnit(medi, medi.Formats[0], &unit.MPEG4AudioGeneric{ Base: unit.Base{ NTP: time.Now(), PTS: decodeTime(pts), }, AUs: aus, }) return nil }) case *mpegts.CodecMPEG1Audio: medi = &description.Media{ Type: description.MediaTypeAudio, Formats: []format.Format{&format.MPEG1Audio{}}, } r.OnDataMPEG1Audio(track, func(pts int64, frames [][]byte) error { stream.WriteUnit(medi, medi.Formats[0], &unit.MPEG1Audio{ Base: unit.Base{ NTP: time.Now(), PTS: decodeTime(pts), }, Frames: frames, }) return nil }) default: continue } medias = append(medias, medi) } if len(medias) == 0 { return fmt.Errorf("no supported tracks found") } rres := path.startPublisher(pathStartPublisherReq{ author: c, desc: &description.Session{Medias: medias}, generateRTPPackets: true, }) if rres.err != nil { return rres.err } stream = rres.stream for { err := r.Read() if err != nil { return err } } } func (c *srtConn) runRead(req srtNewConnReq, pathName string, user string, pass string) (bool, error) { res := c.pathManager.addReader(pathAddReaderReq{ author: c, pathName: pathName, credentials: authCredentials{ ip: c.ip(), user: user, pass: pass, proto: authProtocolSRT, id: &c.uuid, }, }) if res.err != nil { if terr, ok := res.err.(*errAuthentication); ok { // TODO: re-enable. Currently this freezes the listener. // wait some seconds to stop brute force attacks // <-time.After(srtPauseAfterAuthError) return false, terr } return false, res.err } defer res.path.removeReader(pathRemoveReaderReq{author: c}) sconn, err := c.exchangeRequestWithConn(req) if err != nil { return true, err } defer sconn.Close() c.mutex.Lock() c.state = srtConnStateRead c.pathName = pathName c.sconn = sconn c.mutex.Unlock() writer := asyncwriter.New(c.writeQueueSize, c) defer res.stream.RemoveReader(writer) var w *mpegts.Writer var tracks []*mpegts.Track var medias []*description.Media bw := bufio.NewWriterSize(sconn, srtMaxPayloadSize(c.udpMaxPayloadSize)) addTrack := func(medi *description.Media, codec mpegts.Codec) *mpegts.Track { track := &mpegts.Track{ Codec: codec, } tracks = append(tracks, track) medias = append(medias, medi) return track } for _, medi := range res.stream.Desc().Medias { for _, forma := range medi.Formats { switch forma := forma.(type) { case *format.H265: //nolint:dupl track := addTrack(medi, &mpegts.CodecH265{}) var dtsExtractor *h265.DTSExtractor res.stream.AddReader(writer, medi, forma, func(u unit.Unit) error { tunit := u.(*unit.H265) if tunit.AU == nil { return nil } randomAccess := h265.IsRandomAccess(tunit.AU) if dtsExtractor == nil { if !randomAccess { return nil } dtsExtractor = h265.NewDTSExtractor() } dts, err := dtsExtractor.Extract(tunit.AU, tunit.PTS) if err != nil { return err } sconn.SetWriteDeadline(time.Now().Add(time.Duration(c.writeTimeout))) err = w.WriteH26x(track, durationGoToMPEGTS(tunit.PTS), durationGoToMPEGTS(dts), randomAccess, tunit.AU) if err != nil { return err } return bw.Flush() }) case *format.H264: //nolint:dupl track := addTrack(medi, &mpegts.CodecH264{}) var dtsExtractor *h264.DTSExtractor res.stream.AddReader(writer, medi, forma, func(u unit.Unit) error { tunit := u.(*unit.H264) if tunit.AU == nil { return nil } idrPresent := h264.IDRPresent(tunit.AU) if dtsExtractor == nil { if !idrPresent { return nil } dtsExtractor = h264.NewDTSExtractor() } dts, err := dtsExtractor.Extract(tunit.AU, tunit.PTS) if err != nil { return err } sconn.SetWriteDeadline(time.Now().Add(time.Duration(c.writeTimeout))) err = w.WriteH26x(track, durationGoToMPEGTS(tunit.PTS), durationGoToMPEGTS(dts), idrPresent, tunit.AU) if err != nil { return err } return bw.Flush() }) case *format.MPEG4Video: track := addTrack(medi, &mpegts.CodecMPEG4Video{}) firstReceived := false var lastPTS time.Duration res.stream.AddReader(writer, medi, forma, func(u unit.Unit) error { tunit := u.(*unit.MPEG4Video) if tunit.Frame == nil { return nil } if !firstReceived { firstReceived = true } else if tunit.PTS < lastPTS { return fmt.Errorf("MPEG-4 Video streams with B-frames are not supported (yet)") } lastPTS = tunit.PTS sconn.SetWriteDeadline(time.Now().Add(time.Duration(c.writeTimeout))) err = w.WriteMPEG4Video(track, durationGoToMPEGTS(tunit.PTS), tunit.Frame) if err != nil { return err } return bw.Flush() }) case *format.MPEG1Video: track := addTrack(medi, &mpegts.CodecMPEG1Video{}) firstReceived := false var lastPTS time.Duration res.stream.AddReader(writer, medi, forma, func(u unit.Unit) error { tunit := u.(*unit.MPEG1Video) if tunit.Frame == nil { return nil } if !firstReceived { firstReceived = true } else if tunit.PTS < lastPTS { return fmt.Errorf("MPEG-1 Video streams with B-frames are not supported (yet)") } lastPTS = tunit.PTS sconn.SetWriteDeadline(time.Now().Add(time.Duration(c.writeTimeout))) err = w.WriteMPEG1Video(track, durationGoToMPEGTS(tunit.PTS), tunit.Frame) if err != nil { return err } return bw.Flush() }) case *format.MPEG4AudioGeneric: track := addTrack(medi, &mpegts.CodecMPEG4Audio{ Config: *forma.Config, }) res.stream.AddReader(writer, medi, forma, func(u unit.Unit) error { tunit := u.(*unit.MPEG4AudioGeneric) if tunit.AUs == nil { return nil } sconn.SetWriteDeadline(time.Now().Add(time.Duration(c.writeTimeout))) err = w.WriteMPEG4Audio(track, durationGoToMPEGTS(tunit.PTS), tunit.AUs) if err != nil { return err } return bw.Flush() }) case *format.MPEG4AudioLATM: if forma.Config != nil && len(forma.Config.Programs) == 1 && len(forma.Config.Programs[0].Layers) == 1 { track := addTrack(medi, &mpegts.CodecMPEG4Audio{ Config: *forma.Config.Programs[0].Layers[0].AudioSpecificConfig, }) res.stream.AddReader(writer, medi, forma, func(u unit.Unit) error { tunit := u.(*unit.MPEG4AudioLATM) if tunit.AU == nil { return nil } sconn.SetWriteDeadline(time.Now().Add(time.Duration(c.writeTimeout))) err = w.WriteMPEG4Audio(track, durationGoToMPEGTS(tunit.PTS), [][]byte{tunit.AU}) if err != nil { return err } return bw.Flush() }) } case *format.Opus: track := addTrack(medi, &mpegts.CodecOpus{ ChannelCount: func() int { if forma.IsStereo { return 2 } return 1 }(), }) res.stream.AddReader(writer, medi, forma, func(u unit.Unit) error { tunit := u.(*unit.Opus) if tunit.Packets == nil { return nil } sconn.SetWriteDeadline(time.Now().Add(time.Duration(c.writeTimeout))) err = w.WriteOpus(track, durationGoToMPEGTS(tunit.PTS), tunit.Packets) if err != nil { return err } return bw.Flush() }) case *format.MPEG1Audio: track := addTrack(medi, &mpegts.CodecMPEG1Audio{}) res.stream.AddReader(writer, medi, forma, func(u unit.Unit) error { tunit := u.(*unit.MPEG1Audio) if tunit.Frames == nil { return nil } sconn.SetWriteDeadline(time.Now().Add(time.Duration(c.writeTimeout))) err = w.WriteMPEG1Audio(track, durationGoToMPEGTS(tunit.PTS), tunit.Frames) if err != nil { return err } return bw.Flush() }) } } } if len(tracks) == 0 { return true, fmt.Errorf( "the stream doesn't contain any supported codec, which are currently H265, H264, Opus, MPEG-4 Audio") } c.Log(logger.Info, "is reading from path '%s', %s", res.path.name, sourceMediaInfo(medias)) pathConf := res.path.safeConf() if pathConf.RunOnRead != "" { env := res.path.externalCmdEnv() desc := c.apiReaderDescribe() env["MTX_READER_TYPE"] = desc.Type env["MTX_READER_ID"] = desc.ID c.Log(logger.Info, "runOnRead command started") onReadCmd := externalcmd.NewCmd( c.externalCmdPool, pathConf.RunOnRead, pathConf.RunOnReadRestart, env, func(err error) { c.Log(logger.Info, "runOnRead command exited: %v", err) }) defer func() { onReadCmd.Close() c.Log(logger.Info, "runOnRead command stopped") }() } if pathConf.RunOnUnread != "" { defer func() { env := res.path.externalCmdEnv() desc := c.apiReaderDescribe() env["MTX_READER_TYPE"] = desc.Type env["MTX_READER_ID"] = desc.ID c.Log(logger.Info, "runOnUnread command launched") externalcmd.NewCmd( c.externalCmdPool, pathConf.RunOnUnread, false, env, nil) }() } w = mpegts.NewWriter(bw, tracks) // disable read deadline sconn.SetReadDeadline(time.Time{}) writer.Start() select { case <-c.ctx.Done(): writer.Stop() return true, fmt.Errorf("terminated") case err := <-writer.Error(): return true, err } } func (c *srtConn) exchangeRequestWithConn(req srtNewConnReq) (srt.Conn, error) { req.res <- c select { case sconn := <-c.chSetConn: return sconn, nil case <-c.ctx.Done(): return nil, errors.New("terminated") } } // new is called by srtListener through srtServer. func (c *srtConn) new(req srtNewConnReq) *srtConn { select { case c.chNew <- req: return <-req.res case <-c.ctx.Done(): return nil } } // setConn is called by srtListener . func (c *srtConn) setConn(sconn srt.Conn) { select { case c.chSetConn <- sconn: case <-c.ctx.Done(): } } // apiReaderDescribe implements reader. func (c *srtConn) apiReaderDescribe() apiPathSourceOrReader { return apiPathSourceOrReader{ Type: "srtConn", ID: c.uuid.String(), } } // apiSourceDescribe implements source. func (c *srtConn) apiSourceDescribe() apiPathSourceOrReader { return c.apiReaderDescribe() } func (c *srtConn) apiItem() *apiSRTConn { c.mutex.RLock() defer c.mutex.RUnlock() bytesReceived := uint64(0) bytesSent := uint64(0) if c.conn != nil { var s srt.Statistics c.sconn.Stats(&s) bytesReceived = s.Accumulated.ByteRecv bytesSent = s.Accumulated.ByteSent } return &apiSRTConn{ ID: c.uuid, Created: c.created, RemoteAddr: c.connReq.RemoteAddr().String(), State: func() apiSRTConnState { switch c.state { case srtConnStateRead: return apiSRTConnStateRead case srtConnStatePublish: return apiSRTConnStatePublish default: return apiSRTConnStateIdle } }(), Path: c.pathName, BytesReceived: bytesReceived, BytesSent: bytesSent, } }