|
|
@ -324,47 +324,56 @@ class FacebookIE(InfoExtractor):
|
|
|
|
if server_js_data:
|
|
|
|
if server_js_data:
|
|
|
|
video_data = extract_video_data(server_js_data.get('instances', []))
|
|
|
|
video_data = extract_video_data(server_js_data.get('instances', []))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def extract_from_jsmods_instances(js_data):
|
|
|
|
|
|
|
|
if js_data:
|
|
|
|
|
|
|
|
return extract_video_data(try_get(
|
|
|
|
|
|
|
|
js_data, lambda x: x['jsmods']['instances'], list) or [])
|
|
|
|
|
|
|
|
|
|
|
|
if not video_data:
|
|
|
|
if not video_data:
|
|
|
|
server_js_data = self._parse_json(
|
|
|
|
server_js_data = self._parse_json(
|
|
|
|
self._search_regex(
|
|
|
|
self._search_regex(
|
|
|
|
r'bigPipe\.onPageletArrive\(({.+?})\)\s*;\s*}\s*\)\s*,\s*["\']onPageletArrive\s+(?:stream_pagelet|pagelet_group_mall|permalink_video_pagelet)',
|
|
|
|
r'bigPipe\.onPageletArrive\(({.+?})\)\s*;\s*}\s*\)\s*,\s*["\']onPageletArrive\s+(?:stream_pagelet|pagelet_group_mall|permalink_video_pagelet)',
|
|
|
|
webpage, 'js data', default='{}'),
|
|
|
|
webpage, 'js data', default='{}'),
|
|
|
|
video_id, transform_source=js_to_json, fatal=False)
|
|
|
|
video_id, transform_source=js_to_json, fatal=False)
|
|
|
|
if server_js_data:
|
|
|
|
video_data = extract_from_jsmods_instances(server_js_data)
|
|
|
|
video_data = extract_video_data(try_get(
|
|
|
|
|
|
|
|
server_js_data, lambda x: x['jsmods']['instances'],
|
|
|
|
|
|
|
|
list) or [])
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if not video_data:
|
|
|
|
if not video_data:
|
|
|
|
# video info not in first request, do a secondary request using tahoe player specific url
|
|
|
|
if not fatal_if_no_video:
|
|
|
|
|
|
|
|
return webpage, False
|
|
|
|
|
|
|
|
m_msg = re.search(r'class="[^"]*uiInterstitialContent[^"]*"><div>(.*?)</div>', webpage)
|
|
|
|
|
|
|
|
if m_msg is not None:
|
|
|
|
|
|
|
|
raise ExtractorError(
|
|
|
|
|
|
|
|
'The video is not available, Facebook said: "%s"' % m_msg.group(1),
|
|
|
|
|
|
|
|
expected=True)
|
|
|
|
|
|
|
|
elif '>You must log in to continue' in webpage:
|
|
|
|
|
|
|
|
self.raise_login_required()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Video info not in first request, do a secondary request using
|
|
|
|
|
|
|
|
# tahoe player specific URL
|
|
|
|
tahoe_data = self._download_webpage(
|
|
|
|
tahoe_data = self._download_webpage(
|
|
|
|
self._VIDEO_PAGE_TAHOE_TEMPLATE % video_id, video_id,
|
|
|
|
self._VIDEO_PAGE_TAHOE_TEMPLATE % video_id, video_id,
|
|
|
|
data=urlencode_postdata({
|
|
|
|
data=urlencode_postdata({
|
|
|
|
'__user': 0,
|
|
|
|
'__user': 0,
|
|
|
|
'__a': 1,
|
|
|
|
'__a': 1,
|
|
|
|
'__pc': self._search_regex(r'"pkg_cohort":"(.*?)"', webpage, 'pkg cohort', default='PHASED:DEFAULT'),
|
|
|
|
'__pc': self._search_regex(
|
|
|
|
'__rev': self._search_regex(r'"client_revision":(\d+),', webpage, 'client revision', default=3944515),
|
|
|
|
r'pkg_cohort["\']\s*:\s*["\'](.+?)["\']', webpage,
|
|
|
|
|
|
|
|
'pkg cohort', default='PHASED:DEFAULT'),
|
|
|
|
|
|
|
|
'__rev': self._search_regex(
|
|
|
|
|
|
|
|
r'client_revision["\']\s*:\s*(\d+),', webpage,
|
|
|
|
|
|
|
|
'client revision', default='3944515'),
|
|
|
|
}),
|
|
|
|
}),
|
|
|
|
headers={
|
|
|
|
headers={
|
|
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
|
|
})
|
|
|
|
})
|
|
|
|
tahoe_js_data = self._parse_json(self._search_regex(
|
|
|
|
tahoe_js_data = self._parse_json(
|
|
|
|
r'for \(;;\);(.+)', tahoe_data,
|
|
|
|
self._search_regex(
|
|
|
|
'tahoe js data', default='{}'), video_id, fatal=False)
|
|
|
|
r'for\s+\(\s*;\s*;\s*\)\s*;(.+)', tahoe_data,
|
|
|
|
video_data = extract_video_data(tahoe_js_data.get('jsmods', {}).get('instances', []))
|
|
|
|
'tahoe js data', default='{}'),
|
|
|
|
|
|
|
|
video_id, fatal=False)
|
|
|
|
|
|
|
|
video_data = extract_from_jsmods_instances(tahoe_js_data)
|
|
|
|
|
|
|
|
|
|
|
|
if not video_data:
|
|
|
|
if not video_data:
|
|
|
|
if not fatal_if_no_video:
|
|
|
|
raise ExtractorError('Cannot parse data')
|
|
|
|
return webpage, False
|
|
|
|
|
|
|
|
m_msg = re.search(r'class="[^"]*uiInterstitialContent[^"]*"><div>(.*?)</div>', webpage)
|
|
|
|
|
|
|
|
if m_msg is not None:
|
|
|
|
|
|
|
|
raise ExtractorError(
|
|
|
|
|
|
|
|
'The video is not available, Facebook said: "%s"' % m_msg.group(1),
|
|
|
|
|
|
|
|
expected=True)
|
|
|
|
|
|
|
|
elif '>You must log in to continue' in webpage:
|
|
|
|
|
|
|
|
self.raise_login_required()
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
raise ExtractorError('Cannot parse data')
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
formats = []
|
|
|
|
formats = []
|
|
|
|
for f in video_data:
|
|
|
|
for f in video_data:
|
|
|
@ -408,11 +417,10 @@ class FacebookIE(InfoExtractor):
|
|
|
|
video_title = limit_length(video_title, 80)
|
|
|
|
video_title = limit_length(video_title, 80)
|
|
|
|
else:
|
|
|
|
else:
|
|
|
|
video_title = 'Facebook video #%s' % video_id
|
|
|
|
video_title = 'Facebook video #%s' % video_id
|
|
|
|
uploader = clean_html(get_element_by_id('fbPhotoPageAuthorName', webpage))
|
|
|
|
uploader = clean_html(get_element_by_id(
|
|
|
|
if not uploader:
|
|
|
|
'fbPhotoPageAuthorName', webpage)) or self._search_regex(
|
|
|
|
uploader = self._search_regex(
|
|
|
|
r'ownerName\s*:\s*"([^"]+)"', webpage, 'uploader',
|
|
|
|
[r'ownerName\s*:\s*"([^"]+)"', r'property="og:title"\s*content="(.*?)"'],
|
|
|
|
fatal=False) or self._og_search_title(webpage, fatal=False)
|
|
|
|
webpage, 'uploader', fatal=False)
|
|
|
|
|
|
|
|
timestamp = int_or_none(self._search_regex(
|
|
|
|
timestamp = int_or_none(self._search_regex(
|
|
|
|
r'<abbr[^>]+data-utime=["\'](\d+)', webpage,
|
|
|
|
r'<abbr[^>]+data-utime=["\'](\d+)', webpage,
|
|
|
|
'timestamp', default=None))
|
|
|
|
'timestamp', default=None))
|
|
|
|