-
Notifications
You must be signed in to change notification settings - Fork 0
/
packet_stream.py
124 lines (114 loc) · 4.08 KB
/
packet_stream.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
import logging
import pprint
import zlib
import datetime
import packets
class Packet(object):
def __init__(
self,
packet_id,
payload_size,
data,
original_data,
direction,
compressed=False
):
self.id = packet_id
self.payload_size = payload_size
self.data = data
self.original_data = original_data
self.direction = direction
self.compressed = compressed
class PacketStream(object):
logger = logging.getLogger('starrypy.packet_stream.PacketStream')
def __init__(self, protocol):
self._stream = ''
self.id = None
self.payload_size = None
self.header_length = None
self.ready = False
self.payload = None
self.compressed = False
self.packet_size = None
self.protocol = protocol
self.direction = None
self.last_received_timestamp = datetime.datetime.now()
def __add__(self, other):
self._stream += other
try:
self.start_packet()
self.check_packet()
except:
pass
finally:
self.last_received_timestamp = datetime.datetime.now()
return self
def start_packet(self):
try:
if len(self._stream) > 2 and self.payload_size is None:
packet_header = packets.start_packet().parse(self._stream)
self.id = packet_header.id
self.payload_size = abs(packet_header.payload_size)
if packet_header.payload_size < 0:
self.compressed = True
else:
self.compressed = False
self.header_length = 1 + len(packets.SignedVLQ('').build(
packet_header.payload_size)
)
self.packet_size = self.payload_size + self.header_length
return True
except RuntimeError:
self.logger.error('Unknown error in start_packet.')
return False
def check_packet(self):
try:
if (
self.packet_size is not None and
len(self._stream) >= self.packet_size
):
p = self._stream[:self.packet_size]
self._stream = self._stream[self.packet_size:]
if not self._stream:
self._stream = ''
p_parsed = packets.packet().parse(p)
if self.compressed:
try:
z = zlib.decompressobj()
p_parsed.data = z.decompress(p_parsed.data)
except zlib.error:
self.logger.error(
'Decompression error in check_packet.'
)
self.logger.debug('Parsed packet:')
self.logger.debug(pprint.pformat(p_parsed))
self.logger.debug('Packet data:')
self.logger.debug(
pprint.pformat(
p_parsed.original_data.encode('hex')
)
)
self.logger.debug('Following packet data:')
self.logger.debug(
pprint.pformat(self._stream.encode('hex'))
)
raise
packet = Packet(
packet_id=p_parsed.id,
payload_size=p_parsed.payload_size,
data=p_parsed.data,
original_data=p,
direction=self.direction
)
self.compressed = False
self.protocol.string_received(packet)
self.reset()
if self.start_packet():
self.check_packet()
except RuntimeError:
self.logger.error('Unknown error in check_packet')
def reset(self):
self.id = None
self.payload_size = None
self.packet_size = None
self.compressed = False