You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

301 lines
7.9 KiB

from __future__ import annotations
from typing import Optional, Tuple, Union
from a2s.a2s_async import request_async
from a2s.a2s_sync import request_sync
from a2s.datacls import DataclsMeta
from a2s.defaults import DEFAULT_ENCODING, DEFAULT_TIMEOUT
from a2s.exceptions import BufferExhaustedError
from .a2s_protocol import A2SProtocol
from .byteio import ByteReader
A2S_INFO_RESPONSE = 0x49
A2S_INFO_RESPONSE_LEGACY = 0x6D
StrOrBytes = Union[str, bytes]
class SourceInfo(metaclass=DataclsMeta):
protocol: int
"""Protocol version used by the server"""
server_name: StrOrBytes
"""Display name of the server"""
map_name: StrOrBytes
"""The currently loaded map"""
folder: StrOrBytes
"""Name of the game directory"""
game: StrOrBytes
"""Name of the game"""
app_id: int
"""App ID of the game required to connect"""
player_count: int
"""Number of players currently connected"""
max_players: int
"""Number of player slots available"""
bot_count: int
"""Number of bots on the server"""
server_type: StrOrBytes
"""Type of the server:
'd': Dedicated server
'l': Non-dedicated server
'p': SourceTV relay (proxy)"""
platform: StrOrBytes
"""Operating system of the server
'l', 'w', 'm' for Linux, Windows, macOS"""
password_protected: bool
"""Server requires a password to connect"""
vac_enabled: bool
"""Server has VAC enabled"""
version: StrOrBytes
"""Version of the server software"""
# Optional:
edf: int = 0
"""Extra data field, used to indicate if extra values are
included in the response"""
port: Optional[int]
"""Port of the game server."""
steam_id: Optional[int]
"""Steam ID of the server"""
stv_port: Optional[int]
"""Port of the SourceTV server"""
stv_name: Optional[StrOrBytes]
"""Name of the SourceTV server"""
keywords: Optional[StrOrBytes]
"""Tags that describe the gamemode being played"""
game_id: Optional[int]
"""Game ID for games that have an app ID too high for 16bit."""
# Client determined values:
ping: float
"""Round-trip delay time for the request in seconds"""
@property
def has_port(self) -> bool:
return bool(self.edf & 0x80)
@property
def has_steam_id(self) -> bool:
return bool(self.edf & 0x10)
@property
def has_stv(self) -> bool:
return bool(self.edf & 0x40)
@property
def has_keywords(self) -> bool:
return bool(self.edf & 0x20)
@property
def has_game_id(self) -> bool:
return bool(self.edf & 0x01)
class GoldSrcInfo(metaclass=DataclsMeta):
address: StrOrBytes
"""IP Address and port of the server"""
server_name: StrOrBytes
"""Display name of the server"""
map_name: StrOrBytes
"""The currently loaded map"""
folder: StrOrBytes
"""Name of the game directory"""
game: StrOrBytes
"""Name of the game"""
player_count: int
"""Number of players currently connected"""
max_players: int
"""Number of player slots available"""
protocol: int
"""Protocol version used by the server"""
server_type: StrOrBytes
"""Type of the server:
'd': Dedicated server
'l': Non-dedicated server
'p': SourceTV relay (proxy)"""
platform: StrOrBytes
"""Operating system of the server
'l', 'w' for Linux and Windows"""
password_protected: bool
"""Server requires a password to connect"""
"""Server is running a Half-Life mod instead of the base game"""
is_mod: bool
vac_enabled: bool
"""Server has VAC enabled"""
bot_count: int
"""Number of bots on the server"""
# Optional:
mod_website: Optional[StrOrBytes]
"""URL to the mod website"""
mod_download: Optional[StrOrBytes]
"""URL to download the mod"""
mod_version: Optional[int]
"""Version of the mod installed on the server"""
mod_size: Optional[int]
"""Size in bytes of the mod"""
multiplayer_only: bool = False
"""Mod supports multiplayer only"""
uses_custom_dll: bool = True
"""Mod uses a custom DLL"""
# Client determined values:
ping: float
"""Round-trip delay time for the request in seconds"""
def info(
address: Tuple[str, int],
timeout: float = DEFAULT_TIMEOUT,
encoding: str = DEFAULT_ENCODING,
) -> Union[SourceInfo, GoldSrcInfo]:
return request_sync(address, timeout, encoding, InfoProtocol)
async def ainfo(
address: Tuple[str, int],
timeout: float = DEFAULT_TIMEOUT,
encoding: str = DEFAULT_ENCODING,
) -> Union[SourceInfo, GoldSrcInfo]:
return await request_async(address, timeout, encoding, InfoProtocol)
class InfoProtocol(A2SProtocol):
@staticmethod
def validate_response_type(response_type: int) -> bool:
return response_type in (A2S_INFO_RESPONSE, A2S_INFO_RESPONSE_LEGACY)
@staticmethod
def serialize_request(challenge: int) -> bytes:
if challenge:
return b"\x54Source Engine Query\0" + challenge.to_bytes(
4, "little"
)
else:
return b"\x54Source Engine Query\0"
@staticmethod
def deserialize_response(
reader: ByteReader, response_type: int, ping: Optional[float]
) -> Union[SourceInfo, GoldSrcInfo]:
resp: Union[SourceInfo, GoldSrcInfo]
if response_type == A2S_INFO_RESPONSE:
resp = parse_source(reader)
elif response_type == A2S_INFO_RESPONSE_LEGACY:
resp = parse_goldsrc(reader)
else:
raise Exception(str(response_type))
assert ping
resp.ping = ping
return resp
def parse_source(reader: ByteReader) -> SourceInfo:
resp = SourceInfo()
resp.protocol = reader.read_uint8()
resp.server_name = reader.read_cstring()
resp.map_name = reader.read_cstring()
resp.folder = reader.read_cstring()
resp.game = reader.read_cstring()
resp.app_id = reader.read_uint16()
resp.player_count = reader.read_uint8()
resp.max_players = reader.read_uint8()
resp.bot_count = reader.read_uint8()
resp.server_type = reader.read_char().lower()
resp.platform = reader.read_char().lower()
if resp.platform == "o": # Deprecated mac value
resp.platform = "m"
resp.password_protected = reader.read_bool()
resp.vac_enabled = reader.read_bool()
resp.version = reader.read_cstring()
try:
resp.edf = reader.read_uint8()
except BufferExhaustedError:
pass
if resp.has_port:
resp.port = reader.read_uint16()
if resp.has_steam_id:
resp.steam_id = reader.read_uint64()
if resp.has_stv:
resp.stv_port = reader.read_uint16()
resp.stv_name = reader.read_cstring()
if resp.has_keywords:
resp.keywords = reader.read_cstring()
if resp.has_game_id:
resp.game_id = reader.read_uint64()
return resp
def parse_goldsrc(reader: ByteReader) -> GoldSrcInfo:
resp = GoldSrcInfo()
resp.address = reader.read_cstring()
resp.server_name = reader.read_cstring()
resp.map_name = reader.read_cstring()
resp.folder = reader.read_cstring()
resp.game = reader.read_cstring()
resp.player_count = reader.read_uint8()
resp.max_players = reader.read_uint8()
resp.protocol = reader.read_uint8()
resp.server_type = reader.read_char()
resp.platform = reader.read_char()
resp.password_protected = reader.read_bool()
resp.is_mod = reader.read_bool()
# Some games don't send this section
if resp.is_mod and len(reader.peek()) > 2:
resp.mod_website = reader.read_cstring()
resp.mod_download = reader.read_cstring()
reader.read(1) # Skip a NULL byte
resp.mod_version = reader.read_uint32()
resp.mod_size = reader.read_uint32()
resp.multiplayer_only = reader.read_bool()
resp.uses_custom_dll = reader.read_bool()
resp.vac_enabled = reader.read_bool()
resp.bot_count = reader.read_uint8()
return resp