mirror of
https://github.com/bluenviron/mediamtx
synced 2025-01-23 23:42:59 +00:00
620 lines
15 KiB
Go
620 lines
15 KiB
Go
package core
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net"
|
|
"net/http"
|
|
"net/http/httputil"
|
|
"reflect"
|
|
"sync"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
"github.com/aler9/rtsp-simple-server/internal/conf"
|
|
"github.com/aler9/rtsp-simple-server/internal/logger"
|
|
)
|
|
|
|
func interfaceIsEmpty(i interface{}) bool {
|
|
return reflect.ValueOf(i).Kind() != reflect.Ptr || reflect.ValueOf(i).IsNil()
|
|
}
|
|
|
|
func fillStruct(dest interface{}, source interface{}) {
|
|
rvsource := reflect.ValueOf(source)
|
|
rvdest := reflect.ValueOf(dest)
|
|
nf := rvsource.NumField()
|
|
for i := 0; i < nf; i++ {
|
|
fnew := rvsource.Field(i)
|
|
if !fnew.IsNil() {
|
|
f := rvdest.Elem().FieldByName(rvsource.Type().Field(i).Name)
|
|
if f.Kind() == reflect.Ptr {
|
|
f.Set(fnew)
|
|
} else {
|
|
f.Set(fnew.Elem())
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func cloneStruct(dest interface{}, source interface{}) {
|
|
enc, _ := json.Marshal(source)
|
|
_ = json.Unmarshal(enc, dest)
|
|
}
|
|
|
|
func loadConfData(ctx *gin.Context) (interface{}, error) {
|
|
var in struct {
|
|
// general
|
|
LogLevel *conf.LogLevel `json:"logLevel"`
|
|
LogDestinations *conf.LogDestinations `json:"logDestinations"`
|
|
LogFile *string `json:"logFile"`
|
|
ReadTimeout *conf.StringDuration `json:"readTimeout"`
|
|
WriteTimeout *conf.StringDuration `json:"writeTimeout"`
|
|
ReadBufferCount *int `json:"readBufferCount"`
|
|
API *bool `json:"api"`
|
|
APIAddress *string `json:"apiAddress"`
|
|
Metrics *bool `json:"metrics"`
|
|
MetricsAddress *string `json:"metricsAddress"`
|
|
PPROF *bool `json:"pprof"`
|
|
PPROFAddress *string `json:"pprofAddress"`
|
|
RunOnConnect *string `json:"runOnConnect"`
|
|
RunOnConnectRestart *bool `json:"runOnConnectRestart"`
|
|
|
|
// RTSP
|
|
RTSPDisable *bool `json:"rtspDisable"`
|
|
Protocols *conf.Protocols `json:"protocols"`
|
|
Encryption *conf.Encryption `json:"encryption"`
|
|
RTSPAddress *string `json:"rtspAddress"`
|
|
RTSPSAddress *string `json:"rtspsAddress"`
|
|
RTPAddress *string `json:"rtpAddress"`
|
|
RTCPAddress *string `json:"rtcpAddress"`
|
|
MulticastIPRange *string `json:"multicastIPRange"`
|
|
MulticastRTPPort *int `json:"multicastRTPPort"`
|
|
MulticastRTCPPort *int `json:"multicastRTCPPort"`
|
|
ServerKey *string `json:"serverKey"`
|
|
ServerCert *string `json:"serverCert"`
|
|
AuthMethods *conf.AuthMethods `json:"authMethods"`
|
|
ReadBufferSize *int `json:"readBufferSize"`
|
|
|
|
// RTMP
|
|
RTMPDisable *bool `json:"rtmpDisable"`
|
|
RTMPAddress *string `json:"rtmpAddress"`
|
|
|
|
// HLS
|
|
HLSDisable *bool `json:"hlsDisable"`
|
|
HLSAddress *string `json:"hlsAddress"`
|
|
HLSAlwaysRemux *bool `json:"hlsAlwaysRemux"`
|
|
HLSSegmentCount *int `json:"hlsSegmentCount"`
|
|
HLSSegmentDuration *conf.StringDuration `json:"hlsSegmentDuration"`
|
|
HLSAllowOrigin *string `json:"hlsAllowOrigin"`
|
|
}
|
|
err := json.NewDecoder(ctx.Request.Body).Decode(&in)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return in, err
|
|
}
|
|
|
|
func loadConfPathData(ctx *gin.Context) (interface{}, error) {
|
|
var in struct {
|
|
// source
|
|
Source *string `json:"source"`
|
|
SourceProtocol *conf.SourceProtocol `json:"sourceProtocol"`
|
|
SourceAnyPortEnable *bool `json:"sourceAnyPortEnable"`
|
|
SourceFingerprint *string `json:"sourceFingerprint"`
|
|
SourceOnDemand *bool `json:"sourceOnDemand"`
|
|
SourceOnDemandStartTimeout *conf.StringDuration `json:"sourceOnDemandStartTimeout"`
|
|
SourceOnDemandCloseAfter *conf.StringDuration `json:"sourceOnDemandCloseAfter"`
|
|
SourceRedirect *string `json:"sourceRedirect"`
|
|
DisablePublisherOverride *bool `json:"disablePublisherOverride"`
|
|
Fallback *string `json:"fallback"`
|
|
|
|
// authentication
|
|
PublishUser *string `json:"publishUser"`
|
|
PublishPass *string `json:"publishPass"`
|
|
PublishIPs *conf.IPsOrNets `json:"publishIPs"`
|
|
ReadUser *string `json:"readUser"`
|
|
ReadPass *string `json:"readPass"`
|
|
ReadIPs *conf.IPsOrNets `json:"readIPs"`
|
|
|
|
// custom commands
|
|
RunOnInit *string `json:"runOnInit"`
|
|
RunOnInitRestart *bool `json:"runOnInitRestart"`
|
|
RunOnDemand *string `json:"runOnDemand"`
|
|
RunOnDemandRestart *bool `json:"runOnDemandRestart"`
|
|
RunOnDemandStartTimeout *conf.StringDuration `json:"runOnDemandStartTimeout"`
|
|
RunOnDemandCloseAfter *conf.StringDuration `json:"runOnDemandCloseAfter"`
|
|
RunOnPublish *string `json:"runOnPublish"`
|
|
RunOnPublishRestart *bool `json:"runOnPublishRestart"`
|
|
RunOnRead *string `json:"runOnRead"`
|
|
RunOnReadRestart *bool `json:"runOnReadRestart"`
|
|
}
|
|
err := json.NewDecoder(ctx.Request.Body).Decode(&in)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return in, err
|
|
}
|
|
|
|
type apiPathsItem struct {
|
|
ConfName string `json:"confName"`
|
|
Conf *conf.PathConf `json:"conf"`
|
|
Source interface{} `json:"source"`
|
|
SourceReady bool `json:"sourceReady"`
|
|
Readers []interface{} `json:"readers"`
|
|
}
|
|
|
|
type apiPathsListData struct {
|
|
Items map[string]apiPathsItem `json:"items"`
|
|
}
|
|
|
|
type apiPathsListRes1 struct {
|
|
Data *apiPathsListData
|
|
Paths map[string]*path
|
|
Err error
|
|
}
|
|
|
|
type apiPathsListReq1 struct {
|
|
Res chan apiPathsListRes1
|
|
}
|
|
|
|
type apiPathsListReq2 struct {
|
|
Data *apiPathsListData
|
|
Res chan struct{}
|
|
}
|
|
|
|
type apiRTSPSessionsListItem struct {
|
|
RemoteAddr string `json:"remoteAddr"`
|
|
State string `json:"state"`
|
|
}
|
|
|
|
type apiRTSPSessionsListData struct {
|
|
Items map[string]apiRTSPSessionsListItem `json:"items"`
|
|
}
|
|
|
|
type apiRTSPSessionsListRes struct {
|
|
Data *apiRTSPSessionsListData
|
|
Err error
|
|
}
|
|
|
|
type apiRTSPSessionsListReq struct{}
|
|
|
|
type apiRTSPSessionsKickRes struct {
|
|
Err error
|
|
}
|
|
|
|
type apiRTSPSessionsKickReq struct {
|
|
ID string
|
|
}
|
|
|
|
type apiRTMPConnsListItem struct {
|
|
RemoteAddr string `json:"remoteAddr"`
|
|
State string `json:"state"`
|
|
}
|
|
|
|
type apiRTMPConnsListData struct {
|
|
Items map[string]apiRTMPConnsListItem `json:"items"`
|
|
}
|
|
|
|
type apiRTMPConnsListRes struct {
|
|
Data *apiRTMPConnsListData
|
|
Err error
|
|
}
|
|
|
|
type apiRTMPConnsListReq struct {
|
|
Res chan apiRTMPConnsListRes
|
|
}
|
|
|
|
type apiRTMPConnsKickRes struct {
|
|
Err error
|
|
}
|
|
|
|
type apiRTMPConnsKickReq struct {
|
|
ID string
|
|
Res chan apiRTMPConnsKickRes
|
|
}
|
|
|
|
type apiPathManager interface {
|
|
OnAPIPathsList(req apiPathsListReq1) apiPathsListRes1
|
|
}
|
|
|
|
type apiRTSPServer interface {
|
|
OnAPIRTSPSessionsList(req apiRTSPSessionsListReq) apiRTSPSessionsListRes
|
|
OnAPIRTSPSessionsKick(req apiRTSPSessionsKickReq) apiRTSPSessionsKickRes
|
|
}
|
|
|
|
type apiRTMPServer interface {
|
|
OnAPIRTMPConnsList(req apiRTMPConnsListReq) apiRTMPConnsListRes
|
|
OnAPIRTMPConnsKick(req apiRTMPConnsKickReq) apiRTMPConnsKickRes
|
|
}
|
|
|
|
type apiParent interface {
|
|
Log(logger.Level, string, ...interface{})
|
|
OnAPIConfigSet(conf *conf.Conf)
|
|
}
|
|
|
|
type api struct {
|
|
conf *conf.Conf
|
|
pathManager apiPathManager
|
|
rtspServer apiRTSPServer
|
|
rtspsServer apiRTSPServer
|
|
rtmpServer apiRTMPServer
|
|
parent apiParent
|
|
|
|
mutex sync.Mutex
|
|
s *http.Server
|
|
}
|
|
|
|
func newAPI(
|
|
address string,
|
|
conf *conf.Conf,
|
|
pathManager apiPathManager,
|
|
rtspServer apiRTSPServer,
|
|
rtspsServer apiRTSPServer,
|
|
rtmpServer apiRTMPServer,
|
|
parent apiParent,
|
|
) (*api, error) {
|
|
ln, err := net.Listen("tcp", address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
a := &api{
|
|
conf: conf,
|
|
pathManager: pathManager,
|
|
rtspServer: rtspServer,
|
|
rtspsServer: rtspsServer,
|
|
rtmpServer: rtmpServer,
|
|
parent: parent,
|
|
}
|
|
|
|
gin.SetMode(gin.ReleaseMode)
|
|
router := gin.New()
|
|
router.NoRoute(a.mwLog)
|
|
group := router.Group("/", a.mwLog)
|
|
group.GET("/v1/config/get", a.onConfigGet)
|
|
group.POST("/v1/config/set", a.onConfigSet)
|
|
group.POST("/v1/config/paths/add/*name", a.onConfigPathsAdd)
|
|
group.POST("/v1/config/paths/edit/*name", a.onConfigPathsEdit)
|
|
group.POST("/v1/config/paths/remove/*name", a.onConfigPathsDelete)
|
|
group.GET("/v1/paths/list", a.onPathsList)
|
|
group.GET("/v1/rtspsessions/list", a.onRTSPSessionsList)
|
|
group.POST("/v1/rtspsessions/kick/:id", a.onRTSPSessionsKick)
|
|
group.GET("/v1/rtspssessions/list", a.onRTSPSSessionsList)
|
|
group.POST("/v1/rtspssessions/kick/:id", a.onRTSPSSessionsKick)
|
|
group.GET("/v1/rtmpconns/list", a.onRTMPConnsList)
|
|
group.POST("/v1/rtmpconns/kick/:id", a.onRTMPConnsKick)
|
|
|
|
a.s = &http.Server{
|
|
Handler: router,
|
|
}
|
|
|
|
go a.s.Serve(ln)
|
|
|
|
a.log(logger.Info, "listener opened on "+address)
|
|
|
|
return a, nil
|
|
}
|
|
|
|
func (a *api) close() {
|
|
a.s.Shutdown(context.Background())
|
|
a.log(logger.Info, "closed")
|
|
}
|
|
|
|
// Log is the main logging function.
|
|
func (a *api) log(level logger.Level, format string, args ...interface{}) {
|
|
a.parent.Log(level, "[API] "+format, args...)
|
|
}
|
|
|
|
type logWriter struct {
|
|
gin.ResponseWriter
|
|
buf bytes.Buffer
|
|
}
|
|
|
|
func (w *logWriter) Write(b []byte) (int, error) {
|
|
w.buf.Write(b)
|
|
return w.ResponseWriter.Write(b)
|
|
}
|
|
|
|
func (w *logWriter) WriteString(s string) (int, error) {
|
|
w.buf.WriteString(s)
|
|
return w.ResponseWriter.WriteString(s)
|
|
}
|
|
|
|
func (a *api) mwLog(ctx *gin.Context) {
|
|
byts, _ := httputil.DumpRequest(ctx.Request, true)
|
|
a.log(logger.Debug, "[c->s] %s", string(byts))
|
|
|
|
blw := &logWriter{ResponseWriter: ctx.Writer}
|
|
ctx.Writer = blw
|
|
|
|
ctx.Next()
|
|
|
|
ctx.Writer.Header().Set("Server", "rtsp-simple-server")
|
|
|
|
var buf bytes.Buffer
|
|
fmt.Fprintf(&buf, "%s %d %s\n", ctx.Request.Proto, ctx.Writer.Status(), http.StatusText(ctx.Writer.Status()))
|
|
ctx.Writer.Header().Write(&buf)
|
|
buf.Write([]byte("\n"))
|
|
buf.Write(blw.buf.Bytes())
|
|
|
|
a.log(logger.Debug, "[s->c] %s", buf.String())
|
|
}
|
|
|
|
func (a *api) onConfigGet(ctx *gin.Context) {
|
|
a.mutex.Lock()
|
|
c := a.conf
|
|
a.mutex.Unlock()
|
|
|
|
ctx.JSON(http.StatusOK, c)
|
|
}
|
|
|
|
func (a *api) onConfigSet(ctx *gin.Context) {
|
|
in, err := loadConfData(ctx)
|
|
if err != nil {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
a.mutex.Lock()
|
|
defer a.mutex.Unlock()
|
|
|
|
var newConf conf.Conf
|
|
cloneStruct(&newConf, a.conf)
|
|
fillStruct(&newConf, in)
|
|
|
|
err = newConf.CheckAndFillMissing()
|
|
if err != nil {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
a.conf = &newConf
|
|
|
|
// since reloading the configuration can cause the shutdown of the API,
|
|
// call it in a goroutine
|
|
go a.parent.OnAPIConfigSet(&newConf)
|
|
|
|
ctx.Status(http.StatusOK)
|
|
}
|
|
|
|
func (a *api) onConfigPathsAdd(ctx *gin.Context) {
|
|
name := ctx.Param("name")
|
|
if len(name) < 2 || name[0] != '/' {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
name = name[1:]
|
|
|
|
in, err := loadConfPathData(ctx)
|
|
if err != nil {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
a.mutex.Lock()
|
|
defer a.mutex.Unlock()
|
|
|
|
var newConf conf.Conf
|
|
cloneStruct(&newConf, a.conf)
|
|
|
|
if _, ok := newConf.Paths[name]; ok {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
newConfPath := &conf.PathConf{}
|
|
fillStruct(newConfPath, in)
|
|
|
|
newConf.Paths[name] = newConfPath
|
|
|
|
err = newConf.CheckAndFillMissing()
|
|
if err != nil {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
a.conf = &newConf
|
|
|
|
// since reloading the configuration can cause the shutdown of the API,
|
|
// call it in a goroutine
|
|
go a.parent.OnAPIConfigSet(&newConf)
|
|
|
|
ctx.Status(http.StatusOK)
|
|
}
|
|
|
|
func (a *api) onConfigPathsEdit(ctx *gin.Context) {
|
|
name := ctx.Param("name")
|
|
if len(name) < 2 || name[0] != '/' {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
name = name[1:]
|
|
|
|
in, err := loadConfPathData(ctx)
|
|
if err != nil {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
a.mutex.Lock()
|
|
defer a.mutex.Unlock()
|
|
|
|
var newConf conf.Conf
|
|
cloneStruct(&newConf, a.conf)
|
|
|
|
newConfPath, ok := newConf.Paths[name]
|
|
if !ok {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
fillStruct(newConfPath, in)
|
|
|
|
err = newConf.CheckAndFillMissing()
|
|
if err != nil {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
a.conf = &newConf
|
|
|
|
// since reloading the configuration can cause the shutdown of the API,
|
|
// call it in a goroutine
|
|
go a.parent.OnAPIConfigSet(&newConf)
|
|
|
|
ctx.Status(http.StatusOK)
|
|
}
|
|
|
|
func (a *api) onConfigPathsDelete(ctx *gin.Context) {
|
|
name := ctx.Param("name")
|
|
if len(name) < 2 || name[0] != '/' {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
name = name[1:]
|
|
|
|
a.mutex.Lock()
|
|
defer a.mutex.Unlock()
|
|
|
|
var newConf conf.Conf
|
|
cloneStruct(&newConf, a.conf)
|
|
|
|
if _, ok := newConf.Paths[name]; !ok {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
delete(newConf.Paths, name)
|
|
|
|
err := newConf.CheckAndFillMissing()
|
|
if err != nil {
|
|
ctx.AbortWithStatus(http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
a.conf = &newConf
|
|
|
|
// since reloading the configuration can cause the shutdown of the API,
|
|
// call it in a goroutine
|
|
go a.parent.OnAPIConfigSet(&newConf)
|
|
|
|
ctx.Status(http.StatusOK)
|
|
}
|
|
|
|
func (a *api) onPathsList(ctx *gin.Context) {
|
|
res := a.pathManager.OnAPIPathsList(apiPathsListReq1{})
|
|
if res.Err != nil {
|
|
ctx.AbortWithStatus(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
ctx.JSON(http.StatusOK, res.Data)
|
|
}
|
|
|
|
func (a *api) onRTSPSessionsList(ctx *gin.Context) {
|
|
if interfaceIsEmpty(a.rtspServer) {
|
|
ctx.AbortWithStatus(http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
res := a.rtspServer.OnAPIRTSPSessionsList(apiRTSPSessionsListReq{})
|
|
if res.Err != nil {
|
|
ctx.AbortWithStatus(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
ctx.JSON(http.StatusOK, res.Data)
|
|
}
|
|
|
|
func (a *api) onRTSPSessionsKick(ctx *gin.Context) {
|
|
if interfaceIsEmpty(a.rtspServer) {
|
|
ctx.AbortWithStatus(http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
id := ctx.Param("id")
|
|
|
|
res := a.rtspServer.OnAPIRTSPSessionsKick(apiRTSPSessionsKickReq{ID: id})
|
|
if res.Err != nil {
|
|
ctx.AbortWithStatus(http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
ctx.Status(http.StatusOK)
|
|
}
|
|
|
|
func (a *api) onRTSPSSessionsList(ctx *gin.Context) {
|
|
if interfaceIsEmpty(a.rtspsServer) {
|
|
ctx.AbortWithStatus(http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
res := a.rtspsServer.OnAPIRTSPSessionsList(apiRTSPSessionsListReq{})
|
|
if res.Err != nil {
|
|
ctx.AbortWithStatus(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
ctx.JSON(http.StatusOK, res.Data)
|
|
}
|
|
|
|
func (a *api) onRTSPSSessionsKick(ctx *gin.Context) {
|
|
if interfaceIsEmpty(a.rtspsServer) {
|
|
ctx.AbortWithStatus(http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
id := ctx.Param("id")
|
|
|
|
res := a.rtspsServer.OnAPIRTSPSessionsKick(apiRTSPSessionsKickReq{ID: id})
|
|
if res.Err != nil {
|
|
ctx.AbortWithStatus(http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
ctx.Status(http.StatusOK)
|
|
}
|
|
|
|
func (a *api) onRTMPConnsList(ctx *gin.Context) {
|
|
if interfaceIsEmpty(a.rtmpServer) {
|
|
ctx.AbortWithStatus(http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
res := a.rtmpServer.OnAPIRTMPConnsList(apiRTMPConnsListReq{})
|
|
if res.Err != nil {
|
|
ctx.AbortWithStatus(http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
ctx.JSON(http.StatusOK, res.Data)
|
|
}
|
|
|
|
// OnConfReload is called by core.
|
|
func (a *api) OnConfReload(conf *conf.Conf) {
|
|
a.mutex.Lock()
|
|
defer a.mutex.Unlock()
|
|
a.conf = conf
|
|
}
|
|
|
|
func (a *api) onRTMPConnsKick(ctx *gin.Context) {
|
|
if interfaceIsEmpty(a.rtmpServer) {
|
|
ctx.AbortWithStatus(http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
id := ctx.Param("id")
|
|
|
|
res := a.rtmpServer.OnAPIRTMPConnsKick(apiRTMPConnsKickReq{ID: id})
|
|
if res.Err != nil {
|
|
ctx.AbortWithStatus(http.StatusNotFound)
|
|
return
|
|
}
|
|
|
|
ctx.Status(http.StatusOK)
|
|
}
|