|
|
@ -37,7 +37,7 @@ from ..utils import ( |
|
|
|
parse_codecs, |
|
|
|
parse_duration, |
|
|
|
remove_quotes, |
|
|
|
# remove_start, |
|
|
|
remove_start, |
|
|
|
smuggle_url, |
|
|
|
str_to_int, |
|
|
|
try_get, |
|
|
@ -55,13 +55,8 @@ class YoutubeBaseInfoExtractor(InfoExtractor): |
|
|
|
_TWOFACTOR_URL = 'https://accounts.google.com/signin/challenge' |
|
|
|
|
|
|
|
_LOOKUP_URL = 'https://accounts.google.com/_/signin/sl/lookup' |
|
|
|
_LOOKUP_REQ_TEMPLATE = '["{0}",null,[],null,"US",null,null,2,false,true,[null,null,[2,1,null,1,"https://accounts.google.com/ServiceLogin?passive=true&continue=https%3A%2F%2Fwww.youtube.com%2Fsignin%3Fnext%3D%252F%26action_handle_signin%3Dtrue%26hl%3Den%26app%3Ddesktop%26feature%3Dsign_in_button&hl=en&service=youtube&uilel=3&requestPath=%2FServiceLogin&Page=PasswordSeparationSignIn",null,[],4],1,[null,null,[]],null,null,null,true],"{0}"]' |
|
|
|
|
|
|
|
_PASSWORD_CHALLENGE_URL = 'https://accounts.google.com/_/signin/sl/challenge' |
|
|
|
_PASSWORD_CHALLENGE_REQ_TEMPLATE = '["{0}",null,1,null,[1,null,null,null,["{1}",null,true]],[null,null,[2,1,null,1,"https://accounts.google.com/ServiceLogin?passive=true&continue=https%3A%2F%2Fwww.youtube.com%2Fsignin%3Fnext%3D%252F%26action_handle_signin%3Dtrue%26hl%3Den%26app%3Ddesktop%26feature%3Dsign_in_button&hl=en&service=youtube&uilel=3&requestPath=%2FServiceLogin&Page=PasswordSeparationSignIn",null,[],4],1,[null,null,[]],null,null,null,true]]' |
|
|
|
|
|
|
|
_TFA_URL = 'https://accounts.google.com/_/signin/challenge' |
|
|
|
_TFA_REQ_TEMPLATE = '["{0}",null,2,null,[9,null,null,null,null,null,null,null,[null,"{1}",false,2]]]' |
|
|
|
_CHALLENGE_URL = 'https://accounts.google.com/_/signin/sl/challenge' |
|
|
|
_TFA_URL = 'https://accounts.google.com/_/signin/challenge?hl=en&TL={0}' |
|
|
|
|
|
|
|
_NETRC_MACHINE = 'youtube' |
|
|
|
# If True it will raise an error if no login info is provided |
|
|
@ -112,7 +107,7 @@ class YoutubeBaseInfoExtractor(InfoExtractor): |
|
|
|
'checkedDomains': 'youtube', |
|
|
|
'hl': 'en', |
|
|
|
'deviceinfo': '[null,null,null,[],null,"US",null,null,[],"GlifWebSignIn",null,[null,null,[]]]', |
|
|
|
'f.req': f_req, |
|
|
|
'f.req': json.dumps(f_req), |
|
|
|
'flowName': 'GlifWebSignIn', |
|
|
|
'flowEntry': 'ServiceLogin', |
|
|
|
}) |
|
|
@ -125,53 +120,127 @@ class YoutubeBaseInfoExtractor(InfoExtractor): |
|
|
|
'Google-Accounts-XSRF': 1, |
|
|
|
}) |
|
|
|
|
|
|
|
def warn(message): |
|
|
|
self._downloader.report_warning(message) |
|
|
|
|
|
|
|
lookup_req = [ |
|
|
|
username, |
|
|
|
None, [], None, 'US', None, None, 2, False, True, |
|
|
|
[ |
|
|
|
None, None, |
|
|
|
[2, 1, None, 1, |
|
|
|
'https://accounts.google.com/ServiceLogin?passive=true&continue=https%3A%2F%2Fwww.youtube.com%2Fsignin%3Fnext%3D%252F%26action_handle_signin%3Dtrue%26hl%3Den%26app%3Ddesktop%26feature%3Dsign_in_button&hl=en&service=youtube&uilel=3&requestPath=%2FServiceLogin&Page=PasswordSeparationSignIn', |
|
|
|
None, [], 4], |
|
|
|
1, [None, None, []], None, None, None, True |
|
|
|
], |
|
|
|
username, |
|
|
|
] |
|
|
|
|
|
|
|
lookup_results = req( |
|
|
|
self._LOOKUP_URL, self._LOOKUP_REQ_TEMPLATE.format(username), |
|
|
|
self._LOOKUP_URL, lookup_req, |
|
|
|
'Looking up account info', 'Unable to look up account info') |
|
|
|
|
|
|
|
if lookup_results is False: |
|
|
|
return False |
|
|
|
|
|
|
|
user_hash = lookup_results[0][2] |
|
|
|
user_hash = try_get(lookup_results, lambda x: x[0][2], compat_str) |
|
|
|
if not user_hash: |
|
|
|
warn('Unable to extract user hash') |
|
|
|
return False |
|
|
|
|
|
|
|
challenge_req = [ |
|
|
|
user_hash, |
|
|
|
None, 1, None, [1, None, None, None, [password, None, True]], |
|
|
|
[ |
|
|
|
None, None, [2, 1, None, 1, 'https://accounts.google.com/ServiceLogin?passive=true&continue=https%3A%2F%2Fwww.youtube.com%2Fsignin%3Fnext%3D%252F%26action_handle_signin%3Dtrue%26hl%3Den%26app%3Ddesktop%26feature%3Dsign_in_button&hl=en&service=youtube&uilel=3&requestPath=%2FServiceLogin&Page=PasswordSeparationSignIn', None, [], 4], |
|
|
|
1, [None, None, []], None, None, None, True |
|
|
|
]] |
|
|
|
|
|
|
|
password_challenge_results = req( |
|
|
|
self._PASSWORD_CHALLENGE_URL, |
|
|
|
self._PASSWORD_CHALLENGE_REQ_TEMPLATE.format(user_hash, password), |
|
|
|
'Logging in', 'Unable to log in')[0] |
|
|
|
challenge_results = req( |
|
|
|
self._CHALLENGE_URL, challenge_req, |
|
|
|
'Logging in', 'Unable to log in') |
|
|
|
|
|
|
|
if password_challenge_results is False: |
|
|
|
if challenge_results is False: |
|
|
|
return |
|
|
|
|
|
|
|
msg = password_challenge_results[5] |
|
|
|
if msg is not None and isinstance(msg, list): |
|
|
|
raise ExtractorError('Unable to login: %s' % msg[5], expected=True) |
|
|
|
|
|
|
|
password_challenge_results = password_challenge_results[-1] |
|
|
|
|
|
|
|
# tfa = password_challenge_results[0] |
|
|
|
# if isinstance(tfa, list) and tfa[0][2] == 'TWO_STEP_VERIFICATION': |
|
|
|
# tfa_code = self._get_tfa_info('2-step verification code') |
|
|
|
# |
|
|
|
# if not tfa_code: |
|
|
|
# self._downloader.report_warning( |
|
|
|
# 'Two-factor authentication required. Provide it either interactively or with --twofactor <code>' |
|
|
|
# '(Note that only TOTP (Google Authenticator App) codes work at this time.)') |
|
|
|
# return False |
|
|
|
# |
|
|
|
# tfa_code = remove_start(tfa_code, 'G-') |
|
|
|
# print('tfa', tfa_code) |
|
|
|
# tfa_results = req( |
|
|
|
# self._TFA_URL, |
|
|
|
# self._TFA_REQ_TEMPLATE.format(user_hash, tfa_code), |
|
|
|
# 'Submitting TFA code', 'Unable to submit TFA code') |
|
|
|
# |
|
|
|
# TODO |
|
|
|
login_res = try_get(challenge_results, lambda x: x[0][5], list) |
|
|
|
if login_res: |
|
|
|
login_msg = try_get(login_res, lambda x: x[5], compat_str) |
|
|
|
warn( |
|
|
|
'Unable to login: %s' % 'Invalid password' |
|
|
|
if login_msg == 'INCORRECT_ANSWER_ENTERED' else login_msg) |
|
|
|
return False |
|
|
|
|
|
|
|
res = try_get(challenge_results, lambda x: x[0][-1], list) |
|
|
|
if not res: |
|
|
|
warn('Unable to extract result entry') |
|
|
|
return False |
|
|
|
|
|
|
|
tfa = try_get(res, lambda x: x[0][0], list) |
|
|
|
if tfa: |
|
|
|
tfa_str = try_get(tfa, lambda x: x[2], compat_str) |
|
|
|
if tfa_str == 'TWO_STEP_VERIFICATION': |
|
|
|
# SEND_SUCCESS - TFA code has been successfully sent to phone |
|
|
|
# QUOTA_EXCEEDED - reached the limit of TFA codes |
|
|
|
status = try_get(tfa, lambda x: x[5], compat_str) |
|
|
|
if status == 'QUOTA_EXCEEDED': |
|
|
|
warn('Exceeded the limit of TFA codes, try later') |
|
|
|
return False |
|
|
|
|
|
|
|
tl = try_get(challenge_results, lambda x: x[1][2], compat_str) |
|
|
|
if not tl: |
|
|
|
warn('Unable to extract TL') |
|
|
|
return False |
|
|
|
|
|
|
|
tfa_code = self._get_tfa_info('2-step verification code') |
|
|
|
|
|
|
|
if not tfa_code: |
|
|
|
warn( |
|
|
|
'Two-factor authentication required. Provide it either interactively or with --twofactor <code>' |
|
|
|
'(Note that only TOTP (Google Authenticator App) codes work at this time.)') |
|
|
|
return False |
|
|
|
|
|
|
|
tfa_code = remove_start(tfa_code, 'G-') |
|
|
|
|
|
|
|
tfa_req = [ |
|
|
|
user_hash, None, 2, None, |
|
|
|
[ |
|
|
|
9, None, None, None, None, None, None, None, |
|
|
|
[None, tfa_code, True, 2] |
|
|
|
]] |
|
|
|
|
|
|
|
tfa_results = req( |
|
|
|
self._TFA_URL.format(tl), tfa_req, |
|
|
|
'Submitting TFA code', 'Unable to submit TFA code') |
|
|
|
|
|
|
|
if tfa_results is False: |
|
|
|
return False |
|
|
|
|
|
|
|
tfa_res = try_get(tfa_results, lambda x: x[0][5], list) |
|
|
|
if tfa_res: |
|
|
|
tfa_msg = try_get(tfa_res, lambda x: x[5], compat_str) |
|
|
|
warn( |
|
|
|
'Unable to finish TFA: %s' % 'Invalid TFA code' |
|
|
|
if tfa_msg == 'INCORRECT_ANSWER_ENTERED' else tfa_msg) |
|
|
|
return False |
|
|
|
|
|
|
|
check_cookie_url = try_get( |
|
|
|
tfa_results, lambda x: x[0][-1][2], compat_str) |
|
|
|
else: |
|
|
|
check_cookie_url = try_get(res, lambda x: x[2], compat_str) |
|
|
|
|
|
|
|
if not check_cookie_url: |
|
|
|
warn('Unable to extract CheckCookie URL') |
|
|
|
return False |
|
|
|
|
|
|
|
check_cookie_results = self._download_webpage( |
|
|
|
password_challenge_results[2], None, 'Checking cookie') |
|
|
|
check_cookie_url, None, 'Checking cookie', fatal=False) |
|
|
|
|
|
|
|
if check_cookie_results is False: |
|
|
|
return False |
|
|
|
|
|
|
|
if '>Sign out<' not in check_cookie_results: |
|
|
|
self._downloader.report_warning('Unable to log in') |
|
|
|
if 'https://myaccount.google.com/' not in check_cookie_results: |
|
|
|
warn('Unable to log in') |
|
|
|
return False |
|
|
|
|
|
|
|
return True |
|
|
|