Skip to content

Commit

Permalink
Add an end-to-end test for the NACK sender
Browse files Browse the repository at this point in the history
Test that NACKs are negotiated correctly, and that we receive
the expected NACK if we negotiated it.
  • Loading branch information
jech committed Jan 2, 2025
1 parent 92d573c commit 7c1a7aa
Showing 1 changed file with 152 additions and 0 deletions.
152 changes: 152 additions & 0 deletions interceptor_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,12 +9,14 @@ package webrtc
//
import (
"context"
"io"
"sync/atomic"
"testing"
"time"

"github.com/pion/interceptor"
mock_interceptor "github.com/pion/interceptor/pkg/mock"
"github.com/pion/rtcp"
"github.com/pion/rtp"
"github.com/pion/transport/v3/test"
"github.com/pion/webrtc/v4/pkg/media"
Expand Down Expand Up @@ -285,3 +287,153 @@ func Test_Interceptor_ZeroSSRC(t *testing.T) {
<-probeReceiverCreated
closePairNow(t, offerer, answerer)
}

// TestInterceptorNack is an end-to-end test for the NACK sender.
// It test that:
// - we get a NACK if we negotiated generic NACks;
// - we don't get a NACK if we did not negotiate generick NACKs;
// - the NACK corresponds to the missing packet.
func TestInterceptorNack(t *testing.T) {
to := test.TimeOut(time.Second * 20)
defer to.Stop()

t.Run("Nack", func(t *testing.T) { testInterceptorNack(t, true) })
t.Run("NoNack", func(t *testing.T) { testInterceptorNack(t, false) })
}

func testInterceptorNack(t *testing.T, requestNack bool) {
const numPackets = 20

ir := interceptor.Registry{}
m := MediaEngine{}
var capability []RTCPFeedback
if requestNack {
capability = append(capability, RTCPFeedback{"nack", ""})
}
err := m.RegisterCodec(
RTPCodecParameters{
RTPCodecCapability: RTPCodecCapability{
"video/VP8", 90000, 0,
"",
capability,
},
PayloadType: 96,
},
RTPCodecTypeVideo,
)
assert.NoError(t, err)
api := NewAPI(
WithMediaEngine(&m),
WithInterceptorRegistry(&ir),
)

pc1, err := api.NewPeerConnection(Configuration{})
assert.NoError(t, err)

track1, err := NewTrackLocalStaticRTP(
RTPCodecCapability{MimeType: MimeTypeVP8},
"video", "pion",
)
assert.NoError(t, err)
sender, err := pc1.AddTrack(track1)
assert.NoError(t, err)

pc2, err := NewPeerConnection(Configuration{})
assert.NoError(t, err)

offer, err := pc1.CreateOffer(nil)
assert.NoError(t, err)
err = pc1.SetLocalDescription(offer)
assert.NoError(t, err)
<-GatheringCompletePromise(pc1)

err = pc2.SetRemoteDescription(*pc1.LocalDescription())
assert.NoError(t, err)
answer, err := pc2.CreateAnswer(nil)
assert.NoError(t, err)
err = pc2.SetLocalDescription(answer)
assert.NoError(t, err)
<-GatheringCompletePromise(pc2)

err = pc1.SetRemoteDescription(*pc2.LocalDescription())
assert.NoError(t, err)

var gotNack bool
rtcpDone := make(chan struct{})
go func() {
defer close(rtcpDone)
buf := make([]byte, 1500)
for {
n, _, err := sender.Read(buf)

Check failure on line 367 in interceptor_test.go

View workflow job for this annotation

GitHub Actions / lint / Go

shadow: declaration of "err" shadows declaration at line 313 (govet)
// nolint
if err == io.EOF {
break
}
assert.NoError(t, err)
ps, err := rtcp.Unmarshal(buf[:n])
assert.NoError(t, err)
for _, p := range ps {
if pn, ok := p.(*rtcp.TransportLayerNack); ok {
assert.Equal(t, len(pn.Nacks), 1)
assert.Equal(t,
pn.Nacks[0].PacketID, uint16(1),
)
assert.Equal(t,
pn.Nacks[0].LostPackets,
rtcp.PacketBitmap(0),
)
gotNack = true
}
}
}
}()

done := make(chan struct{})
pc2.OnTrack(func(track2 *TrackRemote, _ *RTPReceiver) {
for i := 0; i < numPackets; i++ {
if i == 1 {
continue
}
p, _, err := track2.ReadRTP()

Check failure on line 397 in interceptor_test.go

View workflow job for this annotation

GitHub Actions / lint / Go

shadow: declaration of "err" shadows declaration at line 313 (govet)
assert.NoError(t, err)
assert.Equal(t, p.SequenceNumber, uint16(i))
}
done <- struct{}{}
})

go func() {
for i := 0; i < numPackets; i++ {
time.Sleep(20 * time.Millisecond)
if i == 1 {
continue
}
var p rtp.Packet
p.Version = 2
p.Marker = true
p.PayloadType = 96
p.SequenceNumber = uint16(i)
p.Timestamp = uint32(i * 90000 / 50)
p.Payload = []byte{42}
err := track1.WriteRTP(&p)

Check failure on line 417 in interceptor_test.go

View workflow job for this annotation

GitHub Actions / lint / Go

shadow: declaration of "err" shadows declaration at line 313 (govet)
assert.NoError(t, err)
}
}()

<-done
err = pc1.Close()
assert.NoError(t, err)
err = pc2.Close()
assert.NoError(t, err)
<-rtcpDone


if requestNack {
if !gotNack {
t.Errorf("Expected to get a NACK, got none")
}
} else {
if gotNack {
t.Errorf("Expected to get no NACK, got one")
}
}
}

0 comments on commit 7c1a7aa

Please sign in to comment.