mirror of
https://github.com/bluenviron/mediamtx
synced 2025-01-25 16:33:39 +00:00
475 lines
12 KiB
Go
475 lines
12 KiB
Go
package core
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"errors"
|
|
"fmt"
|
|
"net"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/aler9/gortsplib/v2"
|
|
"github.com/aler9/gortsplib/v2/pkg/base"
|
|
"github.com/aler9/gortsplib/v2/pkg/format"
|
|
"github.com/aler9/gortsplib/v2/pkg/url"
|
|
"github.com/google/uuid"
|
|
"github.com/pion/rtp"
|
|
|
|
"github.com/aler9/rtsp-simple-server/internal/conf"
|
|
"github.com/aler9/rtsp-simple-server/internal/externalcmd"
|
|
"github.com/aler9/rtsp-simple-server/internal/formatprocessor"
|
|
"github.com/aler9/rtsp-simple-server/internal/logger"
|
|
)
|
|
|
|
const (
|
|
pauseAfterAuthError = 2 * time.Second
|
|
)
|
|
|
|
type rtspSessionPathManager interface {
|
|
publisherAdd(req pathPublisherAddReq) pathPublisherAnnounceRes
|
|
readerAdd(req pathReaderAddReq) pathReaderSetupPlayRes
|
|
}
|
|
|
|
type rtspSessionParent interface {
|
|
log(logger.Level, string, ...interface{})
|
|
}
|
|
|
|
type rtspSession struct {
|
|
isTLS bool
|
|
protocols map[conf.Protocol]struct{}
|
|
session *gortsplib.ServerSession
|
|
author *gortsplib.ServerConn
|
|
externalCmdPool *externalcmd.Pool
|
|
pathManager rtspSessionPathManager
|
|
parent rtspSessionParent
|
|
|
|
uuid uuid.UUID
|
|
created time.Time
|
|
path *path
|
|
stream *stream
|
|
state gortsplib.ServerSessionState
|
|
stateMutex sync.Mutex
|
|
onReadCmd *externalcmd.Cmd // read
|
|
}
|
|
|
|
func newRTSPSession(
|
|
isTLS bool,
|
|
protocols map[conf.Protocol]struct{},
|
|
session *gortsplib.ServerSession,
|
|
sc *gortsplib.ServerConn,
|
|
externalCmdPool *externalcmd.Pool,
|
|
pathManager rtspSessionPathManager,
|
|
parent rtspSessionParent,
|
|
) *rtspSession {
|
|
s := &rtspSession{
|
|
isTLS: isTLS,
|
|
protocols: protocols,
|
|
session: session,
|
|
author: sc,
|
|
externalCmdPool: externalCmdPool,
|
|
pathManager: pathManager,
|
|
parent: parent,
|
|
uuid: uuid.New(),
|
|
created: time.Now(),
|
|
}
|
|
|
|
s.log(logger.Info, "created by %v", s.author.NetConn().RemoteAddr())
|
|
|
|
return s
|
|
}
|
|
|
|
// Close closes a Session.
|
|
func (s *rtspSession) close() {
|
|
s.session.Close()
|
|
}
|
|
|
|
func (s *rtspSession) safeState() gortsplib.ServerSessionState {
|
|
s.stateMutex.Lock()
|
|
defer s.stateMutex.Unlock()
|
|
return s.state
|
|
}
|
|
|
|
func (s *rtspSession) remoteAddr() net.Addr {
|
|
return s.author.NetConn().RemoteAddr()
|
|
}
|
|
|
|
func (s *rtspSession) log(level logger.Level, format string, args ...interface{}) {
|
|
id := hex.EncodeToString(s.uuid[:4])
|
|
s.parent.log(level, "[session %s] "+format, append([]interface{}{id}, args...)...)
|
|
}
|
|
|
|
// onClose is called by rtspServer.
|
|
func (s *rtspSession) onClose(err error) {
|
|
if s.session.State() == gortsplib.ServerSessionStatePlay {
|
|
if s.onReadCmd != nil {
|
|
s.onReadCmd.Close()
|
|
s.onReadCmd = nil
|
|
s.log(logger.Info, "runOnRead command stopped")
|
|
}
|
|
}
|
|
|
|
switch s.session.State() {
|
|
case gortsplib.ServerSessionStatePrePlay, gortsplib.ServerSessionStatePlay:
|
|
s.path.readerRemove(pathReaderRemoveReq{author: s})
|
|
|
|
case gortsplib.ServerSessionStatePreRecord, gortsplib.ServerSessionStateRecord:
|
|
s.path.publisherRemove(pathPublisherRemoveReq{author: s})
|
|
}
|
|
|
|
s.path = nil
|
|
s.stream = nil
|
|
|
|
s.log(logger.Info, "destroyed (%v)", err)
|
|
}
|
|
|
|
// onAnnounce is called by rtspServer.
|
|
func (s *rtspSession) onAnnounce(c *rtspConn, ctx *gortsplib.ServerHandlerOnAnnounceCtx) (*base.Response, error) {
|
|
if len(ctx.Path) == 0 || ctx.Path[0] != '/' {
|
|
return &base.Response{
|
|
StatusCode: base.StatusBadRequest,
|
|
}, fmt.Errorf("invalid path")
|
|
}
|
|
ctx.Path = ctx.Path[1:]
|
|
|
|
res := s.pathManager.publisherAdd(pathPublisherAddReq{
|
|
author: s,
|
|
pathName: ctx.Path,
|
|
authenticate: func(
|
|
pathIPs []fmt.Stringer,
|
|
pathUser conf.Credential,
|
|
pathPass conf.Credential,
|
|
) error {
|
|
return c.authenticate(ctx.Path, ctx.Query, pathIPs, pathUser, pathPass, true, ctx.Request, nil)
|
|
},
|
|
})
|
|
|
|
if res.err != nil {
|
|
switch terr := res.err.(type) {
|
|
case pathErrAuthNotCritical:
|
|
s.log(logger.Debug, "non-critical authentication error: %s", terr.message)
|
|
return terr.response, nil
|
|
|
|
case pathErrAuthCritical:
|
|
// wait some seconds to stop brute force attacks
|
|
<-time.After(pauseAfterAuthError)
|
|
|
|
return terr.response, errors.New(terr.message)
|
|
|
|
default:
|
|
return &base.Response{
|
|
StatusCode: base.StatusBadRequest,
|
|
}, res.err
|
|
}
|
|
}
|
|
|
|
s.path = res.path
|
|
|
|
s.stateMutex.Lock()
|
|
s.state = gortsplib.ServerSessionStatePreRecord
|
|
s.stateMutex.Unlock()
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
}
|
|
|
|
// onSetup is called by rtspServer.
|
|
func (s *rtspSession) onSetup(c *rtspConn, ctx *gortsplib.ServerHandlerOnSetupCtx,
|
|
) (*base.Response, *gortsplib.ServerStream, error) {
|
|
if len(ctx.Path) == 0 || ctx.Path[0] != '/' {
|
|
return &base.Response{
|
|
StatusCode: base.StatusBadRequest,
|
|
}, nil, fmt.Errorf("invalid path")
|
|
}
|
|
ctx.Path = ctx.Path[1:]
|
|
|
|
// in case the client is setupping a stream with UDP or UDP-multicast, and these
|
|
// transport protocols are disabled, gortsplib already blocks the request.
|
|
// we have only to handle the case in which the transport protocol is TCP
|
|
// and it is disabled.
|
|
if ctx.Transport == gortsplib.TransportTCP {
|
|
if _, ok := s.protocols[conf.Protocol(gortsplib.TransportTCP)]; !ok {
|
|
return &base.Response{
|
|
StatusCode: base.StatusUnsupportedTransport,
|
|
}, nil, nil
|
|
}
|
|
}
|
|
|
|
switch s.session.State() {
|
|
case gortsplib.ServerSessionStateInitial, gortsplib.ServerSessionStatePrePlay: // play
|
|
res := s.pathManager.readerAdd(pathReaderAddReq{
|
|
author: s,
|
|
pathName: ctx.Path,
|
|
authenticate: func(
|
|
pathIPs []fmt.Stringer,
|
|
pathUser conf.Credential,
|
|
pathPass conf.Credential,
|
|
) error {
|
|
baseURL := &url.URL{
|
|
Scheme: ctx.Request.URL.Scheme,
|
|
Host: ctx.Request.URL.Host,
|
|
Path: ctx.Path,
|
|
RawQuery: ctx.Query,
|
|
}
|
|
|
|
if ctx.Query != "" {
|
|
baseURL.RawQuery += "/"
|
|
} else {
|
|
baseURL.Path += "/"
|
|
}
|
|
|
|
return c.authenticate(ctx.Path, ctx.Query, pathIPs, pathUser, pathPass, false, ctx.Request, baseURL)
|
|
},
|
|
})
|
|
|
|
if res.err != nil {
|
|
switch terr := res.err.(type) {
|
|
case pathErrAuthNotCritical:
|
|
s.log(logger.Debug, "non-critical authentication error: %s", terr.message)
|
|
return terr.response, nil, nil
|
|
|
|
case pathErrAuthCritical:
|
|
// wait some seconds to stop brute force attacks
|
|
<-time.After(pauseAfterAuthError)
|
|
|
|
return terr.response, nil, errors.New(terr.message)
|
|
|
|
case pathErrNoOnePublishing:
|
|
return &base.Response{
|
|
StatusCode: base.StatusNotFound,
|
|
}, nil, res.err
|
|
|
|
default:
|
|
return &base.Response{
|
|
StatusCode: base.StatusBadRequest,
|
|
}, nil, res.err
|
|
}
|
|
}
|
|
|
|
s.path = res.path
|
|
s.stream = res.stream
|
|
|
|
s.stateMutex.Lock()
|
|
s.state = gortsplib.ServerSessionStatePrePlay
|
|
s.stateMutex.Unlock()
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, res.stream.rtspStream, nil
|
|
|
|
default: // record
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil, nil
|
|
}
|
|
}
|
|
|
|
// onPlay is called by rtspServer.
|
|
func (s *rtspSession) onPlay(ctx *gortsplib.ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
h := make(base.Header)
|
|
|
|
if s.session.State() == gortsplib.ServerSessionStatePrePlay {
|
|
s.log(logger.Info, "is reading from path '%s', with %s, %s",
|
|
s.path.Name(),
|
|
s.session.SetuppedTransport(),
|
|
sourceMediaInfo(s.session.SetuppedMedias()))
|
|
|
|
if s.path.Conf().RunOnRead != "" {
|
|
s.log(logger.Info, "runOnRead command started")
|
|
s.onReadCmd = externalcmd.NewCmd(
|
|
s.externalCmdPool,
|
|
s.path.Conf().RunOnRead,
|
|
s.path.Conf().RunOnReadRestart,
|
|
s.path.externalCmdEnv(),
|
|
func(co int) {
|
|
s.log(logger.Info, "runOnRead command exited with code %d", co)
|
|
})
|
|
}
|
|
|
|
s.stateMutex.Lock()
|
|
s.state = gortsplib.ServerSessionStatePlay
|
|
s.stateMutex.Unlock()
|
|
}
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
Header: h,
|
|
}, nil
|
|
}
|
|
|
|
// onRecord is called by rtspServer.
|
|
func (s *rtspSession) onRecord(ctx *gortsplib.ServerHandlerOnRecordCtx) (*base.Response, error) {
|
|
res := s.path.publisherStart(pathPublisherStartReq{
|
|
author: s,
|
|
medias: s.session.AnnouncedMedias(),
|
|
generateRTPPackets: false,
|
|
})
|
|
if res.err != nil {
|
|
return &base.Response{
|
|
StatusCode: base.StatusBadRequest,
|
|
}, res.err
|
|
}
|
|
|
|
s.log(logger.Info, "is publishing to path '%s', with %s, %s",
|
|
s.path.Name(),
|
|
s.session.SetuppedTransport(),
|
|
sourceMediaInfo(s.session.AnnouncedMedias()))
|
|
|
|
s.stream = res.stream
|
|
|
|
for _, medi := range s.session.AnnouncedMedias() {
|
|
for _, forma := range medi.Formats {
|
|
cmedia := medi
|
|
cformat := forma
|
|
|
|
switch forma.(type) {
|
|
case *format.H264:
|
|
ctx.Session.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
|
err := s.stream.writeData(cmedia, cformat, &formatprocessor.DataH264{
|
|
RTPPackets: []*rtp.Packet{pkt},
|
|
NTP: time.Now(),
|
|
})
|
|
if err != nil {
|
|
s.log(logger.Warn, "%v", err)
|
|
}
|
|
})
|
|
|
|
case *format.H265:
|
|
ctx.Session.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
|
err := s.stream.writeData(cmedia, cformat, &formatprocessor.DataH265{
|
|
RTPPackets: []*rtp.Packet{pkt},
|
|
NTP: time.Now(),
|
|
})
|
|
if err != nil {
|
|
s.log(logger.Warn, "%v", err)
|
|
}
|
|
})
|
|
|
|
case *format.VP8:
|
|
ctx.Session.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
|
err := s.stream.writeData(cmedia, cformat, &formatprocessor.DataVP8{
|
|
RTPPackets: []*rtp.Packet{pkt},
|
|
NTP: time.Now(),
|
|
})
|
|
if err != nil {
|
|
s.log(logger.Warn, "%v", err)
|
|
}
|
|
})
|
|
|
|
case *format.VP9:
|
|
ctx.Session.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
|
err := s.stream.writeData(cmedia, cformat, &formatprocessor.DataVP9{
|
|
RTPPackets: []*rtp.Packet{pkt},
|
|
NTP: time.Now(),
|
|
})
|
|
if err != nil {
|
|
s.log(logger.Warn, "%v", err)
|
|
}
|
|
})
|
|
|
|
case *format.MPEG4Audio:
|
|
ctx.Session.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
|
err := s.stream.writeData(cmedia, cformat, &formatprocessor.DataMPEG4Audio{
|
|
RTPPackets: []*rtp.Packet{pkt},
|
|
NTP: time.Now(),
|
|
})
|
|
if err != nil {
|
|
s.log(logger.Warn, "%v", err)
|
|
}
|
|
})
|
|
|
|
case *format.Opus:
|
|
ctx.Session.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
|
err := s.stream.writeData(cmedia, cformat, &formatprocessor.DataOpus{
|
|
RTPPackets: []*rtp.Packet{pkt},
|
|
NTP: time.Now(),
|
|
})
|
|
if err != nil {
|
|
s.log(logger.Warn, "%v", err)
|
|
}
|
|
})
|
|
|
|
default:
|
|
ctx.Session.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
|
err := s.stream.writeData(cmedia, cformat, &formatprocessor.DataGeneric{
|
|
RTPPackets: []*rtp.Packet{pkt},
|
|
NTP: time.Now(),
|
|
})
|
|
if err != nil {
|
|
s.log(logger.Warn, "%v", err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
s.stateMutex.Lock()
|
|
s.state = gortsplib.ServerSessionStateRecord
|
|
s.stateMutex.Unlock()
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
}
|
|
|
|
// onPause is called by rtspServer.
|
|
func (s *rtspSession) onPause(ctx *gortsplib.ServerHandlerOnPauseCtx) (*base.Response, error) {
|
|
switch s.session.State() {
|
|
case gortsplib.ServerSessionStatePlay:
|
|
if s.onReadCmd != nil {
|
|
s.log(logger.Info, "runOnRead command stopped")
|
|
s.onReadCmd.Close()
|
|
}
|
|
|
|
s.stateMutex.Lock()
|
|
s.state = gortsplib.ServerSessionStatePrePlay
|
|
s.stateMutex.Unlock()
|
|
|
|
case gortsplib.ServerSessionStateRecord:
|
|
s.path.publisherStop(pathPublisherStopReq{author: s})
|
|
|
|
s.stateMutex.Lock()
|
|
s.state = gortsplib.ServerSessionStatePreRecord
|
|
s.stateMutex.Unlock()
|
|
}
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
}
|
|
|
|
// apiReaderDescribe implements reader.
|
|
func (s *rtspSession) apiReaderDescribe() interface{} {
|
|
var typ string
|
|
if s.isTLS {
|
|
typ = "rtspsSession"
|
|
} else {
|
|
typ = "rtspSession"
|
|
}
|
|
|
|
return struct {
|
|
Type string `json:"type"`
|
|
ID string `json:"id"`
|
|
}{typ, s.uuid.String()}
|
|
}
|
|
|
|
// apiSourceDescribe implements source.
|
|
func (s *rtspSession) apiSourceDescribe() interface{} {
|
|
var typ string
|
|
if s.isTLS {
|
|
typ = "rtspsSession"
|
|
} else {
|
|
typ = "rtspSession"
|
|
}
|
|
|
|
return struct {
|
|
Type string `json:"type"`
|
|
ID string `json:"id"`
|
|
}{typ, s.uuid.String()}
|
|
}
|
|
|
|
// onDecodeError is called by rtspServer.
|
|
func (s *rtspSession) onDecodeError(ctx *gortsplib.ServerHandlerOnDecodeErrorCtx) {
|
|
s.log(logger.Warn, "%v", ctx.Error)
|
|
}
|