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

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