mirror of
https://github.com/livekit/livekit.git
synced 2026-03-31 19:45:43 +00:00
* Use media payload size in scoring. Subtract out header bytes when calculating score. This does not seem to affect the score (under perfect conditions), but, using header bytes will inflate the bit rate and will affect scoring. * Add header bytes to ToProto * protocol pointer * fix test
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)
|
|
intervalStats := r.getIntervalStats(uint16(r.extStartSN), uint16(r.getExtHighestSN()+1))
|
|
require.Equal(t, uint32(16), intervalStats.packetsLost)
|
|
|
|
r.Stop()
|
|
}
|