| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325 |
- // SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
- // SPDX-License-Identifier: MIT
- //go:build !js
- // +build !js
- package webrtc
- import (
- "bufio"
- "context"
- "errors"
- "io"
- "strconv"
- "strings"
- "sync"
- "sync/atomic"
- "testing"
- "time"
- "github.com/pion/rtp"
- "github.com/pion/transport/v2/test"
- "github.com/pion/webrtc/v3/internal/util"
- "github.com/pion/webrtc/v3/pkg/media"
- "github.com/pion/webrtc/v3/pkg/rtcerr"
- "github.com/stretchr/testify/assert"
- "github.com/stretchr/testify/require"
- )
- func sendVideoUntilDone(done <-chan struct{}, t *testing.T, tracks []*TrackLocalStaticSample) {
- for {
- select {
- case <-time.After(20 * time.Millisecond):
- for _, track := range tracks {
- assert.NoError(t, track.WriteSample(media.Sample{Data: []byte{0x00}, Duration: time.Second}))
- }
- case <-done:
- return
- }
- }
- }
- func sdpMidHasSsrc(offer SessionDescription, mid string, ssrc SSRC) bool {
- for _, media := range offer.parsed.MediaDescriptions {
- cmid, ok := media.Attribute("mid")
- if !ok {
- continue
- }
- if cmid != mid {
- continue
- }
- cssrc, ok := media.Attribute("ssrc")
- if !ok {
- continue
- }
- parts := strings.Split(cssrc, " ")
- ssrcInt64, err := strconv.ParseUint(parts[0], 10, 32)
- if err != nil {
- continue
- }
- if uint32(ssrcInt64) == uint32(ssrc) {
- return true
- }
- }
- return false
- }
- func TestPeerConnection_Renegotiation_AddRecvonlyTransceiver(t *testing.T) {
- type testCase struct {
- name string
- answererSends bool
- }
- testCases := []testCase{
- // Assert the following behaviors:
- // - Offerer can add a recvonly transceiver
- // - During negotiation, answerer peer adds an inactive (or sendonly) transceiver
- // - Offerer can add a track
- // - Answerer can receive the RTP packets.
- {"add recvonly, then receive from answerer", false},
- // Assert the following behaviors:
- // - Offerer can add a recvonly transceiver
- // - During negotiation, answerer peer adds an inactive (or sendonly) transceiver
- // - Answerer can add a track to the existing sendonly transceiver
- // - Offerer can receive the RTP packets.
- {"add recvonly, then send to answerer", true},
- }
- for _, tc := range testCases {
- tc := tc
- t.Run(tc.name, func(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- _, err = pcOffer.AddTransceiverFromKind(
- RTPCodecTypeVideo,
- RTPTransceiverInit{
- Direction: RTPTransceiverDirectionRecvonly,
- },
- )
- assert.NoError(t, err)
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- localTrack, err := NewTrackLocalStaticSample(
- RTPCodecCapability{MimeType: "video/VP8"}, "track-one", "stream-one",
- )
- require.NoError(t, err)
- if tc.answererSends {
- _, err = pcAnswer.AddTrack(localTrack)
- } else {
- _, err = pcOffer.AddTrack(localTrack)
- }
- require.NoError(t, err)
- onTrackFired, onTrackFiredFunc := context.WithCancel(context.Background())
- if tc.answererSends {
- pcOffer.OnTrack(func(track *TrackRemote, r *RTPReceiver) {
- onTrackFiredFunc()
- })
- assert.NoError(t, signalPair(pcAnswer, pcOffer))
- } else {
- pcAnswer.OnTrack(func(track *TrackRemote, r *RTPReceiver) {
- onTrackFiredFunc()
- })
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- }
- sendVideoUntilDone(onTrackFired.Done(), t, []*TrackLocalStaticSample{localTrack})
- closePairNow(t, pcOffer, pcAnswer)
- })
- }
- }
- /*
- * Assert the following behaviors
- * - We are able to call AddTrack after signaling
- * - OnTrack is NOT called on the other side until after SetRemoteDescription
- * - We are able to re-negotiate and AddTrack is properly called
- */
- func TestPeerConnection_Renegotiation_AddTrack(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- haveRenegotiated := &atomicBool{}
- onTrackFired, onTrackFiredFunc := context.WithCancel(context.Background())
- pcAnswer.OnTrack(func(track *TrackRemote, r *RTPReceiver) {
- if !haveRenegotiated.get() {
- t.Fatal("OnTrack was called before renegotiation")
- }
- onTrackFiredFunc()
- })
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- _, err = pcAnswer.AddTransceiverFromKind(RTPCodecTypeVideo, RTPTransceiverInit{Direction: RTPTransceiverDirectionRecvonly})
- assert.NoError(t, err)
- vp8Track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "foo", "bar")
- assert.NoError(t, err)
- sender, err := pcOffer.AddTrack(vp8Track)
- assert.NoError(t, err)
- // Send 10 packets, OnTrack MUST not be fired
- for i := 0; i <= 10; i++ {
- assert.NoError(t, vp8Track.WriteSample(media.Sample{Data: []byte{0x00}, Duration: time.Second}))
- time.Sleep(20 * time.Millisecond)
- }
- haveRenegotiated.set(true)
- assert.False(t, sender.isNegotiated())
- offer, err := pcOffer.CreateOffer(nil)
- assert.True(t, sender.isNegotiated())
- assert.NoError(t, err)
- assert.NoError(t, pcOffer.SetLocalDescription(offer))
- assert.NoError(t, pcAnswer.SetRemoteDescription(offer))
- answer, err := pcAnswer.CreateAnswer(nil)
- assert.NoError(t, err)
- assert.NoError(t, pcAnswer.SetLocalDescription(answer))
- pcOffer.ops.Done()
- assert.Equal(t, 0, len(vp8Track.rtpTrack.bindings))
- assert.NoError(t, pcOffer.SetRemoteDescription(answer))
- pcOffer.ops.Done()
- assert.Equal(t, 1, len(vp8Track.rtpTrack.bindings))
- sendVideoUntilDone(onTrackFired.Done(), t, []*TrackLocalStaticSample{vp8Track})
- closePairNow(t, pcOffer, pcAnswer)
- }
- // Assert that adding tracks across multiple renegotiations performs as expected
- func TestPeerConnection_Renegotiation_AddTrack_Multiple(t *testing.T) {
- addTrackWithLabel := func(trackID string, pcOffer, pcAnswer *PeerConnection) *TrackLocalStaticSample {
- _, err := pcAnswer.AddTransceiverFromKind(RTPCodecTypeVideo, RTPTransceiverInit{Direction: RTPTransceiverDirectionRecvonly})
- assert.NoError(t, err)
- track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, trackID, trackID)
- assert.NoError(t, err)
- _, err = pcOffer.AddTrack(track)
- assert.NoError(t, err)
- return track
- }
- trackIDs := []string{util.MathRandAlpha(16), util.MathRandAlpha(16), util.MathRandAlpha(16)}
- outboundTracks := []*TrackLocalStaticSample{}
- onTrackCount := map[string]int{}
- onTrackChan := make(chan struct{}, 1)
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- pcAnswer.OnTrack(func(track *TrackRemote, r *RTPReceiver) {
- onTrackCount[track.ID()]++
- onTrackChan <- struct{}{}
- })
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- for i := range trackIDs {
- outboundTracks = append(outboundTracks, addTrackWithLabel(trackIDs[i], pcOffer, pcAnswer))
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- sendVideoUntilDone(onTrackChan, t, outboundTracks)
- }
- closePairNow(t, pcOffer, pcAnswer)
- assert.Equal(t, onTrackCount[trackIDs[0]], 1)
- assert.Equal(t, onTrackCount[trackIDs[1]], 1)
- assert.Equal(t, onTrackCount[trackIDs[2]], 1)
- }
- // Assert that renegotiation triggers OnTrack() with correct ID and label from
- // remote side, even when a transceiver was added before the actual track data
- // was received. This happens when we add a transceiver on the server, create
- // an offer on the server and the browser's answer contains the same SSRC, but
- // a track hasn't been added on the browser side yet. The browser can add a
- // track later and renegotiate, and track ID and label will be set by the time
- // first packets are received.
- func TestPeerConnection_Renegotiation_AddTrack_Rename(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- haveRenegotiated := &atomicBool{}
- onTrackFired, onTrackFiredFunc := context.WithCancel(context.Background())
- var atomicRemoteTrack atomic.Value
- pcOffer.OnTrack(func(track *TrackRemote, r *RTPReceiver) {
- if !haveRenegotiated.get() {
- t.Fatal("OnTrack was called before renegotiation")
- }
- onTrackFiredFunc()
- atomicRemoteTrack.Store(track)
- })
- _, err = pcOffer.AddTransceiverFromKind(RTPCodecTypeVideo, RTPTransceiverInit{Direction: RTPTransceiverDirectionRecvonly})
- assert.NoError(t, err)
- vp8Track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "foo1", "bar1")
- assert.NoError(t, err)
- _, err = pcAnswer.AddTrack(vp8Track)
- assert.NoError(t, err)
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- vp8Track.rtpTrack.id = "foo2"
- vp8Track.rtpTrack.streamID = "bar2"
- haveRenegotiated.set(true)
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- sendVideoUntilDone(onTrackFired.Done(), t, []*TrackLocalStaticSample{vp8Track})
- closePairNow(t, pcOffer, pcAnswer)
- remoteTrack, ok := atomicRemoteTrack.Load().(*TrackRemote)
- require.True(t, ok)
- require.NotNil(t, remoteTrack)
- assert.Equal(t, "foo2", remoteTrack.ID())
- assert.Equal(t, "bar2", remoteTrack.StreamID())
- }
- // TestPeerConnection_Transceiver_Mid tests that we'll provide the same
- // transceiver for a media id on successive offer/answer
- func TestPeerConnection_Transceiver_Mid(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- pcAnswer, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- track1, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "video", "pion1")
- require.NoError(t, err)
- sender1, err := pcOffer.AddTrack(track1)
- require.NoError(t, err)
- track2, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "video", "pion2")
- require.NoError(t, err)
- sender2, err := pcOffer.AddTrack(track2)
- require.NoError(t, err)
- // this will create the initial offer using generateUnmatchedSDP
- offer, err := pcOffer.CreateOffer(nil)
- assert.NoError(t, err)
- offerGatheringComplete := GatheringCompletePromise(pcOffer)
- assert.NoError(t, pcOffer.SetLocalDescription(offer))
- <-offerGatheringComplete
- assert.NoError(t, pcAnswer.SetRemoteDescription(*pcOffer.LocalDescription()))
- answer, err := pcAnswer.CreateAnswer(nil)
- assert.NoError(t, err)
- answerGatheringComplete := GatheringCompletePromise(pcAnswer)
- assert.NoError(t, pcAnswer.SetLocalDescription(answer))
- <-answerGatheringComplete
- // apply answer so we'll test generateMatchedSDP
- assert.NoError(t, pcOffer.SetRemoteDescription(*pcAnswer.LocalDescription()))
- pcOffer.ops.Done()
- pcAnswer.ops.Done()
- // Must have 3 media descriptions (2 video channels)
- assert.Equal(t, len(offer.parsed.MediaDescriptions), 2)
- assert.True(t, sdpMidHasSsrc(offer, "0", sender1.trackEncodings[0].ssrc), "Expected mid %q with ssrc %d, offer.SDP: %s", "0", sender1.trackEncodings[0].ssrc, offer.SDP)
- // Remove first track, must keep same number of media
- // descriptions and same track ssrc for mid 1 as previous
- assert.NoError(t, pcOffer.RemoveTrack(sender1))
- offer, err = pcOffer.CreateOffer(nil)
- assert.NoError(t, err)
- assert.NoError(t, pcOffer.SetLocalDescription(offer))
- assert.Equal(t, len(offer.parsed.MediaDescriptions), 2)
- assert.True(t, sdpMidHasSsrc(offer, "1", sender2.trackEncodings[0].ssrc), "Expected mid %q with ssrc %d, offer.SDP: %s", "1", sender2.trackEncodings[0].ssrc, offer.SDP)
- _, err = pcAnswer.CreateAnswer(nil)
- assert.Equal(t, err, &rtcerr.InvalidStateError{Err: ErrIncorrectSignalingState})
- pcOffer.ops.Done()
- pcAnswer.ops.Done()
- assert.NoError(t, pcAnswer.SetRemoteDescription(offer))
- answer, err = pcAnswer.CreateAnswer(nil)
- assert.NoError(t, err)
- assert.NoError(t, pcOffer.SetRemoteDescription(answer))
- track3, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "video", "pion3")
- require.NoError(t, err)
- sender3, err := pcOffer.AddTrack(track3)
- require.NoError(t, err)
- offer, err = pcOffer.CreateOffer(nil)
- assert.NoError(t, err)
- // We reuse the existing non-sending transceiver
- assert.Equal(t, len(offer.parsed.MediaDescriptions), 2)
- assert.True(t, sdpMidHasSsrc(offer, "0", sender3.trackEncodings[0].ssrc), "Expected mid %q with ssrc %d, offer.sdp: %s", "0", sender3.trackEncodings[0].ssrc, offer.SDP)
- assert.True(t, sdpMidHasSsrc(offer, "1", sender2.trackEncodings[0].ssrc), "Expected mid %q with ssrc %d, offer.sdp: %s", "1", sender2.trackEncodings[0].ssrc, offer.SDP)
- closePairNow(t, pcOffer, pcAnswer)
- }
- func TestPeerConnection_Renegotiation_CodecChange(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- pcAnswer, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- track1, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "video1", "pion1")
- require.NoError(t, err)
- track2, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "video2", "pion2")
- require.NoError(t, err)
- sender1, err := pcOffer.AddTrack(track1)
- require.NoError(t, err)
- _, err = pcAnswer.AddTransceiverFromKind(RTPCodecTypeVideo, RTPTransceiverInit{Direction: RTPTransceiverDirectionRecvonly})
- require.NoError(t, err)
- tracksCh := make(chan *TrackRemote)
- tracksClosed := make(chan struct{})
- pcAnswer.OnTrack(func(track *TrackRemote, r *RTPReceiver) {
- tracksCh <- track
- for {
- if _, _, readErr := track.ReadRTP(); errors.Is(readErr, io.EOF) {
- tracksClosed <- struct{}{}
- return
- }
- }
- })
- err = signalPair(pcOffer, pcAnswer)
- require.NoError(t, err)
- transceivers := pcOffer.GetTransceivers()
- require.Equal(t, 1, len(transceivers))
- require.Equal(t, "0", transceivers[0].Mid())
- transceivers = pcAnswer.GetTransceivers()
- require.Equal(t, 1, len(transceivers))
- require.Equal(t, "0", transceivers[0].Mid())
- ctx, cancel := context.WithCancel(context.Background())
- go sendVideoUntilDone(ctx.Done(), t, []*TrackLocalStaticSample{track1})
- remoteTrack1 := <-tracksCh
- cancel()
- assert.Equal(t, "video1", remoteTrack1.ID())
- assert.Equal(t, "pion1", remoteTrack1.StreamID())
- require.NoError(t, pcOffer.RemoveTrack(sender1))
- require.NoError(t, signalPair(pcOffer, pcAnswer))
- <-tracksClosed
- sender2, err := pcOffer.AddTrack(track2)
- require.NoError(t, err)
- require.NoError(t, signalPair(pcOffer, pcAnswer))
- transceivers = pcOffer.GetTransceivers()
- require.Equal(t, 1, len(transceivers))
- require.Equal(t, "0", transceivers[0].Mid())
- transceivers = pcAnswer.GetTransceivers()
- require.Equal(t, 1, len(transceivers))
- require.Equal(t, "0", transceivers[0].Mid())
- ctx, cancel = context.WithCancel(context.Background())
- go sendVideoUntilDone(ctx.Done(), t, []*TrackLocalStaticSample{track2})
- remoteTrack2 := <-tracksCh
- cancel()
- require.NoError(t, pcOffer.RemoveTrack(sender2))
- err = signalPair(pcOffer, pcAnswer)
- require.NoError(t, err)
- <-tracksClosed
- assert.Equal(t, "video2", remoteTrack2.ID())
- assert.Equal(t, "pion2", remoteTrack2.StreamID())
- closePairNow(t, pcOffer, pcAnswer)
- }
- func TestPeerConnection_Renegotiation_RemoveTrack(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- _, err = pcAnswer.AddTransceiverFromKind(RTPCodecTypeVideo, RTPTransceiverInit{Direction: RTPTransceiverDirectionRecvonly})
- assert.NoError(t, err)
- vp8Track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "foo", "bar")
- assert.NoError(t, err)
- sender, err := pcOffer.AddTrack(vp8Track)
- assert.NoError(t, err)
- onTrackFired, onTrackFiredFunc := context.WithCancel(context.Background())
- trackClosed, trackClosedFunc := context.WithCancel(context.Background())
- pcAnswer.OnTrack(func(track *TrackRemote, r *RTPReceiver) {
- onTrackFiredFunc()
- for {
- if _, _, err := track.ReadRTP(); errors.Is(err, io.EOF) {
- trackClosedFunc()
- return
- }
- }
- })
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- sendVideoUntilDone(onTrackFired.Done(), t, []*TrackLocalStaticSample{vp8Track})
- assert.NoError(t, pcOffer.RemoveTrack(sender))
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- <-trackClosed.Done()
- closePairNow(t, pcOffer, pcAnswer)
- }
- func TestPeerConnection_RoleSwitch(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcFirstOfferer, pcSecondOfferer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- onTrackFired, onTrackFiredFunc := context.WithCancel(context.Background())
- pcFirstOfferer.OnTrack(func(track *TrackRemote, r *RTPReceiver) {
- onTrackFiredFunc()
- })
- assert.NoError(t, signalPair(pcFirstOfferer, pcSecondOfferer))
- // Add a new Track to the second offerer
- // This asserts that it will match the ordering of the last RemoteDescription, but then also add new Transceivers to the end
- _, err = pcFirstOfferer.AddTransceiverFromKind(RTPCodecTypeVideo, RTPTransceiverInit{Direction: RTPTransceiverDirectionRecvonly})
- assert.NoError(t, err)
- vp8Track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "foo", "bar")
- assert.NoError(t, err)
- _, err = pcSecondOfferer.AddTrack(vp8Track)
- assert.NoError(t, err)
- assert.NoError(t, signalPair(pcSecondOfferer, pcFirstOfferer))
- sendVideoUntilDone(onTrackFired.Done(), t, []*TrackLocalStaticSample{vp8Track})
- closePairNow(t, pcFirstOfferer, pcSecondOfferer)
- }
- // Assert that renegotiation doesn't attempt to gather ICE twice
- // Before we would attempt to gather multiple times and would put
- // the PeerConnection into a broken state
- func TestPeerConnection_Renegotiation_Trickle(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- settingEngine := SettingEngine{}
- api := NewAPI(WithSettingEngine(settingEngine))
- assert.NoError(t, api.mediaEngine.RegisterDefaultCodecs())
- // Invalid STUN server on purpose, will stop ICE Gathering from completing in time
- pcOffer, pcAnswer, err := api.newPair(Configuration{
- ICEServers: []ICEServer{
- {
- URLs: []string{"stun:127.0.0.1:5000"},
- },
- },
- })
- if err != nil {
- t.Fatal(err)
- }
- _, err = pcOffer.CreateDataChannel("test-channel", nil)
- assert.NoError(t, err)
- var wg sync.WaitGroup
- wg.Add(2)
- pcOffer.OnICECandidate(func(c *ICECandidate) {
- if c != nil {
- assert.NoError(t, pcAnswer.AddICECandidate(c.ToJSON()))
- } else {
- wg.Done()
- }
- })
- pcAnswer.OnICECandidate(func(c *ICECandidate) {
- if c != nil {
- assert.NoError(t, pcOffer.AddICECandidate(c.ToJSON()))
- } else {
- wg.Done()
- }
- })
- negotiate := func() {
- offer, err := pcOffer.CreateOffer(nil)
- assert.NoError(t, err)
- assert.NoError(t, pcAnswer.SetRemoteDescription(offer))
- assert.NoError(t, pcOffer.SetLocalDescription(offer))
- answer, err := pcAnswer.CreateAnswer(nil)
- assert.NoError(t, err)
- assert.NoError(t, pcOffer.SetRemoteDescription(answer))
- assert.NoError(t, pcAnswer.SetLocalDescription(answer))
- }
- negotiate()
- negotiate()
- pcOffer.ops.Done()
- pcAnswer.ops.Done()
- wg.Wait()
- closePairNow(t, pcOffer, pcAnswer)
- }
- func TestPeerConnection_Renegotiation_SetLocalDescription(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- onTrackFired, onTrackFiredFunc := context.WithCancel(context.Background())
- pcOffer.OnTrack(func(track *TrackRemote, r *RTPReceiver) {
- onTrackFiredFunc()
- })
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- pcOffer.ops.Done()
- pcAnswer.ops.Done()
- _, err = pcOffer.AddTransceiverFromKind(RTPCodecTypeVideo, RTPTransceiverInit{Direction: RTPTransceiverDirectionRecvonly})
- assert.NoError(t, err)
- localTrack, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "foo", "bar")
- assert.NoError(t, err)
- sender, err := pcAnswer.AddTrack(localTrack)
- assert.NoError(t, err)
- offer, err := pcOffer.CreateOffer(nil)
- assert.NoError(t, err)
- assert.NoError(t, pcOffer.SetLocalDescription(offer))
- assert.NoError(t, pcAnswer.SetRemoteDescription(offer))
- assert.False(t, sender.isNegotiated())
- answer, err := pcAnswer.CreateAnswer(nil)
- assert.NoError(t, err)
- assert.True(t, sender.isNegotiated())
- pcAnswer.ops.Done()
- assert.Equal(t, 0, len(localTrack.rtpTrack.bindings))
- assert.NoError(t, pcAnswer.SetLocalDescription(answer))
- pcAnswer.ops.Done()
- assert.Equal(t, 1, len(localTrack.rtpTrack.bindings))
- assert.NoError(t, pcOffer.SetRemoteDescription(answer))
- sendVideoUntilDone(onTrackFired.Done(), t, []*TrackLocalStaticSample{localTrack})
- closePairNow(t, pcOffer, pcAnswer)
- }
- // Issue #346, don't start the SCTP Subsystem if the RemoteDescription doesn't contain one
- // Before we would always start it, and re-negotiations would fail because SCTP was in flight
- func TestPeerConnection_Renegotiation_NoApplication(t *testing.T) {
- signalPairExcludeDataChannel := func(pcOffer, pcAnswer *PeerConnection) {
- offer, err := pcOffer.CreateOffer(nil)
- assert.NoError(t, err)
- offerGatheringComplete := GatheringCompletePromise(pcOffer)
- assert.NoError(t, pcOffer.SetLocalDescription(offer))
- <-offerGatheringComplete
- assert.NoError(t, pcAnswer.SetRemoteDescription(*pcOffer.LocalDescription()))
- answer, err := pcAnswer.CreateAnswer(nil)
- assert.NoError(t, err)
- answerGatheringComplete := GatheringCompletePromise(pcAnswer)
- assert.NoError(t, pcAnswer.SetLocalDescription(answer))
- <-answerGatheringComplete
- assert.NoError(t, pcOffer.SetRemoteDescription(*pcAnswer.LocalDescription()))
- }
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- pcOfferConnected, pcOfferConnectedCancel := context.WithCancel(context.Background())
- pcOffer.OnICEConnectionStateChange(func(i ICEConnectionState) {
- if i == ICEConnectionStateConnected {
- pcOfferConnectedCancel()
- }
- })
- pcAnswerConnected, pcAnswerConnectedCancel := context.WithCancel(context.Background())
- pcAnswer.OnICEConnectionStateChange(func(i ICEConnectionState) {
- if i == ICEConnectionStateConnected {
- pcAnswerConnectedCancel()
- }
- })
- _, err = pcOffer.AddTransceiverFromKind(RTPCodecTypeVideo, RTPTransceiverInit{Direction: RTPTransceiverDirectionSendrecv})
- assert.NoError(t, err)
- _, err = pcAnswer.AddTransceiverFromKind(RTPCodecTypeVideo, RTPTransceiverInit{Direction: RTPTransceiverDirectionSendrecv})
- assert.NoError(t, err)
- signalPairExcludeDataChannel(pcOffer, pcAnswer)
- pcOffer.ops.Done()
- pcAnswer.ops.Done()
- signalPairExcludeDataChannel(pcOffer, pcAnswer)
- pcOffer.ops.Done()
- pcAnswer.ops.Done()
- <-pcAnswerConnected.Done()
- <-pcOfferConnected.Done()
- assert.Equal(t, pcOffer.SCTP().State(), SCTPTransportStateConnecting)
- assert.Equal(t, pcAnswer.SCTP().State(), SCTPTransportStateConnecting)
- closePairNow(t, pcOffer, pcAnswer)
- }
- func TestAddDataChannelDuringRenegotiation(t *testing.T) {
- lim := test.TimeOut(time.Second * 10)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- pcAnswer, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "video", "pion")
- assert.NoError(t, err)
- _, err = pcOffer.AddTrack(track)
- assert.NoError(t, err)
- offer, err := pcOffer.CreateOffer(nil)
- assert.NoError(t, err)
- offerGatheringComplete := GatheringCompletePromise(pcOffer)
- assert.NoError(t, pcOffer.SetLocalDescription(offer))
- <-offerGatheringComplete
- assert.NoError(t, pcAnswer.SetRemoteDescription(*pcOffer.LocalDescription()))
- answer, err := pcAnswer.CreateAnswer(nil)
- assert.NoError(t, err)
- answerGatheringComplete := GatheringCompletePromise(pcAnswer)
- assert.NoError(t, pcAnswer.SetLocalDescription(answer))
- <-answerGatheringComplete
- assert.NoError(t, pcOffer.SetRemoteDescription(*pcAnswer.LocalDescription()))
- _, err = pcOffer.CreateDataChannel("data-channel", nil)
- assert.NoError(t, err)
- // Assert that DataChannel is in offer now
- offer, err = pcOffer.CreateOffer(nil)
- assert.NoError(t, err)
- applicationMediaSectionCount := 0
- for _, d := range offer.parsed.MediaDescriptions {
- if d.MediaName.Media == mediaSectionApplication {
- applicationMediaSectionCount++
- }
- }
- assert.Equal(t, applicationMediaSectionCount, 1)
- onDataChannelFired, onDataChannelFiredFunc := context.WithCancel(context.Background())
- pcAnswer.OnDataChannel(func(*DataChannel) {
- onDataChannelFiredFunc()
- })
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- <-onDataChannelFired.Done()
- closePairNow(t, pcOffer, pcAnswer)
- }
- // Assert that CreateDataChannel fires OnNegotiationNeeded
- func TestNegotiationCreateDataChannel(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pc, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- var wg sync.WaitGroup
- wg.Add(1)
- pc.OnNegotiationNeeded(func() {
- defer func() {
- wg.Done()
- }()
- })
- // Create DataChannel, wait until OnNegotiationNeeded is fired
- if _, err = pc.CreateDataChannel("testChannel", nil); err != nil {
- t.Error(err.Error())
- }
- // Wait until OnNegotiationNeeded is fired
- wg.Wait()
- assert.NoError(t, pc.Close())
- }
- func TestNegotiationNeededRemoveTrack(t *testing.T) {
- var wg sync.WaitGroup
- wg.Add(1)
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- pcAnswer, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "video", "pion")
- assert.NoError(t, err)
- pcOffer.OnNegotiationNeeded(func() {
- wg.Add(1)
- offer, createOfferErr := pcOffer.CreateOffer(nil)
- assert.NoError(t, createOfferErr)
- offerGatheringComplete := GatheringCompletePromise(pcOffer)
- assert.NoError(t, pcOffer.SetLocalDescription(offer))
- <-offerGatheringComplete
- assert.NoError(t, pcAnswer.SetRemoteDescription(*pcOffer.LocalDescription()))
- answer, createAnswerErr := pcAnswer.CreateAnswer(nil)
- assert.NoError(t, createAnswerErr)
- answerGatheringComplete := GatheringCompletePromise(pcAnswer)
- assert.NoError(t, pcAnswer.SetLocalDescription(answer))
- <-answerGatheringComplete
- assert.NoError(t, pcOffer.SetRemoteDescription(*pcAnswer.LocalDescription()))
- wg.Done()
- wg.Done()
- })
- sender, err := pcOffer.AddTrack(track)
- assert.NoError(t, err)
- assert.NoError(t, track.WriteSample(media.Sample{Data: []byte{0x00}, Duration: time.Second}))
- wg.Wait()
- wg.Add(1)
- assert.NoError(t, pcOffer.RemoveTrack(sender))
- wg.Wait()
- closePairNow(t, pcOffer, pcAnswer)
- }
- func TestNegotiationNeededStressOneSided(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcA, pcB, err := newPair()
- assert.NoError(t, err)
- const expectedTrackCount = 500
- ctx, done := context.WithCancel(context.Background())
- pcA.OnNegotiationNeeded(func() {
- count := len(pcA.GetTransceivers())
- assert.NoError(t, signalPair(pcA, pcB))
- if count == expectedTrackCount {
- done()
- }
- })
- for i := 0; i < expectedTrackCount; i++ {
- track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "video", "pion")
- assert.NoError(t, err)
- _, err = pcA.AddTrack(track)
- assert.NoError(t, err)
- }
- <-ctx.Done()
- assert.Equal(t, expectedTrackCount, len(pcB.GetTransceivers()))
- closePairNow(t, pcA, pcB)
- }
- // TestPeerConnection_Renegotiation_DisableTrack asserts that if a remote track is set inactive
- // that locally it goes inactive as well
- func TestPeerConnection_Renegotiation_DisableTrack(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- assert.NoError(t, err)
- // Create two transceivers
- _, err = pcOffer.AddTransceiverFromKind(RTPCodecTypeVideo)
- assert.NoError(t, err)
- transceiver, err := pcOffer.AddTransceiverFromKind(RTPCodecTypeVideo)
- assert.NoError(t, err)
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- // Assert we have three active transceivers
- offer, err := pcOffer.CreateOffer(nil)
- assert.NoError(t, err)
- assert.Equal(t, strings.Count(offer.SDP, "a=sendrecv"), 3)
- // Assert we have two active transceivers, one inactive
- assert.NoError(t, transceiver.Stop())
- offer, err = pcOffer.CreateOffer(nil)
- assert.NoError(t, err)
- assert.Equal(t, strings.Count(offer.SDP, "a=sendrecv"), 2)
- assert.Equal(t, strings.Count(offer.SDP, "a=inactive"), 1)
- // Assert that the offer disabled one of our transceivers
- assert.NoError(t, pcAnswer.SetRemoteDescription(offer))
- answer, err := pcAnswer.CreateAnswer(nil)
- assert.NoError(t, err)
- assert.Equal(t, strings.Count(answer.SDP, "a=sendrecv"), 1) // DataChannel
- assert.Equal(t, strings.Count(answer.SDP, "a=recvonly"), 1)
- assert.Equal(t, strings.Count(answer.SDP, "a=inactive"), 1)
- closePairNow(t, pcOffer, pcAnswer)
- }
- func TestPeerConnection_Renegotiation_Simulcast(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- m := &MediaEngine{}
- if err := m.RegisterDefaultCodecs(); err != nil {
- panic(err)
- }
- registerSimulcastHeaderExtensions(m, RTPCodecTypeVideo)
- originalRids := []string{"a", "b", "c"}
- signalWithRids := func(sessionDescription string, rids []string) string {
- sessionDescription = strings.SplitAfter(sessionDescription, "a=end-of-candidates\r\n")[0]
- sessionDescription = filterSsrc(sessionDescription)
- for _, rid := range rids {
- sessionDescription += "a=" + sdpAttributeRid + ":" + rid + " send\r\n"
- }
- return sessionDescription + "a=simulcast:send " + strings.Join(rids, ";") + "\r\n"
- }
- var trackMapLock sync.RWMutex
- trackMap := map[string]*TrackRemote{}
- onTrackHandler := func(track *TrackRemote, _ *RTPReceiver) {
- trackMapLock.Lock()
- defer trackMapLock.Unlock()
- trackMap[track.RID()] = track
- }
- sendUntilAllTracksFired := func(vp8Writer *TrackLocalStaticRTP, rids []string) {
- allTracksFired := func() bool {
- trackMapLock.Lock()
- defer trackMapLock.Unlock()
- return len(trackMap) == len(rids)
- }
- for sequenceNumber := uint16(0); !allTracksFired(); sequenceNumber++ {
- time.Sleep(20 * time.Millisecond)
- for ssrc, rid := range rids {
- header := &rtp.Header{
- Version: 2,
- SSRC: uint32(ssrc),
- SequenceNumber: sequenceNumber,
- PayloadType: 96,
- }
- assert.NoError(t, header.SetExtension(1, []byte("0")))
- assert.NoError(t, header.SetExtension(2, []byte(rid)))
- _, err := vp8Writer.bindings[0].writeStream.WriteRTP(header, []byte{0x00})
- assert.NoError(t, err)
- }
- }
- }
- assertTracksClosed := func(t *testing.T) {
- trackMapLock.Lock()
- defer trackMapLock.Unlock()
- for _, track := range trackMap {
- _, _, err := track.ReadRTP() // Ignore first Read, this is our peeked data
- assert.Nil(t, err)
- _, _, err = track.ReadRTP()
- assert.Equal(t, err, io.EOF)
- }
- }
- t.Run("Disable Transceiver", func(t *testing.T) {
- trackMap = map[string]*TrackRemote{}
- pcOffer, pcAnswer, err := NewAPI(WithMediaEngine(m)).newPair(Configuration{})
- assert.NoError(t, err)
- vp8Writer, err := NewTrackLocalStaticRTP(RTPCodecCapability{MimeType: MimeTypeVP8}, "video", "pion2")
- assert.NoError(t, err)
- rtpTransceiver, err := pcOffer.AddTransceiverFromTrack(
- vp8Writer,
- RTPTransceiverInit{
- Direction: RTPTransceiverDirectionSendonly,
- },
- )
- assert.NoError(t, err)
- assert.NoError(t, signalPairWithModification(pcOffer, pcAnswer, func(sessionDescription string) string {
- return signalWithRids(sessionDescription, originalRids)
- }))
- pcAnswer.OnTrack(onTrackHandler)
- sendUntilAllTracksFired(vp8Writer, originalRids)
- assert.NoError(t, pcOffer.RemoveTrack(rtpTransceiver.Sender()))
- assert.NoError(t, signalPairWithModification(pcOffer, pcAnswer, func(sessionDescription string) string {
- sessionDescription = strings.SplitAfter(sessionDescription, "a=end-of-candidates\r\n")[0]
- return sessionDescription
- }))
- assertTracksClosed(t)
- closePairNow(t, pcOffer, pcAnswer)
- })
- t.Run("Change RID", func(t *testing.T) {
- trackMap = map[string]*TrackRemote{}
- pcOffer, pcAnswer, err := NewAPI(WithMediaEngine(m)).newPair(Configuration{})
- assert.NoError(t, err)
- vp8Writer, err := NewTrackLocalStaticRTP(RTPCodecCapability{MimeType: MimeTypeVP8}, "video", "pion2")
- assert.NoError(t, err)
- _, err = pcOffer.AddTransceiverFromTrack(
- vp8Writer,
- RTPTransceiverInit{
- Direction: RTPTransceiverDirectionSendonly,
- },
- )
- assert.NoError(t, err)
- assert.NoError(t, signalPairWithModification(pcOffer, pcAnswer, func(sessionDescription string) string {
- return signalWithRids(sessionDescription, originalRids)
- }))
- pcAnswer.OnTrack(onTrackHandler)
- sendUntilAllTracksFired(vp8Writer, originalRids)
- newRids := []string{"d", "e", "f"}
- assert.NoError(t, signalPairWithModification(pcOffer, pcAnswer, func(sessionDescription string) string {
- scanner := bufio.NewScanner(strings.NewReader(sessionDescription))
- sessionDescription = ""
- for scanner.Scan() {
- l := scanner.Text()
- if strings.HasPrefix(l, "a=rid") || strings.HasPrefix(l, "a=simulcast") {
- continue
- }
- sessionDescription += l + "\n"
- }
- return signalWithRids(sessionDescription, newRids)
- }))
- assertTracksClosed(t)
- closePairNow(t, pcOffer, pcAnswer)
- })
- }
- func TestPeerConnection_Regegotiation_ReuseTransceiver(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- vp8Track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "foo", "bar")
- assert.NoError(t, err)
- sender, err := pcOffer.AddTrack(vp8Track)
- assert.NoError(t, err)
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- assert.Equal(t, len(pcOffer.GetTransceivers()), 1)
- assert.Equal(t, pcOffer.GetTransceivers()[0].getCurrentDirection(), RTPTransceiverDirectionSendonly)
- assert.NoError(t, pcOffer.RemoveTrack(sender))
- assert.Equal(t, pcOffer.GetTransceivers()[0].getCurrentDirection(), RTPTransceiverDirectionSendonly)
- // should not reuse tranceiver
- vp8Track2, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "foo", "bar")
- assert.NoError(t, err)
- sender2, err := pcOffer.AddTrack(vp8Track2)
- assert.NoError(t, err)
- assert.Equal(t, len(pcOffer.GetTransceivers()), 2)
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- assert.True(t, sender2.rtpTransceiver == pcOffer.GetTransceivers()[1])
- // should reuse first transceiver
- sender, err = pcOffer.AddTrack(vp8Track)
- assert.NoError(t, err)
- assert.Equal(t, len(pcOffer.GetTransceivers()), 2)
- assert.True(t, sender.rtpTransceiver == pcOffer.GetTransceivers()[0])
- closePairNow(t, pcOffer, pcAnswer)
- }
- func TestPeerConnection_Renegotiation_MidConflict(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- offerPC, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- answerPC, err := NewPeerConnection(Configuration{})
- assert.NoError(t, err)
- _, err = offerPC.CreateDataChannel("test", nil)
- assert.NoError(t, err)
- _, err = offerPC.AddTransceiverFromKind(RTPCodecTypeVideo, RtpTransceiverInit{Direction: RTPTransceiverDirectionSendonly})
- assert.NoError(t, err)
- _, err = offerPC.AddTransceiverFromKind(RTPCodecTypeAudio, RtpTransceiverInit{Direction: RTPTransceiverDirectionSendonly})
- assert.NoError(t, err)
- offer, err := offerPC.CreateOffer(nil)
- assert.NoError(t, err)
- assert.NoError(t, offerPC.SetLocalDescription(offer))
- assert.NoError(t, answerPC.SetRemoteDescription(offer), offer.SDP)
- answer, err := answerPC.CreateAnswer(nil)
- assert.NoError(t, err)
- assert.NoError(t, answerPC.SetLocalDescription(answer))
- assert.NoError(t, offerPC.SetRemoteDescription(answer))
- assert.Equal(t, SignalingStateStable, offerPC.SignalingState())
- tr, err := offerPC.AddTransceiverFromKind(RTPCodecTypeVideo, RtpTransceiverInit{Direction: RTPTransceiverDirectionSendonly})
- assert.NoError(t, err)
- assert.NoError(t, tr.SetMid("3"))
- _, err = offerPC.AddTransceiverFromKind(RTPCodecTypeVideo, RtpTransceiverInit{Direction: RTPTransceiverDirectionSendrecv})
- assert.NoError(t, err)
- _, err = offerPC.CreateOffer(nil)
- assert.NoError(t, err)
- assert.NoError(t, offerPC.Close())
- assert.NoError(t, answerPC.Close())
- }
- func TestPeerConnection_Regegotiation_AnswerAddsTrack(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- tracksCh := make(chan *TrackRemote)
- pcOffer.OnTrack(func(track *TrackRemote, r *RTPReceiver) {
- tracksCh <- track
- for {
- if _, _, readErr := track.ReadRTP(); errors.Is(readErr, io.EOF) {
- return
- }
- }
- })
- vp8Track, err := NewTrackLocalStaticSample(RTPCodecCapability{MimeType: MimeTypeVP8}, "foo", "bar")
- assert.NoError(t, err)
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- _, err = pcOffer.AddTransceiverFromKind(RTPCodecTypeVideo, RtpTransceiverInit{
- Direction: RTPTransceiverDirectionRecvonly,
- })
- assert.NoError(t, err)
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- _, err = pcAnswer.AddTransceiverFromKind(RTPCodecTypeVideo, RtpTransceiverInit{
- Direction: RTPTransceiverDirectionSendonly,
- })
- assert.NoError(t, err)
- assert.NoError(t, err)
- _, err = pcAnswer.AddTrack(vp8Track)
- assert.NoError(t, err)
- assert.NoError(t, signalPair(pcOffer, pcAnswer))
- ctx, cancel := context.WithCancel(context.Background())
- go sendVideoUntilDone(ctx.Done(), t, []*TrackLocalStaticSample{vp8Track})
- <-tracksCh
- cancel()
- closePairNow(t, pcOffer, pcAnswer)
- }
- func TestNegotiationNeededWithRecvonlyTrack(t *testing.T) {
- lim := test.TimeOut(time.Second * 30)
- defer lim.Stop()
- report := test.CheckRoutines(t)
- defer report()
- pcOffer, pcAnswer, err := newPair()
- if err != nil {
- t.Fatal(err)
- }
- var wg sync.WaitGroup
- wg.Add(1)
- pcAnswer.OnNegotiationNeeded(wg.Done)
- _, err = pcOffer.AddTransceiverFromKind(RTPCodecTypeVideo, RTPTransceiverInit{Direction: RTPTransceiverDirectionRecvonly})
- if err != nil {
- t.Fatal(err)
- }
- if err := signalPair(pcOffer, pcAnswer); err != nil {
- t.Fatal(err)
- }
- onDataChannel, onDataChannelCancel := context.WithCancel(context.Background())
- pcAnswer.OnDataChannel(func(d *DataChannel) {
- onDataChannelCancel()
- })
- <-onDataChannel.Done()
- wg.Wait()
- closePairNow(t, pcOffer, pcAnswer)
- }
|