diff --git a/steam/client/cm.py b/steam/client/cm.py new file mode 100644 index 0000000..e871f44 --- /dev/null +++ b/steam/client/cm.py @@ -0,0 +1,277 @@ +import struct +import binascii +import logging +import zipfile + +try: + from cStringIO import StringIO +except ImportError: + from StringIO import StringIO + +import gevent +from gevent import event +from gevent import queue +from Crypto.Random import random + +from steam.steamid import SteamID +from steam.enums import EMsg, EResult, EUniverse +from steam.client import crypto +from steam.client.connection import TCPConnection +from steam.client.msg import is_proto, clear_proto_bit +from steam.client.msg import Msg, MsgProto + +server_list = [ + ('162.254.196.41', '27020'), ('162.254.196.40', '27021'), + ('162.254.196.43', '27019'), ('162.254.196.40', '27018'), + ('162.254.196.43', '27020'), ('162.254.196.41', '27019'), + ('162.254.196.41', '27018'), ('162.254.196.42', '27020'), + ('162.254.196.41', '27017'), ('162.254.196.41', '27021'), + ('146.66.152.10', '27017'), ('146.66.152.10', '27018'), + ('146.66.152.11', '27019'), ('146.66.152.11', '27020'), + ('146.66.152.10', '27019'), ('162.254.197.42', '27018'), + ('162.254.197.41', '27019'), ('162.254.197.41', '27017'), + ('208.78.164.14', '27017'), ('208.78.164.14', '27019'), + ('208.78.164.9', '27019'), ('208.78.164.14', '27018'), + ('208.78.164.9', '27018'), ('208.78.164.13', '27017'), +] + +logger = logging.getLogger("CMClient") + + +class CMClient: + TCP = 0 + UDP = 1 + + def __init__(self, protocol=0): + self.reconnect = False + + self._init_attributes() + + self.registered_callbacks = {} + + if protocol == CMClient.TCP: + self.connection = TCPConnection() + # elif protocol == CMClient.UDP: + # self.connection = UDPConnection() + else: + raise ValueError("Only TCP is supported") + + self.event_connected = event.Event() + self.event_ready = event.Event() + self.event_disconnected = event.Event() + + self.register_callback(EMsg.ChannelEncryptRequest, self._handle_encrypt_request), + self.register_callback(EMsg.Multi, self._handle_multi), + self.register_callback(EMsg.ClientLogOnResponse, self._handle_logon), + + def connect(self, reconnect=None): + if reconnect is not None: + self.reconnect = reconnect + + logger.debug("Connect (reconnect=%s)" % self.reconnect) + + while True: + with gevent.Timeout(15, False): + server_addr = random.choice(server_list) + self.connection.connect(server_addr) + + if not self.connection.event_connected.is_set(): + if self.reconnect: + logger.debug("Failed to connect. Retrying...") + continue + + logger.debug("Failed to connect") + return False + break + + logger.debug("Event: Connected") + self.event_connected.set() + self._recv_loop = gevent.spawn(self._recv_messages) + return True + + def disconnect(self): + self.connection.disconnect() + + self._recv_loop.kill(block=False) + self._heartbeat_loop.kill() + + self._init_attributes() + + self.event_connected.clear() + self.event_ready.clear() + self.event_disconnected.set() + + logger.debug("Event: Disconnected") + + def _init_attributes(self): + self.key = None + + self.steam_id = None + self.session_id = None + + self.cell_id = None + self.webapi_nonce = None + + self._recv_loop = None + self._heartbeat_loop = None + + def send_message(self, message): + if not isinstance(message, (Msg, MsgProto)): + raise ValueError("Expected Msg or MsgProto, got %s" % message) + + data = message.serialize() + + if self.key: + data = crypto.encrypt(data, self.key) + + logger.debug("Outgoing: %s", repr(message.msg)) + self.connection.put_message(data) + + def _recv_messages(self): + while True: + try: + message = self.connection.get_message(timeout=1) + except queue.Empty: + if not self.connection.event_connected.is_set(): + self.disconnect() + if self.reconnect: + gevent.spawn(self.connect) + return + continue + + if self.key: + message = crypto.decrypt(message, self.key) + + self._parse_message(message) + + def _parse_message(self, message): + emsg_id, = struct.unpack_from(" 0: + size, = struct.unpack_from(" header_size: + message_length, magic = struct.unpack_from(Connection.FMT, buf) + + if magic != Connection.MAGIC: + raise RuntimeError("invalid magic, got %s" % repr(magic)) + + packet_length = header_size + message_length + + if len(buf) < packet_length: + return + + message = buf[header_size:packet_length] + buf = buf[packet_length:] + + self.recv_queue.put(message) + + self._readbuf = buf + + +class TCPConnection(Connection): + def _new_socket(self): + self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) + + def _connect(self, server_addr): + self.socket.connect(server_addr) + + def _read_data(self): + return self.socket.recv(2048) + + def _write_data(self, data): + self.socket.sendall(data) + + +class UDPConnection(Connection): + def _new_socket(self): + self.socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) + + def _connect(self, server_addr): + pass + + def _read_data(self): + pass + + def _write_data(self, data): + pass diff --git a/steam/client/msg.py b/steam/client/msg.py new file mode 100644 index 0000000..1996cb7 --- /dev/null +++ b/steam/client/msg.py @@ -0,0 +1,265 @@ +import struct +import fnmatch +from steam.enums import EMsg, EUniverse, EResult +from steam.protobufs import steammessages_base_pb2 +from steam.protobufs import steammessages_clientserver_pb2 +from steam.protobufs import steammessages_clientserver_2_pb2 + + +class MsgHdr: + _size = struct.calcsize(" 0 + + +def set_proto_bit(emsg): + return int(emsg) | protobuf_mask + + +def clear_proto_bit(emsg): + return int(emsg) & ~protobuf_mask + + +class MsgHdrProtoBuf: + _size = struct.calcsize(" (3,): + long = int class EasyEnum(Enum): @@ -9,17 +12,20 @@ class EasyEnum(Enum): return self.name def __eq__(self, other): - if isinstance(other, int): + if isinstance(other, (int, long)): return self.value == other else: return NotImplemented def __ne__(self, other): - if isinstance(other, int): + if isinstance(other, (int, long)): return self.value != other else: return NotImplemented + def __call__(self): + return int(self) + class EMsg(EasyEnum): Invalid = 0 @@ -1823,6 +1829,7 @@ class EMsg(EasyEnum): class EResult(EasyEnum): + Invalid = 0 OK = 1 # success Fail = 2 # generic failure NoConnection = 3 # no/failed network connection @@ -1886,6 +1893,7 @@ class EUniverse(EasyEnum): Beta = 2 Internal = 3 Dev = 4 + Max = 5 class EType(EasyEnum): @@ -1900,6 +1908,7 @@ class EType(EasyEnum): Chat = 8 ConsoleUser = 9 AnonUser = 10 + Max = 11 class EServerType(EasyEnum):