mirror of
https://github.com/livekit/livekit.git
synced 2026-04-04 16:55:45 +00:00
* Remove `Head` field from `ExtPacket` structure. Although we do not intend to, but if packets get out-of-order in the forwarding path (maybe reading in multiple goroutines or using some worker pool to distribute packets), the `Head` indicator could lead to wrong behaviour. It is possible that at the receiver, the order is - Seq Num N, Head = true - N + 1, Head = true If the forwarding path sees `N + 1` first, the Head flag when it sees `N` packet is incorrect and will lead to incorrect behaviour. The alternative check is very simple. So, remove `Head` flag. * Remove unused field
130 lines
4.0 KiB
Go
130 lines
4.0 KiB
Go
package buffer
|
|
|
|
import (
|
|
"fmt"
|
|
"math/rand"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/pion/rtp"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func getPacket(sn uint16, ts uint32, payloadSize int) *rtp.Packet {
|
|
return &rtp.Packet{
|
|
Header: rtp.Header{
|
|
SequenceNumber: sn,
|
|
Timestamp: ts,
|
|
},
|
|
Payload: make([]byte, payloadSize),
|
|
}
|
|
}
|
|
|
|
func TestRTPStats(t *testing.T) {
|
|
clockRate := uint32(90000)
|
|
r := NewRTPStats(RTPStatsParams{
|
|
ClockRate: clockRate,
|
|
})
|
|
|
|
totalDuration := 5 * time.Second
|
|
bitrate := 1000000
|
|
packetSize := 1000
|
|
pps := (((bitrate + 7) / 8) + packetSize - 1) / packetSize
|
|
framerate := 30
|
|
sleep := 1000 / framerate
|
|
packetsPerFrame := (pps + framerate - 1) / framerate
|
|
|
|
sequenceNumber := uint16(rand.Float64() * float64(1<<16))
|
|
timestamp := uint32(rand.Float64() * float64(1<<32))
|
|
now := time.Now()
|
|
startTime := now
|
|
lastFrameTime := now
|
|
for now.Sub(startTime) < totalDuration {
|
|
timestamp += uint32(now.Sub(lastFrameTime).Seconds() * float64(clockRate))
|
|
for i := 0; i < packetsPerFrame; i++ {
|
|
packet := getPacket(sequenceNumber, timestamp, packetSize)
|
|
r.Update(&packet.Header, len(packet.Payload), 0, time.Now().UnixNano())
|
|
if (sequenceNumber % 100) == 0 {
|
|
jump := uint16(rand.Float64() * 120.0)
|
|
sequenceNumber += jump
|
|
} else {
|
|
sequenceNumber++
|
|
}
|
|
}
|
|
|
|
lastFrameTime = now
|
|
time.Sleep(time.Duration(sleep) * time.Millisecond)
|
|
now = time.Now()
|
|
}
|
|
|
|
r.Stop()
|
|
fmt.Printf("%s\n", r.ToString())
|
|
}
|
|
|
|
func TestRTPStats_Update(t *testing.T) {
|
|
clockRate := uint32(90000)
|
|
r := NewRTPStats(RTPStatsParams{
|
|
ClockRate: clockRate,
|
|
})
|
|
|
|
sequenceNumber := uint16(rand.Float64() * float64(1<<16))
|
|
timestamp := uint32(rand.Float64() * float64(1<<32))
|
|
packet := getPacket(sequenceNumber, timestamp, 1000)
|
|
flowState := r.Update(&packet.Header, len(packet.Payload), 0, time.Now().UnixNano())
|
|
require.False(t, flowState.HasLoss)
|
|
require.True(t, r.initialized)
|
|
require.Equal(t, sequenceNumber, r.highestSN)
|
|
require.Equal(t, timestamp, r.highestTS)
|
|
|
|
// in-order, no loss
|
|
sequenceNumber++
|
|
timestamp += 3000
|
|
packet = getPacket(sequenceNumber, timestamp, 1000)
|
|
flowState = r.Update(&packet.Header, len(packet.Payload), 0, time.Now().UnixNano())
|
|
require.False(t, flowState.HasLoss)
|
|
require.Equal(t, sequenceNumber, r.highestSN)
|
|
require.Equal(t, timestamp, r.highestTS)
|
|
|
|
// out-of-order
|
|
packet = getPacket(sequenceNumber-10, timestamp-30000, 1000)
|
|
flowState = r.Update(&packet.Header, len(packet.Payload), 0, time.Now().UnixNano())
|
|
require.False(t, flowState.HasLoss)
|
|
require.Equal(t, sequenceNumber, r.highestSN)
|
|
require.Equal(t, timestamp, r.highestTS)
|
|
require.Equal(t, uint32(1), r.packetsOutOfOrder)
|
|
require.Equal(t, uint32(0), r.packetsDuplicate)
|
|
|
|
// duplicate
|
|
packet = getPacket(sequenceNumber-10, timestamp-30000, 1000)
|
|
flowState = r.Update(&packet.Header, len(packet.Payload), 0, time.Now().UnixNano())
|
|
require.False(t, flowState.HasLoss)
|
|
require.Equal(t, sequenceNumber, r.highestSN)
|
|
require.Equal(t, timestamp, r.highestTS)
|
|
require.Equal(t, uint32(2), r.packetsOutOfOrder)
|
|
require.Equal(t, uint32(1), r.packetsDuplicate)
|
|
|
|
// loss
|
|
sequenceNumber += 10
|
|
timestamp += 30000
|
|
packet = getPacket(sequenceNumber, timestamp, 1000)
|
|
flowState = r.Update(&packet.Header, len(packet.Payload), 0, time.Now().UnixNano())
|
|
require.True(t, flowState.HasLoss)
|
|
require.Equal(t, sequenceNumber-9, flowState.LossStartInclusive)
|
|
require.Equal(t, sequenceNumber, flowState.LossEndExclusive)
|
|
require.Equal(t, uint32(17), r.packetsLost)
|
|
|
|
// out-of-order should decrement number of lost packets
|
|
packet = getPacket(sequenceNumber-15, timestamp-45000, 1000)
|
|
flowState = r.Update(&packet.Header, len(packet.Payload), 0, time.Now().UnixNano())
|
|
require.False(t, flowState.HasLoss)
|
|
require.Equal(t, sequenceNumber, r.highestSN)
|
|
require.Equal(t, timestamp, r.highestTS)
|
|
require.Equal(t, uint32(3), r.packetsOutOfOrder)
|
|
require.Equal(t, uint32(1), r.packetsDuplicate)
|
|
require.Equal(t, uint32(16), r.packetsLost)
|
|
_, _, _, _, packetsLost, _ := r.getIntervalStats(uint16(r.extStartSN), uint16(r.getExtHighestSN()+1))
|
|
require.Equal(t, uint32(16), packetsLost)
|
|
|
|
r.Stop()
|
|
}
|