From 1b4dba9c6dcc807d13c730a015d9dc6ec3daca11 Mon Sep 17 00:00:00 2001 From: Rossen Georgiev Date: Sun, 7 Jul 2019 16:52:14 +0100 Subject: [PATCH] update SteamError and add on CDNClient --- docs/api/steam.exceptions.rst | 7 ++++ docs/api/steam.rst | 1 + steam/client/cdn.py | 61 +++++++++++++++++++++-------------- steam/exceptions.py | 6 +++- 4 files changed, 50 insertions(+), 25 deletions(-) create mode 100644 docs/api/steam.exceptions.rst diff --git a/docs/api/steam.exceptions.rst b/docs/api/steam.exceptions.rst new file mode 100644 index 0000000..b7630b9 --- /dev/null +++ b/docs/api/steam.exceptions.rst @@ -0,0 +1,7 @@ +exceptions +========== + +.. automodule:: steam.exceptions + :members: + :undoc-members: + :show-inheritance: diff --git a/docs/api/steam.rst b/docs/api/steam.rst index 24895ca..eb568f8 100644 --- a/docs/api/steam.rst +++ b/docs/api/steam.rst @@ -7,6 +7,7 @@ API Reference steam.client steam.core steam.enums + steam.exceptions steam.game_servers steam.globalid steam.guard diff --git a/steam/client/cdn.py b/steam/client/cdn.py index 337dbd4..79735a5 100644 --- a/steam/client/cdn.py +++ b/steam/client/cdn.py @@ -95,8 +95,9 @@ import vdf from gevent.pool import Pool as GPool from cachetools import LRUCache from steam import webapi +from steam.exceptions import SteamError from steam.core.msg import MsgProto -from steam.enums import EResult, EServerType, EType +from steam.enums import EResult, EType from steam.enums.emsg import EMsg from steam.util.web import make_requests_session from steam.core.crypto import symmetric_decrypt, symmetric_decrypt_ecb @@ -280,7 +281,7 @@ class CDNClient(object): self.servers.extend(servers) if not self.servers: - raise ValueError("Failed to fetch content servers") + raise SteamError("Failed to fetch content servers") def get_content_server(self, rotate=False): """Get a CS server for content download @@ -301,6 +302,7 @@ class CDNClient(object): :type depot_id: int :return: returns decryption key :rtype: bytes + :raises SteamError: error message """ if (app_id, depot_id) not in self.depot_keys: msg = self.steam.get_depot_key(app_id, depot_id) @@ -308,12 +310,22 @@ class CDNClient(object): if msg and msg.eresult == EResult.OK: self.depot_keys[(app_id, depot_id)] = msg.depot_encryption_key else: - raise ValueError("Failed getting depot key: %s" % repr( - EResult.Timeout if msg is None else EResult(msg.eresult))) + raise SteamError("Failed getting depot key", + EResult.Timeout if msg is None else EResult(msg.eresult)) return self.depot_keys[(app_id, depot_id)] - def get(self, command, args): + def cdn_cmd(self, command, args): + """Run CDN command request + + :param command: command name + :type command: str + :param args: args + :type args: str + :returns: requests response + :rtype: :class:`requests.Response` + :raises SteamError: on error + """ server = self.get_content_server() while True: @@ -332,8 +344,8 @@ class CDNClient(object): else: if resp.ok: return resp - elif resp.status_code in (401, 403, 404): - resp.raise_for_status() + elif resp.status_code >= 400: + raise SteamError("HTTP Error %s" % resp.status_code) server = self.get_content_server(rotate=True) @@ -348,17 +360,18 @@ class CDNClient(object): :type chunk_id: int :returns: chunk data :rtype: bytes + :raises SteamError: error message """ if (depot_id, chunk_id) not in self._chunk_cache: - resp = self.get('depot', '%s/chunk/%s' % (depot_id, chunk_id)) + resp = self.cdn_cmd('depot', '%s/chunk/%s' % (depot_id, chunk_id)) data = symmetric_decrypt(resp.content, self.get_depot_key(app_id, depot_id)) if data[:2] == b'VZ': if data[-2:] != b'zv': - raise ValueError("VZ: Invalid footer: %s" % repr(data[-2:])) + raise SteamError("VZ: Invalid footer: %s" % repr(data[-2:])) if data[2:3] != b'a': - raise ValueError("VZ: Invalid version: %s" % repr(data[2:3])) + raise SteamError("VZ: Invalid version: %s" % repr(data[2:3])) vzfilter = lzma._decode_filter_properties(lzma.FILTER_LZMA1, data[7:12]) vzdec = lzma.LZMADecompressor(lzma.FORMAT_RAW, filters=[vzfilter]) @@ -366,7 +379,7 @@ class CDNClient(object): # i have no idea why, but some chunks will decompress with 1 extra byte at the end data = vzdec.decompress(data[12:-10])[:decompressed_size] if crc32(data) != checksum: - raise ValueError("VZ: CRC32 checksum doesn't match for decompressed data") + raise SteamError("VZ: CRC32 checksum doesn't match for decompressed data") else: with ZipFile(BytesIO(data)) as zf: data = zf.read(zf.filelist[0]) @@ -390,7 +403,7 @@ class CDNClient(object): :rtype: :class:`.CDNDepotManifest` """ if (app_id, depot_id, manifest_gid) not in self.manifests: - resp = self.get('depot', '%s/manifest/%s/5' % (depot_id, manifest_gid)) + resp = self.cdn_cmd('depot', '%s/manifest/%s/5' % (depot_id, manifest_gid)) if resp.ok: manifest = CDNDepotManifest(self, app_id, resp.content) @@ -436,6 +449,7 @@ class CDNClient(object): Function to filter depots. ``func(depot_id, depot_info)`` :returns: list of :class:`.CDNDepotManifest` :rtype: :class:`list` [:class:`.CDNDepotManifest`] + :raises SteamError: error message """ if app_id not in self.app_depots: self.app_depots[app_id] = self.steam.get_product_info([app_id])['apps'][app_id]['depots'] @@ -444,21 +458,21 @@ class CDNClient(object): is_enc_branch = False if branch not in depots['branches']: - raise ValueError("No branch named %s for app_id %s" % (repr(branch), app_id)) + raise SteamError("No branch named %s for app_id %s" % (repr(branch), app_id)) elif int(depots['branches'][branch].get('pwdrequired', 0)) > 0: is_enc_branch = True if (app_id, branch) not in self.beta_passwords: if not password: - raise ValueError("Branch %r requires a password" % branch) + raise SteamError("Branch %r requires a password" % branch) result = self.check_beta_password(app_id, password) if result != EResult.OK: - raise ValueError("Branch password is not valid. %r" % result) + raise SteamError("Branch password is not valid. %r" % result) if (app_id, branch) not in self.beta_passwords: - raise ValueError("Incorrect password for branch %r" % branch) + raise SteamError("Incorrect password for branch %r" % branch) def async_fetch_manifest(app_id, depot_id, manifest_gid, name): manifest = self.get_manifest(app_id, depot_id, manifest_gid) @@ -520,7 +534,7 @@ class CDNClient(object): for task in tasks: try: result = task.get() - except ValueError as exp: + except SteamError as exp: self._LOG.error("Depot %s (%s): %s", repr(depot_info['name']), depot_id, @@ -549,7 +563,7 @@ class CDNClient(object): :param filter_func: Function to filter depots. ``func(depot_id, depot_info)`` :returns: generator of of CDN files - :rtype: :class:`.CDNDepotFile` + :rtype: [:class:`.CDNDepotFile`] """ for manifest in self.get_manifests(app_id, branch, password, filter_func): for fp in manifest.iter_files(filename_filter): @@ -562,8 +576,7 @@ class CDNClient(object): :type item_id: int :returns: manifest instance :rtype: :class:`.CDNDepotManifest` - :raises: steam error - :rtype: :class:`.SteamError` + :raises SteamError: error message """ resp = self.steam.send_um_and_wait('PublishedFile.GetDetails#1', { 'publishedfileids': [item_id], @@ -579,15 +592,15 @@ class CDNClient(object): }, timeout=7) if resp.header.eresult != EResult.OK: - raise SteamError(resp.header.error_message, resp.header.eresult) + raise SteamError(resp.header.error_message or 'No message', resp.header.eresult) wf = None if resp is None else resp.body.publishedfiledetails[0] if wf is None or wf.result != EResult.OK: - raise ValueError("Failed getting workshop file info: %s" % repr( - EResult.Timeout if resp is None else EResult(wf.result))) + raise SteamError("Failed getting workshop file info", + EResult.Timeout if resp is None else EResult(wf.result)) elif not wf.hcontent_file: - raise ValueError("Workshop file is not on SteamPipe") + raise SteamError("Workshop file is not on SteamPipe", EResult.FileNotFound) app_id = ws_app_id = wf.consumer_appid diff --git a/steam/exceptions.py b/steam/exceptions.py index 76f23f8..5994e26 100644 --- a/steam/exceptions.py +++ b/steam/exceptions.py @@ -3,5 +3,9 @@ from steam.enums import EResult class SteamError(Exception): def __init__(self, message, eresult=EResult.Fail): - Exception.__init__(self, message) + Exception.__init__(self, message, eresult) + self.message = message self.eresult = EResult(eresult) #: :class:`.EResult` + + def __str__(self): + return "(%s) %s" % (self.eresult, self.message)