mirror of
https://github.com/derrod/legendary.git
synced 2025-01-10 14:25:28 +00:00
[cli/core/models/utils] Add basic cloud save syncing support
This commit is contained in:
parent
0df80773c0
commit
98df2a0a38
132
legendary/cli.py
132
legendary/cli.py
|
@ -17,6 +17,7 @@ from sys import exit, stdout
|
||||||
from legendary import __version__, __codename__
|
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
|
||||||
from legendary.utils.custom_parser import AliasedSubParsersAction
|
from legendary.utils.custom_parser import AliasedSubParsersAction
|
||||||
|
|
||||||
# todo custom formatter for cli logger (clean info, highlighted error/warning)
|
# todo custom formatter for cli logger (clean info, highlighted error/warning)
|
||||||
|
@ -223,10 +224,115 @@ 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)
|
||||||
# then get the saves
|
|
||||||
logger.info(f'Downloading saves to "{self.core.get_default_install_dir()}"')
|
logger.info(f'Downloading saves to "{self.core.get_default_install_dir()}"')
|
||||||
# todo expand this to allow downloading single saves and extracting them to the correct directory
|
self.core.download_saves(args.app_name)
|
||||||
self.core.download_saves()
|
|
||||||
|
def sync_saves(self, args):
|
||||||
|
if not self.core.login():
|
||||||
|
logger.error('Login failed! Cannot continue with download process.')
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
igames = self.core.get_installed_list()
|
||||||
|
if args.app_name:
|
||||||
|
igame = self.core.get_installed_game(args.app_name)
|
||||||
|
if not igame:
|
||||||
|
logger.fatal(f'Game not installed: {args.app_name}')
|
||||||
|
exit(1)
|
||||||
|
igames = [igame]
|
||||||
|
|
||||||
|
# check available saves
|
||||||
|
saves = self.core.get_save_games()
|
||||||
|
latest_save = dict()
|
||||||
|
|
||||||
|
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)')
|
||||||
|
|
||||||
|
# evaluate current save state for each game.
|
||||||
|
for igame in igames:
|
||||||
|
if igame.app_name not in latest_save:
|
||||||
|
continue
|
||||||
|
|
||||||
|
game = self.core.get_game(igame.app_name)
|
||||||
|
if 'CloudSaveFolder' not in game.metadata['customAttributes']:
|
||||||
|
# this should never happen unless cloud save support was removed from a game
|
||||||
|
logger.warning(f'{igame.app_name} has remote save(s) but does not support cloud saves?!')
|
||||||
|
continue
|
||||||
|
|
||||||
|
# override save path only if app name is specified
|
||||||
|
if args.app_name and args.save_path:
|
||||||
|
logger.info(f'Overriding save path with "{args.save_path}"...')
|
||||||
|
igame.save_path = args.save_path
|
||||||
|
self.core.lgd.set_installed_game(igame.app_name, igame)
|
||||||
|
|
||||||
|
# if there is no saved save path, try to get one
|
||||||
|
if not igame.save_path:
|
||||||
|
save_path = self.core.get_save_path(igame.app_name)
|
||||||
|
|
||||||
|
# ask user if path is correct if computing for the first time
|
||||||
|
logger.info(f'Computed save path: "{save_path}"')
|
||||||
|
|
||||||
|
if '%' in save_path or '{' in save_path:
|
||||||
|
logger.warning('Path contains unprocessed variables, please enter the correct path manually.')
|
||||||
|
yn = 'n'
|
||||||
|
else:
|
||||||
|
yn = input('Is this correct? [Y/n] ')
|
||||||
|
|
||||||
|
if yn and yn.lower()[0] != 'y':
|
||||||
|
save_path = input('Please enter the correct path (leave empty to skip): ')
|
||||||
|
if not save_path:
|
||||||
|
logger.info('Empty input, skipping...')
|
||||||
|
continue
|
||||||
|
|
||||||
|
if not os.path.exists(save_path):
|
||||||
|
os.makedirs(save_path)
|
||||||
|
igame.save_path = save_path
|
||||||
|
self.core.lgd.set_installed_game(igame.app_name, igame)
|
||||||
|
|
||||||
|
# check if *any* file in the save game directory is newer than the latest uploaded save
|
||||||
|
res, (dt_l, dt_r) = self.core.check_savegame_state(igame.save_path, latest_save[igame.app_name])
|
||||||
|
|
||||||
|
if res == SaveGameStatus.SAME_AGE and not (args.force_upload or args.force_download):
|
||||||
|
logger.info(f'Save game for "{igame.title}" is up to date, skipping...')
|
||||||
|
continue
|
||||||
|
|
||||||
|
if (res == SaveGameStatus.REMOTE_NEWER and not args.force_upload) or args.force_download:
|
||||||
|
if res == SaveGameStatus.REMOTE_NEWER: # only print this info if not forced
|
||||||
|
logger.info(f'Cloud save for "{igame.title}" is newer:')
|
||||||
|
logger.info(f'- Cloud save date: {dt_l.strftime("%Y-%m-%d %H:%M:%S")}')
|
||||||
|
logger.info(f'- Local save date: {dt_r.strftime("%Y-%m-%d %H:%M:%S")}')
|
||||||
|
|
||||||
|
if args.upload_only:
|
||||||
|
logger.info('Save game downloading is disabled, skipping...')
|
||||||
|
continue
|
||||||
|
|
||||||
|
if not args.yes and not args.force_download:
|
||||||
|
choice = input(f'Download cloud save? [Y/n]: ')
|
||||||
|
if choice and choice.lower()[0] != 'y':
|
||||||
|
logger.info('Not downloading...')
|
||||||
|
continue
|
||||||
|
|
||||||
|
logger.info('Downloading remote savegame...')
|
||||||
|
self.core.download_saves(igame.app_name, save_dir=igame.save_path, clean_dir=True,
|
||||||
|
manifest_name=latest_save[igame.app_name].manifest_name)
|
||||||
|
elif res == SaveGameStatus.LOCAL_NEWER or args.force_upload:
|
||||||
|
if res == SaveGameStatus.LOCAL_NEWER:
|
||||||
|
logger.info(f'Local save for "{igame.title}" is newer')
|
||||||
|
logger.info(f'- Cloud save date: {dt_l.strftime("%Y-%m-%d %H:%M:%S")}')
|
||||||
|
logger.info(f'- Local save date: {dt_r.strftime("%Y-%m-%d %H:%M:%S")}')
|
||||||
|
|
||||||
|
if args.download_only:
|
||||||
|
logger.info('Save game uploading is disabled, skipping...')
|
||||||
|
continue
|
||||||
|
|
||||||
|
if not args.yes and not args.force_upload:
|
||||||
|
choice = input(f'Upload local save? [Y/n]: ')
|
||||||
|
if choice and choice.lower()[0] != 'y':
|
||||||
|
logger.info('Not uploading...')
|
||||||
|
continue
|
||||||
|
logger.info('Uploading local savegame...')
|
||||||
|
self.core.upload_save(igame.app_name, igame.save_path, dt_l)
|
||||||
|
|
||||||
def launch_game(self, args, extra):
|
def launch_game(self, args, extra):
|
||||||
app_name = args.app_name
|
app_name = args.app_name
|
||||||
|
@ -472,6 +578,7 @@ def main():
|
||||||
list_files_parser = subparsers.add_parser('list-files', help='List files in manifest')
|
list_files_parser = subparsers.add_parser('list-files', help='List files in manifest')
|
||||||
list_saves_parser = subparsers.add_parser('list-saves', help='List available cloud saves')
|
list_saves_parser = subparsers.add_parser('list-saves', help='List available cloud saves')
|
||||||
download_saves_parser = subparsers.add_parser('download-saves', help='Download all cloud saves')
|
download_saves_parser = subparsers.add_parser('download-saves', help='Download all cloud saves')
|
||||||
|
sync_saves_parser = subparsers.add_parser('sync-saves', help='Sync cloud saves')
|
||||||
|
|
||||||
install_parser.add_argument('app_name', help='Name of the app', metavar='<App Name>')
|
install_parser.add_argument('app_name', help='Name of the app', metavar='<App Name>')
|
||||||
uninstall_parser.add_argument('app_name', help='Name of the app', metavar='<App Name>')
|
uninstall_parser.add_argument('app_name', help='Name of the app', metavar='<App Name>')
|
||||||
|
@ -480,6 +587,10 @@ def main():
|
||||||
help='Name of the app (optional)')
|
help='Name of the app (optional)')
|
||||||
list_saves_parser.add_argument('app_name', nargs='?', metavar='<App Name>', default='',
|
list_saves_parser.add_argument('app_name', nargs='?', metavar='<App Name>', default='',
|
||||||
help='Name of the app (optional)')
|
help='Name of the app (optional)')
|
||||||
|
download_saves_parser.add_argument('app_name', nargs='?', metavar='<App Name>', default='',
|
||||||
|
help='Name of the app (optional)')
|
||||||
|
sync_saves_parser.add_argument('app_name', nargs='?', metavar='<App Name>', default='',
|
||||||
|
help='Name of the app (optional)')
|
||||||
|
|
||||||
# importing only works on Windows right now
|
# importing only works on Windows right now
|
||||||
if os.name == 'nt':
|
if os.name == 'nt':
|
||||||
|
@ -570,6 +681,17 @@ def main():
|
||||||
list_files_parser.add_argument('--install-tag', dest='install_tag', action='store', metavar='<tag>',
|
list_files_parser.add_argument('--install-tag', dest='install_tag', action='store', metavar='<tag>',
|
||||||
type=str, help='Show only files with specified install tag')
|
type=str, help='Show only files with specified install tag')
|
||||||
|
|
||||||
|
sync_saves_parser.add_argument('--skip-upload', dest='download_only', action='store_true',
|
||||||
|
help='Only download new saves from cloud, don\'t upload')
|
||||||
|
sync_saves_parser.add_argument('--skip-download', dest='upload_only', action='store_true',
|
||||||
|
help='Only upload new saves from cloud, don\'t download')
|
||||||
|
sync_saves_parser.add_argument('--force-upload', dest='force_upload', action='store_true',
|
||||||
|
help='Force upload even if local saves are older')
|
||||||
|
sync_saves_parser.add_argument('--force-download', dest='force_download', action='store_true',
|
||||||
|
help='Force download even if local saves are newer')
|
||||||
|
sync_saves_parser.add_argument('--save-path', dest='save_path', action='store',
|
||||||
|
help='Override savegame path (only if app name is specified)')
|
||||||
|
|
||||||
args, extra = parser.parse_known_args()
|
args, extra = parser.parse_known_args()
|
||||||
|
|
||||||
if args.version:
|
if args.version:
|
||||||
|
@ -578,7 +700,7 @@ def main():
|
||||||
|
|
||||||
if args.subparser_name not in ('auth', 'list-games', 'list-installed', 'list-files',
|
if args.subparser_name not in ('auth', 'list-games', 'list-installed', 'list-files',
|
||||||
'launch', 'download', 'uninstall', 'install', 'update',
|
'launch', 'download', 'uninstall', 'install', 'update',
|
||||||
'list-saves', 'download-saves'):
|
'list-saves', 'download-saves', 'sync-saves'):
|
||||||
print(parser.format_help())
|
print(parser.format_help())
|
||||||
|
|
||||||
# Print the main help *and* the help for all of the subcommands. Thanks stackoverflow!
|
# Print the main help *and* the help for all of the subcommands. Thanks stackoverflow!
|
||||||
|
@ -622,6 +744,8 @@ def main():
|
||||||
cli.list_saves(args)
|
cli.list_saves(args)
|
||||||
elif args.subparser_name == 'download-saves':
|
elif args.subparser_name == 'download-saves':
|
||||||
cli.download_saves(args)
|
cli.download_saves(args)
|
||||||
|
elif args.subparser_name == 'sync-saves':
|
||||||
|
cli.sync_saves(args)
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
logger.info('Command was aborted via KeyboardInterrupt, cleaning up...')
|
logger.info('Command was aborted via KeyboardInterrupt, cleaning up...')
|
||||||
|
|
||||||
|
|
|
@ -8,8 +8,8 @@ import shlex
|
||||||
import shutil
|
import shutil
|
||||||
|
|
||||||
from base64 import b64decode
|
from base64 import b64decode
|
||||||
from collections import defaultdict, namedtuple
|
from collections import defaultdict
|
||||||
from datetime import datetime
|
from datetime import datetime, timezone
|
||||||
from multiprocessing import Queue
|
from multiprocessing import Queue
|
||||||
from random import choice as randchoice
|
from random import choice as randchoice
|
||||||
from requests.exceptions import HTTPError
|
from requests.exceptions import HTTPError
|
||||||
|
@ -27,6 +27,7 @@ from legendary.models.json_manifest import JSONManifest
|
||||||
from legendary.models.manifest import Manifest, ManifestMeta
|
from legendary.models.manifest import Manifest, ManifestMeta
|
||||||
from legendary.models.chunk import Chunk
|
from legendary.models.chunk import Chunk
|
||||||
from legendary.utils.game_workarounds import is_opt_enabled
|
from legendary.utils.game_workarounds import is_opt_enabled
|
||||||
|
from legendary.utils.savegame_helper import SaveGameHelper
|
||||||
|
|
||||||
|
|
||||||
# ToDo: instead of true/false return values for success/failure actually raise an exception that the CLI/GUI
|
# ToDo: instead of true/false return values for success/failure actually raise an exception that the CLI/GUI
|
||||||
|
@ -45,6 +46,8 @@ class LegendaryCore:
|
||||||
self.egs = EPCAPI()
|
self.egs = EPCAPI()
|
||||||
self.lgd = LGDLFS()
|
self.lgd = LGDLFS()
|
||||||
|
|
||||||
|
self.local_timezone = datetime.now().astimezone().tzinfo
|
||||||
|
|
||||||
# epic lfs only works on Windows right now
|
# epic lfs only works on Windows right now
|
||||||
if os.name == 'nt':
|
if os.name == 'nt':
|
||||||
self.egl = EPCLFS()
|
self.egl = EPCLFS()
|
||||||
|
@ -279,33 +282,107 @@ class LegendaryCore:
|
||||||
return params, working_dir, env
|
return params, working_dir, env
|
||||||
|
|
||||||
def get_save_games(self, app_name: str = ''):
|
def get_save_games(self, app_name: str = ''):
|
||||||
# todo make this a proper class in legendary.models.egs or something
|
savegames = self.egs.get_user_cloud_saves(app_name, manifests=not not app_name)
|
||||||
CloudSave = namedtuple('CloudSave', ['filename', 'app_name', 'manifest_name', 'iso_date'])
|
|
||||||
savegames = self.egs.get_user_cloud_saves(app_name)
|
|
||||||
_saves = []
|
_saves = []
|
||||||
for fname, f in savegames['files'].items():
|
for fname, f in savegames['files'].items():
|
||||||
if '.manifest' not in fname:
|
if '.manifest' not in fname:
|
||||||
continue
|
continue
|
||||||
f_parts = fname.split('/')
|
f_parts = fname.split('/')
|
||||||
_saves.append(CloudSave(filename=fname, app_name=f_parts[2],
|
_saves.append(SaveGameFile(app_name=f_parts[2], filename=fname, manifest=f_parts[4],
|
||||||
manifest_name=f_parts[4], iso_date=f['lastModified']))
|
datetime=datetime.fromisoformat(f['lastModified'][:-1])))
|
||||||
|
|
||||||
return _saves
|
return _saves
|
||||||
|
|
||||||
def download_saves(self):
|
def get_save_path(self, app_name, wine_prefix='~/.wine'):
|
||||||
|
game = self.lgd.get_game_meta(app_name)
|
||||||
|
save_path = game.metadata['customAttributes'].get('CloudSaveFolder', {}).get('value')
|
||||||
|
if not save_path:
|
||||||
|
raise ValueError('Game does not support cloud saves')
|
||||||
|
|
||||||
|
igame = self.lgd.get_installed_game(app_name)
|
||||||
|
if not igame:
|
||||||
|
raise ValueError('Game is not installed!')
|
||||||
|
|
||||||
|
# the following variables are known:
|
||||||
|
path_vars = {
|
||||||
|
'{appdata}': os.path.expandvars('%APPDATA%'),
|
||||||
|
'{installdir}': igame.install_path,
|
||||||
|
'{userdir}': os.path.expandvars('%userprofile%/documents'),
|
||||||
|
'{epicid}': self.lgd.userdata['account_id']
|
||||||
|
}
|
||||||
|
# the following variables are in the EGL binary but are not used by any of
|
||||||
|
# my games and I'm not sure where they actually point at:
|
||||||
|
# {UserProfile} (Probably %USERPROFILE%)
|
||||||
|
# {UserSavedGames}
|
||||||
|
|
||||||
|
# these paths should always use a forward slash
|
||||||
|
new_save_path = [path_vars.get(p.lower(), p) for p in save_path.split('/')]
|
||||||
|
return os.path.join(*new_save_path)
|
||||||
|
|
||||||
|
def check_savegame_state(self, path: str, save: SaveGameFile) -> (SaveGameStatus, (datetime, datetime)):
|
||||||
|
latest = 0
|
||||||
|
for _dir, _, _files in os.walk(path):
|
||||||
|
for _file in _files:
|
||||||
|
s = os.stat(os.path.join(_dir, _file))
|
||||||
|
latest = max(latest, s.st_mtime)
|
||||||
|
|
||||||
|
# timezones are fun!
|
||||||
|
dt_local = datetime.fromtimestamp(latest).replace(tzinfo=self.local_timezone).astimezone(timezone.utc)
|
||||||
|
dt_remote = datetime.strptime(save.manifest_name, '%Y.%m.%d-%H.%M.%S.manifest').replace(tzinfo=timezone.utc)
|
||||||
|
self.log.debug(f'Local save date: {str(dt_local)}, Remote save date: {str(dt_remote)}')
|
||||||
|
|
||||||
|
# Ideally we check the files themselves based on manifest,
|
||||||
|
# this is mostly a guess but should be accurate enough.
|
||||||
|
if abs((dt_local - dt_remote).total_seconds()) < 60:
|
||||||
|
return SaveGameStatus.SAME_AGE, (dt_local, dt_remote)
|
||||||
|
elif dt_local > dt_remote:
|
||||||
|
return SaveGameStatus.LOCAL_NEWER, (dt_local, dt_remote)
|
||||||
|
else:
|
||||||
|
return SaveGameStatus.REMOTE_NEWER, (dt_local, dt_remote)
|
||||||
|
|
||||||
|
def upload_save(self, app_name, save_dir, local_dt: datetime = None):
|
||||||
|
game = self.lgd.get_game_meta(app_name)
|
||||||
|
save_path = game.metadata['customAttributes'].get('CloudSaveFolder', {}).get('value')
|
||||||
|
if not save_path:
|
||||||
|
raise ValueError('Game does not support cloud saves')
|
||||||
|
|
||||||
|
sgh = SaveGameHelper()
|
||||||
|
files = sgh.package_savegame(save_dir, app_name, self.egs.user.get('account_id'),
|
||||||
|
save_path, local_dt)
|
||||||
|
|
||||||
|
self.log.debug(f'Packed files: {str(files)}, creating cloud files...')
|
||||||
|
resp = self.egs.create_game_cloud_saves(app_name, list(files.keys()))
|
||||||
|
|
||||||
|
self.log.info('Starting upload...')
|
||||||
|
for remote_path, file_info in resp['files'].items():
|
||||||
|
self.log.debug(f'Uploading "{remote_path}"')
|
||||||
|
f = files.get(remote_path)
|
||||||
|
self.egs.unauth_session.put(file_info['writeLink'], data=f.read())
|
||||||
|
|
||||||
|
self.log.info('Finished uploading savegame.')
|
||||||
|
|
||||||
|
def download_saves(self, app_name='', manifest_name='', save_dir='', clean_dir=False):
|
||||||
save_path = os.path.join(self.get_default_install_dir(), '.saves')
|
save_path = os.path.join(self.get_default_install_dir(), '.saves')
|
||||||
if not os.path.exists(save_path):
|
if not os.path.exists(save_path):
|
||||||
os.makedirs(save_path)
|
os.makedirs(save_path)
|
||||||
|
|
||||||
savegames = self.egs.get_user_cloud_saves()
|
savegames = self.egs.get_user_cloud_saves(app_name=app_name)
|
||||||
files = savegames['files']
|
files = savegames['files']
|
||||||
for fname, f in files.items():
|
for fname, f in files.items():
|
||||||
if '.manifest' not in fname:
|
if '.manifest' not in fname:
|
||||||
continue
|
continue
|
||||||
f_parts = fname.split('/')
|
f_parts = fname.split('/')
|
||||||
save_dir = os.path.join(save_path, f'{f_parts[2]}/{f_parts[4].rpartition(".")[0]}')
|
|
||||||
if not os.path.exists(save_dir):
|
if manifest_name and f_parts[4] != manifest_name:
|
||||||
os.makedirs(save_dir)
|
continue
|
||||||
|
if not save_dir:
|
||||||
|
save_dir = os.path.join(save_path, f'{f_parts[2]}/{f_parts[4].rpartition(".")[0]}')
|
||||||
|
if not os.path.exists(save_dir):
|
||||||
|
os.makedirs(save_dir)
|
||||||
|
|
||||||
|
if clean_dir:
|
||||||
|
self.log.info('Deleting old save files...')
|
||||||
|
delete_folder(save_dir)
|
||||||
|
|
||||||
self.log.info(f'Downloading "{fname.split("/", 2)[2]}"...')
|
self.log.info(f'Downloading "{fname.split("/", 2)[2]}"...')
|
||||||
# download manifest
|
# download manifest
|
||||||
|
@ -315,7 +392,7 @@ class LegendaryCore:
|
||||||
continue
|
continue
|
||||||
m = self.load_manfiest(r.content)
|
m = self.load_manfiest(r.content)
|
||||||
|
|
||||||
# download chunks requierd for extraction
|
# download chunks required for extraction
|
||||||
chunks = dict()
|
chunks = dict()
|
||||||
for chunk in m.chunk_data_list.elements:
|
for chunk in m.chunk_data_list.elements:
|
||||||
cpath_p = fname.split('/', 3)[:3]
|
cpath_p = fname.split('/', 3)[:3]
|
||||||
|
@ -341,6 +418,13 @@ class LegendaryCore:
|
||||||
for cp in fm.chunk_parts:
|
for cp in fm.chunk_parts:
|
||||||
fh.write(chunks[cp.guid_num][cp.offset:cp.offset+cp.size])
|
fh.write(chunks[cp.guid_num][cp.offset:cp.offset+cp.size])
|
||||||
|
|
||||||
|
# set modified time to savegame creation timestamp
|
||||||
|
m_date = datetime.strptime(f_parts[4], '%Y.%m.%d-%H.%M.%S.manifest')
|
||||||
|
m_date = m_date.replace(tzinfo=timezone.utc).astimezone(self.local_timezone)
|
||||||
|
os.utime(fpath, (m_date.timestamp(), m_date.timestamp()))
|
||||||
|
|
||||||
|
self.log.info('Successfully completed savegame download.')
|
||||||
|
|
||||||
def is_offline_game(self, app_name: str) -> bool:
|
def is_offline_game(self, app_name: str) -> bool:
|
||||||
return self.lgd.config.getboolean(app_name, 'offline', fallback=False)
|
return self.lgd.config.getboolean(app_name, 'offline', fallback=False)
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
# coding: utf-8
|
# coding: utf-8
|
||||||
|
|
||||||
|
from enum import Enum
|
||||||
|
|
||||||
|
|
||||||
class GameAsset:
|
class GameAsset:
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
|
@ -73,7 +75,7 @@ class Game:
|
||||||
class InstalledGame:
|
class InstalledGame:
|
||||||
def __init__(self, app_name='', title='', version='', manifest_path='', base_urls=None,
|
def __init__(self, app_name='', title='', version='', manifest_path='', base_urls=None,
|
||||||
install_path='', executable='', launch_parameters='', prereq_info=None,
|
install_path='', executable='', launch_parameters='', prereq_info=None,
|
||||||
can_run_offline=False, requires_ot=False, is_dlc=False):
|
can_run_offline=False, requires_ot=False, is_dlc=False, save_path=None):
|
||||||
self.app_name = app_name
|
self.app_name = app_name
|
||||||
self.title = title
|
self.title = title
|
||||||
self.version = version
|
self.version = version
|
||||||
|
@ -87,6 +89,7 @@ class InstalledGame:
|
||||||
self.can_run_offline = can_run_offline
|
self.can_run_offline = can_run_offline
|
||||||
self.requires_ot = requires_ot
|
self.requires_ot = requires_ot
|
||||||
self.is_dlc = is_dlc
|
self.is_dlc = is_dlc
|
||||||
|
self.save_path = save_path
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_json(cls, json):
|
def from_json(cls, json):
|
||||||
|
@ -105,4 +108,19 @@ class InstalledGame:
|
||||||
tmp.can_run_offline = json.get('can_run_offline', False)
|
tmp.can_run_offline = json.get('can_run_offline', False)
|
||||||
tmp.requires_ot = json.get('requires_ot', False)
|
tmp.requires_ot = json.get('requires_ot', False)
|
||||||
tmp.is_dlc = json.get('is_dlc', False)
|
tmp.is_dlc = json.get('is_dlc', False)
|
||||||
|
tmp.save_path = json.get('save_path', None)
|
||||||
return tmp
|
return tmp
|
||||||
|
|
||||||
|
|
||||||
|
class SaveGameFile:
|
||||||
|
def __init__(self, app_name='', filename='', manifest='', datetime=None):
|
||||||
|
self.app_name = app_name
|
||||||
|
self.filename = filename
|
||||||
|
self.manifest_name = manifest
|
||||||
|
self.datetime = datetime
|
||||||
|
|
||||||
|
|
||||||
|
class SaveGameStatus(Enum):
|
||||||
|
LOCAL_NEWER = 1
|
||||||
|
REMOTE_NEWER = -1
|
||||||
|
SAME_AGE = 0
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
import time
|
|
||||||
|
from datetime import datetime
|
||||||
from hashlib import sha1
|
from hashlib import sha1
|
||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
from tempfile import TemporaryFile
|
from tempfile import TemporaryFile
|
||||||
|
@ -30,12 +31,14 @@ class SaveGameHelper:
|
||||||
return ci
|
return ci
|
||||||
|
|
||||||
def package_savegame(self, input_folder: str, app_name: str = '',
|
def package_savegame(self, input_folder: str, app_name: str = '',
|
||||||
epic_id: str = '', cloud_folder: str = ''):
|
epic_id: str = '', cloud_folder: str = '',
|
||||||
|
manifest_dt: datetime = None):
|
||||||
"""
|
"""
|
||||||
:param input_folder: Folder to be packaged into chunks/manifest
|
:param input_folder: Folder to be packaged into chunks/manifest
|
||||||
:param app_name: App name for savegame being stored
|
:param app_name: App name for savegame being stored
|
||||||
:param epic_id: Epic account ID
|
:param epic_id: Epic account ID
|
||||||
:param cloud_folder: Folder the savegame resides in (based on game metadata)
|
:param cloud_folder: Folder the savegame resides in (based on game metadata)
|
||||||
|
:param manifest_dt: datetime for the manifest name (optional)
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
m = Manifest()
|
m = Manifest()
|
||||||
|
@ -45,7 +48,9 @@ class SaveGameHelper:
|
||||||
m.custom_fields = CustomFields()
|
m.custom_fields = CustomFields()
|
||||||
# create metadata for savegame
|
# create metadata for savegame
|
||||||
m.meta.app_name = f'{app_name}{epic_id}'
|
m.meta.app_name = f'{app_name}{epic_id}'
|
||||||
m.meta.build_version = time.strftime('%Y.%m.%d-%H.%M.%S')
|
if not manifest_dt:
|
||||||
|
manifest_dt = datetime.utcnow()
|
||||||
|
m.meta.build_version = manifest_dt.strftime('%Y.%m.%d-%H.%M.%S')
|
||||||
m.custom_fields['CloudSaveFolder'] = cloud_folder
|
m.custom_fields['CloudSaveFolder'] = cloud_folder
|
||||||
|
|
||||||
self.log.info(f'Packing savegame for "{app_name}", input folder: {input_folder}')
|
self.log.info(f'Packing savegame for "{app_name}", input folder: {input_folder}')
|
||||||
|
|
Loading…
Reference in a new issue