Files
rtsp-simple-server/internal/protocols/rtmp/rawmessage/reader_test.go

373 lines
8.0 KiB
Go

package rawmessage
import (
"bytes"
"errors"
"io"
"testing"
"time"
"github.com/stretchr/testify/require"
"github.com/bluenviron/mediamtx/internal/protocols/rtmp/bytecounter"
"github.com/bluenviron/mediamtx/internal/protocols/rtmp/chunk"
)
var cases = []struct {
name string
messages []*Message
chunks []chunk.Chunk
}{
{
"(chunk0) + (chunk1)",
[]*Message{
{
ChunkStreamID: 27,
Timestamp: 18576 * time.Millisecond,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{0x03}, 64),
},
{
ChunkStreamID: 27,
Timestamp: (18576 + 15) * time.Millisecond,
Type: 5,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{0x04}, 64),
},
},
[]chunk.Chunk{
&chunk.Chunk0{
ChunkStreamID: 27,
Timestamp: 18576,
Type: 6,
MessageStreamID: 3123,
BodyLen: 64,
Body: bytes.Repeat([]byte{0x03}, 64),
},
&chunk.Chunk1{
ChunkStreamID: 27,
TimestampDelta: 15,
Type: 5,
BodyLen: 64,
Body: bytes.Repeat([]byte{0x04}, 64),
},
},
},
{
"(chunk0) + (chunk2) + (chunk3)",
[]*Message{
{
ChunkStreamID: 27,
Timestamp: 18576 * time.Millisecond,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{0x03}, 64),
},
{
ChunkStreamID: 27,
Timestamp: (18576 + 15) * time.Millisecond,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{0x04}, 64),
},
{
ChunkStreamID: 27,
Timestamp: (18576 + 15 + 15) * time.Millisecond,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{0x05}, 64),
},
},
[]chunk.Chunk{
&chunk.Chunk0{
ChunkStreamID: 27,
Timestamp: 18576,
Type: 6,
MessageStreamID: 3123,
BodyLen: 64,
Body: bytes.Repeat([]byte{0x03}, 64),
},
&chunk.Chunk2{
ChunkStreamID: 27,
TimestampDelta: 15,
Body: bytes.Repeat([]byte{0x04}, 64),
},
&chunk.Chunk3{
ChunkStreamID: 27,
Body: bytes.Repeat([]byte{0x05}, 64),
},
},
},
{
"(chunk0 + chunk3) + (chunk1 + chunk3) + (chunk2 + chunk3) + (chunk3 + chunk3)",
[]*Message{
{
ChunkStreamID: 27,
Timestamp: 18576 * time.Millisecond,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{0x03}, 190),
},
{
ChunkStreamID: 27,
Timestamp: 18576 * time.Millisecond,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{0x04}, 192),
},
{
ChunkStreamID: 27,
Timestamp: (18576 + 15) * time.Millisecond,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{0x05}, 192),
},
{
ChunkStreamID: 27,
Timestamp: (18576 + 15 + 15) * time.Millisecond,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{0x06}, 192),
},
},
[]chunk.Chunk{
&chunk.Chunk0{
ChunkStreamID: 27,
Timestamp: 18576,
Type: 6,
MessageStreamID: 3123,
BodyLen: 190,
Body: bytes.Repeat([]byte{0x03}, 128),
},
&chunk.Chunk3{
ChunkStreamID: 27,
Body: bytes.Repeat([]byte{0x03}, 62),
},
&chunk.Chunk1{
ChunkStreamID: 27,
TimestampDelta: 0,
Type: 6,
BodyLen: 192,
Body: bytes.Repeat([]byte{0x04}, 128),
},
&chunk.Chunk3{
ChunkStreamID: 27,
Body: bytes.Repeat([]byte{0x04}, 64),
},
&chunk.Chunk2{
ChunkStreamID: 27,
TimestampDelta: 15,
Body: bytes.Repeat([]byte{0x05}, 128),
},
&chunk.Chunk3{
ChunkStreamID: 27,
Body: bytes.Repeat([]byte{0x05}, 64),
},
&chunk.Chunk3{
ChunkStreamID: 27,
Body: bytes.Repeat([]byte{0x06}, 128),
},
&chunk.Chunk3{
ChunkStreamID: 27,
Body: bytes.Repeat([]byte{0x06}, 64),
},
},
},
{
"(chunk0 + chunk3 with extended timestamp)",
[]*Message{
{
ChunkStreamID: 27,
Timestamp: 0xFF123456 * time.Millisecond,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{5}, 160),
},
},
[]chunk.Chunk{
&chunk.Chunk0{
ChunkStreamID: 27,
Timestamp: 4279383126,
Type: 6,
MessageStreamID: 3123,
BodyLen: 160,
Body: bytes.Repeat([]byte{5}, 128),
},
&chunk.Chunk3{
ChunkStreamID: 27,
Body: bytes.Repeat([]byte{5}, 32),
},
},
},
}
func TestReader(t *testing.T) {
for _, ca := range cases {
t.Run(ca.name, func(t *testing.T) {
var buf bytes.Buffer
br := bytecounter.NewReader(&buf)
r := NewReader(br, br, func(_ uint32) error {
return nil
})
hasExtendedTimestamp := false
for _, cach := range ca.chunks {
buf2, err := cach.Marshal(hasExtendedTimestamp)
require.NoError(t, err)
buf.Write(buf2)
hasExtendedTimestamp = chunkHasExtendedTimestamp(cach)
}
var msgs []*Message
for {
msg, err := r.Read()
if errors.Is(err, io.EOF) {
break
}
require.NoError(t, err)
msgs = append(msgs, msg)
}
require.Equal(t, ca.messages, msgs)
})
}
}
func TestReaderAdditional(t *testing.T) {
for _, ca := range []struct {
name string
messages []*Message
chunks []chunk.Chunk
}{
{
"(chunk0) + (chunk3)",
[]*Message{
{
ChunkStreamID: 27,
Timestamp: 4279383126000000,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{5}, 15),
},
{
ChunkStreamID: 27,
Timestamp: 4279383126000000,
Type: 6,
MessageStreamID: 3123,
Body: bytes.Repeat([]byte{6}, 15),
},
},
[]chunk.Chunk{
&chunk.Chunk0{
ChunkStreamID: 27,
Timestamp: 4279383126,
Type: 6,
MessageStreamID: 3123,
BodyLen: 15,
Body: bytes.Repeat([]byte{5}, 15),
},
&chunk.Chunk3{
ChunkStreamID: 27,
Body: bytes.Repeat([]byte{6}, 15),
},
},
},
} {
t.Run(ca.name, func(t *testing.T) {
var buf bytes.Buffer
br := bytecounter.NewReader(&buf)
r := NewReader(br, br, func(_ uint32) error {
return nil
})
hasExtendedTimestamp := false
for _, cach := range ca.chunks {
buf2, err := cach.Marshal(hasExtendedTimestamp)
require.NoError(t, err)
buf.Write(buf2)
hasExtendedTimestamp = chunkHasExtendedTimestamp(cach)
}
var msgs []*Message
for {
msg, err := r.Read()
if errors.Is(err, io.EOF) {
break
}
require.NoError(t, err)
msgs = append(msgs, msg)
}
require.Equal(t, ca.messages, msgs)
})
}
}
func TestReaderAcknowledge(t *testing.T) {
for _, ca := range []string{"standard", "overflow"} {
t.Run(ca, func(t *testing.T) {
onAckCalled := make(chan struct{})
var buf bytes.Buffer
bc := bytecounter.NewReader(&buf)
r := NewReader(bc, bc, func(_ uint32) error {
close(onAckCalled)
return nil
})
if ca == "overflow" {
bc.SetCount(4294967096)
r.lastAckCount = 4294967096
}
err := r.SetChunkSize(65536)
require.NoError(t, err)
r.SetWindowAckSize(100)
buf2, err := chunk.Chunk0{
ChunkStreamID: 27,
Timestamp: 18576,
Type: 6,
MessageStreamID: 3123,
BodyLen: 200,
Body: bytes.Repeat([]byte{0x03}, 200),
}.Marshal(false)
require.NoError(t, err)
buf.Write(buf2)
_, err = r.Read()
require.NoError(t, err)
<-onAckCalled
})
}
}
func FuzzReader(f *testing.F) {
f.Fuzz(func(_ *testing.T, b []byte) {
bcr := bytecounter.NewReader(bytes.NewReader(b))
r := NewReader(bcr, bcr, func(_ uint32) error {
return nil
})
var buf bytes.Buffer
bcw := bytecounter.NewWriter(&buf)
w := NewWriter(bcw, bcw, true)
for {
msg, err := r.Read()
if err == nil {
w.Write(msg) //nolint:errcheck
} else {
break
}
}
})
}