mirror of
https://github.com/bluenviron/mediamtx
synced 2025-01-09 16:29:29 +00:00
64d9060560
new variables: MTX_CONN_TYPE, MTX_CONN_ID, MTX_SOURCE_TYPE, MTX_SOURCE_ID, MTX_READER_TYPE, MTX_READ_ID
473 lines
11 KiB
Go
473 lines
11 KiB
Go
package core
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"fmt"
|
|
"net"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/bluenviron/gortsplib/v4"
|
|
"github.com/bluenviron/gortsplib/v4/pkg/auth"
|
|
"github.com/bluenviron/gortsplib/v4/pkg/base"
|
|
"github.com/bluenviron/gortsplib/v4/pkg/url"
|
|
"github.com/google/uuid"
|
|
"github.com/pion/rtp"
|
|
|
|
"github.com/bluenviron/mediamtx/internal/conf"
|
|
"github.com/bluenviron/mediamtx/internal/externalcmd"
|
|
"github.com/bluenviron/mediamtx/internal/logger"
|
|
"github.com/bluenviron/mediamtx/internal/stream"
|
|
)
|
|
|
|
type rtspSessionPathManager interface {
|
|
addPublisher(req pathAddPublisherReq) pathAddPublisherRes
|
|
addReader(req pathAddReaderReq) pathAddReaderRes
|
|
}
|
|
|
|
type rtspSessionParent interface {
|
|
logger.Writer
|
|
getISTLS() bool
|
|
getServer() *gortsplib.Server
|
|
}
|
|
|
|
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.Stream
|
|
onReadCmd *externalcmd.Cmd // read
|
|
mutex sync.Mutex
|
|
state gortsplib.ServerSessionState
|
|
transport *gortsplib.Transport
|
|
pathName string
|
|
decodeErrLogger logger.Writer
|
|
writeErrLogger logger.Writer
|
|
}
|
|
|
|
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.decodeErrLogger = logger.NewLimitedLogger(s)
|
|
s.writeErrLogger = logger.NewLimitedLogger(s)
|
|
|
|
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) 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...)...)
|
|
}
|
|
|
|
func (s *rtspSession) onUnread() {
|
|
if s.onReadCmd != nil {
|
|
s.Log(logger.Info, "runOnRead command stopped")
|
|
s.onReadCmd.Close()
|
|
}
|
|
|
|
if s.path.conf.RunOnUnread != "" {
|
|
env := s.path.externalCmdEnv()
|
|
desc := s.apiReaderDescribe()
|
|
env["MTX_READER_TYPE"] = desc.Type
|
|
env["MTX_READER_ID"] = desc.ID
|
|
|
|
s.Log(logger.Info, "runOnUnread command launched")
|
|
externalcmd.NewCmd(
|
|
s.externalCmdPool,
|
|
s.path.conf.RunOnUnread,
|
|
false,
|
|
env,
|
|
nil)
|
|
}
|
|
}
|
|
|
|
// onClose is called by rtspServer.
|
|
func (s *rtspSession) onClose(err error) {
|
|
if s.session.State() == gortsplib.ServerSessionStatePlay {
|
|
s.onUnread()
|
|
}
|
|
|
|
switch s.session.State() {
|
|
case gortsplib.ServerSessionStatePrePlay, gortsplib.ServerSessionStatePlay:
|
|
s.path.removeReader(pathRemoveReaderReq{author: s})
|
|
|
|
case gortsplib.ServerSessionStatePreRecord, gortsplib.ServerSessionStateRecord:
|
|
s.path.removePublisher(pathRemovePublisherReq{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:]
|
|
|
|
if c.authNonce == "" {
|
|
var err error
|
|
c.authNonce, err = auth.GenerateNonce()
|
|
if err != nil {
|
|
return &base.Response{
|
|
StatusCode: base.StatusInternalServerError,
|
|
}, err
|
|
}
|
|
}
|
|
|
|
res := s.pathManager.addPublisher(pathAddPublisherReq{
|
|
author: s,
|
|
pathName: ctx.Path,
|
|
credentials: authCredentials{
|
|
query: ctx.Query,
|
|
ip: c.ip(),
|
|
proto: authProtocolRTSP,
|
|
id: &c.uuid,
|
|
rtspRequest: ctx.Request,
|
|
rtspBaseURL: nil,
|
|
rtspNonce: c.authNonce,
|
|
},
|
|
})
|
|
|
|
if res.err != nil {
|
|
switch terr := res.err.(type) {
|
|
case *errAuthentication:
|
|
return c.handleAuthError(terr)
|
|
|
|
default:
|
|
return &base.Response{
|
|
StatusCode: base.StatusBadRequest,
|
|
}, res.err
|
|
}
|
|
}
|
|
|
|
s.path = res.path
|
|
|
|
s.mutex.Lock()
|
|
s.state = gortsplib.ServerSessionStatePreRecord
|
|
s.pathName = ctx.Path
|
|
s.mutex.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
|
|
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 += "/"
|
|
}
|
|
|
|
if c.authNonce == "" {
|
|
var err error
|
|
c.authNonce, err = auth.GenerateNonce()
|
|
if err != nil {
|
|
return &base.Response{
|
|
StatusCode: base.StatusInternalServerError,
|
|
}, nil, err
|
|
}
|
|
}
|
|
|
|
res := s.pathManager.addReader(pathAddReaderReq{
|
|
author: s,
|
|
pathName: ctx.Path,
|
|
credentials: authCredentials{
|
|
query: ctx.Query,
|
|
ip: c.ip(),
|
|
proto: authProtocolRTSP,
|
|
id: &c.uuid,
|
|
rtspRequest: ctx.Request,
|
|
rtspBaseURL: baseURL,
|
|
rtspNonce: c.authNonce,
|
|
},
|
|
})
|
|
|
|
if res.err != nil {
|
|
switch terr := res.err.(type) {
|
|
case *errAuthentication:
|
|
res, err := c.handleAuthError(terr)
|
|
return res, nil, err
|
|
|
|
case errPathNoOnePublishing:
|
|
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.mutex.Lock()
|
|
s.state = gortsplib.ServerSessionStatePrePlay
|
|
s.pathName = ctx.Path
|
|
s.mutex.Unlock()
|
|
|
|
var stream *gortsplib.ServerStream
|
|
if !s.parent.getISTLS() {
|
|
stream = res.stream.RTSPStream(s.parent.getServer())
|
|
} else {
|
|
stream = res.stream.RTSPSStream(s.parent.getServer())
|
|
}
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
|
|
default: // record
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil, nil
|
|
}
|
|
}
|
|
|
|
// onPlay is called by rtspServer.
|
|
func (s *rtspSession) onPlay(_ *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()))
|
|
|
|
pathConf := s.path.safeConf()
|
|
|
|
if pathConf.RunOnRead != "" {
|
|
env := s.path.externalCmdEnv()
|
|
desc := s.apiReaderDescribe()
|
|
env["MTX_READER_TYPE"] = desc.Type
|
|
env["MTX_READER_ID"] = desc.ID
|
|
|
|
s.Log(logger.Info, "runOnRead command started")
|
|
s.onReadCmd = externalcmd.NewCmd(
|
|
s.externalCmdPool,
|
|
pathConf.RunOnRead,
|
|
pathConf.RunOnReadRestart,
|
|
env,
|
|
func(err error) {
|
|
s.Log(logger.Info, "runOnRead command exited: %v", err)
|
|
})
|
|
}
|
|
|
|
s.mutex.Lock()
|
|
s.state = gortsplib.ServerSessionStatePlay
|
|
s.transport = s.session.SetuppedTransport()
|
|
s.mutex.Unlock()
|
|
}
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
Header: h,
|
|
}, nil
|
|
}
|
|
|
|
// onRecord is called by rtspServer.
|
|
func (s *rtspSession) onRecord(_ *gortsplib.ServerHandlerOnRecordCtx) (*base.Response, error) {
|
|
res := s.path.startPublisher(pathStartPublisherReq{
|
|
author: s,
|
|
desc: s.session.AnnouncedDescription(),
|
|
generateRTPPackets: false,
|
|
})
|
|
if res.err != nil {
|
|
return &base.Response{
|
|
StatusCode: base.StatusBadRequest,
|
|
}, res.err
|
|
}
|
|
|
|
s.stream = res.stream
|
|
|
|
for _, medi := range s.session.AnnouncedDescription().Medias {
|
|
for _, forma := range medi.Formats {
|
|
cmedi := medi
|
|
cforma := forma
|
|
|
|
s.session.OnPacketRTP(cmedi, cforma, func(pkt *rtp.Packet) {
|
|
pts, ok := s.session.PacketPTS(cmedi, pkt)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
res.stream.WriteRTPPacket(cmedi, cforma, pkt, time.Now(), pts)
|
|
})
|
|
}
|
|
}
|
|
|
|
s.mutex.Lock()
|
|
s.state = gortsplib.ServerSessionStateRecord
|
|
s.transport = s.session.SetuppedTransport()
|
|
s.mutex.Unlock()
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
}
|
|
|
|
// onPause is called by rtspServer.
|
|
func (s *rtspSession) onPause(_ *gortsplib.ServerHandlerOnPauseCtx) (*base.Response, error) {
|
|
switch s.session.State() {
|
|
case gortsplib.ServerSessionStatePlay:
|
|
s.onUnread()
|
|
|
|
s.mutex.Lock()
|
|
s.state = gortsplib.ServerSessionStatePrePlay
|
|
s.mutex.Unlock()
|
|
|
|
case gortsplib.ServerSessionStateRecord:
|
|
s.path.stopPublisher(pathStopPublisherReq{author: s})
|
|
|
|
s.mutex.Lock()
|
|
s.state = gortsplib.ServerSessionStatePreRecord
|
|
s.mutex.Unlock()
|
|
}
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
}
|
|
|
|
// apiReaderDescribe implements reader.
|
|
func (s *rtspSession) apiReaderDescribe() apiPathSourceOrReader {
|
|
return apiPathSourceOrReader{
|
|
Type: func() string {
|
|
if s.isTLS {
|
|
return "rtspsSession"
|
|
}
|
|
return "rtspSession"
|
|
}(),
|
|
ID: s.uuid.String(),
|
|
}
|
|
}
|
|
|
|
// apiSourceDescribe implements source.
|
|
func (s *rtspSession) apiSourceDescribe() apiPathSourceOrReader {
|
|
return s.apiReaderDescribe()
|
|
}
|
|
|
|
// onPacketLost is called by rtspServer.
|
|
func (s *rtspSession) onPacketLost(ctx *gortsplib.ServerHandlerOnPacketLostCtx) {
|
|
s.decodeErrLogger.Log(logger.Warn, ctx.Error.Error())
|
|
}
|
|
|
|
// onDecodeError is called by rtspServer.
|
|
func (s *rtspSession) onDecodeError(ctx *gortsplib.ServerHandlerOnDecodeErrorCtx) {
|
|
s.decodeErrLogger.Log(logger.Warn, ctx.Error.Error())
|
|
}
|
|
|
|
// onStreamWriteError is called by rtspServer.
|
|
func (s *rtspSession) onStreamWriteError(ctx *gortsplib.ServerHandlerOnStreamWriteErrorCtx) {
|
|
s.writeErrLogger.Log(logger.Warn, ctx.Error.Error())
|
|
}
|
|
|
|
func (s *rtspSession) apiItem() *apiRTSPSession {
|
|
s.mutex.Lock()
|
|
defer s.mutex.Unlock()
|
|
|
|
return &apiRTSPSession{
|
|
ID: s.uuid,
|
|
Created: s.created,
|
|
RemoteAddr: s.remoteAddr().String(),
|
|
State: func() apiRTSPSessionState {
|
|
switch s.state {
|
|
case gortsplib.ServerSessionStatePrePlay,
|
|
gortsplib.ServerSessionStatePlay:
|
|
return apiRTSPSessionStateRead
|
|
|
|
case gortsplib.ServerSessionStatePreRecord,
|
|
gortsplib.ServerSessionStateRecord:
|
|
return apiRTSPSessionStatePublish
|
|
}
|
|
return apiRTSPSessionStateIdle
|
|
}(),
|
|
Path: s.pathName,
|
|
Transport: func() *string {
|
|
if s.transport == nil {
|
|
return nil
|
|
}
|
|
v := s.transport.String()
|
|
return &v
|
|
}(),
|
|
BytesReceived: s.session.BytesReceived(),
|
|
BytesSent: s.session.BytesSent(),
|
|
}
|
|
}
|