wdr.py 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. # -*- coding: utf-8 -*-
  2. from __future__ import unicode_literals
  3. import re
  4. from .common import InfoExtractor
  5. from ..compat import (
  6. compat_parse_qs,
  7. compat_urlparse,
  8. )
  9. from ..utils import (
  10. unified_strdate,
  11. ExtractorError,
  12. )
  13. class WDRIE(InfoExtractor):
  14. _PAGE_REGEX = r'/mediathek/(?P<media_type>[^/]+)/(?P<type>[^/]+)/(?P<display_id>.+)\.html'
  15. _VALID_URL = r'(?P<page_url>https?://(?:www\d\.)?wdr\d?\.de)' + _PAGE_REGEX
  16. _JS_URL_REGEX = r'(https?://deviceids-medp.wdr.de/ondemand/\d+/\d+\.js)'
  17. _TESTS = [
  18. {
  19. 'url': 'http://www1.wdr.de/mediathek/video/sendungen/doku-am-freitag/video-geheimnis-aachener-dom-100.html',
  20. 'md5': 'e58c39c3e30077141d258bf588700a7b',
  21. 'info_dict': {
  22. 'id': 'mdb-1058683',
  23. 'ext': 'flv',
  24. 'display_id': 'doku-am-freitag/video-geheimnis-aachener-dom-100',
  25. 'title': 'Geheimnis Aachener Dom',
  26. 'alt_title': 'Doku am Freitag',
  27. 'upload_date': '20160304',
  28. 'description': 'md5:87be8ff14d8dfd7a7ee46f0299b52318',
  29. 'is_live': False,
  30. 'subtitles': {'de': [{
  31. 'url': 'http://ondemand-ww.wdr.de/medp/fsk0/105/1058683/1058683_12220974.xml'
  32. }]},
  33. },
  34. 'skip': 'Page Not Found',
  35. },
  36. {
  37. 'url': 'http://www1.wdr.de/mediathek/audio/wdr3/wdr3-gespraech-am-samstag/audio-schriftstellerin-juli-zeh-100.html',
  38. 'md5': 'f4c1f96d01cf285240f53ea4309663d8',
  39. 'info_dict': {
  40. 'id': 'mdb-1072000',
  41. 'ext': 'mp3',
  42. 'display_id': 'wdr3-gespraech-am-samstag/audio-schriftstellerin-juli-zeh-100',
  43. 'title': 'Schriftstellerin Juli Zeh',
  44. 'alt_title': 'WDR 3 Gespräch am Samstag',
  45. 'upload_date': '20160312',
  46. 'description': 'md5:e127d320bc2b1f149be697ce044a3dd7',
  47. 'is_live': False,
  48. 'subtitles': {}
  49. },
  50. 'skip': 'Page Not Found',
  51. },
  52. {
  53. 'url': 'http://www1.wdr.de/mediathek/video/live/index.html',
  54. 'info_dict': {
  55. 'id': 'mdb-103364',
  56. 'ext': 'flv',
  57. 'display_id': 'index',
  58. 'title': r're:^WDR Fernsehen im Livestream [0-9]{4}-[0-9]{2}-[0-9]{2} [0-9]{2}:[0-9]{2}$',
  59. 'alt_title': 'WDR Fernsehen Live',
  60. 'upload_date': None,
  61. 'description': 'md5:ae2ff888510623bf8d4b115f95a9b7c9',
  62. 'is_live': True,
  63. 'subtitles': {}
  64. }
  65. },
  66. {
  67. 'url': 'http://www1.wdr.de/mediathek/video/sendungen/aktuelle-stunde/aktuelle-stunde-120.html',
  68. 'playlist_mincount': 10,
  69. 'info_dict': {
  70. 'id': 'aktuelle-stunde/aktuelle-stunde-120',
  71. },
  72. }
  73. ]
  74. def _real_extract(self, url):
  75. mobj = re.match(self._VALID_URL, url)
  76. url_type = mobj.group('type')
  77. page_url = mobj.group('page_url')
  78. display_id = mobj.group('display_id')
  79. webpage = self._download_webpage(url, display_id)
  80. js_url = self._search_regex(self._JS_URL_REGEX, webpage, 'js_url', default=None)
  81. if not js_url:
  82. entries = [
  83. self.url_result(page_url + href[0], 'WDR')
  84. for href in re.findall(
  85. r'<a href="(%s)"' % self._PAGE_REGEX,
  86. webpage)
  87. ]
  88. if entries: # Playlist page
  89. return self.playlist_result(entries, playlist_id=display_id)
  90. raise ExtractorError('No downloadable streams found', expected=True)
  91. js_data = self._download_webpage(js_url, 'metadata')
  92. json_data = self._search_regex(r'\(({.*})\)', js_data, 'json')
  93. metadata = self._parse_json(json_data, display_id)
  94. metadata_tracker_data = metadata["trackerData"]
  95. metadata_media_resource = metadata["mediaResource"]
  96. formats = []
  97. # check if the metadata contains a direct URL to a file
  98. metadata_media_alt = metadata_media_resource.get("alt")
  99. if metadata_media_alt:
  100. for tag_name in ["videoURL", 'audioURL']:
  101. if tag_name in metadata_media_alt:
  102. formats.append({
  103. 'url': metadata_media_alt[tag_name]
  104. })
  105. # check if there are flash-streams for this video
  106. if "dflt" in metadata_media_resource and "videoURL" in metadata_media_resource["dflt"]:
  107. video_url = metadata_media_resource["dflt"]["videoURL"]
  108. if video_url.endswith('.f4m'):
  109. full_video_url = video_url + '?hdcore=3.2.0&plugin=aasp-3.2.0.77.18'
  110. formats.extend(self._extract_f4m_formats(full_video_url, display_id, f4m_id='hds', fatal=False))
  111. elif video_url.endswith('.smil'):
  112. formats.extend(self._extract_smil_formats(video_url, 'stream', fatal=False))
  113. subtitles = {}
  114. caption_url = metadata_media_resource.get("captionURL")
  115. if caption_url:
  116. subtitles['de'] = [{
  117. 'url': caption_url
  118. }]
  119. title = metadata_tracker_data.get("trackerClipTitle")
  120. is_live = url_type == 'live'
  121. if is_live:
  122. title = self._live_title(title)
  123. upload_date = None
  124. elif 'trackerClipAirTime' in metadata_tracker_data:
  125. upload_date = metadata_tracker_data['trackerClipAirTime']
  126. else:
  127. upload_date = self._html_search_meta('DC.Date', webpage, 'upload date')
  128. if upload_date:
  129. upload_date = unified_strdate(upload_date)
  130. self._sort_formats(formats)
  131. return {
  132. 'id': metadata_tracker_data.get("trackerClipId", display_id),
  133. 'display_id': display_id,
  134. 'title': title,
  135. 'alt_title': metadata_tracker_data.get("trackerClipSubcategory"),
  136. 'formats': formats,
  137. 'upload_date': upload_date,
  138. 'description': self._html_search_meta("Description", webpage),
  139. 'is_live': is_live,
  140. 'subtitles': subtitles,
  141. }
  142. class WDRMobileIE(InfoExtractor):
  143. _VALID_URL = r'''(?x)
  144. https?://mobile-ondemand\.wdr\.de/
  145. .*?/fsk(?P<age_limit>[0-9]+)
  146. /[0-9]+/[0-9]+/
  147. (?P<id>[0-9]+)_(?P<title>[0-9]+)'''
  148. IE_NAME = 'wdr:mobile'
  149. _TEST = {
  150. 'url': 'http://mobile-ondemand.wdr.de/CMS2010/mdb/ondemand/weltweit/fsk0/42/421735/421735_4283021.mp4',
  151. 'info_dict': {
  152. 'title': '4283021',
  153. 'id': '421735',
  154. 'ext': 'mp4',
  155. 'age_limit': 0,
  156. },
  157. 'skip': 'Problems with loading data.'
  158. }
  159. def _real_extract(self, url):
  160. mobj = re.match(self._VALID_URL, url)
  161. return {
  162. 'id': mobj.group('id'),
  163. 'title': mobj.group('title'),
  164. 'age_limit': int(mobj.group('age_limit')),
  165. 'url': url,
  166. 'http_headers': {
  167. 'User-Agent': 'mobile',
  168. },
  169. }
  170. class WDRMausIE(InfoExtractor):
  171. _VALID_URL = r'https?://(?:www\.)?wdrmaus\.de/(?:[^/]+/){,2}(?P<id>[^/?#]+)(?:/index\.php5|(?<!index)\.php5|/(?:$|[?#]))'
  172. IE_DESC = 'Sendung mit der Maus'
  173. _TESTS = [{
  174. 'url': 'http://www.wdrmaus.de/aktuelle-sendung/index.php5',
  175. 'info_dict': {
  176. 'id': 'aktuelle-sendung',
  177. 'ext': 'mp4',
  178. 'thumbnail': 're:^http://.+\.jpg',
  179. 'upload_date': 're:^[0-9]{8}$',
  180. 'title': 're:^[0-9.]{10} - Aktuelle Sendung$',
  181. }
  182. }, {
  183. 'url': 'http://www.wdrmaus.de/sachgeschichten/sachgeschichten/40_jahre_maus.php5',
  184. 'md5': '3b1227ca3ed28d73ec5737c65743b2a3',
  185. 'info_dict': {
  186. 'id': '40_jahre_maus',
  187. 'ext': 'mp4',
  188. 'thumbnail': 're:^http://.+\.jpg',
  189. 'upload_date': '20131007',
  190. 'title': '12.03.2011 - 40 Jahre Maus',
  191. }
  192. }]
  193. def _real_extract(self, url):
  194. video_id = self._match_id(url)
  195. webpage = self._download_webpage(url, video_id)
  196. param_code = self._html_search_regex(
  197. r'<a href="\?startVideo=1&amp;([^"]+)"', webpage, 'parameters')
  198. title_date = self._search_regex(
  199. r'<div class="sendedatum"><p>Sendedatum:\s*([0-9\.]+)</p>',
  200. webpage, 'air date')
  201. title_str = self._html_search_regex(
  202. r'<h1>(.*?)</h1>', webpage, 'title')
  203. title = '%s - %s' % (title_date, title_str)
  204. upload_date = unified_strdate(
  205. self._html_search_meta('dc.date', webpage))
  206. fields = compat_parse_qs(param_code)
  207. video_url = fields['firstVideo'][0]
  208. thumbnail = compat_urlparse.urljoin(url, fields['startPicture'][0])
  209. formats = [{
  210. 'format_id': 'rtmp',
  211. 'url': video_url,
  212. }]
  213. jscode = self._download_webpage(
  214. 'http://www.wdrmaus.de/codebase/js/extended-medien.min.js',
  215. video_id, fatal=False,
  216. note='Downloading URL translation table',
  217. errnote='Could not download URL translation table')
  218. if jscode:
  219. for m in re.finditer(
  220. r"stream:\s*'dslSrc=(?P<stream>[^']+)',\s*download:\s*'(?P<dl>[^']+)'\s*\}",
  221. jscode):
  222. if video_url.startswith(m.group('stream')):
  223. http_url = video_url.replace(
  224. m.group('stream'), m.group('dl'))
  225. formats.append({
  226. 'format_id': 'http',
  227. 'url': http_url,
  228. })
  229. break
  230. self._sort_formats(formats)
  231. return {
  232. 'id': video_id,
  233. 'title': title,
  234. 'formats': formats,
  235. 'thumbnail': thumbnail,
  236. 'upload_date': upload_date,
  237. }