pythonhacktoberfeststeamauthenticationauthenticatorsteam-authenticatorsteam-clientsteam-guard-codessteam-websteamworksvalvewebapi
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.
196 lines
7.8 KiB
196 lines
7.8 KiB
r"""
|
|
Listing and information about game servers through Steam
|
|
|
|
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)
|
|
=========================== =========================================================================================================================
|
|
"""
|
|
from steam.steamid import SteamID
|
|
from steam.core.msg import MsgProto
|
|
from steam.enums.emsg import EMsg
|
|
from steam.util import ip_to_int, ip_from_int, proto_to_dict
|
|
|
|
|
|
class GameServers(object):
|
|
def __init__(self, *args, **kwargs):
|
|
super(GameServers, self).__init__(*args, **kwargs)
|
|
|
|
self.gameservers = SteamGameServers(self) #: instance of :class:`SteamGameServers`
|
|
|
|
|
|
class SteamGameServers(object):
|
|
def __init__(self, steam):
|
|
self.steam = steam
|
|
|
|
def query(self, filter_text, max_servers=10, **kwargs):
|
|
r"""
|
|
Query game servers
|
|
|
|
https://developer.valvesoftware.com/wiki/Master_Server_Query_Protocol
|
|
|
|
.. 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 max_servers: (optional) number of servers to return
|
|
:type max_servers: int
|
|
:param app_id: (optional) app id
|
|
:type app_id: int
|
|
:param geo_location_ip: (optional) ip (e.g. '1.2.3.4')
|
|
:type geo_location_ip: str
|
|
:returns: list of servers, see below. (``None`` is returned steam doesn't respond)
|
|
:rtype: :class:`list`, :class:`None`
|
|
|
|
Sample response:
|
|
|
|
.. code:: python
|
|
|
|
[{'auth_players': 0, 'server_ip': '1.2.3.4', 'server_port': 27015},
|
|
{'auth_players': 6, 'server_ip': '1.2.3.4', 'server_port': 27016},
|
|
...
|
|
]
|
|
"""
|
|
if 'geo_location_ip' in kwargs:
|
|
kwargs['geo_location_ip'] = ip_to_int(kwargs['geo_location_ip'])
|
|
|
|
kwargs['filter_text'] = filter_text
|
|
kwargs['max_servers'] = max_servers
|
|
|
|
resp = self.steam.send_job_and_wait(MsgProto(EMsg.ClientGMSServerQuery), kwargs, timeout=30)
|
|
|
|
if resp is None: return None
|
|
|
|
resp = proto_to_dict(resp)
|
|
|
|
for server in resp['servers']:
|
|
server['server_ip'] = ip_from_int(server['server_ip'])
|
|
|
|
return resp['servers']
|
|
|
|
def get_server_list(self, filter_text, max_servers=10):
|
|
"""
|
|
Get list of servers. Works similiarly to :meth:`query`, but the response has more details.
|
|
|
|
:param filter_text: filter for servers
|
|
:type filter_text: str
|
|
:param max_servers: (optional) number of servers to return
|
|
:type max_servers: int
|
|
:returns: list of servers, see below. (``None`` is returned steam doesn't respond)
|
|
:rtype: :class:`list`, :class:`None`
|
|
|
|
|
|
Sample response:
|
|
|
|
.. code:: python
|
|
|
|
[{'addr': '1.2.3.4:27067',
|
|
'appid': 730,
|
|
'bots': 0,
|
|
'dedicated': True,
|
|
'gamedir': 'csgo',
|
|
'gameport': 27067,
|
|
'gametype': 'valve_ds,empty,secure',
|
|
'map': 'de_dust2',
|
|
'max_players': 10,
|
|
'name': 'Valve CS:GO Asia Server (srcdsXXX.XXX.XXX)',
|
|
'os': 'l',
|
|
'players': 0,
|
|
'product': 'csgo',
|
|
'region': 5,
|
|
'secure': True,
|
|
'steamid': SteamID(id=3279818759, type='AnonGameServer', universe='Public', instance=7011),
|
|
'version': '1.35.4.0'}
|
|
]
|
|
"""
|
|
resp = self.steam.unified_messages.send_and_wait("GameServers.GetServerList#1", {
|
|
"filter": filter_text,
|
|
"limit": max_servers,
|
|
}, timeout=20)
|
|
|
|
if resp is None:
|
|
return None
|
|
|
|
resp = proto_to_dict(resp)
|
|
|
|
if not resp:
|
|
return []
|
|
else:
|
|
for server in resp['servers']:
|
|
server['steamid'] = SteamID(server['steamid'])
|
|
|
|
return resp['servers']
|
|
|
|
def get_ips_from_steamid(self, server_steam_ids):
|
|
"""Resolve IPs from SteamIDs
|
|
|
|
:param server_steam_ids: a list of steamids
|
|
:type server_steam_ids: list
|
|
:return: map of ips to steamids
|
|
:rtype: dict
|
|
|
|
Sample response:
|
|
|
|
.. code:: python
|
|
|
|
{SteamID(id=123456, type='AnonGameServer', universe='Public', instance=1234): '1.2.3.4:27060'}
|
|
"""
|
|
resp = self.steam.unified_messages.send_and_wait("GameServers.GetServerIPsBySteamID#1", {
|
|
"server_steamids": server_steam_ids,
|
|
}, timeout=30)
|
|
|
|
if resp is None: return None
|
|
|
|
return {SteamID(server.steamid): server.addr for server in resp.servers}
|
|
|
|
def get_steamids_from_ip(self, server_ips):
|
|
"""Resolve SteamIDs from IPs
|
|
|
|
:param steam_ids: a list of ips (e.g. ``['1.2.3.4:27015',...]``)
|
|
:type steam_ids: list
|
|
:return: map of steamids to ips
|
|
:rtype: dict
|
|
|
|
Sample response:
|
|
|
|
.. code:: python
|
|
|
|
{'1.2.3.4:27060': SteamID(id=123456, type='AnonGameServer', universe='Public', instance=1234)}
|
|
"""
|
|
resp = self.steam.unified_messages.send_and_wait("GameServers.GetServerSteamIDsByIP#1", {
|
|
"server_ips": server_ips,
|
|
}, timeout=30)
|
|
|
|
if resp is None: return None
|
|
|
|
return {server.addr: SteamID(server.steamid) for server in resp.servers}
|
|
|