mirror of
https://github.com/aler9/rtsp-simple-server
synced 2025-10-15 04:01:15 +08:00
rtmp: rewrite implementation of rtmp connection (#1047)
* rtmp: improve MsgCommandAMF0 * rtmp: fix MsgSetPeerBandwidth * rtmp: add message tests * rtmp: replace implementation with new one * rtmp: rename handshake functions * rtmp: avoid calling useless function * rtmp: use time.Duration for PTSDelta * rtmp: fix decoding chunks with relevant size * rtmp: rewrite implementation of rtmp connection * rtmp: fix tests * rtmp: improve error message * rtmp: replace h264 config implementation * link against github.com/notedit/rtmp * normalize MessageStreamID * rtmp: make acknowledge optional * rtmp: fix decoding of chunk2 + chunk3 * avoid using encoding/binary
This commit is contained in:
@@ -3,6 +3,7 @@ package rawmessage
|
||||
import (
|
||||
"bytes"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
|
||||
@@ -10,151 +11,174 @@ import (
|
||||
"github.com/aler9/rtsp-simple-server/internal/rtmp/chunk"
|
||||
)
|
||||
|
||||
type sequenceEntry struct {
|
||||
chunk chunk.Chunk
|
||||
msg *Message
|
||||
}
|
||||
|
||||
func TestReader(t *testing.T) {
|
||||
testSequence := func(t *testing.T, seq []sequenceEntry) {
|
||||
var buf bytes.Buffer
|
||||
bcr := bytecounter.NewReader(&buf)
|
||||
r := NewReader(bcr, func(count uint32) error {
|
||||
return nil
|
||||
})
|
||||
|
||||
for _, entry := range seq {
|
||||
buf2, err := entry.chunk.Marshal()
|
||||
require.NoError(t, err)
|
||||
buf.Write(buf2)
|
||||
msg, err := r.Read()
|
||||
require.NoError(t, err)
|
||||
require.Equal(t, entry.msg, msg)
|
||||
}
|
||||
type sequenceEntry struct {
|
||||
chunk chunk.Chunk
|
||||
msg *Message
|
||||
}
|
||||
|
||||
t.Run("chunk0 + chunk1", func(t *testing.T) {
|
||||
testSequence(t, []sequenceEntry{
|
||||
{
|
||||
&chunk.Chunk0{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
BodyLen: 64,
|
||||
Body: bytes.Repeat([]byte{0x02}, 64),
|
||||
for _, ca := range []struct {
|
||||
name string
|
||||
sequence []sequenceEntry
|
||||
}{
|
||||
{
|
||||
"chunk0 + chunk1",
|
||||
[]sequenceEntry{
|
||||
{
|
||||
&chunk.Chunk0{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
BodyLen: 64,
|
||||
Body: bytes.Repeat([]byte{0x02}, 64),
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576 * time.Millisecond,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x02}, 64),
|
||||
},
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x02}, 64),
|
||||
{
|
||||
&chunk.Chunk1{
|
||||
ChunkStreamID: 27,
|
||||
TimestampDelta: 15,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
BodyLen: 64,
|
||||
Body: bytes.Repeat([]byte{0x03}, 64),
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: (18576 + 15) * time.Millisecond,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x03}, 64),
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
&chunk.Chunk1{
|
||||
ChunkStreamID: 27,
|
||||
TimestampDelta: 15,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
BodyLen: 64,
|
||||
Body: bytes.Repeat([]byte{0x03}, 64),
|
||||
},
|
||||
{
|
||||
"chunk0 + chunk2 + chunk3",
|
||||
[]sequenceEntry{
|
||||
{
|
||||
&chunk.Chunk0{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
BodyLen: 64,
|
||||
Body: bytes.Repeat([]byte{0x02}, 64),
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576 * time.Millisecond,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x02}, 64),
|
||||
},
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576 + 15,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x03}, 64),
|
||||
{
|
||||
&chunk.Chunk2{
|
||||
ChunkStreamID: 27,
|
||||
TimestampDelta: 15,
|
||||
Body: bytes.Repeat([]byte{0x03}, 64),
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: (18576 + 15) * time.Millisecond,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x03}, 64),
|
||||
},
|
||||
},
|
||||
{
|
||||
&chunk.Chunk3{
|
||||
ChunkStreamID: 27,
|
||||
Body: bytes.Repeat([]byte{0x04}, 64),
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: (18576 + 15 + 15) * time.Millisecond,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x04}, 64),
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
"chunk0 + chunk3 + chunk2 + chunk3",
|
||||
[]sequenceEntry{
|
||||
{
|
||||
&chunk.Chunk0{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
BodyLen: 192,
|
||||
Body: bytes.Repeat([]byte{0x03}, 128),
|
||||
},
|
||||
nil,
|
||||
},
|
||||
{
|
||||
&chunk.Chunk3{
|
||||
ChunkStreamID: 27,
|
||||
Body: bytes.Repeat([]byte{0x03}, 64),
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576 * time.Millisecond,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x03}, 192),
|
||||
},
|
||||
},
|
||||
{
|
||||
&chunk.Chunk2{
|
||||
ChunkStreamID: 27,
|
||||
TimestampDelta: 15,
|
||||
Body: bytes.Repeat([]byte{0x04}, 128),
|
||||
},
|
||||
nil,
|
||||
},
|
||||
{
|
||||
&chunk.Chunk3{
|
||||
ChunkStreamID: 27,
|
||||
Body: bytes.Repeat([]byte{0x04}, 64),
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18591 * time.Millisecond,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x04}, 192),
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
} {
|
||||
t.Run(ca.name, func(t *testing.T) {
|
||||
var buf bytes.Buffer
|
||||
bcr := bytecounter.NewReader(&buf)
|
||||
r := NewReader(bcr, func(count uint32) error {
|
||||
return nil
|
||||
})
|
||||
|
||||
for _, entry := range ca.sequence {
|
||||
buf2, err := entry.chunk.Marshal()
|
||||
require.NoError(t, err)
|
||||
buf.Write(buf2)
|
||||
|
||||
if entry.msg != nil {
|
||||
msg, err := r.Read()
|
||||
require.NoError(t, err)
|
||||
require.Equal(t, entry.msg, msg)
|
||||
}
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
t.Run("chunk0 + chunk2 + chunk3", func(t *testing.T) {
|
||||
testSequence(t, []sequenceEntry{
|
||||
{
|
||||
&chunk.Chunk0{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
BodyLen: 64,
|
||||
Body: bytes.Repeat([]byte{0x02}, 64),
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x02}, 64),
|
||||
},
|
||||
},
|
||||
{
|
||||
&chunk.Chunk2{
|
||||
ChunkStreamID: 27,
|
||||
TimestampDelta: 15,
|
||||
Body: bytes.Repeat([]byte{0x03}, 64),
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576 + 15,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x03}, 64),
|
||||
},
|
||||
},
|
||||
{
|
||||
&chunk.Chunk3{
|
||||
ChunkStreamID: 27,
|
||||
Body: bytes.Repeat([]byte{0x04}, 64),
|
||||
},
|
||||
&Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576 + 15 + 15,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x04}, 64),
|
||||
},
|
||||
},
|
||||
})
|
||||
})
|
||||
|
||||
t.Run("chunk0 + chunk3", func(t *testing.T) {
|
||||
var buf bytes.Buffer
|
||||
bcr := bytecounter.NewReader(&buf)
|
||||
r := NewReader(bcr, func(count uint32) error {
|
||||
return nil
|
||||
})
|
||||
|
||||
buf2, err := chunk.Chunk0{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
BodyLen: 192,
|
||||
Body: bytes.Repeat([]byte{0x03}, 128),
|
||||
}.Marshal()
|
||||
require.NoError(t, err)
|
||||
buf.Write(buf2)
|
||||
|
||||
buf2, err = chunk.Chunk3{
|
||||
ChunkStreamID: 27,
|
||||
Body: bytes.Repeat([]byte{0x03}, 64),
|
||||
}.Marshal()
|
||||
require.NoError(t, err)
|
||||
buf.Write(buf2)
|
||||
|
||||
msg, err := r.Read()
|
||||
require.NoError(t, err)
|
||||
require.Equal(t, &Message{
|
||||
ChunkStreamID: 27,
|
||||
Timestamp: 18576,
|
||||
Type: chunk.MessageTypeSetPeerBandwidth,
|
||||
MessageStreamID: 3123,
|
||||
Body: bytes.Repeat([]byte{0x03}, 192),
|
||||
}, msg)
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestReaderAcknowledge(t *testing.T) {
|
||||
|
Reference in New Issue
Block a user