diff --git a/.travis.yml b/.travis.yml new file mode 100644 index 0000000..67f8f11 --- /dev/null +++ b/.travis.yml @@ -0,0 +1,12 @@ +language: python +python: + - "2.7" + - "3.3" + - "3.4" +install: + - make init + - pip install coveralls +script: + - make test +after_success: + - coveralls diff --git a/Makefile b/Makefile index 9cad602..8117c2a 100644 --- a/Makefile +++ b/Makefile @@ -18,7 +18,7 @@ init: test: rm -f .coverage steam/*.pyc tests/*.pyc - PYTHONHASHSEED=0 nosetests --verbosity 2 --with-coverage --cover-package=steam + PYTHONHASHSEED=0 nosetests --verbosity 1 --with-coverage --cover-package=steam pylint: pylint -r n -f colorized steam || true diff --git a/README.rst b/README.rst index 4194572..aa8f112 100644 --- a/README.rst +++ b/README.rst @@ -1,4 +1,4 @@ -|pypi| |license| +|pypi| |license| |coverage| |master_build| Module for interacting with various Steam_ features @@ -104,7 +104,6 @@ SteamID 'https://steamcommunity.com/gid/103582791429521412' - .. _Steam: https://store.steampowered.com/ .. _Steam Web API: https://developer.valvesoftware.com/wiki/Steam_Web_API .. _API Key: http://steamcommunity.com/dev/apikey @@ -117,3 +116,11 @@ SteamID .. |license| image:: https://img.shields.io/pypi/l/steam.svg?style=flat&label=license :target: https://pypi.python.org/pypi/steam :alt: MIT License + +.. |coverage| image:: https://img.shields.io/coveralls/ValvePython/steam/master.svg?style=flat + :target: https://coveralls.io/r/ValvePython/steam?branch=master + :alt: Test coverage + +.. |master_build| image:: https://img.shields.io/travis/ValvePython/steam/master.svg?style=flat&label=master + :target: http://travis-ci.org/ValvePython/steam + :alt: Build status of master branch diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..92acdd2 --- /dev/null +++ b/requirements.txt @@ -0,0 +1,6 @@ +nose +coverage +mock +requests +enum34 +vcrpy diff --git a/steam/__init__.py b/steam/__init__.py index 97af95f..864d4a7 100644 --- a/steam/__init__.py +++ b/steam/__init__.py @@ -1,4 +1,4 @@ -__version__ = "0.5.1" +__version__ = "0.5.2" __author__ = "Rossen Georgiev" from steam.steamid import SteamID diff --git a/steam/steamid.py b/steam/steamid.py index 3e139d0..5b9652a 100644 --- a/steam/steamid.py +++ b/steam/steamid.py @@ -1,6 +1,6 @@ import re import requests -from .enums import EType, EUniverse +from steam.enums import EType, EUniverse class SteamID(object): @@ -44,12 +44,11 @@ class SteamID(object): """ largs = len(args) - lkwargs = len(kwargs) - if largs == 0 and lkwargs == 0: + if largs == 0 and len(kwargs) == 0: self.id = 0 self.type = EType.Invalid - self.universe = EType.Invalid + self.universe = EUniverse.Invalid self.instance = 0 elif largs > 0: if largs > 1: @@ -71,15 +70,15 @@ class SteamID(object): value = match.group('value') # numeric input - if value.isdigit(): + if value.isdigit() or (value.startswith('-') and value[1:].isdigit()): value = int(value) if 0 > value: raise ValueError("Expected positive int, got %d" % value) - if value > 2**64-1: + if value >= 2**64: raise ValueError("Expected a 32/64 bit int") # 32 bit account id - if value < 2**32-1: + if value < 2**32: self.id = value self.type = EType.Individual self.universe = EUniverse.Public @@ -94,14 +93,14 @@ class SteamID(object): # textual input e.g. [g:1:4] else: # try steam2 - match = re.match(r"^STEAM_(?P\d+)" + match = re.match(r"^STEAM_(?P[01])" r":(?P[0-1])" r":(?P\d+)$", value ) if match: self.id = (int(match.group('id')) << 1) | int(match.group('reminder')) - self.universe = EUniverse(int(match.group('universe'))) + self.universe = EUniverse(1) self.type = EType(1) self.instance = 1 return @@ -134,35 +133,36 @@ class SteamID(object): " (e.g. [g:1:4], STEAM_0:1:1234), got %s" % repr(value) ) - elif lkwargs > 0: + elif len(kwargs): if 'id' not in kwargs: raise ValueError("Expected at least 'id' kwarg") self.id = int(kwargs['id']) + assert self.id <= 0xffffFFFF, "id larger than 32bits" + + value = kwargs.get('type', 0) + if type(value) in (int, EType): + self.type = EType(value) + else: + self.type = EType[value.lower().capitalize()] - for kwname in 'type', 'universe': - if kwname in kwargs: - value = kwargs[kwname] - kwenum = getattr(self, "E%s" % kwname.capitalize()) - - resolved = getattr(kwenum, value, None) - if resolved is None: - try: - resolved = kwenum(value) - except ValueError: - raise ValueError( - "Invalid value for kwarg '%s', see SteamID.E%s" % - (kwname, kwname.capitalize()) - ) - setattr(self, kwname, resolved) - - if self.type in (EType.Individual, EType.GameServer): - self.instance = 1 + value = kwargs.get('universe', 0) + if type(value) in (int, EUniverse): + self.universe = EUniverse(value) else: - self.instance = 0 + self.universe = EUniverse[value.lower().capitalize()] + + if 'instance' in kwargs: + self.instance = kwargs['instance'] + assert self.instance <= 0xffffF, "instance larger than 20bits" + else: + if self.type in (EType.Individual, EType.GameServer): + self.instance = 1 + else: + self.instance = 0 def __repr__(self): - return "%s(id=%s, type=%s, universe=%s, instance=%s)" % ( + return "<%s(id=%s, type=%s, universe=%s, instance=%s)>" % ( self.__class__.__name__, self.id, repr(self.type.name), @@ -175,19 +175,26 @@ class SteamID(object): @property def as_steam2(self): - return "STEAM_%s:%s:%s" % ( - self.universe.value, + return "STEAM_0:%s:%s" % ( self.id % 2, self.id >> 1, ) @property def as_steam3(self): - return "[%s:%s:%s]" % ( - self.ETypeChar[self.type.value], - self.universe.value, - self.id, - ) + if self.type is EType.AnonGameServer: + return "[%s:%s:%s:%s]" % ( + self.ETypeChar[self.type.value], + self.universe.value, + self.id, + self.instance + ) + else: + return "[%s:%s:%s]" % ( + self.ETypeChar[self.type.value], + self.universe.value, + self.id, + ) @property def as_64(self): diff --git a/steam/webapi.py b/steam/webapi.py index b99b922..0668310 100644 --- a/steam/webapi.py +++ b/steam/webapi.py @@ -1,6 +1,63 @@ from __future__ import print_function import requests +DEFAULT_PARAMS = { + # api parameters + 'key': None, + 'format': 'json', + # internal + 'https': True, + 'http_timeout': 30, + 'raw': False, +} + + +def webapi_request(path, method='GET', caller=None, params={}): + """ + Low level function for calling Steam's WebAPI + """ + if method not in ('GET', 'POST'): + raise NotImplemented("HTTP method: %s" % repr(self.method)) + + onetime = {} + for param in DEFAULT_PARAMS: + params[param] = onetime[param] = params.get(param, + DEFAULT_PARAMS[param], + ) + path = "%s://api.steampowered.com/%s" % ('https' if params.get('https', True) else 'http', + path) + del params['raw'] + del params['https'] + del params['http_timeout'] + + if onetime['format'] not in ('json', 'vdf', 'xml'): + raise ValueError("Expected format to be json,vdf or xml; got %s" % onetime['format']) + + # move params to data, if data is not specified for POST + # simplifies code calling this method + kwargs = {'params': params} if method == "GET" else {'data': params} + + f = getattr(requests, method.lower()) + resp = f(path, stream=True, timeout=onetime['http_timeout'], **kwargs) + + if caller is not None: + caller.last_response = resp + + if not resp.ok: + raise requests.exceptions.HTTPError("%s %s" % (resp.status_code, resp.reason)) + + if onetime['raw']: + return resp.content + + if onetime['format'] == 'json': + return resp.json() + elif onetime['format'] == 'xml': + import lxml.etree + return lxml.etree.parse(resp.raw) + elif onetime['format'] == 'vdf': + import vdf + return vdf.loads(resp.text) + class WebAPI(object): """ @@ -12,13 +69,15 @@ class WebAPI(object): More: https://developer.valvesoftware.com/wiki/Steam_Web_API """ - def __init__(self, key, format='json', raw=False, https=True): + def __init__(self, key, format='json', raw=False, https=True, http_timeout=30, auto_load_interfaces=True): """ Optain apikey at https://steamcommunity.com/dev/apikey - key - apikey - format - output format (json, vdf, xml) - raw - whenver to deserialize the response + key - apikey + format - output format (json, vdf, xml) + raw - whenver to deserialize the response + https - whenever to use https or not + auto_load_interfaces - should we load interfaces upon initialization These can be specified per method call for one off calls """ @@ -27,8 +86,11 @@ class WebAPI(object): self.format = format self.raw = raw self.https = https + self.http_timeout = http_timeout self.interfaces = [] - self.load_interfaces() + + if auto_load_interfaces: + self.load_interfaces(self.fetch_interfaces()) def __repr__(self): return "%s(key=%s, https=%s)" % ( @@ -37,22 +99,30 @@ class WebAPI(object): repr(self.https), ) - def load_interfaces(self): + def fetch_interfaces(self): """ - Fetches the available interfaces from the API itself and then - populates the name space under the instance + Returns a dict with the response from GetSupportedAPIList + + This is then feeded into WebAPI.load_interfaces(reponse) + The reponse could be cached/save and used to load interfaces """ - result = self._api_request( - None, - "GET", + return webapi_request( "ISteamWebAPIUtil/GetSupportedAPIList/v1/", - params={'format': 'json'}, + method="GET", + caller=None, + params={'format': 'json', + 'key': self.key, + }, ) - if result.get('apilist', {}).get('interfaces', None) is None: + def load_interfaces(self, interfaces_dict): + """ + Populates the namespace under the instance + """ + if interfaces_dict.get('apilist', {}).get('interfaces', None) is None: raise ValueError("Invalid response for GetSupportedAPIList") - interfaces = result['apilist']['interfaces'] + interfaces = interfaces_dict['apilist']['interfaces'] if len(interfaces) == 0: raise ValueError("API returned not interfaces; probably using invalid key") @@ -77,52 +147,6 @@ class WebAPI(object): interface, method = method_path.split('.', 1) return getattr(getattr(self, interface), method)(**kwargs) - @property - def _url_base(self): - return "%s://api.steampowered.com/" % ('https' if self.https else 'http') - - def _api_request(self, caller, method, path, **kwargs): - if method not in ('GET', 'POST'): - raise NotImplemented("HTTP method: %s" % repr(self.method)) - if 'params' not in kwargs: - kwargs['params'] = {} - - onetime = {} - for param in ('key', 'format', 'raw'): - kwargs['params'][param] = onetime[param] = kwargs['params'].get(param, - getattr(self, param) - ) - del kwargs['params']['raw'] - - if onetime['format'] not in ('json', 'vdf', 'xml'): - raise ValueError("Expected format to be json,vdf or xml; got %s" % onetime['format']) - - # move params to data, if data is not specified for POST - # simplifies code calling this method - if method == 'POST' and 'data' not in kwargs: - kwargs['data'] = kwargs['params'] - del kwargs['params'] - - f = getattr(requests, method.lower()) - resp = f(self._url_base + path, stream=True, **kwargs) - - if caller is not None: - caller.last_response = resp - - if not resp.ok: - raise requests.exceptions.HTTPError("%s %s" % (resp.status_code, resp.reason)) - - if onetime['raw']: - return resp.content - - if onetime['format'] == 'json': - return resp.json() - elif onetime['format'] == 'xml': - import lxml.etree - return lxml.etree.parse(resp.raw) - elif onetime['format'] == 'vdf': - import vdf - return vdf.load(resp.raw) def doc(self): print(self.__doc__) @@ -131,7 +155,7 @@ class WebAPI(object): def __doc__(self): doc = "Steam Web API - List of all interfaces\n\n" for interface in self.interfaces: - doc += interface.doc() + doc += interface.__doc__ return doc @@ -140,7 +164,7 @@ class WebAPIInterface(object): Steam Web API Interface """ - def __init__(self, interface_dict, parent=None): + def __init__(self, interface_dict, parent): self._parent = parent self.name = interface_dict['name'] self.methods = [] @@ -167,10 +191,26 @@ class WebAPIInterface(object): def __iter__(self): return iter(self.methods) + @property + def key(self): + return self._parent.key + @property def https(self): return self._parent.https + @property + def http_timeout(self): + return self._parent.http_timeout + + @property + def format(self): + return self._parent.format + + @property + def raw(self): + return self._parent.raw + def doc(self): print(self.__doc__) @@ -178,7 +218,7 @@ class WebAPIInterface(object): def __doc__(self): doc = "%s\n%s\n" % (self.name, '-'*len(self.name)) for method in self.methods: - doc += " %s\n" % method.doc().replace("\n", "\n ") + doc += " %s\n" % method.__doc__.replace("\n", "\n ") return doc @@ -187,7 +227,7 @@ class WebAPIMethod(object): Steam Web API Interface Method """ - def __init__(self, method_dict, parent=None): + def __init__(self, method_dict, parent): self.last_response = None self._parent = parent self._dict = method_dict @@ -197,8 +237,9 @@ class WebAPIMethod(object): for param in params: # add property indicating param can be a list param['_array'] = param['name'].endswith('[0]') - # fix name - param['name'] = param['name'].rstrip('[0]') + # remove array suffix + if param['_array']: + param['name'] = param['name'][:-3] # turn params from a list to a dict self._dict['parameters'][param['name']] = param @@ -213,17 +254,19 @@ class WebAPIMethod(object): ) def __call__(self, **kwargs): - possible_kwargs = set(self._dict['parameters'].keys()) | set(['key', 'format', 'raw']) + possible_kwargs = set(self._dict['parameters'].keys()) | set(DEFAULT_PARAMS.keys()) unrecognized = set(kwargs.keys()).difference(possible_kwargs) if unrecognized: raise ValueError("Unrecognized parameter %s" % repr(unrecognized.pop())) params = {} # process special case kwargs - for param in ('key', 'format', 'raw'): + for param in DEFAULT_PARAMS.keys(): if param in kwargs: params[param] = kwargs[param] del kwargs[param] + else: + params[param] = getattr(self._parent, param) # process method parameters for param in self.parameters.values(): @@ -252,10 +295,10 @@ class WebAPIMethod(object): params[name] = kwargs[name] # make the request - return self._api_request( - self, - self.method, + return webapi_request( "%s/%s/v%s/" % (self._parent.name, self.name, self.version), + method=self.method, + caller=self, params=params, ) @@ -271,18 +314,10 @@ class WebAPIMethod(object): def parameters(self): return self._dict['parameters'] - @property - def _api_request(self): - return self._parent._parent._api_request - @property def name(self): return self._dict['name'] - @property - def https(self): - return self._parent.https - def doc(self): print(self.__doc__) diff --git a/tests/__init__.py b/tests/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/tests/test_steamid.py b/tests/test_steamid.py new file mode 100644 index 0000000..f3c6f74 --- /dev/null +++ b/tests/test_steamid.py @@ -0,0 +1,245 @@ +import unittest +import mock + +from steam.steamid import SteamID, requests +from steam.enums import EType, EUniverse + + +class SteamID_initialization(unittest.TestCase): + def compare(self, obj, test_list): + self.assertEqual(obj.id, test_list[0]) + self.assertEqual(obj.type, test_list[1]) + self.assertEqual(obj.universe, test_list[2]) + self.assertEqual(obj.instance, test_list[3]) + + def test_arg_toomany(self): + with self.assertRaises(ValueError): + SteamID(1, 2) + with self.assertRaises(ValueError): + SteamID(1, 2, 3) + with self.assertRaises(ValueError): + SteamID(1, 2, 3, 4) + + ###################################################### + # 1 ARG + ###################################################### + @mock.patch.object(requests, "get") + def test_arg_community_url_id(self, mock_requests_get): + ResponseMock = mock.Mock() + ResponseMock.content = 'var asd = {"steamid":"76580280500085312","key":5123}' + mock_requests_get.return_value = ResponseMock + + # http + self.compare(SteamID("http://steamcommunity.com/id/testvanity"), + [123456, EType.Individual, EUniverse.Public, 4444] + ) + + mock_requests_get.assert_called_with("http://steamcommunity.com/id/testvanity") + # https + self.compare(SteamID("https://steamcommunity.com/id/testvanity"), + [123456, EType.Individual, EUniverse.Public, 4444] + ) + mock_requests_get.assert_called_with("https://steamcommunity.com/id/testvanity") + # raise + ResponseMock.content = "no steamid json :(" + with self.assertRaises(ValueError): + self.compare(SteamID("https://steamcommunity.com/id/testvanity"), + [123456, EType.Individual, EUniverse.Public, 4444] + ) + + def test_arg_community_url_profiles(self): + # http + self.compare(SteamID("http://steamcommunity.com/profiles/76580280500085312"), + [123456, EType.Individual, EUniverse.Public, 4444] + ) + # https + self.compare(SteamID("https://steamcommunity.com/profiles/76580280500085312"), + [123456, EType.Individual, EUniverse.Public, 4444] + ) + + def test_arg_number_out_of_range(self): + self.assertRaises(ValueError, SteamID, -1) + self.assertRaises(ValueError, SteamID, '-1') + self.assertRaises(ValueError, SteamID, -5555555) + self.assertRaises(ValueError, SteamID, '-5555555') + self.assertRaises(ValueError, SteamID, 2**64) + self.assertRaises(ValueError, SteamID, str(2**64)) + self.assertRaises(ValueError, SteamID, 2**128) + self.assertRaises(ValueError, SteamID, str(2**128)) + + def test_arg_steam32(self): + self.compare(SteamID(1), + [1, EType.Individual, EUniverse.Public, 1]) + self.compare(SteamID('1'), + [1, EType.Individual, EUniverse.Public, 1]) + self.compare(SteamID(12), + [12, EType.Individual, EUniverse.Public, 1]) + self.compare(SteamID('12'), + [12, EType.Individual, EUniverse.Public, 1]) + self.compare(SteamID(123), + [123, EType.Individual, EUniverse.Public, 1]) + self.compare(SteamID('123'), + [123, EType.Individual, EUniverse.Public, 1]) + self.compare(SteamID(12345678), + [12345678, EType.Individual, EUniverse.Public, 1]) + self.compare(SteamID('12345678'), + [12345678, EType.Individual, EUniverse.Public, 1]) + self.compare(SteamID(0xffffFFFF), + [0xffffFFFF, EType.Individual, EUniverse.Public, 1]) + self.compare(SteamID(str(0xffffFFFF)), + [0xffffFFFF, EType.Individual, EUniverse.Public, 1]) + + def test_arg_steam64(self): + self.compare(SteamID(76580280500085312), + [123456, EType.Individual, EUniverse.Public, 4444] + ) + self.compare(SteamID('76580280500085312'), + [123456, EType.Individual, EUniverse.Public, 4444] + ) + self.compare(SteamID(103582791429521412), + [4, EType.Clan, EUniverse.Public, 0] + ) + self.compare(SteamID('103582791429521412'), + [4, EType.Clan, EUniverse.Public, 0] + ) + + ###################################################### + # 1 arg - steam2/steam3 format + ###################################################### + def test_arg_text_invalid(self): + with self.assertRaises(ValueError): + SteamID("randomtext") + + def test_arg_steam2(self): + self.compare(SteamID("STEAM_0:1:1"), + [3, EType.Individual, EUniverse.Public, 1] + ) + self.compare(SteamID("STEAM_1:1:1"), + [3, EType.Individual, EUniverse.Public, 1] + ) + self.compare(SteamID("STEAM_0:0:4"), + [8, EType.Individual, EUniverse.Public, 1] + ) + self.compare(SteamID("STEAM_1:0:4"), + [8, EType.Individual, EUniverse.Public, 1] + ) + + def test_arg_steam3(self): + self.compare(SteamID("[U:1:1234]"), + [1234, EType.Individual, EUniverse.Public, 1] + ) + self.compare(SteamID("[G:1:1234]"), + [1234, EType.GameServer, EUniverse.Public, 1] + ) + self.compare(SteamID("[g:1:4]"), + [4, EType.Clan, EUniverse.Public, 0] + ) + self.compare(SteamID("[A:1:4]"), + [4, EType.AnonGameServer, EUniverse.Public, 0] + ) + self.compare(SteamID("[A:1:1234:567]"), + [1234, EType.AnonGameServer, EUniverse.Public, 567] + ) + + ###################################################### + # KWARGS + ###################################################### + def test_kwarg_id(self): + # id kwarg is required always + with self.assertRaises(ValueError): + SteamID(instance=0) + SteamID(id=None) + + self.assertEqual(SteamID(id=555).id, 555) + self.assertEqual(SteamID(id='555').id, 555) + + def test_kwarg_type(self): + with self.assertRaises(KeyError): + SteamID(id=5, type="doesn't exist") + with self.assertRaises(ValueError): + SteamID(id=5, type=99999999) + with self.assertRaises(AttributeError): + SteamID(id=5, type=None) + + self.assertEqual(SteamID(id=5, type=1).type, EType.Individual) + self.assertEqual(SteamID(id=5, type='Individual').type, EType.Individual) + self.assertEqual(SteamID(id=5, type='iNDIVIDUAL').type, EType.Individual) + + def test_kwarg_universe(self): + with self.assertRaises(KeyError): + SteamID(id=5, universe="doesn't exist") + with self.assertRaises(ValueError): + SteamID(id=5, universe=99999999) + with self.assertRaises(AttributeError): + SteamID(id=5, universe=None) + + self.assertEqual(SteamID(id=5, universe=1).universe, EUniverse.Public) + self.assertEqual(SteamID(id=5, universe='Public').universe, EUniverse.Public) + self.assertEqual(SteamID(id=5, universe='pUBLIC').universe, EUniverse.Public) + + def test_kwarg_instance(self): + self.assertEqual(SteamID(id=5, instance=1234).instance, 1234) + + for etype in EType: + self.assertEqual(SteamID(id=5, type=etype).instance, + 1 if etype in (EType.Individual, EType.GameServer) else 0) + + def test_kwargs_invalid(self): + invalid = [0, EType.Invalid, EUniverse.Invalid, 0] + + self.compare(SteamID(), invalid) + self.compare(SteamID(id=0, type=0, universe=0, instance=0), invalid) + self.compare(SteamID(id=0, + type=EType.Invalid, + universe=EUniverse.Invalid, + instance=0, + ), invalid) + self.compare(SteamID(id=0, + type='Invalid', + universe='Invalid', + instance=0, + ), invalid) + self.compare(SteamID(id=0, + type='iNVALID', + universe='iNVALID', + instance=0, + ), invalid) + + +class SteamID_properties(unittest.TestCase): + def test_repr(self): + # just to cover in coverage + repr(SteamID()) + + def test_str(self): + self.assertEqual(str(SteamID(76580280500085312)), '76580280500085312') + + def test_as_steam2(self): + self.assertEqual(SteamID('STEAM_0:1:4').as_steam2, 'STEAM_0:1:4') + self.assertEqual(SteamID('STEAM_1:1:4').as_steam2, 'STEAM_0:1:4') + + def test_as_steam3(self): + self.assertEqual(SteamID('[U:1:1234]').as_steam3, '[U:1:1234]') + self.assertEqual(SteamID('[g:1:4]').as_steam3, '[g:1:4]') + self.assertEqual(SteamID('[A:1:1234:567]').as_steam3, '[A:1:1234:567]') + self.assertEqual(SteamID('[G:1:1234:567]').as_steam3, '[G:1:1234]') + + def test_as_32(self): + self.assertEqual(SteamID(76580280500085312).as_32, 123456) + + def test_as_64(self): + self.assertEqual(SteamID(76580280500085312).as_64, 76580280500085312) + + def test_community_url(self): + # user url + self.assertEqual(SteamID(76580280500085312).community_url, + 'https://steamcommunity.com/profiles/76580280500085312' + ) + # group url + self.assertEqual(SteamID('[g:1:4]').community_url, + 'https://steamcommunity.com/gid/103582791429521412' + ) + # else None + self.assertEqual(SteamID('[A:1:4]').community_url, + None + ) diff --git a/tests/test_webapi.py b/tests/test_webapi.py new file mode 100644 index 0000000..37f8e0a --- /dev/null +++ b/tests/test_webapi.py @@ -0,0 +1,14 @@ +import unittest +import mock +import vcr + +from steam.webapi import WebAPI, requests +from steam.enums import EType, EUniverse + +test_api_key = 'test_api_key' + + +class TCwebapi(unittest.TestCase): + @vcr.use_cassette('vcr/webapi_init.json', mode='once', serializer='json') + def test_initialization(self): + api = WebAPI(test_api_key) diff --git a/vcr/webapi_getserverinfo.json b/vcr/webapi_getserverinfo.json new file mode 100644 index 0000000..739acbc --- /dev/null +++ b/vcr/webapi_getserverinfo.json @@ -0,0 +1,49 @@ +{ + "version": 1, + "interactions": [ + { + "request": { + "body": null, + "headers": { + "Connection": [ + "keep-alive" + ], + "Accept-Encoding": [ + "gzip, deflate" + ], + "Accept": [ + "*/*" + ], + "User-Agent": [ + "python-requests/2.7.0 CPython/2.7.10 CYGWIN_NT-10.0/2.2.0(0.289/5/3)" + ] + }, + "method": "GET", + "uri": "https://api.steampowered.com/ISteamWebAPIUtil/GetServerInfo/v1/?key=test_api_key&format=json" + }, + "response": { + "status": { + "message": "OK", + "code": 200 + }, + "headers": { + "date": [ + "Sun, 23 Aug 2015 22:58:10 GMT" + ], + "content-length": [ + "78" + ], + "expires": [ + "Sun, 23 Aug 2015 22:58:10 GMT" + ], + "content-type": [ + "application/json; charset=UTF-8" + ] + }, + "body": { + "string": "{\n\t\"servertime\": 1440370690,\n\t\"servertimestring\": \"Sun Aug 23 15:58:10 2015\"\n}" + } + } + } + ] +} diff --git a/vcr/webapi_init.json b/vcr/webapi_init.json new file mode 100644 index 0000000..f8c2e75 --- /dev/null +++ b/vcr/webapi_init.json @@ -0,0 +1,49 @@ +{ + "version": 1, + "interactions": [ + { + "request": { + "body": null, + "headers": { + "Connection": [ + "keep-alive" + ], + "Accept-Encoding": [ + "gzip, deflate" + ], + "Accept": [ + "*/*" + ], + "User-Agent": [ + "python-requests/2.7.0 CPython/2.7.10 CYGWIN_NT-10.0/2.2.0(0.289/5/3)" + ] + }, + "method": "GET", + "uri": "https://api.steampowered.com/ISteamWebAPIUtil/GetSupportedAPIList/v1/?key=test_api_key&format=json" + }, + "response": { + "status": { + "message": "OK", + "code": 200 + }, + "headers": { + "date": [ + "Sun, 23 Aug 2015 22:58:35 GMT" + ], + "content-length": [ + "13742" + ], + "expires": [ + "Sun, 23 Aug 2015 22:58:35 GMT" + ], + "content-type": [ + "application/json; charset=UTF-8" + ] + }, + "body": { + "string": "{\n\t\"apilist\": {\n\t\t\"interfaces\": [\n\t\t\t{\n\t\t\t\t\"name\": \"IGCVersion_205790\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetClientVersion\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetServerVersion\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"IGCVersion_440\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetClientVersion\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetServerVersion\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"IGCVersion_570\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetClientVersion\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetServerVersion\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"IGCVersion_730\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetServerVersion\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"IPortal2Leaderboards_620\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetBucketizedData\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"leaderboardName\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"The leaderboard name to fetch data for.\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"IPortal2Leaderboards_841\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetBucketizedData\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"leaderboardName\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"The leaderboard name to fetch data for.\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamApps\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetAppList\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetAppList\",\n\t\t\t\t\t\t\"version\": 2,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetServersAtAddress\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"addr\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"IP or IP:queryport to list\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"UpToDateCheck\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"appid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"AppID of game\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"version\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"The installed version of the game\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamDirectory\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetCMList\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"cellid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"Client's Steam cell ID\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"maxcount\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"Max number of servers to return\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamEnvoy\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"PaymentOutNotification\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"POST\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"PaymentOutReversalNotification\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"POST\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamNews\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetNewsForApp\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"appid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"AppID to retrieve news for\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"maxlength\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"Maximum length for the content to return, if this is 0 the full content is returned, if it's less then a blurb is generated to fit.\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"enddate\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"Retrieve posts earlier than this date (unix epoch timestamp)\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"count\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"# of posts to retrieve (default 20)\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetNewsForApp\",\n\t\t\t\t\t\t\"version\": 2,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"appid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"AppID to retrieve news for\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"maxlength\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"Maximum length for the content to return, if this is 0 the full content is returned, if it's less then a blurb is generated to fit.\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"enddate\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"Retrieve posts earlier than this date (unix epoch timestamp)\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"count\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"# of posts to retrieve (default 20)\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"feeds\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"Comma-seperated list of feed names to return news for\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamPayPalPaymentsHub\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"PayPalPaymentsHubPaymentNotification\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"POST\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamRemoteStorage\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetCollectionDetails\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"POST\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"collectioncount\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"Number of collections being requested\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"publishedfileids[0]\",\n\t\t\t\t\t\t\t\t\"type\": \"uint64\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"collection ids to get the details for\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetPublishedFileDetails\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"POST\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"itemcount\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"Number of items being requested\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"publishedfileids[0]\",\n\t\t\t\t\t\t\t\t\"type\": \"uint64\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"published file id to look up\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamUserAuth\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"AuthenticateUser\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"POST\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"steamid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint64\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"Should be the users steamid, unencrypted.\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"sessionkey\",\n\t\t\t\t\t\t\t\t\"type\": \"rawbinary\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"Should be a 32 byte random blob of data, which is then encrypted with RSA using the Steam system's public key. Randomness is important here for security.\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"encrypted_loginkey\",\n\t\t\t\t\t\t\t\t\"type\": \"rawbinary\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"Should be the users hashed loginkey, AES encrypted with the sessionkey.\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamUserOAuth\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetTokenDetails\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"access_token\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"OAuth2 token for which to return details\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamUserStats\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetGlobalAchievementPercentagesForApp\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"gameid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint64\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"GameID to retrieve the achievement percentages for\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetGlobalAchievementPercentagesForApp\",\n\t\t\t\t\t\t\"version\": 2,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"gameid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint64\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"GameID to retrieve the achievement percentages for\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetGlobalStatsForGame\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"appid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"AppID that we're getting global stats for\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"count\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"Number of stats get data for\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"name[0]\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"Names of stat to get data for\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"startdate\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"Start date for daily totals (unix epoch timestamp)\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"enddate\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"End date for daily totals (unix epoch timestamp)\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetNumberOfCurrentPlayers\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"appid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"AppID that we're getting user count for\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamWebAPIUtil\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetServerInfo\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"GetSupportedAPIList\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"GET\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"key\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"access key\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"ISteamWebUserPresenceOAuth\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"PollStatus\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"POST\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"steamid\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"Steam ID of the user\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"umqid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint64\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"UMQ Session ID\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"message\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": false,\n\t\t\t\t\t\t\t\t\"description\": \"Message that was last known to the user\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"pollid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"Caller-specific poll id\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"sectimeout\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"Long-poll timeout in seconds\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"secidletime\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"How many seconds is client considering itself idle, e.g. screen is off\"\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"use_accountids\",\n\t\t\t\t\t\t\t\t\"type\": \"uint32\",\n\t\t\t\t\t\t\t\t\"optional\": true,\n\t\t\t\t\t\t\t\t\"description\": \"Boolean, 0 (default): return steamid_from in output, 1: return accountid_from\"\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"IPlayerService\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"RecordOfflinePlaytime\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"POST\",\n\t\t\t\t\t\t\"description\": \"Tracks playtime for a user when they are offline\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"steamid\",\n\t\t\t\t\t\t\t\t\"type\": \"uint64\",\n\t\t\t\t\t\t\t\t\"optional\": false\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"ticket\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"play_sessions\",\n\t\t\t\t\t\t\t\t\"type\": \"{message}\",\n\t\t\t\t\t\t\t\t\"optional\": false\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t},\n\t\t\t{\n\t\t\t\t\"name\": \"IAccountRecoveryService\",\n\t\t\t\t\"methods\": [\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"ReportAccountRecoveryData\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"POST\",\n\t\t\t\t\t\t\"description\": \"Send account recovery data\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"loginuser_list\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"install_config\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"shasentryfile\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"machineid\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t},\n\t\t\t\t\t{\n\t\t\t\t\t\t\"name\": \"RetrieveAccountRecoveryData\",\n\t\t\t\t\t\t\"version\": 1,\n\t\t\t\t\t\t\"httpmethod\": \"POST\",\n\t\t\t\t\t\t\"description\": \"Send account recovery data\",\n\t\t\t\t\t\t\"parameters\": [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\"name\": \"requesthandle\",\n\t\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\t\"optional\": false\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t]\n\t\t\t\t\t\t\n\t\t\t\t\t}\n\t\t\t\t]\n\t\t\t\t\n\t\t\t}\n\t\t]\n\t\t\n\t}\n}" + } + } + } + ] +} diff --git a/vcr/webapi_resolovevanityurl.json b/vcr/webapi_resolovevanityurl.json new file mode 100644 index 0000000..4d474df --- /dev/null +++ b/vcr/webapi_resolovevanityurl.json @@ -0,0 +1,49 @@ +{ + "version": 1, + "interactions": [ + { + "request": { + "body": null, + "headers": { + "Connection": [ + "keep-alive" + ], + "Accept-Encoding": [ + "gzip, deflate" + ], + "Accept": [ + "*/*" + ], + "User-Agent": [ + "python-requests/2.7.0 CPython/2.7.10 CYGWIN_NT-10.0/2.2.0(0.289/5/3)" + ] + }, + "method": "GET", + "uri": "https://api.steampowered.com/ISteamUser/ResolveVanityURL/v1/?url_type=2&vanityurl=valve&key=test_api_key&format=json" + }, + "response": { + "status": { + "message": "OK", + "code": 200 + }, + "headers": { + "date": [ + "Sun, 23 Aug 2015 22:58:21 GMT" + ], + "content-length": [ + "71" + ], + "expires": [ + "Sun, 23 Aug 2015 22:58:21 GMT" + ], + "content-type": [ + "application/json; charset=UTF-8" + ] + }, + "body": { + "string": "{\n\t\"response\": {\n\t\t\"steamid\": \"103582791429521412\",\n\t\t\"success\": 1\n\t}\n}" + } + } + } + ] +} \ No newline at end of file