forked from StarryPy/StarryPy-Python2-Deprecated
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpacket_stream.py
90 lines (80 loc) · 3.18 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
import logging
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:
self.logger.exception("Unknown error in start_packet.", exc_info=True)
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._stream[:self.packet_size], 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.warning("Decompression error in check_packet.")
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:
self.logger.exception("Unknown error in check_packet", exc_info=True)
def reset(self):
self.id = None
self.payload_size = None
self.packet_size = None
self.compressed = False