2018-05-10 15:22:19 +03:00
|
|
|
import struct
|
2019-03-10 03:00:11 +03:00
|
|
|
import random
|
|
|
|
import os
|
2018-05-10 15:22:19 +03:00
|
|
|
|
2019-06-16 16:01:05 +03:00
|
|
|
from .basecodec import BaseCodec
|
2018-05-10 15:22:19 +03:00
|
|
|
|
|
|
|
|
2019-06-16 16:01:05 +03:00
|
|
|
class IntermediateCodec(BaseCodec):
|
|
|
|
"""
|
|
|
|
Intermediate mode between `FullCodec` and `AbridgedCodec`.
|
|
|
|
Always sends 4 extra bytes for the packet length.
|
|
|
|
"""
|
|
|
|
@staticmethod
|
|
|
|
def header_length():
|
|
|
|
return 4
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def tag():
|
|
|
|
return b'\xee\xee\xee\xee' # same as obfuscate tag
|
2019-03-10 03:00:11 +03:00
|
|
|
|
|
|
|
def encode_packet(self, data):
|
|
|
|
return struct.pack('<i', len(data)) + data
|
|
|
|
|
2019-06-16 16:01:05 +03:00
|
|
|
def decode_header(self, header):
|
|
|
|
return struct.unpack('<i', header)[0]
|
2019-03-10 03:00:11 +03:00
|
|
|
|
|
|
|
|
2019-06-16 16:01:05 +03:00
|
|
|
class RandomizedIntermediateCodec(IntermediateCodec):
|
2019-03-10 03:00:11 +03:00
|
|
|
"""
|
2019-06-16 16:01:05 +03:00
|
|
|
Data packets are aligned to 4 bytes. This codec adds random
|
|
|
|
bytes of size from 0 to 3 bytes, which are ignored by decoder.
|
2019-03-10 03:00:11 +03:00
|
|
|
"""
|
2019-03-12 03:12:55 +03:00
|
|
|
tag = None
|
|
|
|
obfuscate_tag = b'\xdd\xdd\xdd\xdd'
|
2019-03-10 03:00:11 +03:00
|
|
|
|
|
|
|
def encode_packet(self, data):
|
|
|
|
pad_size = random.randint(0, 3)
|
|
|
|
padding = os.urandom(pad_size)
|
|
|
|
return super().encode_packet(data + padding)
|
|
|
|
|
|
|
|
async def read_packet(self, reader):
|
2019-06-16 16:01:05 +03:00
|
|
|
raise NotImplementedError(':)')
|
2019-03-10 03:00:11 +03:00
|
|
|
packet_with_padding = await super().read_packet(reader)
|
|
|
|
pad_size = len(packet_with_padding) % 4
|
|
|
|
if pad_size > 0:
|
|
|
|
return packet_with_padding[:-pad_size]
|
|
|
|
return packet_with_padding
|