mirror of
https://github.com/derrod/legendary.git
synced 2025-01-22 06:50:59 +00:00
[cli/api/models] Add "activate" command to redeem Uplay games
This commit is contained in:
parent
a8e35e9f3b
commit
e71ab3155e
|
@ -7,10 +7,12 @@ import logging
|
|||
from requests.auth import HTTPBasicAuth
|
||||
|
||||
from legendary.models.exceptions import InvalidCredentialsError
|
||||
from legendary.models.gql import *
|
||||
|
||||
|
||||
class EPCAPI:
|
||||
_user_agent = 'UELauncher/11.0.1-14907503+++Portal+Release-Live Windows/10.0.19041.1.256.64bit'
|
||||
_store_user_agent = 'EpicGamesLauncher/11.0.1-14907503+++Portal+Release-Live'
|
||||
# required for the oauth request
|
||||
_user_basic = '34a02cf8f4414e29b15921876da36f9a'
|
||||
_pw_basic = 'daafbccc737745039dffe53d94fc76cf'
|
||||
|
@ -23,6 +25,7 @@ class EPCAPI:
|
|||
_ecommerce_host = 'ecommerceintegration-public-service-ecomprod02.ol.epicgames.com'
|
||||
_datastorage_host = 'datastorage-public-service-liveegs.live.use1a.on.epicgames.com'
|
||||
_library_host = 'library-service.live.use1a.on.epicgames.com'
|
||||
_store_gql_host = 'store-launcher.epicgames.com'
|
||||
|
||||
def __init__(self, lc='en', cc='US'):
|
||||
self.session = requests.session()
|
||||
|
@ -42,6 +45,7 @@ class EPCAPI:
|
|||
# update user-agent
|
||||
if version := egs_params['version']:
|
||||
self._user_agent = f'UELauncher/{version} Windows/10.0.19041.1.256.64bit'
|
||||
self._user_agent = f'EpicGamesLauncher/{version}'
|
||||
self.session.headers['User-Agent'] = self._user_agent
|
||||
self.unauth_session.headers['User-Agent'] = self._user_agent
|
||||
# update label
|
||||
|
@ -112,6 +116,12 @@ class EPCAPI:
|
|||
r.raise_for_status()
|
||||
return r.content
|
||||
|
||||
def get_external_auths(self):
|
||||
user_id = self.user.get('account_id')
|
||||
r = self.session.get(f'https://{self._oauth_host}/account/api/public/account/{user_id}/externalAuths')
|
||||
r.raise_for_status()
|
||||
return r.json()
|
||||
|
||||
def get_game_assets(self, platform='Windows', label='Live'):
|
||||
r = self.session.get(f'https://{self._launcher_host}/launcher/api/public/assets/{platform}',
|
||||
params=dict(label=label))
|
||||
|
@ -188,3 +198,30 @@ class EPCAPI:
|
|||
url = f'https://{self._datastorage_host}/api/v1/data/egstore/{path}'
|
||||
r = self.session.delete(url)
|
||||
r.raise_for_status()
|
||||
|
||||
def store_get_uplay_codes(self):
|
||||
user_id = self.user.get('account_id')
|
||||
r = self.session.post(f'https://{self._store_gql_host}/graphql',
|
||||
json=dict(query=uplay_codes_query,
|
||||
variables=dict(accountId=user_id)))
|
||||
r.raise_for_status()
|
||||
return r.json()
|
||||
|
||||
def store_claim_uplay_code(self, uplay_id, game_id):
|
||||
user_id = self.user.get('account_id')
|
||||
r = self.session.post(f'https://{self._store_gql_host}/graphql',
|
||||
json=dict(query=uplay_claim_query,
|
||||
variables=dict(accountId=user_id,
|
||||
uplayAccountId=uplay_id,
|
||||
gameId=game_id)))
|
||||
r.raise_for_status()
|
||||
return r.json()
|
||||
|
||||
def store_redeem_uplay_codes(self, uplay_id):
|
||||
user_id = self.user.get('account_id')
|
||||
r = self.session.post(f'https://{self._store_gql_host}/graphql',
|
||||
json=dict(query=uplay_redeem_query,
|
||||
variables=dict(accountId=user_id,
|
||||
uplayAccountId=uplay_id)))
|
||||
r.raise_for_status()
|
||||
return r.json()
|
||||
|
|
|
@ -1730,6 +1730,63 @@ class LegendaryCLI:
|
|||
after = self.core.lgd.get_dir_size()
|
||||
logger.info(f'Cleanup complete! Removed {(before - after)/1024/1024:.02f} MiB.')
|
||||
|
||||
def activate(self, args):
|
||||
if not args.uplay:
|
||||
logger.error('Only Uplay is supported.')
|
||||
return
|
||||
if not self.core.login():
|
||||
logger.error('Login failed!')
|
||||
return
|
||||
|
||||
ubi_account_id = ''
|
||||
ext_auths = self.core.egs.get_external_auths()
|
||||
for ext_auth in ext_auths:
|
||||
if ext_auth['type'] != 'ubisoft':
|
||||
continue
|
||||
ubi_account_id = ext_auth['externalAuthId']
|
||||
break
|
||||
else:
|
||||
logger.error('No ubisoft account found! Please link your accounts via the following link: '
|
||||
'https://www.epicgames.com/id/link/ubisoft')
|
||||
return
|
||||
|
||||
uplay_keys = self.core.egs.store_get_uplay_codes()
|
||||
key_list = uplay_keys['data']['PartnerIntegration']['accountUplayCodes']
|
||||
redeemed = {k['gameId'] for k in key_list if k['redeemedOnUplay']}
|
||||
|
||||
games = self.core.get_game_list()
|
||||
uplay_games = []
|
||||
for game in games:
|
||||
if game.metadata.get('customAttributes', {}).get('partnerLinkType', {}).get('value') != 'ubisoft':
|
||||
continue
|
||||
if game.metadata.get('customAttributes', {}).get('partnerLinkId', {}).get('value') in redeemed:
|
||||
continue
|
||||
uplay_games.append(game)
|
||||
|
||||
if not uplay_games:
|
||||
logger.info('No remaining games found.')
|
||||
return
|
||||
|
||||
logger.info(f'Found {len(uplay_games)} games to redeem:')
|
||||
for game in sorted(uplay_games, key=lambda g: g.app_title.lower()):
|
||||
logger.info(f' - {game.app_title}')
|
||||
|
||||
if not args.yes:
|
||||
y_n = get_boolean_choice('Do you want to redeem these games?')
|
||||
if not y_n:
|
||||
logger.info('Aborting.')
|
||||
return
|
||||
|
||||
try:
|
||||
for game in uplay_games:
|
||||
game_id = game.metadata.get('customAttributes', {}).get('partnerLinkId', {}).get('value')
|
||||
self.core.egs.store_claim_uplay_code(ubi_account_id, game_id)
|
||||
self.core.egs.store_redeem_uplay_codes(ubi_account_id)
|
||||
except Exception as e:
|
||||
logger.error(f'Failed to redeem Uplay codes: {e!r}')
|
||||
else:
|
||||
logger.info('Redeemed all outstanding Uplay codes.')
|
||||
|
||||
|
||||
def main():
|
||||
parser = argparse.ArgumentParser(description=f'Legendary v{__version__} - "{__codename__}"')
|
||||
|
@ -1768,6 +1825,7 @@ def main():
|
|||
info_parser = subparsers.add_parser('info', help='Prints info about specified app name or manifest')
|
||||
alias_parser = subparsers.add_parser('alias', help='Manage aliases')
|
||||
clean_parser = subparsers.add_parser('cleanup', help='Remove old temporary, metadata, and manifest files')
|
||||
activate_parser = subparsers.add_parser('activate', help='Activate games on third party launchers')
|
||||
|
||||
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>')
|
||||
|
@ -2017,6 +2075,9 @@ def main():
|
|||
default='Mac' if sys_platform == 'darwin' else 'Windows', type=str,
|
||||
help='Platform to fetch info for (default: installed or Mac on macOS, Windows otherwise)')
|
||||
|
||||
activate_parser.add_argument('--uplay', dest='uplay', action='store_true',
|
||||
help='Activate Uplay titles')
|
||||
|
||||
args, extra = parser.parse_known_args()
|
||||
|
||||
if args.version:
|
||||
|
@ -2027,7 +2088,7 @@ def main():
|
|||
'launch', 'download', 'uninstall', 'install', 'update',
|
||||
'repair', 'list-saves', 'download-saves', 'sync-saves',
|
||||
'clean-saves', 'verify-game', 'import-game', 'egl-sync',
|
||||
'status', 'info', 'alias', 'cleanup'):
|
||||
'status', 'info', 'alias', 'cleanup', 'activate'):
|
||||
print(parser.format_help())
|
||||
|
||||
# Print the main help *and* the help for all of the subcommands. Thanks stackoverflow!
|
||||
|
@ -2094,6 +2155,8 @@ def main():
|
|||
cli.alias(args)
|
||||
elif args.subparser_name == 'cleanup':
|
||||
cli.cleanup(args)
|
||||
elif args.subparser_name == 'activate':
|
||||
cli.activate(args)
|
||||
except KeyboardInterrupt:
|
||||
logger.info('Command was aborted via KeyboardInterrupt, cleaning up...')
|
||||
|
||||
|
|
61
legendary/models/gql.py
Normal file
61
legendary/models/gql.py
Normal file
|
@ -0,0 +1,61 @@
|
|||
# GQL queries needed for the EGS API
|
||||
|
||||
uplay_codes_query = '''
|
||||
query partnerIntegrationQuery($accountId: String!) {
|
||||
PartnerIntegration {
|
||||
accountUplayCodes(accountId: $accountId) {
|
||||
epicAccountId
|
||||
gameId
|
||||
uplayAccountId
|
||||
regionCode
|
||||
redeemedOnUplay
|
||||
redemptionTimestamp
|
||||
}
|
||||
}
|
||||
}
|
||||
'''
|
||||
|
||||
uplay_redeem_query = '''
|
||||
mutation redeemAllPendingCodes($accountId: String!, $uplayAccountId: String!) {
|
||||
PartnerIntegration {
|
||||
redeemAllPendingCodes(accountId: $accountId, uplayAccountId: $uplayAccountId) {
|
||||
data {
|
||||
epicAccountId
|
||||
uplayAccountId
|
||||
redeemedOnUplay
|
||||
redemptionTimestamp
|
||||
}
|
||||
success
|
||||
}
|
||||
}
|
||||
}
|
||||
'''
|
||||
|
||||
uplay_claim_query = '''
|
||||
mutation claimUplayCode($accountId: String!, $uplayAccountId: String!, $gameId: String!) {
|
||||
PartnerIntegration {
|
||||
claimUplayCode(
|
||||
accountId: $accountId
|
||||
uplayAccountId: $uplayAccountId
|
||||
gameId: $gameId
|
||||
) {
|
||||
data {
|
||||
assignmentTimestam
|
||||
epicAccountId
|
||||
epicEntitlement {
|
||||
entitlementId
|
||||
catalogItemId
|
||||
entitlementName
|
||||
country
|
||||
}
|
||||
gameId
|
||||
redeemedOnUplay
|
||||
redemptionTimestamp
|
||||
regionCode
|
||||
uplayAccountId
|
||||
}
|
||||
success
|
||||
}
|
||||
}
|
||||
}
|
||||
'''
|
Loading…
Reference in a new issue