mirror of
https://github.com/aler9/gortsplib
synced 2025-09-27 11:32:08 +08:00
1973 lines
50 KiB
Go
1973 lines
50 KiB
Go
package gortsplib
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/tls"
|
|
"net"
|
|
"strconv"
|
|
"strings"
|
|
"sync/atomic"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/pion/rtcp"
|
|
"github.com/pion/rtp"
|
|
psdp "github.com/pion/sdp/v3"
|
|
"github.com/stretchr/testify/require"
|
|
"golang.org/x/net/ipv4"
|
|
|
|
"github.com/bluenviron/gortsplib/v3/pkg/base"
|
|
"github.com/bluenviron/gortsplib/v3/pkg/conn"
|
|
"github.com/bluenviron/gortsplib/v3/pkg/formats"
|
|
"github.com/bluenviron/gortsplib/v3/pkg/headers"
|
|
"github.com/bluenviron/gortsplib/v3/pkg/media"
|
|
"github.com/bluenviron/gortsplib/v3/pkg/sdp"
|
|
"github.com/bluenviron/gortsplib/v3/pkg/url"
|
|
)
|
|
|
|
func multicastCapableIP(t *testing.T) string {
|
|
intfs, err := net.Interfaces()
|
|
require.NoError(t, err)
|
|
|
|
for _, intf := range intfs {
|
|
if (intf.Flags & net.FlagMulticast) != 0 {
|
|
addrs, err := intf.Addrs()
|
|
if err != nil {
|
|
continue
|
|
}
|
|
|
|
for _, addr := range addrs {
|
|
switch v := addr.(type) {
|
|
case *net.IPNet:
|
|
return v.IP.String()
|
|
case *net.IPAddr:
|
|
return v.IP.String()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
t.Errorf("unable to find a multicast IP")
|
|
return ""
|
|
}
|
|
|
|
func relativeControlAttribute(md *psdp.MediaDescription) string {
|
|
v, _ := md.Attribute("control")
|
|
i := strings.Index(v, "trackID=")
|
|
return v[i:]
|
|
}
|
|
|
|
func absoluteControlAttribute(md *psdp.MediaDescription) string {
|
|
v, _ := md.Attribute("control")
|
|
return v
|
|
}
|
|
|
|
func doDescribe(t *testing.T, conn *conn.Conn) *sdp.SessionDescription {
|
|
res, err := writeReqReadRes(conn, base.Request{
|
|
Method: base.Describe,
|
|
URL: mustParseURL("rtsp://localhost:8554/teststream"),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"1"},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
|
|
var desc sdp.SessionDescription
|
|
err = desc.Unmarshal(res.Body)
|
|
require.NoError(t, err)
|
|
return &desc
|
|
}
|
|
|
|
func doSetup(t *testing.T, conn *conn.Conn, u string, inTH *headers.Transport) (*base.Response, *headers.Transport) {
|
|
res, err := writeReqReadRes(conn, base.Request{
|
|
Method: base.Setup,
|
|
URL: mustParseURL(u),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"1"},
|
|
"Transport": inTH.Marshal(),
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
|
|
var th headers.Transport
|
|
err = th.Unmarshal(res.Header["Transport"])
|
|
require.NoError(t, err)
|
|
|
|
return res, &th
|
|
}
|
|
|
|
func doPlay(t *testing.T, conn *conn.Conn, u string, session string) {
|
|
res, err := writeReqReadRes(conn, base.Request{
|
|
Method: base.Play,
|
|
URL: mustParseURL(u),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"1"},
|
|
"Session": base.HeaderValue{session},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
}
|
|
|
|
func doPause(t *testing.T, conn *conn.Conn, u string, session string) {
|
|
res, err := writeReqReadRes(conn, base.Request{
|
|
Method: base.Pause,
|
|
URL: mustParseURL(u),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"1"},
|
|
"Session": base.HeaderValue{session},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
}
|
|
|
|
func readSession(t *testing.T, res *base.Response) string {
|
|
var sx headers.Session
|
|
err := sx.Unmarshal(res.Header["Session"])
|
|
require.NoError(t, err)
|
|
return sx.Session
|
|
}
|
|
|
|
func TestServerPlayPath(t *testing.T) {
|
|
for _, ca := range []struct {
|
|
name string
|
|
setupURL string
|
|
playURL string
|
|
path string
|
|
}{
|
|
{
|
|
"normal",
|
|
"rtsp://localhost:8554/teststream[control]",
|
|
"rtsp://localhost:8554/teststream/",
|
|
"/teststream",
|
|
},
|
|
{
|
|
"with query",
|
|
"rtsp://localhost:8554/teststream?testing=123[control]",
|
|
"rtsp://localhost:8554/teststream/",
|
|
"/teststream",
|
|
},
|
|
{
|
|
// this is needed to support reading mpegts with ffmpeg
|
|
"without media id",
|
|
"rtsp://localhost:8554/teststream/",
|
|
"rtsp://localhost:8554/teststream/",
|
|
"/teststream",
|
|
},
|
|
{
|
|
"subpath",
|
|
"rtsp://localhost:8554/test/stream[control]",
|
|
"rtsp://localhost:8554/test/stream/",
|
|
"/test/stream",
|
|
},
|
|
{
|
|
"subpath without media id",
|
|
"rtsp://localhost:8554/test/stream/",
|
|
"rtsp://localhost:8554/test/stream/",
|
|
"/test/stream",
|
|
},
|
|
{
|
|
"subpath with query",
|
|
"rtsp://localhost:8554/test/stream?testing=123[control]",
|
|
"rtsp://localhost:8554/test/stream",
|
|
"/test/stream",
|
|
},
|
|
{
|
|
"no slash",
|
|
"rtsp://localhost:8554[control]",
|
|
"rtsp://localhost:8554/",
|
|
"",
|
|
},
|
|
{
|
|
"single slash",
|
|
"rtsp://localhost:8554/[control]",
|
|
"rtsp://localhost:8554//",
|
|
"/",
|
|
},
|
|
} {
|
|
t.Run(ca.name, func(t *testing.T) {
|
|
stream := NewServerStream(media.Medias{
|
|
testH264Media,
|
|
testH264Media,
|
|
testH264Media,
|
|
testH264Media,
|
|
})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
require.Equal(t, ca.path, ctx.Path)
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
require.Equal(t, ca.path, ctx.Path)
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
th := &headers.Transport{
|
|
Protocol: headers.TransportProtocolTCP,
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
InterleavedIDs: &[2]int{0, 1},
|
|
}
|
|
|
|
res, _ := doSetup(t, conn,
|
|
strings.ReplaceAll(ca.setupURL, "[control]", "/"+
|
|
relativeControlAttribute(desc.MediaDescriptions[1])),
|
|
th)
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, ca.playURL, session)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestServerPlaySetupErrors(t *testing.T) {
|
|
for _, ca := range []string{
|
|
"different paths",
|
|
"double setup",
|
|
"closed stream",
|
|
} {
|
|
t.Run(ca, func(t *testing.T) {
|
|
nconnClosed := make(chan struct{})
|
|
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
if ca == "closed stream" {
|
|
stream.Close()
|
|
} else {
|
|
defer stream.Close()
|
|
}
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onConnClose: func(ctx *ServerHandlerOnConnCloseCtx) {
|
|
switch ca {
|
|
case "different paths":
|
|
require.EqualError(t, ctx.Error, "can't setup medias with different paths")
|
|
|
|
case "double setup":
|
|
require.EqualError(t, ctx.Error, "media has already been setup")
|
|
|
|
case "closed stream":
|
|
require.EqualError(t, ctx.Error, "stream is closed")
|
|
}
|
|
close(nconnClosed)
|
|
},
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
},
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
th := &headers.Transport{
|
|
Protocol: headers.TransportProtocolTCP,
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
InterleavedIDs: &[2]int{0, 1},
|
|
}
|
|
|
|
res, err := writeReqReadRes(conn, base.Request{
|
|
Method: base.Setup,
|
|
URL: mustParseURL(absoluteControlAttribute(desc.MediaDescriptions[0])),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"2"},
|
|
"Transport": th.Marshal(),
|
|
},
|
|
})
|
|
|
|
switch ca {
|
|
case "different paths":
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
|
|
session := readSession(t, res)
|
|
|
|
th.InterleavedIDs = &[2]int{2, 3}
|
|
|
|
res, err = writeReqReadRes(conn, base.Request{
|
|
Method: base.Setup,
|
|
URL: mustParseURL("rtsp://localhost:8554/test12stream/" + relativeControlAttribute(desc.MediaDescriptions[0])),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"3"},
|
|
"Transport": th.Marshal(),
|
|
"Session": base.HeaderValue{session},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusBadRequest, res.StatusCode)
|
|
|
|
case "double setup":
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
|
|
session := readSession(t, res)
|
|
|
|
th.InterleavedIDs = &[2]int{2, 3}
|
|
|
|
res, err = writeReqReadRes(conn, base.Request{
|
|
Method: base.Setup,
|
|
URL: mustParseURL(absoluteControlAttribute(desc.MediaDescriptions[0])),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"4"},
|
|
"Transport": th.Marshal(),
|
|
"Session": base.HeaderValue{session},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusBadRequest, res.StatusCode)
|
|
|
|
case "closed stream":
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusBadRequest, res.StatusCode)
|
|
}
|
|
|
|
<-nconnClosed
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestServerPlaySetupErrorSameUDPPortsAndIP(t *testing.T) {
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
first := int32(1)
|
|
errorRecv := make(chan struct{})
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onConnClose: func(ctx *ServerHandlerOnConnCloseCtx) {
|
|
if atomic.SwapInt32(&first, 0) == 1 {
|
|
require.EqualError(t, ctx.Error,
|
|
"UDP ports 35466 and 35467 are already assigned to another reader with the same IP")
|
|
close(errorRecv)
|
|
}
|
|
},
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
UDPRTPAddress: "127.0.0.1:8000",
|
|
UDPRTCPAddress: "127.0.0.1:8001",
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
for i := 0; i < 2; i++ {
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
inTH := &headers.Transport{
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
Protocol: headers.TransportProtocolUDP,
|
|
ClientPorts: &[2]int{35466, 35467},
|
|
}
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
res, err := writeReqReadRes(conn, base.Request{
|
|
Method: base.Setup,
|
|
URL: mustParseURL(absoluteControlAttribute(desc.MediaDescriptions[0])),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"2"},
|
|
"Transport": inTH.Marshal(),
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
if i == 0 {
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
} else {
|
|
require.Equal(t, base.StatusBadRequest, res.StatusCode)
|
|
}
|
|
}
|
|
|
|
<-errorRecv
|
|
}
|
|
|
|
func TestServerPlay(t *testing.T) {
|
|
for _, transport := range []string{
|
|
"udp",
|
|
"tcp",
|
|
"tls",
|
|
"multicast",
|
|
} {
|
|
t.Run(transport, func(t *testing.T) {
|
|
nconnOpened := make(chan struct{})
|
|
nconnClosed := make(chan struct{})
|
|
sessionOpened := make(chan struct{})
|
|
sessionClosed := make(chan struct{})
|
|
framesReceived := make(chan struct{})
|
|
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
counter := uint64(0)
|
|
|
|
listenIP := multicastCapableIP(t)
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onConnOpen: func(ctx *ServerHandlerOnConnOpenCtx) {
|
|
close(nconnOpened)
|
|
},
|
|
onConnClose: func(ctx *ServerHandlerOnConnCloseCtx) {
|
|
close(nconnClosed)
|
|
},
|
|
onSessionOpen: func(ctx *ServerHandlerOnSessionOpenCtx) {
|
|
close(sessionOpened)
|
|
},
|
|
onSessionClose: func(ctx *ServerHandlerOnSessionCloseCtx) {
|
|
close(sessionClosed)
|
|
},
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
// send RTCP packets directly to the session.
|
|
// these are sent after the response, only if onPlay returns StatusOK.
|
|
if transport != "multicast" {
|
|
ctx.Session.WritePacketRTCP(stream.Medias()[0], &testRTCPPacket)
|
|
}
|
|
|
|
ctx.Session.OnPacketRTCPAny(func(medi *media.Media, pkt rtcp.Packet) {
|
|
// ignore multicast loopback
|
|
if transport == "multicast" && atomic.AddUint64(&counter, 1) <= 1 {
|
|
return
|
|
}
|
|
|
|
require.Equal(t, stream.Medias()[0], medi)
|
|
require.Equal(t, &testRTCPPacket, pkt)
|
|
close(framesReceived)
|
|
})
|
|
|
|
// the session is added to the stream only after onPlay returns
|
|
// with StatusOK; therefore we must wait before calling
|
|
// ServerStream.WritePacket*()
|
|
go func() {
|
|
time.Sleep(1 * time.Second)
|
|
stream.WritePacketRTCP(stream.Medias()[0], &testRTCPPacket)
|
|
stream.WritePacketRTP(stream.Medias()[0], &testRTPPacket)
|
|
}()
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
onGetParameter: func(ctx *ServerHandlerOnGetParameterCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
RTSPAddress: listenIP + ":8554",
|
|
}
|
|
|
|
switch transport {
|
|
case "udp":
|
|
s.UDPRTPAddress = "127.0.0.1:8000"
|
|
s.UDPRTCPAddress = "127.0.0.1:8001"
|
|
|
|
case "multicast":
|
|
s.MulticastIPRange = "224.1.0.0/16"
|
|
s.MulticastRTPPort = 8000
|
|
s.MulticastRTCPPort = 8001
|
|
|
|
case "tls":
|
|
cert, err := tls.X509KeyPair(serverCert, serverKey)
|
|
require.NoError(t, err)
|
|
s.TLSConfig = &tls.Config{Certificates: []tls.Certificate{cert}}
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", listenIP+":8554")
|
|
require.NoError(t, err)
|
|
|
|
nconn = func() net.Conn {
|
|
if transport == "tls" {
|
|
return tls.Client(nconn, &tls.Config{InsecureSkipVerify: true})
|
|
}
|
|
return nconn
|
|
}()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
<-nconnOpened
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
}
|
|
|
|
switch transport {
|
|
case "udp":
|
|
v := headers.TransportDeliveryUnicast
|
|
inTH.Delivery = &v
|
|
inTH.Protocol = headers.TransportProtocolUDP
|
|
inTH.ClientPorts = &[2]int{35466, 35467}
|
|
|
|
case "multicast":
|
|
v := headers.TransportDeliveryMulticast
|
|
inTH.Delivery = &v
|
|
inTH.Protocol = headers.TransportProtocolUDP
|
|
|
|
default:
|
|
v := headers.TransportDeliveryUnicast
|
|
inTH.Delivery = &v
|
|
inTH.Protocol = headers.TransportProtocolTCP
|
|
inTH.InterleavedIDs = &[2]int{4, 5}
|
|
}
|
|
|
|
res, th := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
var l1 net.PacketConn
|
|
var l2 net.PacketConn
|
|
|
|
switch transport {
|
|
case "udp":
|
|
require.Equal(t, headers.TransportProtocolUDP, th.Protocol)
|
|
require.Equal(t, headers.TransportDeliveryUnicast, *th.Delivery)
|
|
|
|
l1, err = net.ListenPacket("udp", listenIP+":35466")
|
|
require.NoError(t, err)
|
|
defer l1.Close()
|
|
|
|
l2, err = net.ListenPacket("udp", listenIP+":35467")
|
|
require.NoError(t, err)
|
|
defer l2.Close()
|
|
|
|
case "multicast":
|
|
require.Equal(t, headers.TransportProtocolUDP, th.Protocol)
|
|
require.Equal(t, headers.TransportDeliveryMulticast, *th.Delivery)
|
|
|
|
l1, err = net.ListenPacket("udp", "224.0.0.0:"+strconv.FormatInt(int64(th.Ports[0]), 10))
|
|
require.NoError(t, err)
|
|
defer l1.Close()
|
|
|
|
p := ipv4.NewPacketConn(l1)
|
|
|
|
intfs, err := net.Interfaces()
|
|
require.NoError(t, err)
|
|
|
|
for _, intf := range intfs {
|
|
err := p.JoinGroup(&intf, &net.UDPAddr{IP: *th.Destination})
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
l2, err = net.ListenPacket("udp", "224.0.0.0:"+strconv.FormatInt(int64(th.Ports[1]), 10))
|
|
require.NoError(t, err)
|
|
defer l2.Close()
|
|
|
|
p = ipv4.NewPacketConn(l2)
|
|
|
|
intfs, err = net.Interfaces()
|
|
require.NoError(t, err)
|
|
|
|
for _, intf := range intfs {
|
|
err := p.JoinGroup(&intf, &net.UDPAddr{IP: *th.Destination})
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
default:
|
|
require.Equal(t, headers.TransportProtocolTCP, th.Protocol)
|
|
require.Equal(t, headers.TransportDeliveryUnicast, *th.Delivery)
|
|
}
|
|
|
|
<-sessionOpened
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://"+listenIP+":8554/teststream", session)
|
|
|
|
// server -> client (direct)
|
|
switch transport {
|
|
case "udp":
|
|
buf := make([]byte, 2048)
|
|
n, _, err := l2.ReadFrom(buf)
|
|
require.NoError(t, err)
|
|
require.Equal(t, testRTCPPacketMarshaled, buf[:n])
|
|
|
|
case "tcp", "tls":
|
|
f, err := conn.ReadInterleavedFrame()
|
|
require.NoError(t, err)
|
|
|
|
switch f.Channel {
|
|
case 4:
|
|
require.Equal(t, testRTPPacketMarshaled, f.Payload)
|
|
|
|
case 5:
|
|
require.Equal(t, testRTCPPacketMarshaled, f.Payload)
|
|
|
|
default:
|
|
t.Errorf("should not happen")
|
|
}
|
|
}
|
|
|
|
// server -> client (through stream)
|
|
if transport == "udp" || transport == "multicast" {
|
|
buf := make([]byte, 2048)
|
|
n, _, err := l1.ReadFrom(buf)
|
|
require.NoError(t, err)
|
|
require.Equal(t, testRTPPacketMarshaled, buf[:n])
|
|
|
|
buf = make([]byte, 2048)
|
|
n, _, err = l2.ReadFrom(buf)
|
|
require.NoError(t, err)
|
|
require.Equal(t, testRTCPPacketMarshaled, buf[:n])
|
|
} else {
|
|
for i := 0; i < 2; i++ {
|
|
f, err := conn.ReadInterleavedFrame()
|
|
require.NoError(t, err)
|
|
|
|
switch f.Channel {
|
|
case 4:
|
|
require.Equal(t, testRTPPacketMarshaled, f.Payload)
|
|
|
|
case 5:
|
|
require.Equal(t, testRTCPPacketMarshaled, f.Payload)
|
|
|
|
default:
|
|
t.Errorf("should not happen")
|
|
}
|
|
}
|
|
}
|
|
|
|
// client -> server (RTCP)
|
|
switch transport {
|
|
case "udp":
|
|
l2.WriteTo(testRTCPPacketMarshaled, &net.UDPAddr{
|
|
IP: net.ParseIP("127.0.0.1"),
|
|
Port: th.ServerPorts[1],
|
|
})
|
|
<-framesReceived
|
|
|
|
case "multicast":
|
|
l2.WriteTo(testRTCPPacketMarshaled, &net.UDPAddr{
|
|
IP: *th.Destination,
|
|
Port: th.Ports[1],
|
|
})
|
|
<-framesReceived
|
|
|
|
default:
|
|
err := conn.WriteInterleavedFrame(&base.InterleavedFrame{
|
|
Channel: 5,
|
|
Payload: testRTCPPacketMarshaled,
|
|
}, make([]byte, 1024))
|
|
require.NoError(t, err)
|
|
<-framesReceived
|
|
}
|
|
|
|
if transport == "udp" || transport == "multicast" {
|
|
// ping with OPTIONS
|
|
res, err = writeReqReadRes(conn, base.Request{
|
|
Method: base.Options,
|
|
URL: mustParseURL("rtsp://" + listenIP + ":8554/teststream"),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"4"},
|
|
"Session": base.HeaderValue{session},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
|
|
// ping with GET_PARAMETER
|
|
res, err = writeReqReadRes(conn, base.Request{
|
|
Method: base.GetParameter,
|
|
URL: mustParseURL("rtsp://" + listenIP + ":8554/teststream"),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"5"},
|
|
"Session": base.HeaderValue{session},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
}
|
|
|
|
res, err = writeReqReadRes(conn, base.Request{
|
|
Method: base.Teardown,
|
|
URL: mustParseURL("rtsp://" + listenIP + ":8554/teststream"),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"6"},
|
|
"Session": base.HeaderValue{session},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
|
|
<-sessionClosed
|
|
|
|
nconn.Close()
|
|
<-nconnClosed
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestServerPlayDecodeErrors(t *testing.T) {
|
|
for _, ca := range []struct {
|
|
proto string
|
|
name string
|
|
}{
|
|
{"udp", "rtcp invalid"},
|
|
{"udp", "rtcp too big"},
|
|
{"tcp", "rtcp invalid"},
|
|
{"tcp", "rtcp too big"},
|
|
} {
|
|
t.Run(ca.proto+" "+ca.name, func(t *testing.T) {
|
|
errorRecv := make(chan struct{})
|
|
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
onDecodeError: func(ctx *ServerHandlerOnDecodeErrorCtx) {
|
|
switch {
|
|
case ca.proto == "udp" && ca.name == "rtcp invalid":
|
|
require.EqualError(t, ctx.Error, "rtcp: packet too short")
|
|
|
|
case ca.proto == "udp" && ca.name == "rtcp too big":
|
|
require.EqualError(t, ctx.Error, "RTCP packet is too big to be read with UDP")
|
|
|
|
case ca.proto == "tcp" && ca.name == "rtcp invalid":
|
|
require.EqualError(t, ctx.Error, "rtcp: packet too short")
|
|
|
|
case ca.proto == "tcp" && ca.name == "rtcp too big":
|
|
require.EqualError(t, ctx.Error, "RTCP packet size (2000) is greater than maximum allowed (1472)")
|
|
}
|
|
close(errorRecv)
|
|
},
|
|
},
|
|
UDPRTPAddress: "127.0.0.1:8000",
|
|
UDPRTCPAddress: "127.0.0.1:8001",
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
}
|
|
|
|
if ca.proto == "udp" {
|
|
inTH.Protocol = headers.TransportProtocolUDP
|
|
inTH.ClientPorts = &[2]int{35466, 35467}
|
|
} else {
|
|
inTH.Protocol = headers.TransportProtocolTCP
|
|
inTH.InterleavedIDs = &[2]int{0, 1}
|
|
}
|
|
|
|
res, resTH := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
var l1 net.PacketConn
|
|
var l2 net.PacketConn
|
|
|
|
if ca.proto == "udp" {
|
|
l1, err = net.ListenPacket("udp", "127.0.0.1:35466")
|
|
require.NoError(t, err)
|
|
defer l1.Close()
|
|
|
|
l2, err = net.ListenPacket("udp", "127.0.0.1:35467")
|
|
require.NoError(t, err)
|
|
defer l2.Close()
|
|
}
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
|
|
switch { //nolint:dupl
|
|
case ca.proto == "udp" && ca.name == "rtcp invalid":
|
|
l2.WriteTo([]byte{0x01, 0x02}, &net.UDPAddr{
|
|
IP: net.ParseIP("127.0.0.1"),
|
|
Port: resTH.ServerPorts[1],
|
|
})
|
|
|
|
case ca.proto == "udp" && ca.name == "rtcp too big":
|
|
l2.WriteTo(bytes.Repeat([]byte{0x01, 0x02}, 2000/2), &net.UDPAddr{
|
|
IP: net.ParseIP("127.0.0.1"),
|
|
Port: resTH.ServerPorts[1],
|
|
})
|
|
|
|
case ca.proto == "tcp" && ca.name == "rtcp invalid":
|
|
err = conn.WriteInterleavedFrame(&base.InterleavedFrame{
|
|
Channel: 1,
|
|
Payload: []byte{0x01, 0x02},
|
|
}, make([]byte, 2048))
|
|
require.NoError(t, err)
|
|
|
|
case ca.proto == "tcp" && ca.name == "rtcp too big":
|
|
err = conn.WriteInterleavedFrame(&base.InterleavedFrame{
|
|
Channel: 1,
|
|
Payload: bytes.Repeat([]byte{0x01, 0x02}, 2000/2),
|
|
}, make([]byte, 2048))
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
<-errorRecv
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestServerPlayRTCPReport(t *testing.T) {
|
|
for _, ca := range []string{"udp", "tcp"} {
|
|
t.Run(ca, func(t *testing.T) {
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
senderReportPeriod: 1 * time.Second,
|
|
RTSPAddress: "localhost:8554",
|
|
UDPRTPAddress: "127.0.0.1:8000",
|
|
UDPRTCPAddress: "127.0.0.1:8001",
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
}
|
|
|
|
if ca == "udp" {
|
|
inTH.Protocol = headers.TransportProtocolUDP
|
|
inTH.ClientPorts = &[2]int{35466, 35467}
|
|
} else {
|
|
inTH.Protocol = headers.TransportProtocolTCP
|
|
inTH.InterleavedIDs = &[2]int{0, 1}
|
|
}
|
|
|
|
res, _ := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
var l1 net.PacketConn
|
|
var l2 net.PacketConn
|
|
if ca == "udp" {
|
|
l1, err = net.ListenPacket("udp", "localhost:35466")
|
|
require.NoError(t, err)
|
|
defer l1.Close()
|
|
|
|
l2, err = net.ListenPacket("udp", "localhost:35467")
|
|
require.NoError(t, err)
|
|
defer l2.Close()
|
|
}
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
|
|
for i := 0; i < 2; i++ {
|
|
stream.WritePacketRTP(stream.Medias()[0], &rtp.Packet{
|
|
Header: rtp.Header{
|
|
Version: 2,
|
|
PayloadType: 96,
|
|
SSRC: 0x38F27A2F,
|
|
},
|
|
Payload: []byte{0x05}, // IDR
|
|
})
|
|
}
|
|
|
|
var buf []byte
|
|
|
|
if ca == "udp" {
|
|
buf = make([]byte, 2048)
|
|
var n int
|
|
n, _, err = l2.ReadFrom(buf)
|
|
require.NoError(t, err)
|
|
buf = buf[:n]
|
|
} else {
|
|
for i := 0; i < 2; i++ {
|
|
_, err := conn.ReadInterleavedFrame()
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
f, err := conn.ReadInterleavedFrame()
|
|
require.NoError(t, err)
|
|
require.Equal(t, 1, f.Channel)
|
|
buf = f.Payload
|
|
}
|
|
|
|
packets, err := rtcp.Unmarshal(buf)
|
|
require.NoError(t, err)
|
|
require.Equal(t, &rtcp.SenderReport{
|
|
SSRC: 0x38F27A2F,
|
|
NTPTime: packets[0].(*rtcp.SenderReport).NTPTime,
|
|
RTPTime: packets[0].(*rtcp.SenderReport).RTPTime,
|
|
PacketCount: 2,
|
|
OctetCount: 2,
|
|
}, packets[0])
|
|
|
|
res, err = writeReqReadRes(conn, base.Request{
|
|
Method: base.Teardown,
|
|
URL: mustParseURL("rtsp://localhost:8554/teststream"),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"4"},
|
|
"Session": base.HeaderValue{session},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestServerPlayVLCMulticast(t *testing.T) {
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
listenIP := multicastCapableIP(t)
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
},
|
|
RTSPAddress: listenIP + ":8554",
|
|
MulticastIPRange: "224.1.0.0/16",
|
|
MulticastRTPPort: 8000,
|
|
MulticastRTCPPort: 8001,
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", listenIP+":8554")
|
|
require.NoError(t, err)
|
|
conn := conn.NewConn(nconn)
|
|
defer nconn.Close()
|
|
|
|
res, err := writeReqReadRes(conn, base.Request{
|
|
Method: base.Describe,
|
|
URL: mustParseURL("rtsp://" + listenIP + ":8554/teststream?vlcmulticast"),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"1"},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
|
|
var desc sdp.SessionDescription
|
|
err = desc.Unmarshal(res.Body)
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, "224.1.0.0", desc.ConnectionInformation.Address.Address)
|
|
}
|
|
|
|
func TestServerPlayTCPResponseBeforeFrames(t *testing.T) {
|
|
writerDone := make(chan struct{})
|
|
writerTerminate := make(chan struct{})
|
|
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
RTSPAddress: "localhost:8554",
|
|
Handler: &testServerHandler{
|
|
onConnClose: func(ctx *ServerHandlerOnConnCloseCtx) {
|
|
close(writerTerminate)
|
|
<-writerDone
|
|
},
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
go func() {
|
|
defer close(writerDone)
|
|
|
|
stream.WritePacketRTP(stream.Medias()[0], &testRTPPacket)
|
|
|
|
t := time.NewTicker(50 * time.Millisecond)
|
|
defer t.Stop()
|
|
|
|
for {
|
|
select {
|
|
case <-t.C:
|
|
stream.WritePacketRTP(stream.Medias()[0], &testRTPPacket)
|
|
case <-writerTerminate:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
time.Sleep(50 * time.Millisecond)
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Protocol: headers.TransportProtocolTCP,
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
InterleavedIDs: &[2]int{0, 1},
|
|
}
|
|
|
|
res, _ := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
|
|
_, err = conn.ReadInterleavedFrame()
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
func TestServerPlayPlayPlay(t *testing.T) {
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
UDPRTPAddress: "127.0.0.1:8000",
|
|
UDPRTCPAddress: "127.0.0.1:8001",
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Protocol: headers.TransportProtocolUDP,
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
ClientPorts: &[2]int{30450, 30451},
|
|
}
|
|
|
|
res, _ := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
}
|
|
|
|
func TestServerPlayPlayPausePlay(t *testing.T) {
|
|
writerStarted := false
|
|
writerDone := make(chan struct{})
|
|
writerTerminate := make(chan struct{})
|
|
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onConnClose: func(ctx *ServerHandlerOnConnCloseCtx) {
|
|
close(writerTerminate)
|
|
<-writerDone
|
|
},
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
if !writerStarted {
|
|
writerStarted = true
|
|
go func() {
|
|
defer close(writerDone)
|
|
|
|
t := time.NewTicker(50 * time.Millisecond)
|
|
defer t.Stop()
|
|
|
|
for {
|
|
select {
|
|
case <-t.C:
|
|
stream.WritePacketRTP(stream.Medias()[0], &testRTPPacket)
|
|
case <-writerTerminate:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
onPause: func(ctx *ServerHandlerOnPauseCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Protocol: headers.TransportProtocolTCP,
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
InterleavedIDs: &[2]int{0, 1},
|
|
}
|
|
|
|
res, _ := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
doPause(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
}
|
|
|
|
func TestServerPlayPlayPausePause(t *testing.T) {
|
|
writerDone := make(chan struct{})
|
|
writerTerminate := make(chan struct{})
|
|
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onConnClose: func(ctx *ServerHandlerOnConnCloseCtx) {
|
|
close(writerTerminate)
|
|
<-writerDone
|
|
},
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
go func() {
|
|
defer close(writerDone)
|
|
|
|
t := time.NewTicker(50 * time.Millisecond)
|
|
defer t.Stop()
|
|
|
|
for {
|
|
select {
|
|
case <-t.C:
|
|
stream.WritePacketRTP(stream.Medias()[0], &testRTPPacket)
|
|
case <-writerTerminate:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
onPause: func(ctx *ServerHandlerOnPauseCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Protocol: headers.TransportProtocolTCP,
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
InterleavedIDs: &[2]int{0, 1},
|
|
}
|
|
|
|
res, _ := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
|
|
doPause(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
|
|
doPause(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
}
|
|
|
|
func TestServerPlayTimeout(t *testing.T) {
|
|
for _, transport := range []string{
|
|
"udp",
|
|
"multicast",
|
|
// there's no timeout when reading with TCP
|
|
} {
|
|
t.Run(transport, func(t *testing.T) {
|
|
sessionClosed := make(chan struct{})
|
|
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onSessionClose: func(ctx *ServerHandlerOnSessionCloseCtx) {
|
|
close(sessionClosed)
|
|
},
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
ReadTimeout: 1 * time.Second,
|
|
sessionTimeout: 1 * time.Second,
|
|
RTSPAddress: "localhost:8554",
|
|
checkStreamPeriod: 500 * time.Millisecond,
|
|
}
|
|
|
|
switch transport {
|
|
case "udp":
|
|
s.UDPRTPAddress = "127.0.0.1:8000"
|
|
s.UDPRTCPAddress = "127.0.0.1:8001"
|
|
|
|
case "multicast":
|
|
s.MulticastIPRange = "224.1.0.0/16"
|
|
s.MulticastRTPPort = 8000
|
|
s.MulticastRTCPPort = 8001
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
}
|
|
|
|
switch transport {
|
|
case "udp":
|
|
v := headers.TransportDeliveryUnicast
|
|
inTH.Delivery = &v
|
|
inTH.Protocol = headers.TransportProtocolUDP
|
|
inTH.ClientPorts = &[2]int{35466, 35467}
|
|
|
|
case "multicast":
|
|
v := headers.TransportDeliveryMulticast
|
|
inTH.Delivery = &v
|
|
inTH.Protocol = headers.TransportProtocolUDP
|
|
}
|
|
|
|
res, _ := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
|
|
<-sessionClosed
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestServerPlayWithoutTeardown(t *testing.T) {
|
|
for _, transport := range []string{
|
|
"udp",
|
|
"tcp",
|
|
} {
|
|
t.Run(transport, func(t *testing.T) {
|
|
nconnClosed := make(chan struct{})
|
|
sessionClosed := make(chan struct{})
|
|
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onConnClose: func(ctx *ServerHandlerOnConnCloseCtx) {
|
|
close(nconnClosed)
|
|
},
|
|
onSessionClose: func(ctx *ServerHandlerOnSessionCloseCtx) {
|
|
close(sessionClosed)
|
|
},
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
ReadTimeout: 1 * time.Second,
|
|
sessionTimeout: 1 * time.Second,
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
if transport == "udp" {
|
|
s.UDPRTPAddress = "127.0.0.1:8000"
|
|
s.UDPRTCPAddress = "127.0.0.1:8001"
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
}
|
|
|
|
if transport == "udp" {
|
|
inTH.Protocol = headers.TransportProtocolUDP
|
|
inTH.ClientPorts = &[2]int{35466, 35467}
|
|
} else {
|
|
inTH.Protocol = headers.TransportProtocolTCP
|
|
inTH.InterleavedIDs = &[2]int{0, 1}
|
|
}
|
|
|
|
res, _ := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
|
|
nconn.Close()
|
|
|
|
<-sessionClosed
|
|
<-nconnClosed
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestServerPlayUDPChangeConn(t *testing.T) {
|
|
stream := NewServerStream(media.Medias{testH264Media})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
onGetParameter: func(ctx *ServerHandlerOnGetParameterCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
UDPRTPAddress: "127.0.0.1:8000",
|
|
UDPRTCPAddress: "127.0.0.1:8001",
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
sxID := ""
|
|
|
|
func() {
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
Protocol: headers.TransportProtocolUDP,
|
|
ClientPorts: &[2]int{35466, 35467},
|
|
}
|
|
|
|
res, _ := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
|
|
sxID = session
|
|
}()
|
|
|
|
func() {
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
res, err := writeReqReadRes(conn, base.Request{
|
|
Method: base.GetParameter,
|
|
URL: mustParseURL("rtsp://localhost:8554/teststream/"),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"4"},
|
|
"Session": base.HeaderValue{sxID},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
}()
|
|
}
|
|
|
|
func TestServerPlayPartialMedias(t *testing.T) {
|
|
stream := NewServerStream(media.Medias{testH264Media, testH264Media})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
go func() {
|
|
time.Sleep(1 * time.Second)
|
|
stream.WritePacketRTP(stream.Medias()[0], &testRTPPacket)
|
|
stream.WritePacketRTP(stream.Medias()[1], &testRTPPacket)
|
|
}()
|
|
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
err := s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
Protocol: headers.TransportProtocolTCP,
|
|
InterleavedIDs: &[2]int{4, 5},
|
|
}
|
|
|
|
res, _ := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
session := readSession(t, res)
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
|
|
f, err := conn.ReadInterleavedFrame()
|
|
require.NoError(t, err)
|
|
require.Equal(t, 4, f.Channel)
|
|
require.Equal(t, testRTPPacketMarshaled, f.Payload)
|
|
}
|
|
|
|
func TestServerPlayAdditionalInfos(t *testing.T) {
|
|
getInfos := func() (*headers.RTPInfo, []*uint32) {
|
|
nconn, err := net.Dial("tcp", "localhost:8554")
|
|
require.NoError(t, err)
|
|
defer nconn.Close()
|
|
conn := conn.NewConn(nconn)
|
|
|
|
desc := doDescribe(t, conn)
|
|
|
|
inTH := &headers.Transport{
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
Protocol: headers.TransportProtocolTCP,
|
|
InterleavedIDs: &[2]int{0, 1},
|
|
}
|
|
|
|
res, th := doSetup(t, conn, absoluteControlAttribute(desc.MediaDescriptions[0]), inTH)
|
|
|
|
ssrcs := make([]*uint32, 2)
|
|
ssrcs[0] = th.SSRC
|
|
|
|
inTH = &headers.Transport{
|
|
Delivery: func() *headers.TransportDelivery {
|
|
v := headers.TransportDeliveryUnicast
|
|
return &v
|
|
}(),
|
|
Mode: func() *headers.TransportMode {
|
|
v := headers.TransportModePlay
|
|
return &v
|
|
}(),
|
|
Protocol: headers.TransportProtocolTCP,
|
|
InterleavedIDs: &[2]int{2, 3},
|
|
}
|
|
|
|
session := readSession(t, res)
|
|
|
|
res, err = writeReqReadRes(conn, base.Request{
|
|
Method: base.Setup,
|
|
URL: mustParseURL(absoluteControlAttribute(desc.MediaDescriptions[1])),
|
|
Header: base.Header{
|
|
"CSeq": base.HeaderValue{"3"},
|
|
"Transport": inTH.Marshal(),
|
|
"Session": base.HeaderValue{session},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, base.StatusOK, res.StatusCode)
|
|
|
|
th = &headers.Transport{}
|
|
err = th.Unmarshal(res.Header["Transport"])
|
|
require.NoError(t, err)
|
|
ssrcs[1] = th.SSRC
|
|
|
|
doPlay(t, conn, "rtsp://localhost:8554/teststream", session)
|
|
|
|
var ri headers.RTPInfo
|
|
err = ri.Unmarshal(res.Header["RTP-Info"])
|
|
require.NoError(t, err)
|
|
|
|
return &ri, ssrcs
|
|
}
|
|
|
|
forma := &formats.Generic{
|
|
PayloadTyp: 96,
|
|
RTPMap: "private/90000",
|
|
}
|
|
err := forma.Init()
|
|
require.NoError(t, err)
|
|
|
|
stream := NewServerStream(media.Medias{
|
|
&media.Media{
|
|
Type: "application",
|
|
Formats: []formats.Format{forma},
|
|
},
|
|
&media.Media{
|
|
Type: "application",
|
|
Formats: []formats.Format{forma},
|
|
},
|
|
})
|
|
defer stream.Close()
|
|
|
|
s := &Server{
|
|
Handler: &testServerHandler{
|
|
onDescribe: func(ctx *ServerHandlerOnDescribeCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onSetup: func(ctx *ServerHandlerOnSetupCtx) (*base.Response, *ServerStream, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, stream, nil
|
|
},
|
|
onPlay: func(ctx *ServerHandlerOnPlayCtx) (*base.Response, error) {
|
|
return &base.Response{
|
|
StatusCode: base.StatusOK,
|
|
}, nil
|
|
},
|
|
},
|
|
RTSPAddress: "localhost:8554",
|
|
}
|
|
|
|
err = s.Start()
|
|
require.NoError(t, err)
|
|
defer s.Close()
|
|
|
|
stream.WritePacketRTP(stream.Medias()[0], &rtp.Packet{
|
|
Header: rtp.Header{
|
|
Version: 2,
|
|
PayloadType: 96,
|
|
SequenceNumber: 556,
|
|
Timestamp: 984512368,
|
|
SSRC: 96342362,
|
|
},
|
|
Payload: []byte{0x01, 0x02, 0x03, 0x04},
|
|
})
|
|
|
|
rtpInfo, ssrcs := getInfos()
|
|
require.True(t, strings.HasPrefix(mustParseURL((*rtpInfo)[0].URL).Path, "/teststream/trackID="))
|
|
require.Equal(t, &headers.RTPInfo{
|
|
&headers.RTPInfoEntry{
|
|
URL: (&url.URL{
|
|
Scheme: "rtsp",
|
|
Host: "localhost:8554",
|
|
Path: mustParseURL((*rtpInfo)[0].URL).Path,
|
|
}).String(),
|
|
SequenceNumber: func() *uint16 {
|
|
v := uint16(557)
|
|
return &v
|
|
}(),
|
|
Timestamp: (*rtpInfo)[0].Timestamp,
|
|
},
|
|
}, rtpInfo)
|
|
require.Equal(t, []*uint32{
|
|
func() *uint32 {
|
|
v := uint32(96342362)
|
|
return &v
|
|
}(),
|
|
nil,
|
|
}, ssrcs)
|
|
|
|
stream.WritePacketRTP(stream.Medias()[1], &rtp.Packet{
|
|
Header: rtp.Header{
|
|
Version: 2,
|
|
PayloadType: 96,
|
|
SequenceNumber: 87,
|
|
Timestamp: 756436454,
|
|
SSRC: 536474323,
|
|
},
|
|
Payload: []byte{0x01, 0x02, 0x03, 0x04},
|
|
})
|
|
|
|
rtpInfo, ssrcs = getInfos()
|
|
require.True(t, strings.HasPrefix(mustParseURL((*rtpInfo)[0].URL).Path, "/teststream/trackID="))
|
|
require.Equal(t, &headers.RTPInfo{
|
|
&headers.RTPInfoEntry{
|
|
URL: (&url.URL{
|
|
Scheme: "rtsp",
|
|
Host: "localhost:8554",
|
|
Path: mustParseURL((*rtpInfo)[0].URL).Path,
|
|
}).String(),
|
|
SequenceNumber: func() *uint16 {
|
|
v := uint16(557)
|
|
return &v
|
|
}(),
|
|
Timestamp: (*rtpInfo)[0].Timestamp,
|
|
},
|
|
&headers.RTPInfoEntry{
|
|
URL: (&url.URL{
|
|
Scheme: "rtsp",
|
|
Host: "localhost:8554",
|
|
Path: mustParseURL((*rtpInfo)[1].URL).Path,
|
|
}).String(),
|
|
SequenceNumber: func() *uint16 {
|
|
v := uint16(88)
|
|
return &v
|
|
}(),
|
|
Timestamp: (*rtpInfo)[1].Timestamp,
|
|
},
|
|
}, rtpInfo)
|
|
require.Equal(t, []*uint32{
|
|
func() *uint32 {
|
|
v := uint32(96342362)
|
|
return &v
|
|
}(),
|
|
func() *uint32 {
|
|
v := uint32(536474323)
|
|
return &v
|
|
}(),
|
|
}, ssrcs)
|
|
}
|