Browse Source

update SteamError and add on CDNClient

pull/214/head
Rossen Georgiev 6 years ago
parent
commit
1b4dba9c6d
  1. 7
      docs/api/steam.exceptions.rst
  2. 1
      docs/api/steam.rst
  3. 61
      steam/client/cdn.py
  4. 6
      steam/exceptions.py

7
docs/api/steam.exceptions.rst

@ -0,0 +1,7 @@
exceptions
==========
.. automodule:: steam.exceptions
:members:
:undoc-members:
:show-inheritance:

1
docs/api/steam.rst

@ -7,6 +7,7 @@ API Reference
steam.client steam.client
steam.core steam.core
steam.enums steam.enums
steam.exceptions
steam.game_servers steam.game_servers
steam.globalid steam.globalid
steam.guard steam.guard

61
steam/client/cdn.py

@ -95,8 +95,9 @@ import vdf
from gevent.pool import Pool as GPool from gevent.pool import Pool as GPool
from cachetools import LRUCache from cachetools import LRUCache
from steam import webapi from steam import webapi
from steam.exceptions import SteamError
from steam.core.msg import MsgProto 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.enums.emsg import EMsg
from steam.util.web import make_requests_session from steam.util.web import make_requests_session
from steam.core.crypto import symmetric_decrypt, symmetric_decrypt_ecb from steam.core.crypto import symmetric_decrypt, symmetric_decrypt_ecb
@ -280,7 +281,7 @@ class CDNClient(object):
self.servers.extend(servers) self.servers.extend(servers)
if not self.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): def get_content_server(self, rotate=False):
"""Get a CS server for content download """Get a CS server for content download
@ -301,6 +302,7 @@ class CDNClient(object):
:type depot_id: int :type depot_id: int
:return: returns decryption key :return: returns decryption key
:rtype: bytes :rtype: bytes
:raises SteamError: error message
""" """
if (app_id, depot_id) not in self.depot_keys: if (app_id, depot_id) not in self.depot_keys:
msg = self.steam.get_depot_key(app_id, depot_id) msg = self.steam.get_depot_key(app_id, depot_id)
@ -308,12 +310,22 @@ class CDNClient(object):
if msg and msg.eresult == EResult.OK: if msg and msg.eresult == EResult.OK:
self.depot_keys[(app_id, depot_id)] = msg.depot_encryption_key self.depot_keys[(app_id, depot_id)] = msg.depot_encryption_key
else: else:
raise ValueError("Failed getting depot key: %s" % repr( raise SteamError("Failed getting depot key",
EResult.Timeout if msg is None else EResult(msg.eresult))) EResult.Timeout if msg is None else EResult(msg.eresult))
return self.depot_keys[(app_id, depot_id)] 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() server = self.get_content_server()
while True: while True:
@ -332,8 +344,8 @@ class CDNClient(object):
else: else:
if resp.ok: if resp.ok:
return resp return resp
elif resp.status_code in (401, 403, 404): elif resp.status_code >= 400:
resp.raise_for_status() raise SteamError("HTTP Error %s" % resp.status_code)
server = self.get_content_server(rotate=True) server = self.get_content_server(rotate=True)
@ -348,17 +360,18 @@ class CDNClient(object):
:type chunk_id: int :type chunk_id: int
:returns: chunk data :returns: chunk data
:rtype: bytes :rtype: bytes
:raises SteamError: error message
""" """
if (depot_id, chunk_id) not in self._chunk_cache: 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)) data = symmetric_decrypt(resp.content, self.get_depot_key(app_id, depot_id))
if data[:2] == b'VZ': if data[:2] == b'VZ':
if data[-2:] != b'zv': 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': 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]) vzfilter = lzma._decode_filter_properties(lzma.FILTER_LZMA1, data[7:12])
vzdec = lzma.LZMADecompressor(lzma.FORMAT_RAW, filters=[vzfilter]) 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 # 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] data = vzdec.decompress(data[12:-10])[:decompressed_size]
if crc32(data) != checksum: 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: else:
with ZipFile(BytesIO(data)) as zf: with ZipFile(BytesIO(data)) as zf:
data = zf.read(zf.filelist[0]) data = zf.read(zf.filelist[0])
@ -390,7 +403,7 @@ class CDNClient(object):
:rtype: :class:`.CDNDepotManifest` :rtype: :class:`.CDNDepotManifest`
""" """
if (app_id, depot_id, manifest_gid) not in self.manifests: 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: if resp.ok:
manifest = CDNDepotManifest(self, app_id, resp.content) manifest = CDNDepotManifest(self, app_id, resp.content)
@ -436,6 +449,7 @@ class CDNClient(object):
Function to filter depots. ``func(depot_id, depot_info)`` Function to filter depots. ``func(depot_id, depot_info)``
:returns: list of :class:`.CDNDepotManifest` :returns: list of :class:`.CDNDepotManifest`
:rtype: :class:`list` [:class:`.CDNDepotManifest`] :rtype: :class:`list` [:class:`.CDNDepotManifest`]
:raises SteamError: error message
""" """
if app_id not in self.app_depots: if app_id not in self.app_depots:
self.app_depots[app_id] = self.steam.get_product_info([app_id])['apps'][app_id]['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 is_enc_branch = False
if branch not in depots['branches']: 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: elif int(depots['branches'][branch].get('pwdrequired', 0)) > 0:
is_enc_branch = True is_enc_branch = True
if (app_id, branch) not in self.beta_passwords: if (app_id, branch) not in self.beta_passwords:
if not password: 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) result = self.check_beta_password(app_id, password)
if result != EResult.OK: 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: 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): def async_fetch_manifest(app_id, depot_id, manifest_gid, name):
manifest = self.get_manifest(app_id, depot_id, manifest_gid) manifest = self.get_manifest(app_id, depot_id, manifest_gid)
@ -520,7 +534,7 @@ class CDNClient(object):
for task in tasks: for task in tasks:
try: try:
result = task.get() result = task.get()
except ValueError as exp: except SteamError as exp:
self._LOG.error("Depot %s (%s): %s", self._LOG.error("Depot %s (%s): %s",
repr(depot_info['name']), repr(depot_info['name']),
depot_id, depot_id,
@ -549,7 +563,7 @@ class CDNClient(object):
:param filter_func: :param filter_func:
Function to filter depots. ``func(depot_id, depot_info)`` Function to filter depots. ``func(depot_id, depot_info)``
:returns: generator of of CDN files :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 manifest in self.get_manifests(app_id, branch, password, filter_func):
for fp in manifest.iter_files(filename_filter): for fp in manifest.iter_files(filename_filter):
@ -562,8 +576,7 @@ class CDNClient(object):
:type item_id: int :type item_id: int
:returns: manifest instance :returns: manifest instance
:rtype: :class:`.CDNDepotManifest` :rtype: :class:`.CDNDepotManifest`
:raises: steam error :raises SteamError: error message
:rtype: :class:`.SteamError`
""" """
resp = self.steam.send_um_and_wait('PublishedFile.GetDetails#1', { resp = self.steam.send_um_and_wait('PublishedFile.GetDetails#1', {
'publishedfileids': [item_id], 'publishedfileids': [item_id],
@ -579,15 +592,15 @@ class CDNClient(object):
}, timeout=7) }, timeout=7)
if resp.header.eresult != EResult.OK: 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] wf = None if resp is None else resp.body.publishedfiledetails[0]
if wf is None or wf.result != EResult.OK: if wf is None or wf.result != EResult.OK:
raise ValueError("Failed getting workshop file info: %s" % repr( raise SteamError("Failed getting workshop file info",
EResult.Timeout if resp is None else EResult(wf.result))) EResult.Timeout if resp is None else EResult(wf.result))
elif not wf.hcontent_file: 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 app_id = ws_app_id = wf.consumer_appid

6
steam/exceptions.py

@ -3,5 +3,9 @@ from steam.enums import EResult
class SteamError(Exception): class SteamError(Exception):
def __init__(self, message, eresult=EResult.Fail): 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` self.eresult = EResult(eresult) #: :class:`.EResult`
def __str__(self):
return "(%s) %s" % (self.eresult, self.message)

Loading…
Cancel
Save