ivfreader_test.go 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. // SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
  2. // SPDX-License-Identifier: MIT
  3. package ivfreader
  4. import (
  5. "bytes"
  6. "io"
  7. "testing"
  8. "github.com/stretchr/testify/assert"
  9. )
  10. // buildIVFContainer takes frames and prepends valid IVF file header
  11. func buildIVFContainer(frames ...*[]byte) *bytes.Buffer {
  12. // Valid IVF file header taken from: https://github.com/webmproject/...
  13. // vp8-test-vectors/blob/master/vp80-00-comprehensive-001.ivf
  14. // Video Image Width - 176
  15. // Video Image Height - 144
  16. // Frame Rate Rate - 30000
  17. // Frame Rate Scale - 1000
  18. // Video Length in Frames - 29
  19. // BitRate: 64.01 kb/s
  20. ivf := []byte{
  21. 0x44, 0x4b, 0x49, 0x46, 0x00, 0x00, 0x20, 0x00,
  22. 0x56, 0x50, 0x38, 0x30, 0xb0, 0x00, 0x90, 0x00,
  23. 0x30, 0x75, 0x00, 0x00, 0xe8, 0x03, 0x00, 0x00,
  24. 0x1d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  25. }
  26. for f := range frames {
  27. ivf = append(ivf, *frames[f]...)
  28. }
  29. return bytes.NewBuffer(ivf)
  30. }
  31. func TestIVFReader_ParseValidFileHeader(t *testing.T) {
  32. assert := assert.New(t)
  33. ivf := buildIVFContainer(&[]byte{})
  34. reader, header, err := NewWith(ivf)
  35. assert.Nil(err, "IVFReader should be created")
  36. assert.NotNil(reader, "Reader shouldn't be nil")
  37. assert.NotNil(header, "Header shouldn't be nil")
  38. assert.Equal("DKIF", header.signature, "signature is 'DKIF'")
  39. assert.Equal(uint16(0), header.version, "version should be 0")
  40. assert.Equal("VP80", header.FourCC, "FourCC should be 'VP80'")
  41. assert.Equal(uint16(176), header.Width, "width should be 176")
  42. assert.Equal(uint16(144), header.Height, "height should be 144")
  43. assert.Equal(uint32(30000), header.TimebaseDenominator, "timebase denominator should be 30000")
  44. assert.Equal(uint32(1000), header.TimebaseNumerator, "timebase numerator should be 1000")
  45. assert.Equal(uint32(29), header.NumFrames, "number of frames should be 29")
  46. assert.Equal(uint32(0), header.unused, "bytes should be unused")
  47. }
  48. func TestIVFReader_ParseValidFrames(t *testing.T) {
  49. assert := assert.New(t)
  50. // Frame Length - 4
  51. // Timestamp - None
  52. // Frame Payload - 0xDEADBEEF
  53. validFrame1 := []byte{
  54. 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  55. 0x00, 0x00, 0x00, 0x00, 0xDE, 0xAD, 0xBE, 0xEF,
  56. }
  57. // Frame Length - 12
  58. // Timestamp - None
  59. // Frame Payload - 0xDEADBEEFDEADBEEF
  60. validFrame2 := []byte{
  61. 0x0C, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  62. 0x00, 0x00, 0x00, 0x00, 0xDE, 0xAD, 0xBE, 0xEF,
  63. 0xDE, 0xAD, 0xBE, 0xEF, 0xDE, 0xAD, 0xBE, 0xEF,
  64. }
  65. ivf := buildIVFContainer(&validFrame1, &validFrame2)
  66. reader, _, err := NewWith(ivf)
  67. assert.Nil(err, "IVFReader should be created")
  68. assert.NotNil(reader, "Reader shouldn't be nil")
  69. // Parse Frame #1
  70. payload, header, err := reader.ParseNextFrame()
  71. assert.Nil(err, "Should have parsed frame #1 without error")
  72. assert.Equal(uint32(4), header.FrameSize, "Frame header frameSize should be 4")
  73. assert.Equal(4, len(payload), "Payload should be length 4")
  74. assert.Equal(
  75. payload,
  76. []byte{
  77. 0xDE, 0xAD, 0xBE, 0xEF,
  78. },
  79. "Payload value should be 0xDEADBEEF")
  80. assert.Equal(int64(ivfFrameHeaderSize+ivfFileHeaderSize+header.FrameSize), reader.bytesReadSuccesfully)
  81. previousBytesRead := reader.bytesReadSuccesfully
  82. // Parse Frame #2
  83. payload, header, err = reader.ParseNextFrame()
  84. assert.Nil(err, "Should have parsed frame #2 without error")
  85. assert.Equal(uint32(12), header.FrameSize, "Frame header frameSize should be 4")
  86. assert.Equal(12, len(payload), "Payload should be length 12")
  87. assert.Equal(
  88. payload,
  89. []byte{
  90. 0xDE, 0xAD, 0xBE, 0xEF, 0xDE, 0xAD,
  91. 0xBE, 0xEF, 0xDE, 0xAD, 0xBE, 0xEF,
  92. },
  93. "Payload value should be 0xDEADBEEFDEADBEEF")
  94. assert.Equal(int64(ivfFrameHeaderSize+header.FrameSize)+previousBytesRead, reader.bytesReadSuccesfully)
  95. }
  96. func TestIVFReader_ParseIncompleteFrameHeader(t *testing.T) {
  97. assert := assert.New(t)
  98. // frame with 11-byte header (missing 1 byte)
  99. incompleteFrame := []byte{
  100. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  101. 0x00, 0x00, 0x00,
  102. }
  103. ivf := buildIVFContainer(&incompleteFrame)
  104. reader, _, err := NewWith(ivf)
  105. assert.Nil(err, "IVFReader should be created")
  106. assert.NotNil(reader, "Reader shouldn't be nil")
  107. // Parse Frame #1
  108. payload, header, err := reader.ParseNextFrame()
  109. assert.Nil(payload, "Payload should be nil")
  110. assert.Nil(header, "Incomplete header should be nil")
  111. assert.Equal(errIncompleteFrameHeader, err)
  112. }
  113. func TestIVFReader_ParseIncompleteFramePayload(t *testing.T) {
  114. assert := assert.New(t)
  115. // frame with header defining frameSize of 4
  116. // but only 2 bytes available (missing 2 bytes)
  117. incompleteFrame := []byte{
  118. 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  119. 0x00, 0x00, 0x00, 0x00, 0xDE, 0xAD,
  120. }
  121. ivf := buildIVFContainer(&incompleteFrame)
  122. reader, _, err := NewWith(ivf)
  123. assert.Nil(err, "IVFReader should be created")
  124. assert.NotNil(reader, "Reader shouldn't be nil")
  125. // Parse Frame #1
  126. payload, header, err := reader.ParseNextFrame()
  127. assert.Nil(payload, "Incomplete payload should be nil")
  128. assert.Nil(header, "Header should be nil")
  129. assert.Equal(errIncompleteFrameData, err)
  130. }
  131. func TestIVFReader_EOFWhenNoFramesLeft(t *testing.T) {
  132. assert := assert.New(t)
  133. ivf := buildIVFContainer(&[]byte{})
  134. reader, _, err := NewWith(ivf)
  135. assert.Nil(err, "IVFReader should be created")
  136. assert.NotNil(reader, "Reader shouldn't be nil")
  137. _, _, err = reader.ParseNextFrame()
  138. assert.Equal(io.EOF, err)
  139. }