mirror of https://github.com/conqp/rcon
7 changed files with 95 additions and 94 deletions
@ -1,8 +1,8 @@ |
|||||
"""RCON client library.""" |
"""RCON client library.""" |
||||
|
|
||||
from rcon.asyncio import rcon |
from rcon.asyncio import rcon |
||||
|
from rcon.client import Client |
||||
from rcon.exceptions import RequestIdMismatch, WrongPassword |
from rcon.exceptions import RequestIdMismatch, WrongPassword |
||||
from rcon.proto import Client |
|
||||
|
|
||||
|
|
||||
__all__ = ['RequestIdMismatch', 'WrongPassword', 'Client', 'rcon'] |
__all__ = ['RequestIdMismatch', 'WrongPassword', 'Client', 'rcon'] |
||||
|
@ -0,0 +1,86 @@ |
|||||
|
"""Synchronous client.""" |
||||
|
|
||||
|
from socket import socket |
||||
|
from typing import Optional |
||||
|
|
||||
|
from rcon.exceptions import RequestIdMismatch, WrongPassword |
||||
|
from rcon.proto import Packet |
||||
|
|
||||
|
|
||||
|
__all__ = ['Client'] |
||||
|
|
||||
|
|
||||
|
class Client: |
||||
|
"""An RCON client.""" |
||||
|
|
||||
|
__slots__ = ('_socket', 'host', 'port', 'passwd') |
||||
|
|
||||
|
def __init__(self, host: str, port: int, *, |
||||
|
timeout: Optional[float] = None, |
||||
|
passwd: Optional[str] = None): |
||||
|
"""Initializes the base client with the SOCK_STREAM socket type.""" |
||||
|
self._socket = socket() |
||||
|
self.host = host |
||||
|
self.port = port |
||||
|
self.timeout = timeout |
||||
|
self.passwd = passwd |
||||
|
|
||||
|
def __enter__(self): |
||||
|
"""Attempts an auto-login if a password is set.""" |
||||
|
self._socket.__enter__() |
||||
|
self.connect(login=True) |
||||
|
return self |
||||
|
|
||||
|
def __exit__(self, typ, value, traceback): |
||||
|
"""Delegates to the underlying socket's exit method.""" |
||||
|
return self._socket.__exit__(typ, value, traceback) |
||||
|
|
||||
|
@property |
||||
|
def timeout(self) -> float: |
||||
|
"""Returns the socket timeout.""" |
||||
|
return self._socket.gettimeout() |
||||
|
|
||||
|
@timeout.setter |
||||
|
def timeout(self, timeout: float): |
||||
|
"""Sets the socket timeout.""" |
||||
|
self._socket.settimeout(timeout) |
||||
|
|
||||
|
def connect(self, login: bool = False) -> None: |
||||
|
"""Connects the socket and attempts a |
||||
|
login if wanted and a password is set. |
||||
|
""" |
||||
|
self._socket.connect((self.host, self.port)) |
||||
|
|
||||
|
if login and self.passwd is not None: |
||||
|
self.login(self.passwd) |
||||
|
|
||||
|
def close(self) -> None: |
||||
|
"""Closes the socket connection.""" |
||||
|
self._socket.close() |
||||
|
|
||||
|
def communicate(self, packet: Packet) -> Packet: |
||||
|
"""Sends and receives a packet.""" |
||||
|
with self._socket.makefile('wb') as file: |
||||
|
file.write(bytes(packet)) |
||||
|
|
||||
|
with self._socket.makefile('rb') as file: |
||||
|
return Packet.read(file) |
||||
|
|
||||
|
def login(self, passwd: str) -> bool: |
||||
|
"""Performs a login.""" |
||||
|
response = self.communicate(Packet.make_login(passwd)) |
||||
|
|
||||
|
if response.id == -1: |
||||
|
raise WrongPassword() |
||||
|
|
||||
|
return True |
||||
|
|
||||
|
def run(self, command: str, *arguments: str, raw: bool = False) -> str: |
||||
|
"""Runs a command.""" |
||||
|
request = Packet.make_command(command, *arguments) |
||||
|
response = self.communicate(request) |
||||
|
|
||||
|
if response.id != request.id: |
||||
|
raise RequestIdMismatch(request.id, response.id) |
||||
|
|
||||
|
return response if raw else response.payload |
Loading…
Reference in new issue