""" WebAPI provides a thin gevent cooperative wrapper over Steam's WebAPI Calling an endpoint .. code:: python >>> api = WebAPI(key) >>> api.ISteamUser.ResolveVanityURL(vanityurl="valve", url_type=2) >>> api.call('ISteamUser.ResolveVanityURL', vanityurl="valve", url_type=2) {u'response': {u'steamid': u'103582791429521412', u'success': 1}} All globals params (``key``, ``https``, ``format``, ``raw``) can be specified on per call basis. .. code:: python >>> print a.ISteamUser.ResolveVanityURL(format='vdf', raw=True, vanityurl="valve", url_type=2) "response" { "steamid" "103582791429521412" "success" "1" } """ from steam.util.web import make_requests_session 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={}, session=None): """ Low level function for calling Steam's WebAPI :param path: request url :type path: :class:`str` :param method: HTTP method (GET or POST) :type method: :class:`str` :param caller: caller reference, caller.last_response is set to the last response :param params: dict of WebAPI and endpoint specific params :type params: :class:`dict` :param session: an instance requests session, or one is created per call :type session: :class:`requests.Session` :return: response based on paramers :rtype: :class:`dict`, :class:`lxml.etree.Element`, :class:`str` """ 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(session, method.lower()) resp = f(path, stream=False, timeout=onetime['http_timeout'], **kwargs) # we keep a reference of the last response instance on the caller if caller is not None: caller.last_response = resp # 4XX and 5XX will cause this to raise resp.raise_for_status() # response if onetime['raw']: return resp.text if onetime['format'] == 'json': return resp.json() elif onetime['format'] == 'xml': import lxml.etree return lxml.etree.fromstring(resp.content) elif onetime['format'] == 'vdf': import vdf return vdf.loads(resp.text) class WebAPI(object): """ Steam WebAPI wrapper. See https://developer.valvesoftware.com/wiki/Steam_Web_API .. note:: Interfaces and methods are populated automatically from WebAPI. :param key: api key from https://steamcommunity.com/dev/apikey :type key: :class:`str` :param format: response format, either (``json``, ``vdf``, or ``xml``) only when ``raw=False`` :type format: :class:`str` :param raw: return raw response :type raw: class:`bool` :param https: use ``https`` :type https: :class:`bool` :param http_timeout: HTTP timeout in seconds :type http_timeout: :class:`int` :param auto_load_interfaces: load interfaces from the WebAPI :type auto_load_interfaces: :class:`bool` These can be specified per method call for one off calls """ key = DEFAULT_PARAMS['key'] format = DEFAULT_PARAMS['format'] raw = DEFAULT_PARAMS['raw'] https = DEFAULT_PARAMS['https'] http_timeout = DEFAULT_PARAMS['http_timeout'] interfaces = [] def __init__(self, key, format = DEFAULT_PARAMS['format'], raw = DEFAULT_PARAMS['raw'], https = DEFAULT_PARAMS['https'], http_timeout = DEFAULT_PARAMS['http_timeout'], auto_load_interfaces = True): self.key = key #: api key self.format = format #: format (``json``, ``vdf``, or ``xml``) self.raw = raw #: return raw reponse or parse self.https = https #: use https or not self.http_timeout = http_timeout #: HTTP timeout in seconds self.interfaces = [] #: list of all interfaces self.session = make_requests_session() #: :class:`requests.Session` from :func:`steam.util.web.make_requests_session` if auto_load_interfaces: self.load_interfaces(self.fetch_interfaces()) def __repr__(self): return "%s(key=%s, https=%s)" % ( self.__class__.__name__, repr(self.key), repr(self.https), ) def fetch_interfaces(self): """ Returns a dict with the response from ``GetSupportedAPIList`` :return: :class:`dict` of all interfaces and methods The returned value can passed to :py:func:`WebAPI.load_interfaces` """ return webapi_request( "ISteamWebAPIUtil/GetSupportedAPIList/v1/", method="GET", caller=None, params={'format': 'json', 'key': self.key, }, session=self.session, ) 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 = interfaces_dict['apilist']['interfaces'] if len(interfaces) == 0: raise ValueError("API returned not interfaces; probably using invalid key") # clear existing interface instances for interface in self.interfaces: delattr(self, interface.name) self.interfaces = [] # create interface instances from response for interface in interfaces: obj = WebAPIInterface(interface, parent=self) self.interfaces.append(obj) setattr(self, obj.name, obj) def call(self, method_path, **kwargs): """ Make an API call for specific method :param method_path: format ``Interface.Method`` (e.g. ``ISteamWebAPIUtil.GetServerInfo``) :type method_path: :class:`str` :param kwargs: keyword arguments for the specific method :return: response :rtype: :class:`dict`, :class:`lxml.etree.Element` or :class:`str` """ interface, method = method_path.split('.', 1) return getattr(getattr(self, interface), method)(**kwargs) def doc(self): """ Print documentation for all interfaces and their methods """ doc = "Steam Web API - List of all interfaces\n\n" for interface in self.interfaces: doc += interface.__doc__ print(doc) class WebAPIInterface(object): """ Steam Web API Interface """ def __init__(self, interface_dict, parent): self._parent = parent self.name = interface_dict['name'] self.methods = [] for method in interface_dict['methods']: obj = WebAPIMethod(method, parent=self) self.methods.append(obj) # map the method name as attribute including version setattr(self, "%s_v%d" % (obj.name, obj.version), obj) # without version, but th refernce of latest version current_obj = getattr(self, obj.name, None) if current_obj is None or current_obj.version < obj.version: setattr(self, obj.name, obj) def __repr__(self): return "<%s %s with %s methods>" % ( self.__class__.__name__, repr(self.name), repr(len(list(self))), ) 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 @property def session(self): return self._parent.session def doc(self): print(self.__doc__) @property 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 ") return doc class WebAPIMethod(object): """ Steam Web API Interface Method """ def __init__(self, method_dict, parent): self.last_response = None self._parent = parent self._dict = method_dict params = method_dict['parameters'] self._dict['parameters'] = {} for param in params: # add property indicating param can be a list param['_array'] = param['name'].endswith('[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 def __repr__(self): return "<%s %s>" % ( self.__class__.__name__, repr("%s.%s_v%d" % ( self._parent.name, self.name, self.version, )), ) def __call__(self, **kwargs): 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 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(): name = param['name'] islist = param['_array'] optional = param['optional'] # raise if we are missing a required parameter if not optional and name not in kwargs and name != 'key': raise ValueError("Method requires %s to be set" % repr(name)) # populate params that will be passed to _api_request if name in kwargs: # some parameters can be an array, they need to be send as seperate field # the array index is append to the name (e.g. name[0], name[1] etc) if islist: if not isinstance(kwargs[name], list): raise ValueError("Expected %s to be a list, got %s" % ( repr(name), repr(type(kwargs[name]))) ) for idx, value in enumerate(kwargs[name]): params['%s[%d]' % (name, idx)] = value else: params[name] = kwargs[name] # make the request return webapi_request( "%s/%s/v%s/" % (self._parent.name, self.name, self.version), method=self.method, caller=self, params=params, session=self._parent.session, ) @property def version(self): return self._dict['version'] @property def method(self): return self._dict['httpmethod'] @property def parameters(self): return self._dict['parameters'] @property def name(self): return self._dict['name'] def doc(self): print(self.__doc__) @property def __doc__(self): doc = "%(httpmethod)s %(name)s (v%(version)04d)\n" % self._dict if 'description' in self._dict: doc += "\n %(description)s\n" % self._dict if len(self.parameters): doc += " \n Parameters:\n" for param in sorted(self.parameters.values(), key=lambda x: x['name']): doc += " %s %s %s%s\n" % ( param['name'].ljust(25), ((param['type']+"[]") if param['_array'] else param['type'] ).ljust(8), 'optional' if param['optional'] else 'required', (("\n - " + param['description']) if 'description' in param and param['description'] else '' ), ) return doc