mediamtx/main.go

191 lines
4.3 KiB
Go
Raw Normal View History

2019-12-28 21:07:03 +00:00
package main
import (
"fmt"
"log"
"os"
2020-01-03 21:39:55 +00:00
"regexp"
"strings"
"time"
2019-12-28 21:07:03 +00:00
"gopkg.in/alecthomas/kingpin.v2"
)
var Version string = "v0.0.0"
const (
_READ_TIMEOUT = 5 * time.Second
_WRITE_TIMEOUT = 5 * time.Second
)
2019-12-31 12:48:17 +00:00
type trackFlow int
const (
_TRACK_FLOW_RTP trackFlow = iota
_TRACK_FLOW_RTCP
)
type track struct {
rtpPort int
rtcpPort int
}
type streamProtocol int
const (
2020-01-20 16:20:53 +00:00
_STREAM_PROTOCOL_UDP streamProtocol = iota
2019-12-31 12:48:17 +00:00
_STREAM_PROTOCOL_TCP
)
func (s streamProtocol) String() string {
if s == _STREAM_PROTOCOL_UDP {
return "udp"
}
return "tcp"
}
2020-05-10 13:33:42 +00:00
type args struct {
version bool
protocolsStr string
rtspPort int
rtpPort int
rtcpPort int
publishUser string
publishPass string
preScript string
postScript string
2019-12-28 21:07:03 +00:00
}
2020-05-10 13:33:42 +00:00
type program struct {
args args
protocols map[streamProtocol]struct{}
rtspl *serverTcpListener
rtpl *serverUdpListener
rtcpl *serverUdpListener
2020-05-10 13:33:42 +00:00
}
2020-05-10 13:33:42 +00:00
func newProgram(args args) (*program, error) {
if args.version == true {
fmt.Println("rtsp-simple-server " + Version)
os.Exit(0)
}
2020-01-26 11:58:56 +00:00
2020-05-10 13:33:42 +00:00
if args.rtspPort == 0 {
2020-01-26 11:58:56 +00:00
return nil, fmt.Errorf("rtsp port not provided")
}
2020-05-10 13:33:42 +00:00
if args.rtpPort == 0 {
2020-01-26 11:58:56 +00:00
return nil, fmt.Errorf("rtp port not provided")
}
2020-05-10 13:33:42 +00:00
if args.rtcpPort == 0 {
2020-01-26 11:58:56 +00:00
return nil, fmt.Errorf("rtcp port not provided")
}
2020-05-10 13:33:42 +00:00
if (args.rtpPort % 2) != 0 {
2020-01-26 11:58:56 +00:00
return nil, fmt.Errorf("rtp port must be even")
}
2020-05-10 13:33:42 +00:00
if args.rtcpPort != (args.rtpPort + 1) {
return nil, fmt.Errorf("rtcp and rtp ports must be consecutive")
2020-01-26 11:58:56 +00:00
}
protocols := make(map[streamProtocol]struct{})
2020-05-10 13:33:42 +00:00
for _, proto := range strings.Split(args.protocolsStr, ",") {
switch proto {
case "udp":
protocols[_STREAM_PROTOCOL_UDP] = struct{}{}
case "tcp":
protocols[_STREAM_PROTOCOL_TCP] = struct{}{}
default:
return nil, fmt.Errorf("unsupported protocol: %s", proto)
}
}
if len(protocols) == 0 {
2020-01-26 11:58:56 +00:00
return nil, fmt.Errorf("no protocols provided")
}
2020-05-10 13:33:42 +00:00
if args.publishUser != "" {
if !regexp.MustCompile("^[a-zA-Z0-9]+$").MatchString(args.publishUser) {
return nil, fmt.Errorf("publish username must be alphanumeric")
}
}
2020-05-10 13:33:42 +00:00
if args.publishPass != "" {
if !regexp.MustCompile("^[a-zA-Z0-9]+$").MatchString(args.publishPass) {
return nil, fmt.Errorf("publish password must be alphanumeric")
2020-01-03 21:39:55 +00:00
}
}
2020-05-10 13:33:42 +00:00
if args.publishUser != "" && args.publishPass == "" || args.publishUser == "" && args.publishPass != "" {
return nil, fmt.Errorf("publish username and password must be both filled")
}
2020-01-03 21:39:55 +00:00
log.Printf("rtsp-simple-server %s", Version)
2019-12-28 21:07:03 +00:00
p := &program{
args: args,
protocols: protocols,
2019-12-28 21:07:03 +00:00
}
var err error
2020-05-10 13:33:42 +00:00
p.rtpl, err = newServerUdpListener(p, args.rtpPort, _TRACK_FLOW_RTP)
2019-12-28 21:07:03 +00:00
if err != nil {
return nil, err
}
2020-05-10 13:33:42 +00:00
p.rtcpl, err = newServerUdpListener(p, args.rtcpPort, _TRACK_FLOW_RTCP)
2019-12-28 21:07:03 +00:00
if err != nil {
return nil, err
}
2020-01-20 14:22:01 +00:00
p.rtspl, err = newServerTcpListener(p)
2019-12-28 21:07:03 +00:00
if err != nil {
return nil, err
}
2019-12-29 00:52:52 +00:00
go p.rtpl.run()
go p.rtcpl.run()
go p.rtspl.run()
2019-12-28 21:07:03 +00:00
2020-05-10 13:33:42 +00:00
return p, nil
2019-12-28 21:07:03 +00:00
}
func main() {
2020-05-10 13:33:42 +00:00
kingpin.CommandLine.Help = "rtsp-simple-server " + Version + "\n\n" +
"RTSP server."
argVersion := kingpin.Flag("version", "print rtsp-simple-server version").Bool()
argProtocolsStr := kingpin.Flag("protocols", "supported protocols").Default("udp,tcp").String()
argRtspPort := kingpin.Flag("rtsp-port", "port of the RTSP TCP listener").Default("8554").Int()
argRtpPort := kingpin.Flag("rtp-port", "port of the RTP UDP listener").Default("8000").Int()
argRtcpPort := kingpin.Flag("rtcp-port", "port of the RTCP UDP listener").Default("8001").Int()
argPublishUser := kingpin.Flag("publish-user", "optional username required to publish").Default("").String()
argPublishPass := kingpin.Flag("publish-pass", "optional password required to publish").Default("").String()
argPreScript := kingpin.Flag("pre-script", "optional script to run on client connect").Default("").String()
argPostScript := kingpin.Flag("post-script", "optional script to run on client disconnect").Default("").String()
kingpin.Parse()
_, err := newProgram(args{
version: *argVersion,
protocolsStr: *argProtocolsStr,
rtspPort: *argRtspPort,
rtpPort: *argRtpPort,
rtcpPort: *argRtcpPort,
publishUser: *argPublishUser,
publishPass: *argPublishPass,
preScript: *argPreScript,
postScript: *argPostScript,
})
2019-12-28 21:07:03 +00:00
if err != nil {
2019-12-29 11:32:54 +00:00
log.Fatal("ERR: ", err)
2019-12-28 21:07:03 +00:00
}
2020-05-10 13:33:42 +00:00
infty := make(chan struct{})
<-infty
2019-12-28 21:07:03 +00:00
}