mediamtx/internal/core/rtsp_conn.go

304 lines
7.4 KiB
Go
Raw Normal View History

package core
2019-12-31 12:48:17 +00:00
import (
2020-05-10 20:56:46 +00:00
"errors"
"fmt"
2019-12-31 12:48:17 +00:00
"net"
"time"
2019-12-31 12:48:17 +00:00
2020-01-20 09:21:05 +00:00
"github.com/aler9/gortsplib"
2020-11-15 16:56:54 +00:00
"github.com/aler9/gortsplib/pkg/auth"
"github.com/aler9/gortsplib/pkg/base"
"github.com/aler9/gortsplib/pkg/headers"
"github.com/aler9/rtsp-simple-server/internal/conf"
2020-11-01 21:56:56 +00:00
"github.com/aler9/rtsp-simple-server/internal/externalcmd"
"github.com/aler9/rtsp-simple-server/internal/logger"
2019-12-31 12:48:17 +00:00
)
const (
rtspConnPauseAfterAuthError = 2 * time.Second
)
type rtspConnParent interface {
2021-10-27 19:01:00 +00:00
log(logger.Level, string, ...interface{})
2020-10-19 20:17:48 +00:00
}
type rtspConn struct {
id string
externalAuthenticationURL string
rtspAddress string
authMethods []headers.AuthMethod
readTimeout conf.StringDuration
runOnConnect string
runOnConnectRestart bool
externalCmdPool *externalcmd.Pool
pathManager *pathManager
conn *gortsplib.ServerConn
parent rtspConnParent
2020-10-19 20:17:48 +00:00
created time.Time
2021-05-07 21:07:31 +00:00
onConnectCmd *externalcmd.Cmd
authUser string
authPass string
authValidator *auth.Validator
authFailures int
2019-12-31 12:48:17 +00:00
}
func newRTSPConn(
id string,
externalAuthenticationURL string,
rtspAddress string,
authMethods []headers.AuthMethod,
readTimeout conf.StringDuration,
2020-10-19 20:17:48 +00:00
runOnConnect string,
runOnConnectRestart bool,
externalCmdPool *externalcmd.Pool,
pathManager *pathManager,
2020-12-06 17:01:10 +00:00
conn *gortsplib.ServerConn,
2022-04-07 10:50:35 +00:00
parent rtspConnParent,
) *rtspConn {
c := &rtspConn{
id: id,
externalAuthenticationURL: externalAuthenticationURL,
rtspAddress: rtspAddress,
authMethods: authMethods,
readTimeout: readTimeout,
runOnConnect: runOnConnect,
runOnConnectRestart: runOnConnectRestart,
externalCmdPool: externalCmdPool,
pathManager: pathManager,
conn: conn,
parent: parent,
created: time.Now(),
2019-12-31 12:48:17 +00:00
}
2021-05-09 12:41:18 +00:00
c.log(logger.Info, "opened")
2021-05-07 21:07:31 +00:00
if c.runOnConnect != "" {
c.log(logger.Info, "runOnConnect command started")
2021-05-07 21:07:31 +00:00
_, port, _ := net.SplitHostPort(c.rtspAddress)
c.onConnectCmd = externalcmd.NewCmd(
c.externalCmdPool,
c.runOnConnect,
c.runOnConnectRestart,
externalcmd.Environment{
"RTSP_PATH": "",
"RTSP_PORT": port,
},
func(co int) {
c.log(logger.Info, "runOnInit command exited with code %d", co)
})
2021-05-07 21:07:31 +00:00
}
2020-10-19 20:17:48 +00:00
return c
2019-12-31 12:48:17 +00:00
}
func (c *rtspConn) log(level logger.Level, format string, args ...interface{}) {
2021-10-27 19:01:00 +00:00
c.parent.log(level, "[conn %v] "+format, append([]interface{}{c.conn.NetConn().RemoteAddr()}, args...)...)
2020-10-19 20:17:48 +00:00
}
2021-05-07 21:07:31 +00:00
// Conn returns the RTSP connection.
func (c *rtspConn) Conn() *gortsplib.ServerConn {
2021-05-07 21:07:31 +00:00
return c.conn
}
func (c *rtspConn) remoteAddr() net.Addr {
return c.conn.NetConn().RemoteAddr()
}
func (c *rtspConn) ip() net.IP {
return c.conn.NetConn().RemoteAddr().(*net.TCPAddr).IP
}
func (c *rtspConn) authenticate(
pathName string,
pathIPs []fmt.Stringer,
pathUser conf.Credential,
pathPass conf.Credential,
2022-08-22 09:24:21 +00:00
isPublishing bool,
2021-05-07 21:07:31 +00:00
req *base.Request,
query string,
2021-05-07 21:07:31 +00:00
) error {
if c.externalAuthenticationURL != "" {
username := ""
password := ""
var auth headers.Authorization
2022-06-27 15:47:07 +00:00
err := auth.Unmarshal(req.Header["Authorization"])
if err == nil && auth.Method == headers.AuthBasic {
username = auth.BasicUser
password = auth.BasicPass
}
err = externalAuth(
c.externalAuthenticationURL,
c.ip().String(),
username,
password,
pathName,
2022-08-22 09:24:21 +00:00
isPublishing,
query)
if err != nil {
c.authFailures++
// VLC with login prompt sends 4 requests:
// 1) without credentials
// 2) with password but without username
// 3) without credentials
// 4) with password and username
// therefore we must allow up to 3 failures
if c.authFailures > 3 {
return pathErrAuthCritical{
2022-01-14 22:42:41 +00:00
message: "unauthorized: " + err.Error(),
response: &base.Response{
StatusCode: base.StatusUnauthorized,
},
}
}
v := "IPCAM"
return pathErrAuthNotCritical{
2022-01-14 22:42:41 +00:00
message: "unauthorized: " + err.Error(),
response: &base.Response{
StatusCode: base.StatusUnauthorized,
Header: base.Header{
"WWW-Authenticate": headers.Authenticate{
Method: headers.AuthBasic,
Realm: &v,
2022-06-27 15:47:07 +00:00
}.Marshal(),
},
},
}
}
2021-05-07 21:07:31 +00:00
}
if pathIPs != nil {
ip := c.ip()
if !ipEqualOrInRange(ip, pathIPs) {
return pathErrAuthCritical{
2022-01-14 22:42:41 +00:00
message: fmt.Sprintf("IP '%s' not allowed", ip),
response: &base.Response{
2020-12-13 12:58:56 +00:00
StatusCode: base.StatusUnauthorized,
2021-05-07 21:07:31 +00:00
},
2020-12-13 12:58:56 +00:00
}
2021-05-07 21:07:31 +00:00
}
}
2020-12-13 12:58:56 +00:00
if pathUser != "" {
// reset authValidator every time the credentials change
if c.authValidator == nil || c.authUser != string(pathUser) || c.authPass != string(pathPass) {
c.authUser = string(pathUser)
c.authPass = string(pathPass)
c.authValidator = auth.NewValidator(string(pathUser), string(pathPass), c.authMethods)
2020-12-13 12:58:56 +00:00
}
2021-05-07 21:07:31 +00:00
err := c.authValidator.ValidateRequest(req)
if err != nil {
c.authFailures++
// VLC with login prompt sends 4 requests:
// 1) without credentials
// 2) with password but without username
// 3) without credentials
// 4) with password and username
// therefore we must allow up to 3 failures
if c.authFailures > 3 {
return pathErrAuthCritical{
2022-01-14 22:42:41 +00:00
message: "unauthorized: " + err.Error(),
response: &base.Response{
StatusCode: base.StatusUnauthorized,
},
}
}
return pathErrAuthNotCritical{
2022-01-14 22:42:41 +00:00
response: &base.Response{
StatusCode: base.StatusUnauthorized,
Header: base.Header{
"WWW-Authenticate": c.authValidator.Header(),
},
},
}
}
2020-12-13 12:58:56 +00:00
// login successful, reset authFailures
c.authFailures = 0
}
2020-12-13 12:58:56 +00:00
return nil
}
2021-10-27 19:01:00 +00:00
// onClose is called by rtspServer.
func (c *rtspConn) onClose(err error) {
c.log(logger.Info, "closed (%v)", err)
if c.onConnectCmd != nil {
c.onConnectCmd.Close()
c.log(logger.Info, "runOnConnect command stopped")
}
}
2021-10-27 19:01:00 +00:00
// onRequest is called by rtspServer.
func (c *rtspConn) onRequest(req *base.Request) {
c.log(logger.Debug, "[c->s] %v", req)
}
// OnResponse is called by rtspServer.
func (c *rtspConn) OnResponse(res *base.Response) {
c.log(logger.Debug, "[s->c] %v", res)
}
2021-10-27 19:01:00 +00:00
// onDescribe is called by rtspServer.
func (c *rtspConn) onDescribe(ctx *gortsplib.ServerHandlerOnDescribeCtx,
2021-09-09 21:05:54 +00:00
) (*base.Response, *gortsplib.ServerStream, error) {
res := c.pathManager.describe(pathDescribeReq{
2022-01-14 22:42:41 +00:00
pathName: ctx.Path,
2022-02-19 22:06:24 +00:00
url: ctx.Request.URL,
2022-01-14 22:42:41 +00:00
authenticate: func(
pathIPs []fmt.Stringer,
pathUser conf.Credential,
2022-04-07 10:50:35 +00:00
pathPass conf.Credential,
) error {
2022-08-22 09:24:21 +00:00
return c.authenticate(ctx.Path, pathIPs, pathUser, pathPass, false, ctx.Request, ctx.Query)
},
})
2022-01-14 22:42:41 +00:00
if res.err != nil {
switch terr := res.err.(type) {
case pathErrAuthNotCritical:
2022-01-14 22:42:41 +00:00
c.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(rtspConnPauseAfterAuthError)
2022-01-14 22:42:41 +00:00
return terr.response, nil, errors.New(terr.message)
case pathErrNoOnePublishing:
return &base.Response{
StatusCode: base.StatusNotFound,
2022-01-14 22:42:41 +00:00
}, nil, res.err
default:
return &base.Response{
StatusCode: base.StatusBadRequest,
2022-01-14 22:42:41 +00:00
}, nil, res.err
}
}
2022-01-14 22:42:41 +00:00
if res.redirect != "" {
return &base.Response{
StatusCode: base.StatusMovedPermanently,
Header: base.Header{
2022-01-14 22:42:41 +00:00
"Location": base.HeaderValue{res.redirect},
},
}, nil, nil
}
return &base.Response{
StatusCode: base.StatusOK,
2022-01-14 22:42:41 +00:00
}, res.stream.rtspStream, nil
}