|
@@ -11,7 +11,100 @@ from ..utils import (
|
|
|
|
|
|
|
|
|
|
class WashingtonPostIE(InfoExtractor):
|
|
class WashingtonPostIE(InfoExtractor):
|
|
- _VALID_URL = r'https?://(?:www\.)?washingtonpost\.com/.*?/(?P<id>[^/]+)/(?:$|[?#])'
|
|
|
|
|
|
+ IE_NAME = 'washingtonpost'
|
|
|
|
+ _VALID_URL = r'(?:washingtonpost:|https?://(?:www\.)?washingtonpost\.com/video/(?:[^/]+/)*)(?P<id>[\da-f]{8}-[\da-f]{4}-[\da-f]{4}-[\da-f]{4}-[\da-f]{12})'
|
|
|
|
+ _TEST = {
|
|
|
|
+ 'url': 'https://www.washingtonpost.com/video/c/video/480ba4ee-1ec7-11e6-82c2-a7dcb313287d',
|
|
|
|
+ 'md5': '6f537e1334b714eb15f9563bd4b9cdfa',
|
|
|
|
+ 'info_dict': {
|
|
|
|
+ 'id': '480ba4ee-1ec7-11e6-82c2-a7dcb313287d',
|
|
|
|
+ 'ext': 'mp4',
|
|
|
|
+ 'title': 'Egypt finds belongings, debris from plane crash',
|
|
|
|
+ 'description': 'md5:a17ceee432f215a5371388c1f680bd86',
|
|
|
|
+ 'upload_date': '20160520',
|
|
|
|
+ 'uploader': 'Reuters',
|
|
|
|
+ 'timestamp': 1463778452,
|
|
|
|
+ },
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ def _real_extract(self, url):
|
|
|
|
+ video_id = self._match_id(url)
|
|
|
|
+ video_data = self._download_json(
|
|
|
|
+ 'http://www.washingtonpost.com/posttv/c/videojson/%s?resType=jsonp' % video_id,
|
|
|
|
+ video_id, transform_source=strip_jsonp)[0]['contentConfig']
|
|
|
|
+ title = video_data['title']
|
|
|
|
+
|
|
|
|
+ urls = []
|
|
|
|
+ formats = []
|
|
|
|
+ for s in video_data.get('streams', []):
|
|
|
|
+ s_url = s.get('url')
|
|
|
|
+ if not s_url or s_url in urls:
|
|
|
|
+ continue
|
|
|
|
+ urls.append(s_url)
|
|
|
|
+ video_type = s.get('type')
|
|
|
|
+ if video_type == 'smil':
|
|
|
|
+ continue
|
|
|
|
+ elif video_type in ('ts', 'hls'):
|
|
|
|
+ m3u8_formats = self._extract_m3u8_formats(
|
|
|
|
+ s_url, video_id, 'mp4', 'm3u8_native', m3u8_id='hls', fatal=False)
|
|
|
|
+ for m3u8_format in m3u8_formats:
|
|
|
|
+ width = m3u8_format.get('width')
|
|
|
|
+ if not width:
|
|
|
|
+ continue
|
|
|
|
+ vbr = self._search_regex(
|
|
|
|
+ r'%d_%d_(\d+)' % (width, m3u8_format['height']), m3u8_format['url'], 'vbr', default=None)
|
|
|
|
+ if vbr:
|
|
|
|
+ m3u8_format.update({
|
|
|
|
+ 'vbr': int_or_none(vbr),
|
|
|
|
+ })
|
|
|
|
+ formats.extend(m3u8_formats)
|
|
|
|
+ else:
|
|
|
|
+ width = int_or_none(s.get('width'))
|
|
|
|
+ vbr = int_or_none(s.get('bitrate'))
|
|
|
|
+ has_width = width != 0
|
|
|
|
+ formats.append({
|
|
|
|
+ 'format_id': (
|
|
|
|
+ '%s-%d-%d' % (video_type, width, vbr)
|
|
|
|
+ if width
|
|
|
|
+ else video_type),
|
|
|
|
+ 'vbr': vbr if has_width else None,
|
|
|
|
+ 'width': width,
|
|
|
|
+ 'height': int_or_none(s.get('height')),
|
|
|
|
+ 'acodec': s.get('audioCodec'),
|
|
|
|
+ 'vcodec': s.get('videoCodec') if has_width else 'none',
|
|
|
|
+ 'filesize': int_or_none(s.get('fileSize')),
|
|
|
|
+ 'url': s_url,
|
|
|
|
+ 'ext': 'mp4',
|
|
|
|
+ 'protocol': {
|
|
|
|
+ 'mp4': 'http',
|
|
|
|
+ 'ts': 'm3u8_native',
|
|
|
|
+ 'hls': 'm3u8_native',
|
|
|
|
+ }.get(s.get('type')),
|
|
|
|
+ })
|
|
|
|
+ source_media_url = video_data.get('sourceMediaURL')
|
|
|
|
+ if source_media_url:
|
|
|
|
+ formats.append({
|
|
|
|
+ 'format_id': 'source_media',
|
|
|
|
+ 'url': source_media_url,
|
|
|
|
+ })
|
|
|
|
+ self._sort_formats(
|
|
|
|
+ formats, ('width', 'height', 'vbr', 'filesize', 'tbr', 'format_id'))
|
|
|
|
+
|
|
|
|
+ return {
|
|
|
|
+ 'id': video_id,
|
|
|
|
+ 'title': title,
|
|
|
|
+ 'description': video_data.get('blurb'),
|
|
|
|
+ 'uploader': video_data.get('credits', {}).get('source'),
|
|
|
|
+ 'formats': formats,
|
|
|
|
+ 'duration': int_or_none(video_data.get('videoDuration'), 100),
|
|
|
|
+ 'timestamp': int_or_none(
|
|
|
|
+ video_data.get('dateConfig', {}).get('dateFirstPublished'), 1000),
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+class WashingtonPostArticleIE(InfoExtractor):
|
|
|
|
+ IE_NAME = 'washingtonpost:article'
|
|
|
|
+ _VALID_URL = r'https?://(?:www\.)?washingtonpost\.com/(?:[^/]+/)*(?P<id>[^/?#]+)'
|
|
_TESTS = [{
|
|
_TESTS = [{
|
|
'url': 'http://www.washingtonpost.com/sf/national/2014/03/22/sinkhole-of-bureaucracy/',
|
|
'url': 'http://www.washingtonpost.com/sf/national/2014/03/22/sinkhole-of-bureaucracy/',
|
|
'info_dict': {
|
|
'info_dict': {
|
|
@@ -63,6 +156,10 @@ class WashingtonPostIE(InfoExtractor):
|
|
}]
|
|
}]
|
|
}]
|
|
}]
|
|
|
|
|
|
|
|
+ @classmethod
|
|
|
|
+ def suitable(cls, url):
|
|
|
|
+ return False if WashingtonPostIE.suitable(url) else super(WashingtonPostArticleIE, cls).suitable(url)
|
|
|
|
+
|
|
def _real_extract(self, url):
|
|
def _real_extract(self, url):
|
|
page_id = self._match_id(url)
|
|
page_id = self._match_id(url)
|
|
webpage = self._download_webpage(url, page_id)
|
|
webpage = self._download_webpage(url, page_id)
|
|
@@ -74,54 +171,7 @@ class WashingtonPostIE(InfoExtractor):
|
|
<div\s+class="posttv-video-embed[^>]*?data-uuid=|
|
|
<div\s+class="posttv-video-embed[^>]*?data-uuid=|
|
|
data-video-uuid=
|
|
data-video-uuid=
|
|
)"([^"]+)"''', webpage)
|
|
)"([^"]+)"''', webpage)
|
|
- entries = []
|
|
|
|
- for i, uuid in enumerate(uuids, start=1):
|
|
|
|
- vinfo_all = self._download_json(
|
|
|
|
- 'http://www.washingtonpost.com/posttv/c/videojson/%s?resType=jsonp' % uuid,
|
|
|
|
- page_id,
|
|
|
|
- transform_source=strip_jsonp,
|
|
|
|
- note='Downloading information of video %d/%d' % (i, len(uuids))
|
|
|
|
- )
|
|
|
|
- vinfo = vinfo_all[0]['contentConfig']
|
|
|
|
- uploader = vinfo.get('credits', {}).get('source')
|
|
|
|
- timestamp = int_or_none(
|
|
|
|
- vinfo.get('dateConfig', {}).get('dateFirstPublished'), 1000)
|
|
|
|
-
|
|
|
|
- formats = [{
|
|
|
|
- 'format_id': (
|
|
|
|
- '%s-%s-%s' % (s.get('type'), s.get('width'), s.get('bitrate'))
|
|
|
|
- if s.get('width')
|
|
|
|
- else s.get('type')),
|
|
|
|
- 'vbr': s.get('bitrate') if s.get('width') != 0 else None,
|
|
|
|
- 'width': s.get('width'),
|
|
|
|
- 'height': s.get('height'),
|
|
|
|
- 'acodec': s.get('audioCodec'),
|
|
|
|
- 'vcodec': s.get('videoCodec') if s.get('width') != 0 else 'none',
|
|
|
|
- 'filesize': s.get('fileSize'),
|
|
|
|
- 'url': s.get('url'),
|
|
|
|
- 'ext': 'mp4',
|
|
|
|
- 'preference': -100 if s.get('type') == 'smil' else None,
|
|
|
|
- 'protocol': {
|
|
|
|
- 'MP4': 'http',
|
|
|
|
- 'F4F': 'f4m',
|
|
|
|
- }.get(s.get('type')),
|
|
|
|
- } for s in vinfo.get('streams', [])]
|
|
|
|
- source_media_url = vinfo.get('sourceMediaURL')
|
|
|
|
- if source_media_url:
|
|
|
|
- formats.append({
|
|
|
|
- 'format_id': 'source_media',
|
|
|
|
- 'url': source_media_url,
|
|
|
|
- })
|
|
|
|
- self._sort_formats(formats)
|
|
|
|
- entries.append({
|
|
|
|
- 'id': uuid,
|
|
|
|
- 'title': vinfo['title'],
|
|
|
|
- 'description': vinfo.get('blurb'),
|
|
|
|
- 'uploader': uploader,
|
|
|
|
- 'formats': formats,
|
|
|
|
- 'duration': int_or_none(vinfo.get('videoDuration'), 100),
|
|
|
|
- 'timestamp': timestamp,
|
|
|
|
- })
|
|
|
|
|
|
+ entries = [self.url_result('washingtonpost:%s' % uuid, 'WashingtonPost', uuid) for uuid in uuids]
|
|
|
|
|
|
return {
|
|
return {
|
|
'_type': 'playlist',
|
|
'_type': 'playlist',
|