test_datagram.py 5.7 KB

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