test_datagram.py 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. import pytest
  2. from panda3d import core
  3. import tempfile
  4. # Fixtures for generating interesting datagrams (and verification functions) on
  5. # the fly...
  6. @pytest.fixture(scope='module',
  7. params=[False, True],
  8. ids=['stdfloat_float', 'stdfloat_double'])
  9. def datagram_small(request):
  10. """Returns a small datagram, along with a verification function."""
  11. dg = core.Datagram()
  12. dg.set_stdfloat_double(request.param)
  13. dg.add_uint8(3)
  14. dg.add_uint16(14159)
  15. dg.add_uint32(0xDEADBEEF)
  16. dg.add_uint64(0x0123456789ABCDEF)
  17. dg.add_int8(-77)
  18. dg.add_int16(-1)
  19. dg.add_int32(-972965890)
  20. dg.add_int64(-1001001001001001)
  21. dg.add_string('this is a string')
  22. dg.add_string32('this is another string')
  23. dg.add_string('this is yet a third string')
  24. dg.add_blob(b'blob data \x00\xf2\xa0\x00\x00')
  25. dg.add_blob32(b'\xc9\x8f\x00 test blob32')
  26. dg.add_stdfloat(800.2)
  27. dg.add_stdfloat(3.1415926)
  28. dg.add_stdfloat(2.7182818)
  29. def readback_function(dgi):
  30. assert dgi.get_remaining_size() > 0
  31. assert dgi.get_uint8() == 3
  32. assert dgi.get_uint16() == 14159
  33. assert dgi.get_uint32() == 0xDEADBEEF
  34. assert dgi.get_uint64() == 0x0123456789ABCDEF
  35. assert dgi.get_int8() == -77
  36. assert dgi.get_int16() == -1
  37. assert dgi.get_int32() == -972965890
  38. assert dgi.get_int64() == -1001001001001001
  39. assert dgi.get_string() == 'this is a string'
  40. assert dgi.get_string32() == 'this is another string'
  41. assert dgi.get_string() == 'this is yet a third string'
  42. assert dgi.get_blob() == b'blob data \x00\xf2\xa0\x00\x00'
  43. assert dgi.get_blob32() == b'\xc9\x8f\x00 test blob32'
  44. assert dgi.get_stdfloat() == pytest.approx(800.2)
  45. assert dgi.get_stdfloat() == pytest.approx(3.1415926)
  46. assert dgi.get_stdfloat() == pytest.approx(2.7182818)
  47. assert dgi.get_remaining_size() == 0
  48. return dg, readback_function
  49. @pytest.fixture(scope='module')
  50. def datagram_large():
  51. """Returns a big datagram, along with a verification function."""
  52. dg = core.Datagram()
  53. for x in range(2000000):
  54. dg.add_uint32(x)
  55. dg.add_string('the magic words are squeamish ossifrage')
  56. def readback_function(dgi):
  57. assert dgi.get_remaining_size() > 0
  58. for x in range(2000000):
  59. assert dgi.get_uint32() == x
  60. assert dgi.get_string() == 'the magic words are squeamish ossifrage'
  61. assert dgi.get_remaining_size() == 0
  62. return dg, readback_function
  63. def test_datagram_bytes():
  64. """Tests that we can put and get a bytes object on Datagram."""
  65. dg = core.Datagram(b'abc\x00')
  66. dg.append_data(b'\xff123')
  67. assert bytes(dg) == b'abc\x00\xff123'
  68. dgi = core.DatagramIterator(dg)
  69. dgi.get_remaining_bytes() == b'abc\x00\xff123'
  70. def test_datagram_get_message():
  71. dg = core.Datagram(b'abc\x00')
  72. dg.append_data(b'\xff123')
  73. assert dg.get_message() == b'abc\x00\xff123'
  74. def test_datagram_pickle():
  75. import pickle
  76. dg = core.Datagram()
  77. assert pickle.loads(pickle.dumps(dg, -1)) == dg
  78. dg = core.Datagram(b'abc\x00')
  79. assert pickle.loads(pickle.dumps(dg, -1)) == dg
  80. def test_datagram_cow():
  81. dg1 = core.Datagram()
  82. dg1.append_data(b'1234')
  83. dg2 = core.Datagram(dg1)
  84. dg2.append_data(b'5678')
  85. assert dg1.get_message() == b'1234'
  86. assert dg2.get_message() == b'12345678'
  87. def test_iterator(datagram_small):
  88. """This tests Datagram/DatagramIterator, and sort of serves as a self-check
  89. of the test fixtures too."""
  90. dg, verify = datagram_small
  91. dgi = core.DatagramIterator(dg)
  92. verify(dgi)
  93. # This tests the copy constructor:
  94. def test_copy(datagram_small):
  95. dg, verify = datagram_small
  96. dg2 = core.Datagram(dg)
  97. dgi = core.DatagramIterator(dg2)
  98. verify(dgi)
  99. def test_assign(datagram_small):
  100. dg, verify = datagram_small
  101. dg2 = core.Datagram()
  102. dg2.assign(dg)
  103. dgi = core.DatagramIterator(dg2)
  104. verify(dgi)
  105. # These test DatagramInputFile/DatagramOutputFile:
  106. def do_file_test(dg, verify, filename):
  107. dof = core.DatagramOutputFile()
  108. dof.open(filename)
  109. dof.put_datagram(dg)
  110. dof.close()
  111. dg2 = core.Datagram()
  112. dif = core.DatagramInputFile()
  113. dif.open(filename)
  114. assert dif.get_datagram(dg2)
  115. dif.close()
  116. # This is normally saved by the DatagramOutputFile header. We cheat here.
  117. dg2.set_stdfloat_double(dg.get_stdfloat_double())
  118. dgi = core.DatagramIterator(dg2)
  119. verify(dgi)
  120. def test_file_small(datagram_small):
  121. """This tests DatagramOutputFile/DatagramInputFile on small datagrams."""
  122. dg, verify = datagram_small
  123. file = tempfile.NamedTemporaryFile(suffix='.bin')
  124. filename = core.Filename.from_os_specific(file.name)
  125. filename.make_true_case()
  126. do_file_test(dg, verify, filename)
  127. def test_file_large(datagram_large):
  128. """This tests DatagramOutputFile/DatagramInputFile on very large datagrams."""
  129. dg, verify = datagram_large
  130. file = tempfile.NamedTemporaryFile(suffix='.bin')
  131. filename = core.Filename.from_os_specific(file.name)
  132. filename.make_true_case()
  133. do_file_test(dg, verify, filename)
  134. def test_file_corrupt(datagram_small):
  135. """This tests DatagramInputFile's handling of a corrupt size header."""
  136. dg, verify = datagram_small
  137. file = tempfile.NamedTemporaryFile(suffix='.bin')
  138. filename = core.Filename.from_os_specific(file.name)
  139. filename.make_true_case()
  140. dof = core.DatagramOutputFile()
  141. dof.open(filename)
  142. dof.put_datagram(dg)
  143. dof.close()
  144. # Corrupt the size header to 1GB
  145. file.seek(0)
  146. file.write(b'\xFF\xFF\xFF\x4F')
  147. file.flush()
  148. dg2 = core.Datagram()
  149. dif = core.DatagramInputFile()
  150. dif.open(filename)
  151. assert not dif.get_datagram(dg2)
  152. dif.close()
  153. # Truncate the file
  154. for size in [12, 8, 4, 3, 2, 1, 0]:
  155. file.truncate(size)
  156. dg2 = core.Datagram()
  157. dif = core.DatagramInputFile()
  158. dif.open(filename)
  159. assert not dif.get_datagram(dg2)
  160. dif.close()
  161. # Should we test that dg2 is unmodified?