mirror of
https://github.com/derrod/legendary.git
synced 2025-01-08 13:55:28 +00:00
[core/utils] More generic/robust selective dl, add support for Fortnite
Existing installations should ask for the install tags that should be used on first update. It will now be easier to add more games as well.
This commit is contained in:
parent
fd004db4b9
commit
7609553b12
|
@ -21,10 +21,10 @@ from legendary import __version__, __codename__
|
||||||
from legendary.core import LegendaryCore
|
from legendary.core import LegendaryCore
|
||||||
from legendary.models.exceptions import InvalidCredentialsError
|
from legendary.models.exceptions import InvalidCredentialsError
|
||||||
from legendary.models.game import SaveGameStatus, VerifyResult
|
from legendary.models.game import SaveGameStatus, VerifyResult
|
||||||
from legendary.utils.cli import get_boolean_choice
|
from legendary.utils.cli import get_boolean_choice, sdl_prompt
|
||||||
from legendary.utils.custom_parser import AliasedSubParsersAction
|
from legendary.utils.custom_parser import AliasedSubParsersAction
|
||||||
from legendary.utils.lfs import validate_files
|
from legendary.utils.lfs import validate_files
|
||||||
from legendary.utils.game_workarounds import cyber_prompt_2077
|
from legendary.utils.selective_dl import get_sdl_appname
|
||||||
|
|
||||||
# todo custom formatter for cli logger (clean info, highlighted error/warning)
|
# todo custom formatter for cli logger (clean info, highlighted error/warning)
|
||||||
logging.basicConfig(
|
logging.basicConfig(
|
||||||
|
@ -579,14 +579,15 @@ class LegendaryCLI:
|
||||||
logger.info(f'Using existing repair file: {repair_file}')
|
logger.info(f'Using existing repair file: {repair_file}')
|
||||||
|
|
||||||
# Workaround for Cyberpunk 2077 preload
|
# Workaround for Cyberpunk 2077 preload
|
||||||
if game.app_name.startswith('Ginger'):
|
if not args.install_tag and ((sdl_name := get_sdl_appname(game.app_name)) is not None):
|
||||||
if not self.core.is_installed(game.app_name):
|
config_tags = self.core.lgd.config.get(game.app_name, 'install_tags', fallback=None)
|
||||||
args.install_tag = cyber_prompt_2077()
|
if not self.core.is_installed(game.app_name) or config_tags is None:
|
||||||
|
args.install_tag = sdl_prompt(sdl_name, game.app_title)
|
||||||
if game.app_name not in self.core.lgd.config:
|
if game.app_name not in self.core.lgd.config:
|
||||||
self.core.lgd.config[game.app_name] = dict()
|
self.core.lgd.config[game.app_name] = dict()
|
||||||
self.core.lgd.config.set(game.app_name, 'install_tags', ','.join(args.install_tag))
|
self.core.lgd.config.set(game.app_name, 'install_tags', ','.join(args.install_tag))
|
||||||
else:
|
else:
|
||||||
args.install_tag = self.core.lgd.config.get(game.app_name, 'install_tags', fallback='').split(',')
|
args.install_tag = config_tags.split(',')
|
||||||
|
|
||||||
logger.info('Preparing download...')
|
logger.info('Preparing download...')
|
||||||
# todo use status queue to print progress from CLI
|
# todo use status queue to print progress from CLI
|
||||||
|
|
|
@ -1,3 +1,6 @@
|
||||||
|
from legendary.utils.selective_dl import games
|
||||||
|
|
||||||
|
|
||||||
def get_boolean_choice(prompt, default=True):
|
def get_boolean_choice(prompt, default=True):
|
||||||
if default:
|
if default:
|
||||||
yn = 'Y/n'
|
yn = 'Y/n'
|
||||||
|
@ -11,3 +14,31 @@ def get_boolean_choice(prompt, default=True):
|
||||||
return True
|
return True
|
||||||
else:
|
else:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def sdl_prompt(app_name, title):
|
||||||
|
tags = ['']
|
||||||
|
if '__required' in games[app_name]:
|
||||||
|
tags.extend(games[app_name]['__required']['tags'])
|
||||||
|
|
||||||
|
print(f'You are about to install {title}, this game supports selective downloads.')
|
||||||
|
print('The following optional packs are available:')
|
||||||
|
for tag, info in games[app_name].items():
|
||||||
|
if tag == '__required':
|
||||||
|
continue
|
||||||
|
print(' *', tag, '-', info['name'])
|
||||||
|
|
||||||
|
print('Please enter a comma-separated list of optional packs to install (leave blank for defaults)')
|
||||||
|
examples = ','.join([g for g in games[app_name].keys() if g != '__required'][:2])
|
||||||
|
choices = input(f'Additional packs [e.g. {examples}]: ')
|
||||||
|
if not choices:
|
||||||
|
return tags
|
||||||
|
|
||||||
|
for c in choices.split(','):
|
||||||
|
c = c.strip()
|
||||||
|
if c in games[app_name]:
|
||||||
|
tags.extend(games[app_name][c]['tags'])
|
||||||
|
else:
|
||||||
|
print('Invalid tag:', c)
|
||||||
|
|
||||||
|
return tags
|
||||||
|
|
|
@ -17,39 +17,3 @@ def is_opt_enabled(app_name, version):
|
||||||
if version in versions or not versions:
|
if version in versions or not versions:
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
_cyberpunk_sdl = {
|
|
||||||
'de': {'tags': ['voice_de_de'], 'name': 'Deutsch'},
|
|
||||||
'es': {'tags': ['voice_es_es'], 'name': 'español (España)'},
|
|
||||||
'fr': {'tags': ['voice_fr_fr'], 'name': 'français'},
|
|
||||||
'it': {'tags': ['voice_it_it'], 'name': 'italiano'},
|
|
||||||
'ja': {'tags': ['voice_ja_jp'], 'name': '日本語'},
|
|
||||||
'ko': {'tags': ['voice_ko_kr'], 'name': '한국어'},
|
|
||||||
'pl': {'tags': ['voice_pl_pl'], 'name': 'polski'},
|
|
||||||
'pt': {'tags': ['voice_pt_br'], 'name': 'português brasileiro'},
|
|
||||||
'ru': {'tags': ['voice_ru_ru'], 'name': 'русский'},
|
|
||||||
'zh': {'tags': ['voice_zh_cn'], 'name': '中文(中国)'}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def cyber_prompt_2077():
|
|
||||||
print('You are about to install Cyberpunk 2077, this game supports selective downloads for langauge packs.')
|
|
||||||
print('The following language packs are available:')
|
|
||||||
for tag, info in _cyberpunk_sdl.items():
|
|
||||||
print(' *', tag, '-', info['name'])
|
|
||||||
|
|
||||||
print('Please enter a comma-separated list of language packs to install (leave blank for english only)')
|
|
||||||
choices = input('Additional languages [e.g. de,fr]: ')
|
|
||||||
if not choices:
|
|
||||||
return ['']
|
|
||||||
|
|
||||||
tags = ['']
|
|
||||||
for c in choices.split(','):
|
|
||||||
c = c.strip()
|
|
||||||
if c in _cyberpunk_sdl:
|
|
||||||
tags.extend(_cyberpunk_sdl[c]['tags'])
|
|
||||||
else:
|
|
||||||
print('Invalid tag:', c)
|
|
||||||
|
|
||||||
return tags
|
|
||||||
|
|
35
legendary/utils/selective_dl.py
Normal file
35
legendary/utils/selective_dl.py
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
_cyberpunk_sdl = {
|
||||||
|
'de': {'tags': ['voice_de_de'], 'name': 'Deutsch'},
|
||||||
|
'es': {'tags': ['voice_es_es'], 'name': 'español (España)'},
|
||||||
|
'fr': {'tags': ['voice_fr_fr'], 'name': 'français'},
|
||||||
|
'it': {'tags': ['voice_it_it'], 'name': 'italiano'},
|
||||||
|
'ja': {'tags': ['voice_ja_jp'], 'name': '日本語'},
|
||||||
|
'ko': {'tags': ['voice_ko_kr'], 'name': '한국어'},
|
||||||
|
'pl': {'tags': ['voice_pl_pl'], 'name': 'polski'},
|
||||||
|
'pt': {'tags': ['voice_pt_br'], 'name': 'português brasileiro'},
|
||||||
|
'ru': {'tags': ['voice_ru_ru'], 'name': 'русский'},
|
||||||
|
'cn': {'tags': ['voice_zh_cn'], 'name': '中文(中国)'}
|
||||||
|
}
|
||||||
|
|
||||||
|
_fortnite_sdl = {
|
||||||
|
'__required': {'tags': ['chunk0', 'chunk10'], 'name': 'Fortnite Core'},
|
||||||
|
'stw': {'tags': ['chunk11', 'chunk11optional'], 'name': 'Fortnite Save the World'},
|
||||||
|
'hd_textures': {'tags': ['chunk10optional'], 'name': 'High Resolution Textures'},
|
||||||
|
'lang_de': {'tags': ['chunk2'], 'name': '(Language Pack) Deutsch'},
|
||||||
|
'lang_fr': {'tags': ['chunk5'], 'name': '(Language Pack) français'},
|
||||||
|
'lang_pl': {'tags': ['chunk7'], 'name': '(Language Pack) polski'},
|
||||||
|
'lang_ru': {'tags': ['chunk8'], 'name': '(Language Pack) русский'},
|
||||||
|
'lang_cn': {'tags': ['chunk9'], 'name': '(Language Pack) 中文(中国)'}
|
||||||
|
}
|
||||||
|
|
||||||
|
games = {
|
||||||
|
'Fortnite': _fortnite_sdl,
|
||||||
|
'Ginger': _cyberpunk_sdl
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def get_sdl_appname(app_name):
|
||||||
|
for k in games.keys():
|
||||||
|
if app_name.startswith(k):
|
||||||
|
return k
|
||||||
|
return None
|
Loading…
Reference in a new issue