// +build !js package webrtc import ( "bytes" "context" "errors" "io" "sync" "sync/atomic" "testing" "time" "github.com/pion/transport/packetio" "github.com/pion/transport/test" "github.com/pion/webrtc/v3/pkg/media" "github.com/stretchr/testify/assert" ) func untilConnectionState(state PeerConnectionState, peers ...*PeerConnection) *sync.WaitGroup { var triggered sync.WaitGroup triggered.Add(len(peers)) hdlr := func(p PeerConnectionState) { if p == state { triggered.Done() } } for _, p := range peers { p.OnConnectionStateChange(hdlr) } return &triggered } func Test_RTPSender_ReplaceTrack(t *testing.T) { lim := test.TimeOut(time.Second * 10) defer lim.Stop() report := test.CheckRoutines(t) defer report() t.Run("Basic", func(t *testing.T) { s := SettingEngine{} s.DisableSRTPReplayProtection(true) m := &MediaEngine{} assert.NoError(t, m.RegisterDefaultCodecs()) sender, receiver, err := NewAPI(WithMediaEngine(m), WithSettingEngine(s)).newPair(Configuration{}) assert.NoError(t, err) trackA, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: "video/vp8"}, "video", "pion") assert.NoError(t, err) trackB, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: "video/vp8"}, "video", "pion") assert.NoError(t, err) rtpSender, err := sender.AddTrack(trackA) assert.NoError(t, err) seenPacketA, seenPacketACancel := context.WithCancel(context.Background()) seenPacketB, seenPacketBCancel := context.WithCancel(context.Background()) var onTrackCount uint64 receiver.OnTrack(func(track *TrackRemote, _ *RTPReceiver) { assert.Equal(t, uint64(1), atomic.AddUint64(&onTrackCount, 1)) for { pkt, _, err := track.ReadRTP() if err != nil { assert.True(t, errors.Is(io.EOF, err)) return } switch { case bytes.Equal(pkt.Payload, []byte{0x10, 0xAA}): seenPacketACancel() case bytes.Equal(pkt.Payload, []byte{0x10, 0xBB}): seenPacketBCancel() } } }) assert.NoError(t, signalPair(sender, receiver)) // Block Until packet with 0xAA has been seen func() { for range time.Tick(time.Millisecond * 20) { select { case <-seenPacketA.Done(): return default: assert.NoError(t, trackA.WriteSample(media.Sample{Data: []byte{0xAA}, Duration: time.Second})) } } }() assert.NoError(t, rtpSender.ReplaceTrack(trackB)) // Block Until packet with 0xBB has been seen func() { for range time.Tick(time.Millisecond * 20) { select { case <-seenPacketB.Done(): return default: assert.NoError(t, trackB.WriteSample(media.Sample{Data: []byte{0xBB}, Duration: time.Second})) } } }() closePairNow(t, sender, receiver) }) t.Run("Invalid Codec Change", func(t *testing.T) { sender, receiver, err := newPair() assert.NoError(t, err) trackA, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: "video/vp8"}, "video", "pion") assert.NoError(t, err) trackB, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: "video/h264", SDPFmtpLine: "level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=42001f"}, "video", "pion") assert.NoError(t, err) rtpSender, err := sender.AddTrack(trackA) assert.NoError(t, err) assert.NoError(t, signalPair(sender, receiver)) seenPacket, seenPacketCancel := context.WithCancel(context.Background()) receiver.OnTrack(func(_ *TrackRemote, _ *RTPReceiver) { seenPacketCancel() }) func() { for range time.Tick(time.Millisecond * 20) { select { case <-seenPacket.Done(): return default: assert.NoError(t, trackA.WriteSample(media.Sample{Data: []byte{0xAA}, Duration: time.Second})) } } }() assert.True(t, errors.Is(rtpSender.ReplaceTrack(trackB), ErrUnsupportedCodec)) closePairNow(t, sender, receiver) }) } func Test_RTPSender_GetParameters(t *testing.T) { lim := test.TimeOut(time.Second * 10) defer lim.Stop() report := test.CheckRoutines(t) defer report() offerer, answerer, err := newPair() assert.NoError(t, err) rtpTransceiver, err := offerer.AddTransceiverFromKind(RTPCodecTypeVideo) assert.NoError(t, err) assert.NoError(t, signalPair(offerer, answerer)) parameters := rtpTransceiver.Sender().GetParameters() assert.NotEqual(t, 0, len(parameters.Codecs)) assert.Equal(t, 1, len(parameters.Encodings)) assert.Equal(t, rtpTransceiver.Sender().ssrc, parameters.Encodings[0].SSRC) closePairNow(t, offerer, answerer) } func Test_RTPSender_SetReadDeadline(t *testing.T) { lim := test.TimeOut(time.Second * 30) defer lim.Stop() report := test.CheckRoutines(t) defer report() sender, receiver, wan := createVNetPair(t) track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: "video/vp8"}, "video", "pion") assert.NoError(t, err) rtpSender, err := sender.AddTrack(track) assert.NoError(t, err) peerConnectionsConnected := untilConnectionState(PeerConnectionStateConnected, sender, receiver) assert.NoError(t, signalPair(sender, receiver)) peerConnectionsConnected.Wait() assert.NoError(t, rtpSender.SetReadDeadline(time.Now().Add(1*time.Second))) _, _, err = rtpSender.ReadRTCP() assert.Error(t, err, packetio.ErrTimeout) assert.NoError(t, wan.Stop()) closePairNow(t, sender, receiver) }