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.
353 lines
8.9 KiB
353 lines
8.9 KiB
import re
|
|
from steam.enums.base import SteamIntEnum
|
|
from steam.enums import EType, EUniverse
|
|
|
|
class ETypeChar(SteamIntEnum):
|
|
I = EType.Invalid
|
|
U = EType.Individual
|
|
M = EType.Multiseat
|
|
G = EType.GameServer
|
|
A = EType.AnonGameServer
|
|
P = EType.Pending
|
|
C = EType.ContentServer
|
|
g = EType.Clan
|
|
T = EType.Chat
|
|
c = EType.Chat
|
|
L = EType.Chat
|
|
a = EType.AnonUser
|
|
|
|
def __str__(self):
|
|
return self.name
|
|
|
|
ETypeChars = ''.join(map(str, list(ETypeChar)))
|
|
|
|
|
|
class SteamID(int):
|
|
"""
|
|
Object for converting steamID to its' various representations
|
|
|
|
.. code:: python
|
|
|
|
SteamID() # invalid steamid
|
|
SteamID(12345) # accountid
|
|
SteamID('12345')
|
|
SteamID(id=12345, type='Invalid', universe='Invalid', instance=0)
|
|
SteamID(103582791429521412) # steam64
|
|
SteamID('103582791429521412')
|
|
SteamID('STEAM_1:0:2') # steam2
|
|
SteamID('[g:1:4]') # steam3
|
|
"""
|
|
|
|
def __new__(cls, *args, **kwargs):
|
|
steam64 = make_steam64(*args, **kwargs)
|
|
return super(SteamID, cls).__new__(cls, steam64)
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
pass
|
|
|
|
def __repr__(self):
|
|
return "<%s(id=%s, type=%s, universe=%s, instance=%s)>" % (
|
|
self.__class__.__name__,
|
|
self.id,
|
|
repr(self.type.name),
|
|
repr(self.universe.name),
|
|
self.instance,
|
|
)
|
|
|
|
@property
|
|
def id(self):
|
|
"""
|
|
:return: account id
|
|
:rtype: int
|
|
"""
|
|
return int(self) & 0xFFffFFff
|
|
|
|
@property
|
|
def instance(self):
|
|
"""
|
|
:rtype: int
|
|
"""
|
|
return (int(self) >> 32) & 0xFFffF
|
|
|
|
@property
|
|
def type(self):
|
|
"""
|
|
:rtype: :py:class:`steam.enum.EType`
|
|
"""
|
|
return EType((int(self) >> 52) & 0xF)
|
|
|
|
@property
|
|
def universe(self):
|
|
"""
|
|
:rtype: :py:class:`steam.enum.EUniverse`
|
|
"""
|
|
return EUniverse((int(self) >> 56) & 0xFF)
|
|
|
|
@property
|
|
def as_32(self):
|
|
"""
|
|
:return: account id
|
|
:rtype: int
|
|
"""
|
|
return self.id
|
|
|
|
@property
|
|
def as_64(self):
|
|
"""
|
|
:return: steam64 format
|
|
:rtype: int
|
|
"""
|
|
return int(self)
|
|
|
|
@property
|
|
def as_steam2(self):
|
|
"""
|
|
:return: steam2 format (e.g ``STEAM_0:0:1234``)
|
|
:rtype: str
|
|
"""
|
|
return "STEAM_0:%s:%s" % (
|
|
self.id % 2,
|
|
self.id >> 1,
|
|
)
|
|
|
|
@property
|
|
def as_steam3(self):
|
|
"""
|
|
:return: steam3 format (e.g ``[U:1:1234]``)
|
|
:rtype: str
|
|
"""
|
|
if self.type is EType.AnonGameServer:
|
|
return "[%s:%s:%s:%s]" % (
|
|
str(ETypeChar(self.type)),
|
|
int(self.universe),
|
|
self.id,
|
|
self.instance
|
|
)
|
|
else:
|
|
return "[%s:%s:%s]" % (
|
|
str(ETypeChar(self.type)),
|
|
int(self.universe),
|
|
self.id,
|
|
)
|
|
|
|
@property
|
|
def community_url(self):
|
|
"""
|
|
:return: e.g https://steamcommunity.com/profiles/123456789
|
|
:rtype: str
|
|
"""
|
|
suffix = {
|
|
EType.Individual: "profiles/%s",
|
|
EType.Clan: "gid/%s",
|
|
}
|
|
if self.type in suffix:
|
|
url = "https://steamcommunity.com/%s" % suffix[self.type]
|
|
return url % self.as_64
|
|
|
|
return None
|
|
|
|
def is_valid(self):
|
|
"""
|
|
:rtype: :py:class:`bool`
|
|
"""
|
|
return (self.id > 0
|
|
and self.type is not EType.Invalid
|
|
and self.universe is not EUniverse.Invalid
|
|
)
|
|
|
|
|
|
def make_steam64(id=0, *args, **kwargs):
|
|
"""
|
|
Returns steam64 from various other representations.
|
|
|
|
.. code:: python
|
|
|
|
make_steam64() # invalid steamid
|
|
make_steam64(12345) # accountid
|
|
make_steam64('12345')
|
|
make_steam64(id=12345, type='Invalid', universe='Invalid', instance=0)
|
|
make_steam64(103582791429521412) # steam64
|
|
make_steam64('103582791429521412')
|
|
make_steam64('STEAM_1:0:2') # steam2
|
|
make_steam64('[g:1:4]') # steam3
|
|
"""
|
|
|
|
accountid = id
|
|
etype = EType.Invalid
|
|
universe = EUniverse.Invalid
|
|
instance = None
|
|
|
|
if len(args) == 0 and len(kwargs) == 0:
|
|
value = str(accountid)
|
|
|
|
# numeric input
|
|
if value.isdigit():
|
|
value = int(value)
|
|
|
|
# 32 bit account id
|
|
if 0 < value < 2**32:
|
|
accountid = value
|
|
etype = EType.Individual
|
|
universe = EUniverse.Public
|
|
# 64 bit
|
|
elif value < 2**64:
|
|
return value
|
|
|
|
# textual input e.g. [g:1:4]
|
|
else:
|
|
result = steam2_to_tuple(value) or steam3_to_tuple(value)
|
|
|
|
if result:
|
|
(accountid,
|
|
etype,
|
|
universe,
|
|
instance,
|
|
) = result
|
|
else:
|
|
accountid = 0
|
|
|
|
elif len(args) > 0:
|
|
length = len(args)
|
|
if length == 1:
|
|
etype, = args
|
|
elif length == 2:
|
|
etype, universe = args
|
|
elif length == 3:
|
|
etype, universe, instance = args
|
|
else:
|
|
raise TypeError("Takes at most 4 arguments (%d given)" % length)
|
|
|
|
if len(kwargs) > 0:
|
|
etype = kwargs.get('type', etype)
|
|
universe = kwargs.get('universe', universe)
|
|
instance = kwargs.get('instance', instance)
|
|
|
|
etype = (EType(etype)
|
|
if isinstance(etype, (int, EType))
|
|
else EType[etype]
|
|
)
|
|
|
|
universe = (EUniverse(universe)
|
|
if isinstance(universe, (int, EUniverse))
|
|
else EUniverse[universe]
|
|
)
|
|
|
|
if instance is None:
|
|
instance = 1 if etype in (EType.Individual, EType.GameServer) else 0
|
|
|
|
assert instance <= 0xffffF, "instance larger than 20bits"
|
|
|
|
return (universe << 56) | (etype << 52) | (instance << 32) | accountid
|
|
|
|
|
|
def steam2_to_tuple(value):
|
|
"""
|
|
:param value: steam2 (e.g. ``STEAM_0:0:1234``)
|
|
:type value: str
|
|
:return: (accountid, type, universe, instance)
|
|
:rtype: ``tuple`` or ``None``
|
|
"""
|
|
match = re.match(r"^STEAM_(?P<universe>[01])"
|
|
r":(?P<reminder>[0-1])"
|
|
r":(?P<id>\d+)$", value
|
|
)
|
|
|
|
if not match:
|
|
return None
|
|
|
|
steam32 = (int(match.group('id')) << 1) | int(match.group('reminder'))
|
|
|
|
return (steam32, EType(1), EUniverse(1), 1)
|
|
|
|
|
|
def steam3_to_tuple(value):
|
|
"""
|
|
:param value: steam3 (e.g. ``[U:1:1234]``)
|
|
:type value: str
|
|
:return: (accountid, type, universe, instance)
|
|
:rtype: ``tuple`` or ``None``
|
|
"""
|
|
match = re.match(r"^\["
|
|
r"(?P<type>[%s]):" # type char
|
|
r"(?P<universe>\d+):" # universe
|
|
r"(?P<id>\d+)" # accountid
|
|
r"(:(?P<instance>\d+))?" # instance
|
|
r"\]$" % ETypeChars,
|
|
value
|
|
)
|
|
if not match:
|
|
return None
|
|
|
|
steam32 = int(match.group('id'))
|
|
universe = EUniverse(int(match.group('universe')))
|
|
etype = ETypeChar[match.group('type')]
|
|
instance = match.group('instance')
|
|
|
|
if instance is None:
|
|
if etype in (EType.Individual, EType.GameServer):
|
|
instance = 1
|
|
else:
|
|
instance = 0
|
|
else:
|
|
instance = int(instance)
|
|
|
|
return (steam32, etype, universe, instance)
|
|
|
|
def steam64_from_url(url):
|
|
"""
|
|
Takes a Steam Community url and returns steam64 or None
|
|
|
|
:param url: steam community url
|
|
:type url: str
|
|
:return: steam64
|
|
:rtype: ``int`` or ``None``
|
|
|
|
Example URLs::
|
|
|
|
https://steamcommunity.com/gid/[g:1:4]
|
|
https://steamcommunity.com/gid/103582791429521412
|
|
https://steamcommunity.com/groups/Valve
|
|
https://steamcommunity.com/profiles/[U:1:12]
|
|
https://steamcommunity.com/profiles/76561197960265740
|
|
https://steamcommunity.com/id/johnc
|
|
"""
|
|
|
|
match = re.match(r'^https?://steamcommunity.com/'
|
|
r'(?P<type>profiles|id|gid|groups)/(?P<value>.*)/?$', url)
|
|
|
|
if not match:
|
|
return None
|
|
|
|
import requests
|
|
|
|
if match.group('type') in ('id', 'profiles'):
|
|
xml = requests.get("%s/?xml=1" % url).text
|
|
match = re.findall('<steamID64>(\d+)</steamID64>', xml)
|
|
else:
|
|
xml = requests.get("%s/memberslistxml/?xml=1" % url).text
|
|
match = re.findall('<groupID64>(\d+)</groupID64>', xml)
|
|
|
|
if not match:
|
|
return None
|
|
|
|
return match[0] # return matched steam64
|
|
|
|
|
|
def from_url(url):
|
|
"""
|
|
Takes Steam community url and returns a SteamID instance or None
|
|
|
|
:param url: steam community url
|
|
:type url: str
|
|
:return: `SteamID` instance
|
|
:rtype: :py:class:`steam.SteamID` or ``None``
|
|
|
|
See :py:func:`steam64_from_url`
|
|
"""
|
|
|
|
steam64 = steam64_from_url(url)
|
|
|
|
if steam64:
|
|
return SteamID(steam64)
|
|
|
|
return None
|
|
|