forked from LiamBindle/PyVESC
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtest.py
426 lines (389 loc) · 15.7 KB
/
test.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
from unittest import TestCase
class TestPacket(TestCase):
def exact_single_frame(self, length):
"""
Simplest test possible. Create a packet, then parse a buffer containing this packet. Size of buffer is exactly
one packet (no excess).
:param length: Number of bytes in payload.
"""
import random
import pyvesc.protocol.packet.codec as vesc_packet
correct_payload_index = None
if length < 256:
correct_payload_index = 2
else:
correct_payload_index = 3
test_payload = bytes(random.getrandbits(8) for i in range(length))
# test framing
packet = vesc_packet.frame(test_payload)
self.assertEqual(len(packet), correct_payload_index + length + 3, "size of packet")
buffer = bytearray(packet)
# test Parser
parsed, consumed = vesc_packet.unframe(buffer)
buffer = buffer[consumed:]
self.assertEqual(parsed, test_payload)
self.assertEqual(len(buffer), 0)
def exact_two_frames(self, length1, length2):
"""
Check that if there is more than one packet in a buffer, that the unpacker will properly unpack the packets.
Size of buffer for this test is exactly two packets.
:param length1: Length of first payload
:param length2: Length of second payload
"""
import random
import pyvesc.protocol.packet.codec as vesc_packet
correct_payload_index1 = None
correct_payload_index2 = None
if length1 < 256:
correct_payload_index1 = 2
else:
correct_payload_index1 = 3
if length2 < 256:
correct_payload_index2 = 2
else:
correct_payload_index2 = 3
test_payload1 = bytes(random.getrandbits(8) for i in range(length1))
test_payload2 = bytes(random.getrandbits(8) for i in range(length2))
# test framing
packet1 = vesc_packet.frame(test_payload1)
packet2 = vesc_packet.frame(test_payload2)
self.assertEqual(len(packet1), correct_payload_index1 + length1 + 3, "size of packet")
self.assertEqual(len(packet2), correct_payload_index2 + length2 + 3, "size of packet")
buffer = bytearray(packet1 + packet2)
# test Parser
parsed, consumed = vesc_packet.unframe(buffer)
buffer = buffer[consumed:]
self.assertEqual(parsed, test_payload1)
self.assertEqual(len(buffer), len(packet2))
parsed, consumed = vesc_packet.unframe(buffer)
buffer = buffer[consumed:]
self.assertEqual(parsed, test_payload2)
self.assertEqual(len(buffer), 0)
def parse_buffer(self, length):
import random
import pyvesc.protocol.packet.codec as vesc_packet
correct_payload_index = None
if length < 256:
correct_payload_index = 2
else:
correct_payload_index = 3
test_payload = bytes(random.getrandbits(8) for i in range(length))
packet = vesc_packet.frame(test_payload)
# test on small buffers
for n in range(0, 5):
in_buffer = bytearray(packet[:n])
parsed, consumed = vesc_packet.unframe(in_buffer)
out_buffer = in_buffer[consumed:]
self.assertEqual(parsed, None)
self.assertEqual(in_buffer, out_buffer)
# test on buffer almost big enough
for n in range(len(packet) - 4, len(packet)):
in_buffer = bytearray(packet[:n])
parsed, consumed = vesc_packet.unframe(in_buffer)
out_buffer = in_buffer[consumed:]
self.assertEqual(parsed, None)
self.assertEqual(in_buffer, out_buffer)
# test on buffer slightly too big
extension = b'\x02\x04\x07'
extended_packet = packet + b'\x02\x04\x07'
for n in range(len(packet) + 1, len(packet) + 4):
in_buffer = bytearray(extended_packet[:n])
parsed, consumed = vesc_packet.unframe(in_buffer)
out_buffer = in_buffer[consumed:]
self.assertEqual(parsed, test_payload)
self.assertEqual(out_buffer, extension[:n - len(packet)])
def test_small_packets(self):
for length in range(1, 5):
self.exact_single_frame(length)
self.exact_two_frames(length, length + 2)
self.parse_buffer(length)
def test_med_packets(self):
for length in range(254, 258):
self.exact_single_frame(length)
self.exact_two_frames(length, length - 23)
self.parse_buffer(length)
def test_large_packets(self):
for length in range(1022, 1024):
self.exact_single_frame(length)
self.exact_two_frames(length, length + 100)
self.parse_buffer(length)
def test_corrupt_detection(self):
import pyvesc.protocol.packet.codec as vesc_packet
# make a good packet
test_payload = b'Te!'
good_packet = b'\x02\x03Te!B\x92\x03'
corrupt_packets = []
# corrupt first byte
corrupt = b'\x01\x03Te!B\x92\x03'
corrupt_packets.append(corrupt)
# corrupt payload_length (to be smaller and larger)
smaller_corrupt = b'\x02\x02Te!B\x92\x03'
larger_corrupt = b'\x02\x04Te!B\x92\x03\x03'
corrupt_packets.append(smaller_corrupt)
corrupt_packets.append(larger_corrupt)
# corrupt first byte in payload
corrupt = b'\x02\x03se!B\x92\x03'
corrupt_packets.append(corrupt)
# corrupt last byte in payload
corrupt = b'\x02\x03TeyB\x92\x03'
corrupt_packets.append(corrupt)
# corrupt crc
corrupt = b'\x02\x03Te!\xaa\x91\x03'
corrupt_packets.append(corrupt)
# corrupt terminator
corrupt = b'\x02\x03Te!B\x92\x09'
corrupt_packets.append(corrupt)
# check that exceptions are given on each corrupt packet
for corrupt in corrupt_packets:
in_buffer = bytearray(corrupt)
parsed, consumed = vesc_packet.unframe(in_buffer)
out_buffer = in_buffer[consumed:]
self.assertEqual(parsed, None)
self.assertTrue(consumed > 0) # if a packet is corrupt then at least something should be consumed
# get correct out_cuffer (in all of these cases it is just consuming to the next valid start byte (no more no less)
self.assertEqual(consumed, vesc_packet.Stateless._next_possible_packet_index(in_buffer))
# check that the good packet is parsed
in_buffer = bytearray(good_packet)
parsed, consumed = vesc_packet.unframe(in_buffer)
out_buffer = in_buffer[consumed:]
self.assertEqual(parsed, test_payload)
self.assertEqual(out_buffer, b'')
def test_corrupt_recovery(self):
import pyvesc.protocol.packet.codec as vesc_packet
# make a good packet
test_payload = b'Te!'
good_packet = b'\x02\x03Te!B\x92\x03'
packet_to_recover = b'\x02\x04!\xe1$ 8\xbb\x03' # goal is to recover this packet
payload_to_recover = b'!\xe1$ '
after_goal = b'\x05\x09\x01' # mimic another corrupt packet after
corrupt_packets = []
# corrupt first byte
corrupt = b'\x01\x03Te!B\x92\x03'
corrupt_packets.append(corrupt + packet_to_recover + after_goal)
# corrupt payload_length (to be smaller and larger)
smaller_corrupt = b'\x02\x02Te!B\x92\x03'
larger_corrupt = b'\x02\x04Te!B\x92\x03\x03'
corrupt_packets.append(smaller_corrupt + packet_to_recover + after_goal)
corrupt_packets.append(larger_corrupt + packet_to_recover + after_goal)
# corrupt first byte in payload
corrupt = b'\x02\x03se!B\x92\x03'
corrupt_packets.append(corrupt + packet_to_recover + after_goal)
# corrupt last byte in payload
corrupt = b'\x02\x03TeyB\x92\x03'
corrupt_packets.append(corrupt + packet_to_recover + after_goal)
# corrupt crc
corrupt = b'\x02\x03Te!\xaa\x91\x03'
corrupt_packets.append(corrupt + packet_to_recover + after_goal)
# corrupt terminator
corrupt = b'\x02\x03Te!B\x92\x09'
corrupt_packets.append(corrupt + packet_to_recover + after_goal)
# check that exceptions are given on each corrupt packet
for corrupt in corrupt_packets:
in_buffer = bytearray(corrupt)
parsed, consumed = vesc_packet.unframe(in_buffer)
out_buffer = in_buffer[consumed:]
self.assertEqual(parsed, payload_to_recover)
found_packet_start = corrupt.find(packet_to_recover)
self.assertTrue(consumed == (found_packet_start + len(packet_to_recover)))
# check that the good packet is parsed
in_buffer = bytearray(good_packet)
parsed, consumed = vesc_packet.unframe(in_buffer)
out_buffer = in_buffer[consumed:]
self.assertEqual(parsed, test_payload)
self.assertEqual(out_buffer, b'')
class TestMsg(TestCase):
def setUp(self):
import copy
from pyvesc.protocol.base import VESCMessage
self._initial_registry = copy.deepcopy(VESCMessage._msg_registry)
def tearDown(self):
from pyvesc.protocol.base import VESCMessage
VESCMessage._msg_registry = self._initial_registry
self._initial_registry = None
def verify_packing_and_unpacking(self, msg):
from pyvesc.protocol.base import VESCMessage
payload_bytestring = VESCMessage.pack(msg)
parsed_msg = VESCMessage.unpack(payload_bytestring)
self.assertEqual(parsed_msg.id, msg.id)
for name in [names[0] for names in msg.fields]:
self.assertEqual(getattr(parsed_msg, name), getattr(msg, name))
def test_single_message(self):
from pyvesc.protocol.base import VESCMessage
class TestMsg1(metaclass=VESCMessage):
id = 0x12
fields = [
('f1', 'B'),
('f2', 'H'),
('f3', 'i'),
('f4', 'L'),
('f5', 'b'),
('f6', 'I'),
]
test_message = TestMsg1(27, 25367, -1124192846, 2244862237, 17, 73262)
self.verify_packing_and_unpacking(test_message)
def test_multiple_messages(self):
from pyvesc.protocol.base import VESCMessage
class testMsg1(metaclass=VESCMessage):
id = 0x45
fields = [
('f1', 'B'),
('f2', 'H'),
('f3', 'i'),
('f4', 'L'),
('f5', 'b'),
('f6', 'I'),
]
class testMsg2(metaclass=VESCMessage):
id = 0x19
fields = [
('f1', 'B'),
('f2', 'B'),
]
class testMsg3(metaclass=VESCMessage):
id = 0x11
fields = [
('f1', 'i'),
('f2', 'i'),
]
class testMsg4(metaclass=VESCMessage):
id = 0x24
fields = [
('f1', 'i'),
('f2', 's'),
('f3', 'i'),
('f4', 'B'),
('f5', 'i'),
]
test_message1 = testMsg1(27, 25367, -1124192846, 2244862237, 17, 73262)
test_message12 = testMsg1(82, 45132, 382136436, 27374, 18, 72134)
test_message2 = testMsg2(27, 13)
test_message22 = testMsg2(52, 19)
test_message3 = testMsg3(-7841, 4611)
test_message32 = testMsg3(-123, 4123)
test_message4 = testMsg4(4531, 'hello world', 1421, 34, 14215)
self.verify_packing_and_unpacking(test_message1)
self.verify_packing_and_unpacking(test_message2)
self.verify_packing_and_unpacking(test_message3)
self.verify_packing_and_unpacking(test_message4)
def test_errors(self):
from pyvesc.protocol.base import VESCMessage
# try to make two messages with the same ID
class testMsg1(metaclass=VESCMessage):
id = 0x01
fields = [
('f1', 'H'),
('f2', 'H'),
]
caught = False
try:
class testMsg2(metaclass=VESCMessage):
id = 0x01
fields = [
('f1', 'B'),
('f2', 'B'),
]
except TypeError as e:
caught = True
self.assertTrue(caught)
# check that message classes are final
caught = False
try:
class testMsg4(testMsg1):
id = 0x01
fields = [
('f1', 'B'),
('f2', 'B'),
]
except TypeError as e:
caught = True
self.assertTrue(caught)
# check that no more than 1 string field is allowed
caught = False
try:
class testMsg7(metaclass=VESCMessage):
id = 0x02
fields = [
('f1', 's'),
('f2', 's'),
]
except TypeError as e:
caught = True
self.assertTrue(caught)
# check that 's' is used instead of 'p'
caught = False
try:
class testMsg8(metaclass=VESCMessage):
id = 0x31
fields = [
('f1', 'p'),
]
except TypeError as e:
caught = True
self.assertTrue(caught)
# try to fill a message with the wrong number of arguments
caught = False
try:
testmessage1 = testMsg1(2, 4, 5) # should be 2 args
except AttributeError as e:
caught = True
self.assertTrue(caught)
class TestInterface(TestCase):
def setUp(self):
import copy
from pyvesc.protocol.base import VESCMessage
self._initial_registry = copy.deepcopy(VESCMessage._msg_registry)
def tearDown(self):
from pyvesc.protocol.base import VESCMessage
VESCMessage._msg_registry = self._initial_registry
self._initial_registry = None
def verify_encode_decode(self, msg):
import pyvesc
encoded = pyvesc.encode(msg)
decoded, consumed = pyvesc.decode(encoded)
self.assertEqual(consumed, len(encoded))
for field in msg._field_names:
self.assertEqual(getattr(msg, field), getattr(decoded, field))
def test_interface(self):
from pyvesc.VESCMotor.messages import VESCMessage
class testMsg1(metaclass=VESCMessage):
id = 0x45
fields = [
('f1', 'B'),
('f2', 'H'),
('f3', 'i'),
('f4', 'L'),
('f5', 'b'),
('f6', 'I'),
]
class testMsg2(metaclass=VESCMessage):
id = 0x19
fields = [
('f1', 'B'),
('f2', 'B'),
]
class testMsg3(metaclass=VESCMessage):
id = 0x11
fields = [
('f1', 'i'),
('f2', 'i'),
]
class testMsg4(metaclass=VESCMessage):
id = 0x24
fields = [
('f1', 'i'),
('f2', 's'),
('f3', 'i'),
('f4', 'B'),
('f5', 'i'),
]
test_message1 = testMsg1(27, 25367, -1124192846, 2244862237, 17, 73262)
test_message12 = testMsg1(82, 45132, 382136436, 27374, 18, 72134)
test_message2 = testMsg2(27, 13)
test_message22 = testMsg2(52, 19)
test_message3 = testMsg3(-7841, 4611)
test_message32 = testMsg3(-123, 4123)
test_message4 = testMsg4(4531, 'hello world', 1421, 34, 14215)
self.verify_encode_decode(test_message1)
self.verify_encode_decode(test_message2)
self.verify_encode_decode(test_message3)
self.verify_encode_decode(test_message4)