2018-05-10 15:22:19 +03:00
|
|
|
import struct
|
|
|
|
|
2018-09-27 20:22:35 +03:00
|
|
|
from .connection import Connection
|
2018-05-10 15:22:19 +03:00
|
|
|
|
|
|
|
|
2018-09-27 20:22:35 +03:00
|
|
|
class ConnectionTcpAbridged(Connection):
|
2018-05-10 15:22:19 +03:00
|
|
|
"""
|
|
|
|
This is the mode with the lowest overhead, as it will
|
|
|
|
only require 1 byte if the packet length is less than
|
|
|
|
508 bytes (127 << 2, which is very common).
|
|
|
|
"""
|
2018-10-04 18:11:31 +03:00
|
|
|
async def connect(self, timeout=None, ssl=None):
|
|
|
|
await super().connect(timeout=timeout, ssl=ssl)
|
2018-10-04 18:34:58 +03:00
|
|
|
self._writer.write(b'\xef')
|
|
|
|
await self._writer.drain()
|
2018-05-10 15:22:19 +03:00
|
|
|
|
2018-09-27 20:22:35 +03:00
|
|
|
def _write(self, data):
|
|
|
|
"""
|
|
|
|
Define wrapper write methods for `TcpObfuscated` to override.
|
|
|
|
"""
|
|
|
|
self._writer.write(data)
|
2018-05-10 15:22:19 +03:00
|
|
|
|
2018-09-27 20:22:35 +03:00
|
|
|
async def _read(self, n):
|
|
|
|
"""
|
|
|
|
Define wrapper read methods for `TcpObfuscated` to override.
|
|
|
|
"""
|
|
|
|
return await self._reader.readexactly(n)
|
2018-05-10 15:22:19 +03:00
|
|
|
|
2018-09-27 20:22:35 +03:00
|
|
|
def _send(self, data):
|
|
|
|
length = len(data) >> 2
|
2018-05-10 15:22:19 +03:00
|
|
|
if length < 127:
|
|
|
|
length = struct.pack('B', length)
|
|
|
|
else:
|
|
|
|
length = b'\x7f' + int.to_bytes(length, 3, 'little')
|
|
|
|
|
2018-09-27 20:22:35 +03:00
|
|
|
self._write(length + data)
|
|
|
|
|
|
|
|
async def _recv(self):
|
|
|
|
length = struct.unpack('<B', await self._read(1))[0]
|
|
|
|
if length >= 127:
|
|
|
|
length = struct.unpack(
|
|
|
|
'<i', await self._read(3) + b'\0')[0]
|
|
|
|
|
|
|
|
return await self._read(length << 2)
|