mirror of
https://github.com/derrod/legendary.git
synced 2024-12-22 17:55:27 +00:00
Cherry-pick some Sourcery refactoring suggestions
This commit is contained in:
parent
85f6bd3220
commit
0e23b8e4f0
|
@ -51,10 +51,7 @@ class EPCAPI:
|
||||||
self.language_code = lc
|
self.language_code = lc
|
||||||
self.country_code = cc
|
self.country_code = cc
|
||||||
|
|
||||||
if timeout > 0:
|
self.request_timeout = timeout if timeout > 0 else None
|
||||||
self.request_timeout = timeout
|
|
||||||
else:
|
|
||||||
self.request_timeout = None
|
|
||||||
|
|
||||||
def get_auth_url(self):
|
def get_auth_url(self):
|
||||||
login_url = 'https://www.epicgames.com/id/login?redirectUrl='
|
login_url = 'https://www.epicgames.com/id/login?redirectUrl='
|
||||||
|
@ -236,10 +233,8 @@ class EPCAPI:
|
||||||
return records
|
return records
|
||||||
|
|
||||||
def get_user_cloud_saves(self, app_name='', manifests=False, filenames=None):
|
def get_user_cloud_saves(self, app_name='', manifests=False, filenames=None):
|
||||||
if app_name and manifests:
|
if app_name:
|
||||||
app_name += '/manifests/'
|
app_name += '/manifests/' if manifests else '/'
|
||||||
elif app_name:
|
|
||||||
app_name += '/'
|
|
||||||
|
|
||||||
user_id = self.user.get('account_id')
|
user_id = self.user.get('account_id')
|
||||||
|
|
||||||
|
|
|
@ -247,7 +247,7 @@ class LegendaryCLI:
|
||||||
elif _store:
|
elif _store:
|
||||||
print(f' ! This game has to be installed through a third-party store ({_store}, not supported)')
|
print(f' ! This game has to be installed through a third-party store ({_store}, not supported)')
|
||||||
else:
|
else:
|
||||||
print(f' ! No version information (unknown cause)')
|
print(' ! No version information (unknown cause)')
|
||||||
# Games that have assets, but only require a one-time activation before they can be independently installed
|
# Games that have assets, but only require a one-time activation before they can be independently installed
|
||||||
# via a third-party platform (e.g. Uplay)
|
# via a third-party platform (e.g. Uplay)
|
||||||
if game.partner_link_type:
|
if game.partner_link_type:
|
||||||
|
@ -380,15 +380,16 @@ class LegendaryCLI:
|
||||||
writer.writerow(['path', 'hash', 'size', 'install_tags'])
|
writer.writerow(['path', 'hash', 'size', 'install_tags'])
|
||||||
writer.writerows((fm.filename, fm.hash.hex(), fm.file_size, '|'.join(fm.install_tags)) for fm in files)
|
writer.writerows((fm.filename, fm.hash.hex(), fm.file_size, '|'.join(fm.install_tags)) for fm in files)
|
||||||
elif args.json:
|
elif args.json:
|
||||||
_files = []
|
_files = [
|
||||||
for fm in files:
|
dict(
|
||||||
_files.append(dict(
|
|
||||||
filename=fm.filename,
|
filename=fm.filename,
|
||||||
sha_hash=fm.hash.hex(),
|
sha_hash=fm.hash.hex(),
|
||||||
install_tags=fm.install_tags,
|
install_tags=fm.install_tags,
|
||||||
file_size=fm.file_size,
|
file_size=fm.file_size,
|
||||||
flags=fm.flags,
|
flags=fm.flags
|
||||||
))
|
)
|
||||||
|
for fm in files
|
||||||
|
]
|
||||||
return self._print_json(_files, args.pretty_json)
|
return self._print_json(_files, args.pretty_json)
|
||||||
else:
|
else:
|
||||||
install_tags = set()
|
install_tags = set()
|
||||||
|
@ -430,7 +431,7 @@ class LegendaryCLI:
|
||||||
if not self.core.login():
|
if not self.core.login():
|
||||||
logger.error('Login failed! Cannot continue with download process.')
|
logger.error('Login failed! Cannot continue with download process.')
|
||||||
exit(1)
|
exit(1)
|
||||||
logger.info(f'Cleaning saves...')
|
logger.info('Cleaning saves...')
|
||||||
self.core.clean_saves(self._resolve_aliases(args.app_name), args.delete_incomplete)
|
self.core.clean_saves(self._resolve_aliases(args.app_name), args.delete_incomplete)
|
||||||
|
|
||||||
def sync_saves(self, args):
|
def sync_saves(self, args):
|
||||||
|
@ -449,10 +450,9 @@ class LegendaryCLI:
|
||||||
|
|
||||||
# check available saves
|
# check available saves
|
||||||
saves = self.core.get_save_games()
|
saves = self.core.get_save_games()
|
||||||
latest_save = dict()
|
latest_save = {
|
||||||
|
save.app_name: save for save in sorted(saves, key=lambda a: a.datetime)
|
||||||
for save in sorted(saves, key=lambda a: a.datetime):
|
}
|
||||||
latest_save[save.app_name] = save
|
|
||||||
|
|
||||||
logger.info(f'Got {len(latest_save)} remote save game(s)')
|
logger.info(f'Got {len(latest_save)} remote save game(s)')
|
||||||
|
|
||||||
|
@ -475,7 +475,7 @@ class LegendaryCLI:
|
||||||
# if there is no saved save path, try to get one
|
# if there is no saved save path, try to get one
|
||||||
if not igame.save_path:
|
if not igame.save_path:
|
||||||
if args.yes:
|
if args.yes:
|
||||||
logger.info(f'Save path for this title has not been set, skipping due to --yes')
|
logger.info('Save path for this title has not been set, skipping due to --yes')
|
||||||
continue
|
continue
|
||||||
|
|
||||||
save_path = self.core.get_save_path(igame.app_name, platform=igame.platform)
|
save_path = self.core.get_save_path(igame.app_name, platform=igame.platform)
|
||||||
|
|
|
@ -85,7 +85,7 @@ class LegendaryCore:
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.log.warning(f'Getting locale failed: {e!r}, falling back to using en-US.')
|
self.log.warning(f'Getting locale failed: {e!r}, falling back to using en-US.')
|
||||||
elif system() != 'Darwin': # macOS doesn't have a default locale we can query
|
elif system() != 'Darwin': # macOS doesn't have a default locale we can query
|
||||||
self.log.warning(f'Could not determine locale, falling back to en-US')
|
self.log.warning('Could not determine locale, falling back to en-US')
|
||||||
|
|
||||||
self.update_available = False
|
self.update_available = False
|
||||||
self.force_show_update = False
|
self.force_show_update = False
|
||||||
|
@ -123,9 +123,9 @@ class LegendaryCore:
|
||||||
|
|
||||||
if r.status_code == 200:
|
if r.status_code == 200:
|
||||||
return r.json()['code']
|
return r.json()['code']
|
||||||
else:
|
|
||||||
self.log.error(f'Getting exchange code failed: {r.json()}')
|
self.log.error(f'Getting exchange code failed: {r.json()}')
|
||||||
return ''
|
return ''
|
||||||
|
|
||||||
def auth_code(self, code) -> bool:
|
def auth_code(self, code) -> bool:
|
||||||
"""
|
"""
|
||||||
|
@ -274,10 +274,10 @@ class LegendaryCore:
|
||||||
"""Applies configuration options returned by update API"""
|
"""Applies configuration options returned by update API"""
|
||||||
if not version_info:
|
if not version_info:
|
||||||
version_info = self.lgd.get_cached_version()['data']
|
version_info = self.lgd.get_cached_version()['data']
|
||||||
# if cached data is invalid
|
# if cached data is invalid
|
||||||
if not version_info:
|
if not version_info:
|
||||||
self.log.debug('No cached legendary config to apply.')
|
self.log.debug('No cached legendary config to apply.')
|
||||||
return
|
return
|
||||||
|
|
||||||
if 'egl_config' in version_info:
|
if 'egl_config' in version_info:
|
||||||
self.egs.update_egs_params(version_info['egl_config'])
|
self.egs.update_egs_params(version_info['egl_config'])
|
||||||
|
@ -342,10 +342,7 @@ class LegendaryCore:
|
||||||
if not self.egs.user:
|
if not self.egs.user:
|
||||||
return []
|
return []
|
||||||
|
|
||||||
if self.lgd.assets:
|
assets = self.lgd.assets.copy() if self.lgd.assets else dict()
|
||||||
assets = self.lgd.assets.copy()
|
|
||||||
else:
|
|
||||||
assets = dict()
|
|
||||||
|
|
||||||
assets.update({
|
assets.update({
|
||||||
platform: [
|
platform: [
|
||||||
|
@ -579,9 +576,17 @@ class LegendaryCore:
|
||||||
# get environment overrides from config
|
# get environment overrides from config
|
||||||
env = dict()
|
env = dict()
|
||||||
if 'default.env' in self.lgd.config:
|
if 'default.env' in self.lgd.config:
|
||||||
env.update({k: v for k, v in self.lgd.config[f'default.env'].items() if v and not k.startswith(';')})
|
env |= {
|
||||||
|
k: v
|
||||||
|
for k, v in self.lgd.config['default.env'].items()
|
||||||
|
if v and not k.startswith(';')
|
||||||
|
}
|
||||||
if f'{app_name}.env' in self.lgd.config:
|
if f'{app_name}.env' in self.lgd.config:
|
||||||
env.update({k: v for k, v in self.lgd.config[f'{app_name}.env'].items() if v and not k.startswith(';')})
|
env |= {
|
||||||
|
k: v
|
||||||
|
for k, v in self.lgd.config[f'{app_name}.env'].items()
|
||||||
|
if v and not k.startswith(';')
|
||||||
|
}
|
||||||
|
|
||||||
if disable_wine:
|
if disable_wine:
|
||||||
return env
|
return env
|
||||||
|
@ -719,10 +724,8 @@ class LegendaryCore:
|
||||||
elif not install.can_run_offline:
|
elif not install.can_run_offline:
|
||||||
self.log.warning('Game is not approved for offline use and may not work correctly.')
|
self.log.warning('Game is not approved for offline use and may not work correctly.')
|
||||||
|
|
||||||
user_name = self.lgd.userdata['displayName']
|
|
||||||
account_id = self.lgd.userdata['account_id']
|
account_id = self.lgd.userdata['account_id']
|
||||||
if user:
|
user_name = user or self.lgd.userdata['displayName']
|
||||||
user_name = user
|
|
||||||
|
|
||||||
params.egl_parameters.extend([
|
params.egl_parameters.extend([
|
||||||
'-AUTH_LOGIN=unused',
|
'-AUTH_LOGIN=unused',
|
||||||
|
@ -760,10 +763,7 @@ class LegendaryCore:
|
||||||
return params
|
return params
|
||||||
|
|
||||||
def get_origin_uri(self, app_name: str, offline: bool = False) -> str:
|
def get_origin_uri(self, app_name: str, offline: bool = False) -> str:
|
||||||
if offline:
|
token = '0' if offline else self.egs.get_game_token()['code']
|
||||||
token = '0'
|
|
||||||
else:
|
|
||||||
token = self.egs.get_game_token()['code']
|
|
||||||
|
|
||||||
user_name = self.lgd.userdata['displayName']
|
user_name = self.lgd.userdata['displayName']
|
||||||
account_id = self.lgd.userdata['account_id']
|
account_id = self.lgd.userdata['account_id']
|
||||||
|
@ -819,18 +819,18 @@ class LegendaryCore:
|
||||||
}
|
}
|
||||||
|
|
||||||
if sys_platform == 'win32':
|
if sys_platform == 'win32':
|
||||||
path_vars.update({
|
path_vars |= {
|
||||||
'{appdata}': os.path.expandvars('%LOCALAPPDATA%'),
|
'{appdata}': os.path.expandvars('%LOCALAPPDATA%'),
|
||||||
'{userdir}': os.path.expandvars('%userprofile%/documents'),
|
'{userdir}': os.path.expandvars('%userprofile%/documents'),
|
||||||
'{userprofile}': os.path.expandvars('%userprofile%'),
|
'{userprofile}': os.path.expandvars('%userprofile%'),
|
||||||
'{usersavedgames}': os.path.expandvars('%userprofile%/Saved Games')
|
'{usersavedgames}': os.path.expandvars('%userprofile%/Saved Games'),
|
||||||
})
|
}
|
||||||
elif sys_platform == 'darwin' and platform == 'Mac':
|
elif sys_platform == 'darwin' and platform == 'Mac':
|
||||||
path_vars.update({
|
path_vars |= {
|
||||||
'{appdata}': os.path.expanduser('~/Library/Application Support'),
|
'{appdata}': os.path.expanduser('~/Library/Application Support'),
|
||||||
'{userdir}': os.path.expanduser('~/Documents'),
|
'{userdir}': os.path.expanduser('~/Documents'),
|
||||||
'{userlibrary}': os.path.expanduser('~/Library')
|
'{userlibrary}': os.path.expanduser('~/Library'),
|
||||||
})
|
}
|
||||||
else:
|
else:
|
||||||
wine_pfx = None
|
wine_pfx = None
|
||||||
# on mac CrossOver takes precedence so check for a bottle first
|
# on mac CrossOver takes precedence so check for a bottle first
|
||||||
|
@ -868,10 +868,10 @@ class LegendaryCore:
|
||||||
wine_pfx = mac_get_bottle_path(cx_bottle)
|
wine_pfx = mac_get_bottle_path(cx_bottle)
|
||||||
|
|
||||||
if not wine_pfx:
|
if not wine_pfx:
|
||||||
proton_pfx = os.getenv('STEAM_COMPAT_DATA_PATH')
|
if proton_pfx := os.getenv('STEAM_COMPAT_DATA_PATH'):
|
||||||
if proton_pfx:
|
|
||||||
wine_pfx = f'{proton_pfx}/pfx'
|
wine_pfx = f'{proton_pfx}/pfx'
|
||||||
wine_pfx = os.getenv('WINEPREFIX', wine_pfx)
|
else:
|
||||||
|
wine_pfx = os.getenv('WINEPREFIX', wine_pfx)
|
||||||
|
|
||||||
# if all else fails, use the WINE default
|
# if all else fails, use the WINE default
|
||||||
if not wine_pfx:
|
if not wine_pfx:
|
||||||
|
@ -1111,7 +1111,7 @@ class LegendaryCore:
|
||||||
missing_chunks += 1
|
missing_chunks += 1
|
||||||
|
|
||||||
if (0 < missing_chunks < total_chunks and delete_incomplete) or missing_chunks == total_chunks:
|
if (0 < missing_chunks < total_chunks and delete_incomplete) or missing_chunks == total_chunks:
|
||||||
self.log.error(f'Chunk(s) missing, marking manifest for deletion.')
|
self.log.error('Chunk(s) missing, marking manifest for deletion.')
|
||||||
deletion_list.append(fname)
|
deletion_list.append(fname)
|
||||||
continue
|
continue
|
||||||
elif 0 < missing_chunks < total_chunks:
|
elif 0 < missing_chunks < total_chunks:
|
||||||
|
@ -1153,10 +1153,7 @@ class LegendaryCore:
|
||||||
|
|
||||||
for ass in self.get_assets(True):
|
for ass in self.get_assets(True):
|
||||||
if ass.app_name == app_name:
|
if ass.app_name == app_name:
|
||||||
if ass.build_version != installed.version:
|
return ass.build_version == installed.version
|
||||||
return False
|
|
||||||
else:
|
|
||||||
return True
|
|
||||||
# if we get here something is very wrong
|
# if we get here something is very wrong
|
||||||
raise ValueError(f'Could not find {app_name} in asset list!')
|
raise ValueError(f'Could not find {app_name} in asset list!')
|
||||||
|
|
||||||
|
@ -1167,10 +1164,10 @@ class LegendaryCore:
|
||||||
return self._get_installed_game(app_name) is not None
|
return self._get_installed_game(app_name) is not None
|
||||||
|
|
||||||
def is_dlc(self, app_name: str) -> bool:
|
def is_dlc(self, app_name: str) -> bool:
|
||||||
meta = self.lgd.get_game_meta(app_name)
|
if meta := self.lgd.get_game_meta(app_name):
|
||||||
if not meta:
|
return meta.is_dlc
|
||||||
|
else:
|
||||||
raise ValueError('Game unknown!')
|
raise ValueError('Game unknown!')
|
||||||
return meta.is_dlc
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def load_manifest(data: bytes) -> Manifest:
|
def load_manifest(data: bytes) -> Manifest:
|
||||||
|
@ -1252,10 +1249,7 @@ class LegendaryCore:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
r = self.egs.unauth_session.get(f'{base_url}/Deltas/{new_build_id}/{old_build_id}.delta')
|
r = self.egs.unauth_session.get(f'{base_url}/Deltas/{new_build_id}/{old_build_id}.delta')
|
||||||
if r.status_code == 200:
|
return r.content if r.status_code == 200 else None
|
||||||
return r.content
|
|
||||||
else:
|
|
||||||
return None
|
|
||||||
|
|
||||||
def prepare_download(self, game: Game, base_game: Game = None, base_path: str = '',
|
def prepare_download(self, game: Game, base_game: Game = None, base_path: str = '',
|
||||||
status_q: Queue = None, max_shm: int = 0, max_workers: int = 0,
|
status_q: Queue = None, max_shm: int = 0, max_workers: int = 0,
|
||||||
|
|
|
@ -29,7 +29,7 @@ class DLManager(Process):
|
||||||
|
|
||||||
self.base_url = base_url
|
self.base_url = base_url
|
||||||
self.dl_dir = download_dir
|
self.dl_dir = download_dir
|
||||||
self.cache_dir = cache_dir if cache_dir else os.path.join(download_dir, '.cache')
|
self.cache_dir = cache_dir or os.path.join(download_dir, '.cache')
|
||||||
|
|
||||||
# All the queues!
|
# All the queues!
|
||||||
self.logging_queue = None
|
self.logging_queue = None
|
||||||
|
@ -37,7 +37,7 @@ class DLManager(Process):
|
||||||
self.writer_queue = None
|
self.writer_queue = None
|
||||||
self.dl_result_q = None
|
self.dl_result_q = None
|
||||||
self.writer_result_q = None
|
self.writer_result_q = None
|
||||||
self.max_workers = max_workers if max_workers else min(cpu_count() * 2, 16)
|
self.max_workers = max_workers or min(cpu_count() * 2, 16)
|
||||||
self.dl_timeout = dl_timeout
|
self.dl_timeout = dl_timeout
|
||||||
|
|
||||||
# Analysis stuff
|
# Analysis stuff
|
||||||
|
|
|
@ -51,12 +51,12 @@ class DLWorker(Process):
|
||||||
empty = False
|
empty = False
|
||||||
except Empty:
|
except Empty:
|
||||||
if not empty:
|
if not empty:
|
||||||
logger.debug(f'Queue Empty, waiting for more...')
|
logger.debug('Queue Empty, waiting for more...')
|
||||||
empty = True
|
empty = True
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if isinstance(job, TerminateWorkerTask): # let worker die
|
if isinstance(job, TerminateWorkerTask): # let worker die
|
||||||
logger.debug(f'Worker received termination signal, shutting down...')
|
logger.debug('Worker received termination signal, shutting down...')
|
||||||
break
|
break
|
||||||
|
|
||||||
tries = 0
|
tries = 0
|
||||||
|
@ -99,7 +99,7 @@ class DLWorker(Process):
|
||||||
break
|
break
|
||||||
|
|
||||||
if not chunk:
|
if not chunk:
|
||||||
logger.warning(f'Chunk somehow None?')
|
logger.warning('Chunk somehow None?')
|
||||||
self.o_q.put(DownloaderTaskResult(success=False, **job.__dict__))
|
self.o_q.put(DownloaderTaskResult(success=False, **job.__dict__))
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
@ -107,7 +107,7 @@ class DLWorker(Process):
|
||||||
try:
|
try:
|
||||||
size = len(chunk.data)
|
size = len(chunk.data)
|
||||||
if size > job.shm.size:
|
if size > job.shm.size:
|
||||||
logger.fatal(f'Downloaded chunk is longer than SharedMemorySegment!')
|
logger.fatal('Downloaded chunk is longer than SharedMemorySegment!')
|
||||||
|
|
||||||
self.shm.buf[job.shm.offset:job.shm.offset + size] = bytes(chunk.data)
|
self.shm.buf[job.shm.offset:job.shm.offset + size] = bytes(chunk.data)
|
||||||
del chunk
|
del chunk
|
||||||
|
@ -130,7 +130,7 @@ class FileWorker(Process):
|
||||||
self.q = queue
|
self.q = queue
|
||||||
self.o_q = out_queue
|
self.o_q = out_queue
|
||||||
self.base_path = base_path
|
self.base_path = base_path
|
||||||
self.cache_path = cache_path if cache_path else os.path.join(base_path, '.cache')
|
self.cache_path = cache_path or os.path.join(base_path, '.cache')
|
||||||
self.shm = SharedMemory(name=shm)
|
self.shm = SharedMemory(name=shm)
|
||||||
self.log_level = logging.getLogger().level
|
self.log_level = logging.getLogger().level
|
||||||
self.logging_queue = logging_queue
|
self.logging_queue = logging_queue
|
||||||
|
@ -143,7 +143,7 @@ class FileWorker(Process):
|
||||||
|
|
||||||
logger = logging.getLogger(self.name)
|
logger = logging.getLogger(self.name)
|
||||||
logger.setLevel(self.log_level)
|
logger.setLevel(self.log_level)
|
||||||
logger.debug(f'Download worker reporting for duty!')
|
logger.debug('Download worker reporting for duty!')
|
||||||
|
|
||||||
last_filename = ''
|
last_filename = ''
|
||||||
current_file = None
|
current_file = None
|
||||||
|
@ -159,7 +159,7 @@ class FileWorker(Process):
|
||||||
if isinstance(j, TerminateWorkerTask):
|
if isinstance(j, TerminateWorkerTask):
|
||||||
if current_file:
|
if current_file:
|
||||||
current_file.close()
|
current_file.close()
|
||||||
logger.debug(f'Worker received termination signal, shutting down...')
|
logger.debug('Worker received termination signal, shutting down...')
|
||||||
# send termination task to results halnder as well
|
# send termination task to results halnder as well
|
||||||
self.o_q.put(TerminateWorkerTask())
|
self.o_q.put(TerminateWorkerTask())
|
||||||
break
|
break
|
||||||
|
|
|
@ -90,7 +90,7 @@ class LGDLFS:
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.log.error(f'Unable to read configuration file, please ensure that file is valid! '
|
self.log.error(f'Unable to read configuration file, please ensure that file is valid! '
|
||||||
f'(Error: {repr(e)})')
|
f'(Error: {repr(e)})')
|
||||||
self.log.warning(f'Continuing with blank config in safe-mode...')
|
self.log.warning('Continuing with blank config in safe-mode...')
|
||||||
self.config.read_only = True
|
self.config.read_only = True
|
||||||
|
|
||||||
# make sure "Legendary" section exists
|
# make sure "Legendary" section exists
|
||||||
|
@ -221,8 +221,7 @@ class LGDLFS:
|
||||||
f.write(manifest_data)
|
f.write(manifest_data)
|
||||||
|
|
||||||
def get_game_meta(self, app_name):
|
def get_game_meta(self, app_name):
|
||||||
_meta = self._game_metadata.get(app_name, None)
|
if _meta := self._game_metadata.get(app_name, None):
|
||||||
if _meta:
|
|
||||||
return Game.from_json(_meta)
|
return Game.from_json(_meta)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
@ -233,14 +232,14 @@ class LGDLFS:
|
||||||
json.dump(json_meta, open(meta_file, 'w'), indent=2, sort_keys=True)
|
json.dump(json_meta, open(meta_file, 'w'), indent=2, sort_keys=True)
|
||||||
|
|
||||||
def delete_game_meta(self, app_name):
|
def delete_game_meta(self, app_name):
|
||||||
if app_name in self._game_metadata:
|
if app_name not in self._game_metadata:
|
||||||
del self._game_metadata[app_name]
|
|
||||||
meta_file = os.path.join(self.path, 'metadata', f'{app_name}.json')
|
|
||||||
if os.path.exists(meta_file):
|
|
||||||
os.remove(meta_file)
|
|
||||||
else:
|
|
||||||
raise ValueError(f'Game {app_name} does not exist in metadata DB!')
|
raise ValueError(f'Game {app_name} does not exist in metadata DB!')
|
||||||
|
|
||||||
|
del self._game_metadata[app_name]
|
||||||
|
meta_file = os.path.join(self.path, 'metadata', f'{app_name}.json')
|
||||||
|
if os.path.exists(meta_file):
|
||||||
|
os.remove(meta_file)
|
||||||
|
|
||||||
def get_game_app_names(self):
|
def get_game_app_names(self):
|
||||||
return sorted(self._game_metadata.keys())
|
return sorted(self._game_metadata.keys())
|
||||||
|
|
||||||
|
@ -264,9 +263,16 @@ class LGDLFS:
|
||||||
self.log.warning(f'Failed to delete file "{f}": {e!r}')
|
self.log.warning(f'Failed to delete file "{f}": {e!r}')
|
||||||
|
|
||||||
def clean_manifests(self, in_use):
|
def clean_manifests(self, in_use):
|
||||||
in_use_files = set(f'{clean_filename(f"{app_name}_{version}")}.manifest' for app_name, version, _ in in_use)
|
in_use_files = {
|
||||||
in_use_files |= set(f'{clean_filename(f"{app_name}_{platform}_{version}")}.manifest'
|
f'{clean_filename(f"{app_name}_{version}")}.manifest'
|
||||||
for app_name, version, platform in in_use)
|
for app_name, version, _ in in_use
|
||||||
|
}
|
||||||
|
|
||||||
|
in_use_files |= {
|
||||||
|
f'{clean_filename(f"{app_name}_{platform}_{version}")}.manifest'
|
||||||
|
for app_name, version, platform in in_use
|
||||||
|
}
|
||||||
|
|
||||||
for f in os.listdir(os.path.join(self.path, 'manifests')):
|
for f in os.listdir(os.path.join(self.path, 'manifests')):
|
||||||
if f not in in_use_files:
|
if f not in in_use_files:
|
||||||
try:
|
try:
|
||||||
|
@ -282,8 +288,7 @@ class LGDLFS:
|
||||||
self.log.debug(f'Failed to load installed game data: {e!r}')
|
self.log.debug(f'Failed to load installed game data: {e!r}')
|
||||||
return None
|
return None
|
||||||
|
|
||||||
game_json = self._installed.get(app_name, None)
|
if game_json := self._installed.get(app_name, None):
|
||||||
if game_json:
|
|
||||||
return InstalledGame.from_json(game_json)
|
return InstalledGame.from_json(game_json)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
@ -392,7 +397,7 @@ class LGDLFS:
|
||||||
def get_overlay_install_info(self):
|
def get_overlay_install_info(self):
|
||||||
if not self._overlay_install_info:
|
if not self._overlay_install_info:
|
||||||
try:
|
try:
|
||||||
data = json.load(open(os.path.join(self.path, f'overlay_install.json')))
|
data = json.load(open(os.path.join(self.path, 'overlay_install.json')))
|
||||||
self._overlay_install_info = InstalledGame.from_json(data)
|
self._overlay_install_info = InstalledGame.from_json(data)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.log.debug(f'Failed to load overlay install data: {e!r}')
|
self.log.debug(f'Failed to load overlay install data: {e!r}')
|
||||||
|
@ -440,9 +445,7 @@ class LGDLFS:
|
||||||
|
|
||||||
def serialise_sets(obj):
|
def serialise_sets(obj):
|
||||||
"""Turn sets into sorted lists for storage"""
|
"""Turn sets into sorted lists for storage"""
|
||||||
if isinstance(obj, set):
|
return sorted(obj) if isinstance(obj, set) else obj
|
||||||
return sorted(obj)
|
|
||||||
return obj
|
|
||||||
|
|
||||||
json.dump(alias_map, open(os.path.join(self.path, 'aliases.json'), 'w', newline='\n'),
|
json.dump(alias_map, open(os.path.join(self.path, 'aliases.json'), 'w', newline='\n'),
|
||||||
indent=2, sort_keys=True, default=serialise_sets)
|
indent=2, sort_keys=True, default=serialise_sets)
|
||||||
|
|
|
@ -113,10 +113,7 @@ class Chunk:
|
||||||
return _chunk
|
return _chunk
|
||||||
|
|
||||||
def write(self, fp=None, compress=True):
|
def write(self, fp=None, compress=True):
|
||||||
if not fp:
|
bio = fp or BytesIO()
|
||||||
bio = BytesIO()
|
|
||||||
else:
|
|
||||||
bio = fp
|
|
||||||
|
|
||||||
self.uncompressed_size = self.compressed_size = len(self.data)
|
self.uncompressed_size = self.compressed_size = len(self.data)
|
||||||
if compress or self.compressed:
|
if compress or self.compressed:
|
||||||
|
@ -143,7 +140,4 @@ class Chunk:
|
||||||
# finally, add the data
|
# finally, add the data
|
||||||
bio.write(self._data)
|
bio.write(self._data)
|
||||||
|
|
||||||
if not fp:
|
return bio.tell() if fp else bio.getvalue()
|
||||||
return bio.getvalue()
|
|
||||||
else:
|
|
||||||
return bio.tell()
|
|
||||||
|
|
|
@ -145,9 +145,9 @@ class EGLManifest:
|
||||||
tmp.executable = igame.executable
|
tmp.executable = igame.executable
|
||||||
tmp.main_game_appname = game.app_name # todo for DLC support this needs to be the base game
|
tmp.main_game_appname = game.app_name # todo for DLC support this needs to be the base game
|
||||||
tmp.app_folder_name = game.metadata.get('customAttributes', {}).get('FolderName', {}).get('value', '')
|
tmp.app_folder_name = game.metadata.get('customAttributes', {}).get('FolderName', {}).get('value', '')
|
||||||
tmp.manifest_location = igame.install_path + '/.egstore'
|
tmp.manifest_location = f'{igame.install_path}/.egstore'
|
||||||
tmp.ownership_token = igame.requires_ot
|
tmp.ownership_token = igame.requires_ot
|
||||||
tmp.staging_location = igame.install_path + '/.egstore/bps'
|
tmp.staging_location = f'{igame.install_path}/.egstore/bps'
|
||||||
tmp.can_run_offline = igame.can_run_offline
|
tmp.can_run_offline = igame.can_run_offline
|
||||||
tmp.is_incomplete_install = False
|
tmp.is_incomplete_install = False
|
||||||
tmp.needs_validation = igame.needs_verification
|
tmp.needs_validation = igame.needs_verification
|
||||||
|
|
|
@ -92,8 +92,7 @@ class Manifest:
|
||||||
_m.file_manifest_list = FML.read(_tmp)
|
_m.file_manifest_list = FML.read(_tmp)
|
||||||
_m.custom_fields = CustomFields.read(_tmp)
|
_m.custom_fields = CustomFields.read(_tmp)
|
||||||
|
|
||||||
unhandled_data = _tmp.read()
|
if unhandled_data := _tmp.read():
|
||||||
if unhandled_data:
|
|
||||||
logger.warning(f'Did not read {len(unhandled_data)} remaining bytes in manifest! '
|
logger.warning(f'Did not read {len(unhandled_data)} remaining bytes in manifest! '
|
||||||
f'This may not be a problem.')
|
f'This may not be a problem.')
|
||||||
|
|
||||||
|
@ -152,10 +151,7 @@ class Manifest:
|
||||||
self.data = zlib.compress(self.data)
|
self.data = zlib.compress(self.data)
|
||||||
self.size_compressed = len(self.data)
|
self.size_compressed = len(self.data)
|
||||||
|
|
||||||
if not fp:
|
bio = fp or BytesIO()
|
||||||
bio = BytesIO()
|
|
||||||
else:
|
|
||||||
bio = fp
|
|
||||||
|
|
||||||
bio.write(struct.pack('<I', self.header_magic))
|
bio.write(struct.pack('<I', self.header_magic))
|
||||||
bio.write(struct.pack('<I', self.header_size))
|
bio.write(struct.pack('<I', self.header_size))
|
||||||
|
@ -166,10 +162,7 @@ class Manifest:
|
||||||
bio.write(struct.pack('<I', self.serialisation_version))
|
bio.write(struct.pack('<I', self.serialisation_version))
|
||||||
bio.write(self.data)
|
bio.write(self.data)
|
||||||
|
|
||||||
if not fp:
|
return bio.tell() if fp else bio.getvalue()
|
||||||
return bio.getvalue()
|
|
||||||
else:
|
|
||||||
return bio.tell()
|
|
||||||
|
|
||||||
|
|
||||||
class ManifestMeta:
|
class ManifestMeta:
|
||||||
|
@ -226,7 +219,7 @@ class ManifestMeta:
|
||||||
|
|
||||||
# This is a list though I've never seen more than one entry
|
# This is a list though I've never seen more than one entry
|
||||||
entries = struct.unpack('<I', bio.read(4))[0]
|
entries = struct.unpack('<I', bio.read(4))[0]
|
||||||
for i in range(entries):
|
for _ in range(entries):
|
||||||
_meta.prereq_ids.append(read_fstring(bio))
|
_meta.prereq_ids.append(read_fstring(bio))
|
||||||
|
|
||||||
_meta.prereq_name = read_fstring(bio)
|
_meta.prereq_name = read_fstring(bio)
|
||||||
|
@ -348,7 +341,7 @@ class CDL:
|
||||||
|
|
||||||
# the way this data is stored is rather odd, maybe there's a nicer way to write this...
|
# the way this data is stored is rather odd, maybe there's a nicer way to write this...
|
||||||
|
|
||||||
for i in range(_cdl.count):
|
for _ in range(_cdl.count):
|
||||||
_cdl.elements.append(ChunkInfo(manifest_version=manifest_version))
|
_cdl.elements.append(ChunkInfo(manifest_version=manifest_version))
|
||||||
|
|
||||||
# guid, doesn't seem to be a standard like UUID but is fairly straightfoward, 4 bytes, 128 bit.
|
# guid, doesn't seem to be a standard like UUID but is fairly straightfoward, 4 bytes, 128 bit.
|
||||||
|
@ -495,7 +488,7 @@ class FML:
|
||||||
_fml.version = struct.unpack('B', bio.read(1))[0]
|
_fml.version = struct.unpack('B', bio.read(1))[0]
|
||||||
_fml.count = struct.unpack('<I', bio.read(4))[0]
|
_fml.count = struct.unpack('<I', bio.read(4))[0]
|
||||||
|
|
||||||
for i in range(_fml.count):
|
for _ in range(_fml.count):
|
||||||
_fml.elements.append(FileManifest())
|
_fml.elements.append(FileManifest())
|
||||||
|
|
||||||
for fm in _fml.elements:
|
for fm in _fml.elements:
|
||||||
|
@ -516,14 +509,14 @@ class FML:
|
||||||
# install tags, no idea what they do, I've only seen them in the Fortnite manifest
|
# install tags, no idea what they do, I've only seen them in the Fortnite manifest
|
||||||
for fm in _fml.elements:
|
for fm in _fml.elements:
|
||||||
_elem = struct.unpack('<I', bio.read(4))[0]
|
_elem = struct.unpack('<I', bio.read(4))[0]
|
||||||
for i in range(_elem):
|
for _ in range(_elem):
|
||||||
fm.install_tags.append(read_fstring(bio))
|
fm.install_tags.append(read_fstring(bio))
|
||||||
|
|
||||||
# Each file is made up of "Chunk Parts" that can be spread across the "chunk stream"
|
# Each file is made up of "Chunk Parts" that can be spread across the "chunk stream"
|
||||||
for fm in _fml.elements:
|
for fm in _fml.elements:
|
||||||
_elem = struct.unpack('<I', bio.read(4))[0]
|
_elem = struct.unpack('<I', bio.read(4))[0]
|
||||||
_offset = 0
|
_offset = 0
|
||||||
for i in range(_elem):
|
for _ in range(_elem):
|
||||||
chunkp = ChunkPart()
|
chunkp = ChunkPart()
|
||||||
_start = bio.tell()
|
_start = bio.tell()
|
||||||
_size = struct.unpack('<I', bio.read(4))[0]
|
_size = struct.unpack('<I', bio.read(4))[0]
|
||||||
|
@ -709,15 +702,8 @@ class CustomFields:
|
||||||
_cf.version = struct.unpack('B', bio.read(1))[0]
|
_cf.version = struct.unpack('B', bio.read(1))[0]
|
||||||
_cf.count = struct.unpack('<I', bio.read(4))[0]
|
_cf.count = struct.unpack('<I', bio.read(4))[0]
|
||||||
|
|
||||||
_keys = []
|
_keys = [read_fstring(bio) for _ in range(_cf.count)]
|
||||||
_values = []
|
_values = [read_fstring(bio) for _ in range(_cf.count)]
|
||||||
|
|
||||||
for i in range(_cf.count):
|
|
||||||
_keys.append(read_fstring(bio))
|
|
||||||
|
|
||||||
for i in range(_cf.count):
|
|
||||||
_values.append(read_fstring(bio))
|
|
||||||
|
|
||||||
_cf._dict = dict(zip(_keys, _values))
|
_cf._dict = dict(zip(_keys, _values))
|
||||||
|
|
||||||
if (size_read := bio.tell() - cf_start) != _cf.size:
|
if (size_read := bio.tell() - cf_start) != _cf.size:
|
||||||
|
@ -766,8 +752,7 @@ class ManifestComparison:
|
||||||
old_files = {fm.filename: fm.hash for fm in old_manifest.file_manifest_list.elements}
|
old_files = {fm.filename: fm.hash for fm in old_manifest.file_manifest_list.elements}
|
||||||
|
|
||||||
for fm in manifest.file_manifest_list.elements:
|
for fm in manifest.file_manifest_list.elements:
|
||||||
old_file_hash = old_files.pop(fm.filename, None)
|
if old_file_hash := old_files.pop(fm.filename, None):
|
||||||
if old_file_hash:
|
|
||||||
if fm.hash == old_file_hash:
|
if fm.hash == old_file_hash:
|
||||||
comp.unchanged.add(fm.filename)
|
comp.unchanged.add(fm.filename)
|
||||||
else:
|
else:
|
||||||
|
|
|
@ -1,8 +1,5 @@
|
||||||
def get_boolean_choice(prompt, default=True):
|
def get_boolean_choice(prompt, default=True):
|
||||||
if default:
|
yn = 'Y/n' if default else 'y/N'
|
||||||
yn = 'Y/n'
|
|
||||||
else:
|
|
||||||
yn = 'y/N'
|
|
||||||
|
|
||||||
choice = input(f'{prompt} [{yn}]: ')
|
choice = input(f'{prompt} [{yn}]: ')
|
||||||
if not choice:
|
if not choice:
|
||||||
|
@ -21,10 +18,10 @@ def get_int_choice(prompt, default=None, min_choice=None, max_choice=None, retur
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
try:
|
try:
|
||||||
inp = input(prompt)
|
if inp := input(prompt):
|
||||||
if not inp:
|
choice = int(inp)
|
||||||
|
else:
|
||||||
return default
|
return default
|
||||||
choice = int(inp)
|
|
||||||
except ValueError:
|
except ValueError:
|
||||||
if return_on_invalid:
|
if return_on_invalid:
|
||||||
return None
|
return None
|
||||||
|
@ -61,7 +58,7 @@ def sdl_prompt(sdl_data, title):
|
||||||
examples = ', '.join([g for g in sdl_data.keys() if g != '__required'][:2])
|
examples = ', '.join([g for g in sdl_data.keys() if g != '__required'][:2])
|
||||||
print(f'Please enter tags of pack(s) to install (space/comma-separated, e.g. "{examples}")')
|
print(f'Please enter tags of pack(s) to install (space/comma-separated, e.g. "{examples}")')
|
||||||
print('Leave blank to use defaults (only required data will be downloaded).')
|
print('Leave blank to use defaults (only required data will be downloaded).')
|
||||||
choices = input(f'Additional packs [Enter to confirm]: ')
|
choices = input('Additional packs [Enter to confirm]: ')
|
||||||
if not choices:
|
if not choices:
|
||||||
return tags
|
return tags
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ class HiddenAliasSubparsersAction(argparse._SubParsersAction):
|
||||||
def add_parser(self, name, **kwargs):
|
def add_parser(self, name, **kwargs):
|
||||||
# set prog from the existing prefix
|
# set prog from the existing prefix
|
||||||
if kwargs.get('prog') is None:
|
if kwargs.get('prog') is None:
|
||||||
kwargs['prog'] = '%s %s' % (self._prog_prefix, name)
|
kwargs['prog'] = f'{self._prog_prefix} {name}'
|
||||||
|
|
||||||
aliases = kwargs.pop('aliases', ())
|
aliases = kwargs.pop('aliases', ())
|
||||||
hide_aliases = kwargs.pop('hide_aliases', False)
|
hide_aliases = kwargs.pop('hide_aliases', False)
|
||||||
|
|
|
@ -22,7 +22,7 @@ except Exception as e:
|
||||||
|
|
||||||
login_url = 'https://www.epicgames.com/id/login'
|
login_url = 'https://www.epicgames.com/id/login'
|
||||||
sid_url = 'https://www.epicgames.com/id/api/redirect?'
|
sid_url = 'https://www.epicgames.com/id/api/redirect?'
|
||||||
logout_url = 'https://www.epicgames.com/id/logout?productName=epic-games&redirectUrl=' + login_url
|
logout_url = f'https://www.epicgames.com/id/logout?productName=epic-games&redirectUrl={login_url}'
|
||||||
goodbye_url = 'https://legendary.gl/goodbye'
|
goodbye_url = 'https://legendary.gl/goodbye'
|
||||||
window_js = '''
|
window_js = '''
|
||||||
window.ue = {
|
window.ue = {
|
||||||
|
@ -102,7 +102,7 @@ class MockLauncher:
|
||||||
def trigger_sid_exchange(self, *args, **kwargs):
|
def trigger_sid_exchange(self, *args, **kwargs):
|
||||||
# check if code-based login hasn't already set the destroy flag
|
# check if code-based login hasn't already set the destroy flag
|
||||||
if not self.destroy_on_load:
|
if not self.destroy_on_load:
|
||||||
logger.debug(f'Injecting SID JS')
|
logger.debug('Injecting SID JS')
|
||||||
# inject JS to get SID API response and call our API
|
# inject JS to get SID API response and call our API
|
||||||
self.window.evaluate_js(get_sid_js)
|
self.window.evaluate_js(get_sid_js)
|
||||||
|
|
||||||
|
@ -139,6 +139,6 @@ def do_webview_login(callback_sid=None, callback_code=None):
|
||||||
return None
|
return None
|
||||||
|
|
||||||
if api.callback_result is None:
|
if api.callback_result is None:
|
||||||
logger.error(f'Login aborted by user.')
|
logger.error('Login aborted by user.')
|
||||||
|
|
||||||
return api.callback_result
|
return api.callback_result
|
||||||
|
|
Loading…
Reference in a new issue