2014-01-07 18:04:48 +09:00
from __future__ import unicode_literals
2013-08-27 18:56:48 +09:00
import re
from . common import InfoExtractor
2021-06-10 23:38:48 +09:00
from . . compat import (
compat_parse_qs ,
compat_str ,
compat_urllib_parse_urlparse ,
)
2016-08-29 00:43:15 +09:00
from . turner import TurnerBaseIE
2021-06-10 23:38:48 +09:00
from . . utils import (
clean_podcast_url ,
int_or_none ,
parse_iso8601 ,
try_get ,
url_basename ,
)
2013-08-27 18:56:48 +09:00
2013-08-28 07:14:19 +09:00
2016-08-29 00:43:15 +09:00
class CNNIE ( TurnerBaseIE ) :
2016-08-21 03:00:25 +09:00
_VALID_URL = r ''' (?x)https?://(?:(?P<sub_domain>edition|www|money) \ .)?cnn \ .com/(?:video/(?:data/.+?| \ ?)/)?videos?/
2015-05-24 03:04:02 +09:00
( ? P < path > . + ? / ( ? P < title > [ ^ / ] + ? ) ( ? : \. ( ? : [ a - z \- ] + ) | ( ? = & ) ) ) '''
2013-08-27 18:56:48 +09:00
2013-08-28 07:14:19 +09:00
_TESTS = [ {
2014-01-07 18:04:48 +09:00
' url ' : ' http://edition.cnn.com/video/?/video/sports/2013/06/09/nadal-1-on-1.cnn ' ,
' md5 ' : ' 3e6121ea48df7e2259fe73a0628605c4 ' ,
' info_dict ' : {
2016-08-30 18:46:48 +09:00
' id ' : ' sports/2013/06/09/nadal-1-on-1.cnn ' ,
2014-11-05 06:25:08 +09:00
' ext ' : ' mp4 ' ,
2014-01-07 18:04:48 +09:00
' title ' : ' Nadal wins 8th French Open title ' ,
' description ' : ' World Sport \' s Amanda Davies chats with 2013 French Open champion Rafael Nadal. ' ,
' duration ' : 135 ,
' upload_date ' : ' 20130609 ' ,
2013-08-27 18:56:48 +09:00
} ,
2016-08-29 00:43:15 +09:00
' expected_warnings ' : [ ' Failed to download m3u8 information ' ] ,
2014-11-24 05:39:15 +09:00
} , {
2016-02-14 18:37:17 +09:00
' url ' : ' http://edition.cnn.com/video/?/video/us/2013/08/21/sot-student-gives-epic-speech.georgia-institute-of-technology&utm_source=feedburner&utm_medium=feed&utm_campaign=Feed % 3A+rss %2F cnn_topstories+ % 28RSS % 3A+Top+Stories % 29 ' ,
' md5 ' : ' b5cc60c60a3477d185af8f19a2a26f4e ' ,
' info_dict ' : {
2016-08-30 18:46:48 +09:00
' id ' : ' us/2013/08/21/sot-student-gives-epic-speech.georgia-institute-of-technology ' ,
2014-11-05 06:25:08 +09:00
' ext ' : ' mp4 ' ,
2016-02-14 18:37:17 +09:00
' title ' : " Student ' s epic speech stuns new freshmen " ,
' description ' : " A Georgia Tech student welcomes the incoming freshmen with an epic speech backed by music from \" 2001: A Space Odyssey. \" " ,
' upload_date ' : ' 20130821 ' ,
2016-08-29 00:43:15 +09:00
} ,
' expected_warnings ' : [ ' Failed to download m3u8 information ' ] ,
2014-12-30 04:50:28 +09:00
} , {
' url ' : ' http://www.cnn.com/video/data/2.0/video/living/2014/12/22/growing-america-nashville-salemtown-board-episode-1.hln.html ' ,
' md5 ' : ' f14d02ebd264df951feb2400e2c25a1b ' ,
' info_dict ' : {
2016-08-30 18:46:48 +09:00
' id ' : ' living/2014/12/22/growing-america-nashville-salemtown-board-episode-1.hln ' ,
2014-12-30 04:50:28 +09:00
' ext ' : ' mp4 ' ,
' title ' : ' Nashville Ep. 1: Hand crafted skateboards ' ,
' description ' : ' md5:e7223a503315c9f150acac52e76de086 ' ,
' upload_date ' : ' 20141222 ' ,
2016-08-29 00:43:15 +09:00
} ,
' expected_warnings ' : [ ' Failed to download m3u8 information ' ] ,
2016-08-21 03:00:25 +09:00
} , {
' url ' : ' http://money.cnn.com/video/news/2016/08/19/netflix-stunning-stats.cnnmoney/index.html ' ,
' md5 ' : ' 52a515dc1b0f001cd82e4ceda32be9d1 ' ,
' info_dict ' : {
2016-08-30 18:46:48 +09:00
' id ' : ' /video/news/2016/08/19/netflix-stunning-stats.cnnmoney ' ,
2016-08-21 03:00:25 +09:00
' ext ' : ' mp4 ' ,
' title ' : ' 5 stunning stats about Netflix ' ,
' description ' : ' Did you know that Netflix has more than 80 million members? Here are five facts about the online video distributor that you probably didn \' t know. ' ,
' upload_date ' : ' 20160819 ' ,
2016-08-29 00:43:15 +09:00
} ,
' params ' : {
# m3u8 download
' skip_download ' : True ,
} ,
2015-03-29 07:39:41 +09:00
} , {
' url ' : ' http://cnn.com/video/?/video/politics/2015/03/27/pkg-arizona-senator-church-attendance-mandatory.ktvk ' ,
' only_matching ' : True ,
2015-04-07 21:59:13 +09:00
} , {
' url ' : ' http://cnn.com/video/?/video/us/2015/04/06/dnt-baker-refuses-anti-gay-order.wkmg ' ,
' only_matching ' : True ,
2016-08-21 03:00:25 +09:00
} , {
' url ' : ' http://edition.cnn.com/videos/arts/2016/04/21/olympic-games-cultural-a-z-brazil.cnn ' ,
' only_matching ' : True ,
2013-08-28 07:14:19 +09:00
} ]
2013-08-27 18:56:48 +09:00
2016-08-21 03:00:25 +09:00
_CONFIG = {
# http://edition.cnn.com/.element/apps/cvp/3.0/cfg/spider/cnn/expansion/config.xml
' edition ' : {
' data_src ' : ' http://edition.cnn.com/video/data/3.0/video/ %s /index.xml ' ,
' media_src ' : ' http://pmd.cdn.turner.com/cnn/big ' ,
} ,
# http://money.cnn.com/.element/apps/cvp2/cfg/config.xml
' money ' : {
' data_src ' : ' http://money.cnn.com/video/data/4.0/video/ %s .xml ' ,
' media_src ' : ' http://ht3.cdn.turner.com/money/big ' ,
} ,
}
2016-08-30 03:26:53 +09:00
def _extract_timestamp ( self , video_data ) :
# TODO: fix timestamp extraction
return None
2013-08-27 18:56:48 +09:00
def _real_extract ( self , url ) :
2016-08-21 03:00:25 +09:00
sub_domain , path , page_title = re . match ( self . _VALID_URL , url ) . groups ( )
if sub_domain not in ( ' money ' , ' edition ' ) :
sub_domain = ' edition '
config = self . _CONFIG [ sub_domain ]
2016-08-29 00:43:15 +09:00
return self . _extract_cvp_info (
config [ ' data_src ' ] % path , page_title , {
' default ' : {
' media_src ' : config [ ' media_src ' ] ,
2020-12-20 04:12:01 +09:00
} ,
' f4m ' : {
' host ' : ' cnn-vh.akamaihd.net ' ,
} ,
2016-08-29 00:43:15 +09:00
} )
2014-02-11 22:38:17 +09:00
class CNNBlogsIE ( InfoExtractor ) :
_VALID_URL = r ' https?://[^ \ .]+ \ .blogs \ .cnn \ .com/.+ '
_TEST = {
' url ' : ' http://reliablesources.blogs.cnn.com/2014/02/09/criminalizing-journalism/ ' ,
' md5 ' : ' 3e56f97b0b6ffb4b79f4ea0749551084 ' ,
' info_dict ' : {
2016-08-30 18:46:48 +09:00
' id ' : ' bestoftv/2014/02/09/criminalizing-journalism.cnn ' ,
2014-02-11 22:38:17 +09:00
' ext ' : ' mp4 ' ,
' title ' : ' Criminalizing journalism? ' ,
' description ' : ' Glenn Greenwald responds to comments made this week on Capitol Hill that journalists could be criminal accessories. ' ,
' upload_date ' : ' 20140209 ' ,
} ,
2016-08-30 02:23:21 +09:00
' expected_warnings ' : [ ' Failed to download m3u8 information ' ] ,
2014-02-11 22:38:17 +09:00
' add_ie ' : [ ' CNN ' ] ,
}
def _real_extract ( self , url ) :
webpage = self . _download_webpage ( url , url_basename ( url ) )
cnn_url = self . _html_search_regex ( r ' data-url= " (.+?) " ' , webpage , ' cnn url ' )
2019-01-02 01:56:05 +09:00
return self . url_result ( cnn_url , CNNIE . ie_key ( ) )
2014-12-23 02:40:36 +09:00
class CNNArticleIE ( InfoExtractor ) :
2021-06-10 23:38:48 +09:00
_VALID_URL = r ' https?://(?:(?:edition|www) \ .)?cnn \ .com/(?!(?:videos?|audio/podcasts)/) '
2014-12-23 02:40:36 +09:00
_TEST = {
' url ' : ' http://www.cnn.com/2014/12/21/politics/obama-north-koreas-hack-not-war-but-cyber-vandalism/ ' ,
2015-01-21 18:27:18 +09:00
' md5 ' : ' 689034c2a3d9c6dc4aa72d65a81efd01 ' ,
2014-12-23 02:40:36 +09:00
' info_dict ' : {
2016-08-30 18:46:48 +09:00
' id ' : ' bestoftv/2014/12/21/ip-north-korea-obama.cnn ' ,
2014-12-23 02:40:36 +09:00
' ext ' : ' mp4 ' ,
2015-01-21 18:27:18 +09:00
' title ' : ' Obama: Cyberattack not an act of war ' ,
2016-08-30 02:23:21 +09:00
' description ' : ' md5:0a802a40d2376f60e6b04c8d5bcebc4b ' ,
2015-01-21 18:27:18 +09:00
' upload_date ' : ' 20141221 ' ,
2014-12-23 02:40:36 +09:00
} ,
2016-08-30 02:23:21 +09:00
' expected_warnings ' : [ ' Failed to download m3u8 information ' ] ,
2014-12-23 02:40:36 +09:00
' add_ie ' : [ ' CNN ' ] ,
}
def _real_extract ( self , url ) :
webpage = self . _download_webpage ( url , url_basename ( url ) )
cnn_url = self . _html_search_regex ( r " video: \ s* ' ([^ ' ]+) ' " , webpage , ' cnn url ' )
2019-01-02 01:56:05 +09:00
return self . url_result ( ' http://cnn.com/video/?/video/ ' + cnn_url , CNNIE . ie_key ( ) )
2021-06-10 23:38:48 +09:00
class CNNPodcastsIE ( InfoExtractor ) :
_VALID_URL = r ' https?://(?:(?:edition|us|www) \ .)?cnn \ .com/audio/podcasts/ '
_TESTS = [ {
' url ' : ' https://edition.cnn.com/audio/podcasts/lincoln?episodeguid=4780950f-e269-407b-9ea1-accc01762945 ' ,
' info_dict ' : {
' id ' : ' 4780950f-e269-407b-9ea1-accc01762945 ' ,
' ext ' : ' mp3 ' ,
' title ' : ' Rising Star ' ,
' description ' : ' md5:cc953c3786761333e0829608d2437aba ' ,
' timestamp ' : 1613361600 ,
' upload_date ' : ' 20210215 ' ,
} ,
} , {
# playlist
' url ' : ' https://edition.cnn.com/audio/podcasts/lincoln ' ,
' info_dict ' : {
' id ' : ' lincoln ' ,
' title ' : ' Lincoln: Divided We Stand ' ,
' description ' : ' md5:9e122d8d05d58464fc2d5346d84671df ' ,
} ,
' playlist_count ' : 7 ,
} ]
def _real_extract ( self , url ) :
episode_id = None
query = compat_urllib_parse_urlparse ( url ) . query
if query :
episode_id = compat_parse_qs ( query ) . get ( ' episodeguid ' , [ None ] ) [ 0 ]
playlist_id = url_basename ( url )
video_id = episode_id or playlist_id
webpage = self . _download_webpage ( url , video_id )
episode_data = self . _parse_json ( self . _search_regex (
r ' EPISODE_DATA \ s*= \ s*( \ { .+? \ }); ' , webpage , ' episode data ' ) , video_id )
episodes = episode_data . get ( ' episodes ' ) or [ ]
def entry_info_dict ( episode ) :
description = episode . get ( ' summary ' )
if description :
# remove extra note
description = re . sub ( r ' \ r \ n.* ' , ' ' , description , flags = re . DOTALL ) . strip ( )
return {
' id ' : episode . get ( ' guid ' ) ,
' title ' : episode . get ( ' title ' ) ,
' url ' : clean_podcast_url ( try_get ( episode , lambda x : x [ ' enclosure ' ] [ ' url ' ] , compat_str ) ) ,
' description ' : description ,
' duration ' : int_or_none ( episode . get ( ' duration ' ) ) ,
' timestamp ' : parse_iso8601 ( episode . get ( ' publishedDate ' ) ) ,
}
if episode_id :
for episode in episodes :
if episode . get ( ' guid ' ) == episode_id :
return entry_info_dict ( episode )
else :
entries = [ ]
for episode in episodes :
entries . append ( entry_info_dict ( episode ) )
playlist_title = episode_data . get ( ' name ' )
playlist_description = episode_data . get ( ' description ' )
return self . playlist_result (
entries , playlist_id , playlist_title , playlist_description )