Browse Source

Improve BattlEye RCon client

pull/8/head 2.0.3
Richard Neumann 3 years ago
parent
commit
87f3c19787
  1. 45
      rcon/battleye/client.py
  2. 56
      rcon/battleye/proto.py

45
rcon/battleye/client.py

@ -4,8 +4,13 @@ from ipaddress import IPv4Address
from socket import SOCK_DGRAM from socket import SOCK_DGRAM
from typing import Union from typing import Union
from rcon.battleye.proto import Command, LoginRequest from rcon.battleye.proto import Command
from rcon.battleye.proto import CommandResponse
from rcon.battleye.proto import Header
from rcon.battleye.proto import LoginRequest
from rcon.battleye.proto import LoginResponse
from rcon.client import BaseClient from rcon.client import BaseClient
from rcon.exceptions import WrongPassword
__all__ = ['Client'] __all__ = ['Client']
@ -17,18 +22,36 @@ Host = Union[str, IPv4Address]
class Client(BaseClient, socket_type=SOCK_DGRAM): class Client(BaseClient, socket_type=SOCK_DGRAM):
"""BattlEye RCon client.""" """BattlEye RCon client."""
def communicate(self, data: bytes, *, read: int = 4096) -> bytes: def send(self, data: bytes) -> None:
"""Sends and receives packets.""" """Sends bytes."""
self._socket.send(data) with self._socket.makefile('wb') as file:
return self._socket.recv(read) file.write(data)
def login(self, passwd: str) -> bytes: def _login(self, login_request: LoginRequest) -> LoginResponse:
"""Logs the user in.""" """Logs the user in."""
return self.communicate(bytes(LoginRequest.from_passwd(passwd))) self.send(bytes(login_request))
with self._socket.makefile('rb') as file:
return LoginResponse.read(file)
def login(self, passwd: str) -> bool:
"""Logs the user in."""
if not self._login(LoginRequest.from_passwd(passwd)).success:
raise WrongPassword()
return True
def _run(self, command: Command) -> CommandResponse:
"""Executes a command."""
self.send(bytes(command))
with self._socket.makefile('rb') as file:
header = Header.read(file)
# TODO: Can we determine the packet size?
remainder = self._socket.recv(4096)
return CommandResponse.from_bytes(header, remainder)
def run(self, command: str, *args: str) -> str: def run(self, command: str, *args: str) -> str:
"""Executes a command.""" """Executes a command."""
packet = Command.from_command(command, *args) return self._run(Command.from_command(command, *args)).message
_ = self.communicate(bytes(packet))
# TODO: Process response
return ''

56
rcon/battleye/proto.py

@ -4,7 +4,13 @@ from typing import IO, NamedTuple
from zlib import crc32 from zlib import crc32
__all__ = ['LoginRequest', 'Command'] __all__ = [
'Header',
'LoginRequest',
'LoginResponse',
'Command',
'CommandResponse'
]
PREFIX = 'BE' PREFIX = 'BE'
@ -45,7 +51,7 @@ class Header(NamedTuple):
@classmethod @classmethod
def read(cls, file: IO): def read(cls, file: IO):
"""Reads the packet from a socket.""" """Reads the packet from a file-like object."""
return cls.from_bytes(file.read(2), file.read(4), file.read(1)) return cls.from_bytes(file.read(2), file.read(4), file.read(1))
@ -74,6 +80,28 @@ class LoginRequest(NamedTuple):
return cls(0x00, passwd) return cls(0x00, passwd)
class LoginResponse(NamedTuple):
"""A login response."""
header: Header
type: int
success: bool
@classmethod
def from_bytes(cls, header: Header, typ: bytes, success: bytes):
"""Creates a login response from the given bytes."""
return cls(
header,
int.from_bytes(typ, 'little'),
bool(int.from_bytes(success, 'little'))
)
@classmethod
def read(cls, file: IO):
"""Reads a login response from a file-like object."""
return cls.from_bytes(Header.read(file), file.read(1), file.read(1))
class Command(NamedTuple): class Command(NamedTuple):
"""Command packet.""" """Command packet."""
@ -107,3 +135,27 @@ class Command(NamedTuple):
def from_command(cls, command: str, *args: str): def from_command(cls, command: str, *args: str):
"""Creates a command packet from the command and arguments.""" """Creates a command packet from the command and arguments."""
return cls.from_string(' '.join([command, *args])) return cls.from_string(' '.join([command, *args]))
class CommandResponse(NamedTuple):
"""A command response."""
header: Header
type: int
seq: int
payload: bytes
@classmethod
def from_bytes(cls, header: Header, data: bytes):
"""Creates a command response from the given bytes."""
return cls(
header,
int.from_bytes(data[:1], 'little'),
int.from_bytes(data[1:2], 'little'),
data[2:]
)
@property
def message(self) -> str:
"""Returns the text message."""
return self.payload.decode('ascii')

Loading…
Cancel
Save