| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132 | # encoding: utf-8import reimport jsonimport xml.etree.ElementTreefrom .common import InfoExtractorfrom ..utils import (    compat_urllib_parse,    find_xpath_attr,    compat_urlparse,    ExtractorError,)class BrightcoveIE(InfoExtractor):    _VALID_URL = r'https?://.*brightcove\.com/(services|viewer).*\?(?P<query>.*)'    _FEDERATED_URL_TEMPLATE = 'http://c.brightcove.com/services/viewer/htmlFederated?%s'    _PLAYLIST_URL_TEMPLATE = 'http://c.brightcove.com/services/json/experience/runtime/?command=get_programming_for_experience&playerKey=%s'    _TESTS = [        {            # From http://www.8tv.cat/8aldia/videos/xavier-sala-i-martin-aquesta-tarda-a-8-al-dia/            u'url': u'http://c.brightcove.com/services/viewer/htmlFederated?playerID=1654948606001&flashID=myExperience&%40videoPlayer=2371591881001',            u'file': u'2371591881001.mp4',            u'md5': u'8eccab865181d29ec2958f32a6a754f5',            u'note': u'Test Brightcove downloads and detection in GenericIE',            u'info_dict': {                u'title': u'Xavier Sala i Martín: “Un banc que no presta és un banc zombi que no serveix per a res”',                u'uploader': u'8TV',                u'description': u'md5:a950cc4285c43e44d763d036710cd9cd',            }        },        {            # From http://medianetwork.oracle.com/video/player/1785452137001            u'url': u'http://c.brightcove.com/services/viewer/htmlFederated?playerID=1217746023001&flashID=myPlayer&%40videoPlayer=1785452137001',            u'file': u'1785452137001.flv',            u'info_dict': {                u'title': u'JVMLS 2012: Arrays 2.0 - Opportunities and Challenges',                u'description': u'John Rose speaks at the JVM Language Summit, August 1, 2012.',                u'uploader': u'Oracle',            },        },    ]    @classmethod    def _build_brighcove_url(cls, object_str):        """        Build a Brightcove url from a xml string containing        <object class="BrightcoveExperience">{params}</object>        """        # Fix up some stupid HTML, see https://github.com/rg3/youtube-dl/issues/1553        object_str = re.sub(r'(<param name="[^"]+" value="[^"]+")>',                            lambda m: m.group(1) + '/>', object_str)        # Fix up some stupid XML, see https://github.com/rg3/youtube-dl/issues/1608        object_str = object_str.replace(u'<--', u'<!--')        object_doc = xml.etree.ElementTree.fromstring(object_str)        assert u'BrightcoveExperience' in object_doc.attrib['class']        params = {'flashID': object_doc.attrib['id'],                  'playerID': find_xpath_attr(object_doc, './param', 'name', 'playerID').attrib['value'],                  }        playerKey = find_xpath_attr(object_doc, './param', 'name', 'playerKey')        # Not all pages define this value        if playerKey is not None:            params['playerKey'] = playerKey.attrib['value']        videoPlayer = find_xpath_attr(object_doc, './param', 'name', '@videoPlayer')        if videoPlayer is not None:            params['@videoPlayer'] = videoPlayer.attrib['value']        data = compat_urllib_parse.urlencode(params)        return cls._FEDERATED_URL_TEMPLATE % data    def _real_extract(self, url):        mobj = re.match(self._VALID_URL, url)        query_str = mobj.group('query')        query = compat_urlparse.parse_qs(query_str)        videoPlayer = query.get('@videoPlayer')        if videoPlayer:            return self._get_video_info(videoPlayer[0], query_str)        else:            player_key = query['playerKey']            return self._get_playlist_info(player_key[0])    def _get_video_info(self, video_id, query):        request_url = self._FEDERATED_URL_TEMPLATE % query        webpage = self._download_webpage(request_url, video_id)        self.report_extraction(video_id)        info = self._search_regex(r'var experienceJSON = ({.*?});', webpage, 'json')        info = json.loads(info)['data']        video_info = info['programmedContent']['videoPlayer']['mediaDTO']        return self._extract_video_info(video_info)    def _get_playlist_info(self, player_key):        playlist_info = self._download_webpage(self._PLAYLIST_URL_TEMPLATE % player_key,                                               player_key, u'Downloading playlist information')        json_data = json.loads(playlist_info)        if 'videoList' not in json_data:            raise ExtractorError(u'Empty playlist')        playlist_info = json_data['videoList']        videos = [self._extract_video_info(video_info) for video_info in playlist_info['mediaCollectionDTO']['videoDTOs']]        return self.playlist_result(videos, playlist_id=playlist_info['id'],                                    playlist_title=playlist_info['mediaCollectionDTO']['displayName'])    def _extract_video_info(self, video_info):        info = {            'id': video_info['id'],            'title': video_info['displayName'],            'description': video_info.get('shortDescription'),            'thumbnail': video_info.get('videoStillURL') or video_info.get('thumbnailURL'),            'uploader': video_info.get('publisherName'),        }        renditions = video_info.get('renditions')        if renditions:            renditions = sorted(renditions, key=lambda r: r['size'])            best_format = renditions[-1]            info.update({                'url': best_format['defaultURL'],            })        elif video_info.get('FLVFullLengthURL') is not None:            info.update({                'url': video_info['FLVFullLengthURL'],            })        else:            raise ExtractorError(u'Unable to extract video url for %s' % info['id'])        return info
 |