import sys import re from steam.enums.base import SteamIntEnum from steam.enums import EType, EUniverse from steam.util.web import make_requests_session if sys.version_info < (3,): intBase = long else: intBase = int 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(intBase): """ 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_1:0:1234``) :rtype: str .. note:: ``STEAM_X:Y:Z``. The value of ``X`` should represent the universe, or ``1`` for ``Public``. However, there was a bug in GoldSrc and Orange Box games and ``X`` was ``0``. If you need that format use :attr:`SteamID.as_steam2_zero` """ return "STEAM_1:%s:%s" % ( self.id % 2, self.id >> 1, ) @property def as_steam2_zero(self): """ For GoldSrc and Orange Box games. See :attr:`SteamID.as_steam2` :return: steam2 format (e.g ``STEAM_0:0:1234``) :rtype: str """ return self.as_steam2.replace("_1", "_0") @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_1:0:1234`) :type value: str :return: (accountid, type, universe, instance) :rtype: ``tuple`` or ``None`` .. note:: The universe will be always set to ``1``. See :attr:`SteamID.as_steam2` """ match = re.match(r"^STEAM_(?P[01])" r":(?P[0-1])" r":(?P\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[%s]):" # type char r"(?P\d+):" # universe r"(?P\d+)" # accountid r"(:(?P\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'(?Pprofiles|id|gid|groups)/(?P.*)/?$', url) if not match: return None web = make_requests_session() if match.group('type') in ('id', 'profiles'): xml = web.get("%s/?xml=1" % url).text match = re.findall('(\d+)', xml) else: xml = web.get("%s/memberslistxml/?xml=1" % url).text match = re.findall('(\d+)', 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