diff options
Diffstat (limited to 'youtube_dl/extractor/bbc.py')
| -rw-r--r-- | youtube_dl/extractor/bbc.py | 45 | 
1 files changed, 26 insertions, 19 deletions
| diff --git a/youtube_dl/extractor/bbc.py b/youtube_dl/extractor/bbc.py index 2cdce1eb9..923273fb2 100644 --- a/youtube_dl/extractor/bbc.py +++ b/youtube_dl/extractor/bbc.py @@ -2,7 +2,6 @@  from __future__ import unicode_literals  import re -import xml.etree.ElementTree  from .common import InfoExtractor  from ..utils import ( @@ -14,18 +13,22 @@ from ..utils import (      remove_end,      unescapeHTML,  ) -from ..compat import compat_HTTPError +from ..compat import ( +    compat_etree_fromstring, +    compat_HTTPError, +)  class BBCCoUkIE(InfoExtractor):      IE_NAME = 'bbc.co.uk'      IE_DESC = 'BBC iPlayer' -    _VALID_URL = r'https?://(?:www\.)?bbc\.co\.uk/(?:(?:programmes/(?!articles/)|iplayer(?:/[^/]+)?/(?:episode/|playlist/))|music/clips[/#])(?P<id>[\da-z]{8})' +    _ID_REGEX = r'[pb][\da-z]{7}' +    _VALID_URL = r'https?://(?:www\.)?bbc\.co\.uk/(?:(?:programmes/(?!articles/)|iplayer(?:/[^/]+)?/(?:episode/|playlist/))|music/clips[/#])(?P<id>%s)' % _ID_REGEX      _MEDIASELECTOR_URLS = [          # Provides HQ HLS streams with even better quality that pc mediaset but fails          # with geolocation in some cases when it's even not geo restricted at all (e.g. -        # http://www.bbc.co.uk/programmes/b06bp7lf) +        # http://www.bbc.co.uk/programmes/b06bp7lf). Also may fail with selectionunavailable.          'http://open.live.bbc.co.uk/mediaselector/5/select/version/2.0/mediaset/iptv-all/vpid/%s',          'http://open.live.bbc.co.uk/mediaselector/5/select/version/2.0/mediaset/pc/vpid/%s',      ] @@ -44,9 +47,8 @@ class BBCCoUkIE(InfoExtractor):              'info_dict': {                  'id': 'b039d07m',                  'ext': 'flv', -                'title': 'Kaleidoscope, Leonard Cohen', +                'title': 'Leonard Cohen, Kaleidoscope - BBC Radio 4',                  'description': 'The Canadian poet and songwriter reflects on his musical career.', -                'duration': 1740,              },              'params': {                  # rtmp download @@ -109,7 +111,8 @@ class BBCCoUkIE(InfoExtractor):              'params': {                  # rtmp download                  'skip_download': True, -            } +            }, +            'skip': 'Episode is no longer available on BBC iPlayer Radio',          }, {              'url': 'http://www.bbc.co.uk/music/clips/p02frcc3',              'note': 'Audio', @@ -220,11 +223,9 @@ class BBCCoUkIE(InfoExtractor):              elif transfer_format == 'dash':                  pass              elif transfer_format == 'hls': -                m3u8_formats = self._extract_m3u8_formats( +                formats.extend(self._extract_m3u8_formats(                      href, programme_id, ext='mp4', entry_protocol='m3u8_native', -                    m3u8_id=supplier, fatal=False) -                if m3u8_formats: -                    formats.extend(m3u8_formats) +                    m3u8_id=supplier, fatal=False))              # Direct link              else:                  formats.append({ @@ -332,7 +333,7 @@ class BBCCoUkIE(InfoExtractor):                  return self._download_media_selector_url(                      mediaselector_url % programme_id, programme_id)              except BBCCoUkIE.MediaSelectionError as e: -                if e.id in ('notukerror', 'geolocation'): +                if e.id in ('notukerror', 'geolocation', 'selectionunavailable'):                      last_exception = e                      continue                  self._raise_extractor_error(e) @@ -343,8 +344,8 @@ class BBCCoUkIE(InfoExtractor):              media_selection = self._download_xml(                  url, programme_id, 'Downloading media selection XML')          except ExtractorError as ee: -            if isinstance(ee.cause, compat_HTTPError) and ee.cause.code == 403: -                media_selection = xml.etree.ElementTree.fromstring(ee.cause.read().decode('utf-8')) +            if isinstance(ee.cause, compat_HTTPError) and ee.cause.code in (403, 404): +                media_selection = compat_etree_fromstring(ee.cause.read().decode('utf-8'))              else:                  raise          return self._process_media_selector(media_selection, programme_id) @@ -451,6 +452,7 @@ class BBCCoUkIE(InfoExtractor):          webpage = self._download_webpage(url, group_id, 'Downloading video page')          programme_id = None +        duration = None          tviplayer = self._search_regex(              r'mediator\.bind\(({.+?})\s*,\s*document\.getElementById', @@ -463,14 +465,16 @@ class BBCCoUkIE(InfoExtractor):          if not programme_id:              programme_id = self._search_regex( -                r'"vpid"\s*:\s*"([\da-z]{8})"', webpage, 'vpid', fatal=False, default=None) +                r'"vpid"\s*:\s*"(%s)"' % self._ID_REGEX, webpage, 'vpid', fatal=False, default=None)          if programme_id:              formats, subtitles = self._download_media_selector(programme_id)              title = self._og_search_title(webpage)              description = self._search_regex(                  r'<p class="[^"]*medium-description[^"]*">([^<]+)</p>', -                webpage, 'description', fatal=False) +                webpage, 'description', default=None) +            if not description: +                description = self._html_search_meta('description', webpage)          else:              programme_id, title, description, duration, formats, subtitles = self._download_playlist(group_id) @@ -584,6 +588,7 @@ class BBCIE(BBCCoUkIE):              'ext': 'mp4',              'title': '''Judge Mindy Glazer: "I'm sorry to see you here... I always wondered what happened to you"''',              'duration': 56, +            'description': '''Judge Mindy Glazer: "I'm sorry to see you here... I always wondered what happened to you"''',          },          'params': {              'skip_download': True, @@ -726,6 +731,7 @@ class BBCIE(BBCCoUkIE):          # article with multiple videos embedded with playlist.sxml (e.g.          # http://www.bbc.com/sport/0/football/34475836)          playlists = re.findall(r'<param[^>]+name="playlist"[^>]+value="([^"]+)"', webpage) +        playlists.extend(re.findall(r'data-media-id="([^"]+/playlist\.sxml)"', webpage))          if playlists:              entries = [                  self._extract_from_playlist_sxml(playlist_url, playlist_id, timestamp) @@ -778,8 +784,9 @@ class BBCIE(BBCCoUkIE):          # single video story (e.g. http://www.bbc.com/travel/story/20150625-sri-lankas-spicy-secret)          programme_id = self._search_regex( -            [r'data-video-player-vpid="([\da-z]{8})"', -             r'<param[^>]+name="externalIdentifier"[^>]+value="([\da-z]{8})"'], +            [r'data-video-player-vpid="(%s)"' % self._ID_REGEX, +             r'<param[^>]+name="externalIdentifier"[^>]+value="(%s)"' % self._ID_REGEX, +             r'videoId\s*:\s*["\'](%s)["\']' % self._ID_REGEX],              webpage, 'vpid', default=None)          if programme_id: @@ -814,7 +821,7 @@ class BBCIE(BBCCoUkIE):          # Multiple video article (e.g.          # http://www.bbc.co.uk/blogs/adamcurtis/entries/3662a707-0af9-3149-963f-47bea720b460) -        EMBED_URL = r'https?://(?:www\.)?bbc\.co\.uk/(?:[^/]+/)+[\da-z]{8}(?:\b[^"]+)?' +        EMBED_URL = r'https?://(?:www\.)?bbc\.co\.uk/(?:[^/]+/)+%s(?:\b[^"]+)?' % self._ID_REGEX          entries = []          for match in extract_all(r'new\s+SMP\(({.+?})\)'):              embed_url = match.get('playerSettings', {}).get('externalEmbedUrl') | 
