|
@ -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 '' |
|
|
|
|
|