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
10 KiB

r""" Master Server Query Protocol
This module implements the legacy Steam master server protocol.
https://developer.valvesoftware.com/wiki/Master_Server_Query_Protocol
Nowadays games query servers through Steam. See :any:`steam.client.builtins.gameservers`
Filters
-------
.. note::
Multiple filters can be joined to together (Eg. ``\app\730\white\1\empty\1``)
=========================== =========================================================================================================================
Filter code What it does
=========================== =========================================================================================================================
\\nor\\[x] A special filter, specifies that servers matching any of the following [x] conditions should not be returned
\\nand\\[x] A special filter, specifies that servers matching all of the following [x] conditions should not be returned
\\dedicated\\1 Servers running dedicated
\\secure\\1 Servers using anti-cheat technology (VAC, but potentially others as well)
\\gamedir\\[mod] Servers running the specified modification (ex. cstrike)
\\map\\[map] Servers running the specified map (ex. cs_italy)
\\linux\\1 Servers running on a Linux platform
\\password\\0 Servers that are not password protected
\\empty\\1 Servers that are not empty
\\full\\1 Servers that are not full
\\proxy\\1 Servers that are spectator proxies
\\appid\\[appid] Servers that are running game [appid]
\\napp\\[appid] Servers that are NOT running game [appid] (This was introduced to block Left 4 Dead games from the Steam Server Browser)
\\noplayers\\1 Servers that are empty
\\white\\1 Servers that are whitelisted
\\gametype\\[tag,...] Servers with all of the given tag(s) in sv_tags
\\gamedata\\[tag,...] Servers with all of the given tag(s) in their 'hidden' tags (L4D2)
\\gamedataor\\[tag,...] Servers with any of the given tag(s) in their 'hidden' tags (L4D2)
\\name_match\\[hostname] Servers with their hostname matching [hostname] (can use * as a wildcard)
\\version_match\\[version] Servers running version [version] (can use * as a wildcard)
\\collapse_addr_hash\\1 Return only one server for each unique IP address matched
\\gameaddr\\[ip] Return only servers on the specified IP address (port supported and optional)
=========================== =========================================================================================================================
"""
import socket
from struct import pack as _pack, unpack_from as _unpack_from
from time import time as _time
from enum import IntEnum, Enum
from steam.util.binary import StructReader
class MSRegion(IntEnum):
US_East = 0x00
US_West = 0x01
South_America = 0x02
Europe = 0x03
Asia = 0x04
Australia = 0x05
Middle_East = 0x06
Africa = 0x07
World = 0xFF
class MSServer:
GoldSrc = ('hl1master.steampowered.com', 27010) #: These have been shutdown
Source = ('hl2master.steampowered.com', 27011)
Source_27015 = ('208.64.200.65', 27015) #: ``hl2master`` but on different port
def query_master(filter_text=r'\napp\500', region=MSRegion.World, master=MSServer.Source):
r"""Generator that returns (IP,port) pairs of serveras
.. warning::
Valve's master servers seem to be heavily rate limited.
Queries that return a large numbers IPs will timeout before returning everything.
There is no way to resume the query.
Use :class:`SteamClient` to access game servers in a reliable way.
.. note::
When specifying ``filter_text`` use *raw strings* otherwise python won't treat backslashes
as literal characters (e.g. ``query(r'\appid\730\white\1')``)
:param filter_text: filter for servers
:type filter_text: str
:param region: (optional) region code
:type region: :class:`.MSRegion`
:param master: (optional) master server to query
:type master: (:class:`str`, :class:`int`)
:raises: This function will raise in various situations
:returns: a generator yielding (ip, port) pairs
:rtype: :class:`generator`
"""
if not isinstance(region, MSRegion):
raise TypeError("region_code is not of type MSRegion")
ms = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
ms.connect(master)
ms.settimeout(8)
next_ip = b'0.0.0.0:0'
req_prefix = b'1' + _pack('>B', region)
req_suffix = b'\x00' + filter_text.encode('utf-8') + b'\x00'
while True:
ms.send(req_prefix + next_ip + req_suffix)
data = ms.recv(2048)
data = StructReader(data)
# verify response header
if data.read(6) != b'\xFF\xFF\xFF\xFF\x66\x0A':
raise RuntimeError("Invalid response from master server")
# read list of servers
while data.rlen():
ip = '.'.join(map(str, data.unpack('>BBBB')))
port, = data.unpack('>H')
# check if we've reach the end of the list
if ip == '0.0.0.0' and port == 0:
return
yield ip, port
next_ip = '{}:{}'.format(ip, port).encode('utf-8')
def _handle_a2s_response(sock):
packet = sock.recv(2048)
header, = _unpack_from('<l', packet)
if header == -1: # single packet response
return packet[4:]
elif header == -2: # multi packet response
raise RuntimeError("Multi packet response not implemented yet")
else:
raise RuntimeError("Invalid reponse header")
def a2s_info(server_addr, goldsrc=False, timeout=6):
"""Get information from a server
:param server_addr: (ip, port) for the server
:type server_addr: tuple
:param goldsrc: (optional) Weather to expect GoldSrc or Source response format
:type goldsrc: :class:`bool`
:param timeout: (optional) timeout in seconds
:type timeout: int
:returns: a dict with information or `None` on timeout
:rtype: :class:`dict`, :class:`None`
"""
ss = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
ss.connect(server_addr)
ss.settimeout(timeout)
# request server info
ss.send(_pack('<lc', -1, b'T') + b'Source Engine Query\x00')
resp_header = b'm' if goldsrc else b'I'
while True:
try:
data = _handle_a2s_response(ss)
except socket.timeout:
return None
else:
if data[0:1] == resp_header:
break
data = StructReader(data)
data.skip(1) # header
# GoldSrc response
if goldsrc:
info = {
'address': data.read_cstring(),
'name': data.read_cstring(),
'map': data.read_cstring(),
'folder': data.read_cstring(),
'game': data.read_cstring(),
}
(info['players'],
info['max_players'],
info['protocol'],
info['server_type'],
info['environment'],
info['visibility'],
info['mod'],
) = data.unpack('<BBBccBB')
if info['mod'] == 1:
info['link'] = data.read_cstring(),
info['download_link'] = data.read_cstring(),
(info['version'],
info['size'],
info['type'],
info['ddl'],
) = data.unpack('<xLLBB')
info['vac'], info['bots'] = data.unpack('<BB')
# Source response
else:
info = {
'protocol': data.unpack('<b')[0],
'name': data.read_cstring(),
'map': data.read_cstring(),
'folder': data.read_cstring(),
'game': data.read_cstring(),
}
(info['app_id'],
info['players'],
info['max_players'],
info['bots'],
info['server_type'],
info['environment'],
info['visibility'],
info['vac'],
) = data.unpack('<HBBBccBB')
return info
def a2s_player(server_addr, challenge=0, timeout=8):
"""Get list of players and their info
:param server_addr: (ip, port) for the server
:type server_addr: tuple
:param challenge: (optional) challenge number
:type challenge: int
:param timeout: (optional) timeout in seconds
:type timeout: int
:returns: a list of players
:rtype: :class:`list`, :class:`None`
"""
ss = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
ss.connect(server_addr)
ss.settimeout(timeout)
# request challenge number
if challenge in (-1, 0):
ss.send(_pack('<lci', -1, b'U', challenge))
_, header, challange = _unpack_from('<lcl', ss.recv(512))
if header != b'A':
raise RuntimeError("Unexpected challange response")
# request player info
ss.send(_pack('<lci', -1, b'U', challange))
try:
data = StructReader(_handle_a2s_response(ss))
except socket.timeout:
return None
header, num_players = data.unpack('<BB')
players = []
while len(players) != num_players:
player = dict()
player['index'] = data.unpack('<B')[0]
player['name'] = data.read_cstring()
player['score'], player['duration'] = data.unpack('<lf')
players.append(player)
if data.rlen() / 8 == num_players: # assume the ship server
for player in players:
player['deaths'], player['money'] = data.unpack('<ll')
return players
def a2s_ping(server_addr, timeout=8):
"""Ping a server
.. warning::
This method for pinging is considered deprecated and will not work on newer sources games
:param server_addr: (ip, port) for the server
:type server_addr: tuple
:param timeout: (optional) timeout in seconds
:type timeout: int
:returns: ping response in seconds or `None` for timeout
:rtype: :class:`float`, :class:`None`
"""
ss = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
ss.connect(server_addr)
ss.settimeout(timeout)
ss.send(_pack('<lc', -1, b'i'))
start = _time()
try:
data = _handle_a2s_response(ss)
except socket.timeout:
return None
diff = _time() - start
if data[0:1] == b'j':
return diff