2016-10-02 19:39:18 +08:00
# coding: utf-8
2014-01-30 13:13:57 +08:00
from __future__ import unicode_literals
2013-09-09 03:55:11 +08:00
import re
from . common import InfoExtractor
2015-07-19 23:54:12 +08:00
from . . compat import compat_urlparse
2014-12-13 19:24:42 +08:00
from . . utils import (
2014-09-17 00:47:59 +08:00
clean_html ,
2014-12-13 19:24:42 +08:00
ExtractorError ,
2014-09-17 00:47:59 +08:00
int_or_none ,
2014-12-13 19:24:42 +08:00
parse_duration ,
2015-04-05 01:02:04 +08:00
determine_ext ,
2013-09-09 03:55:11 +08:00
)
2017-12-06 17:54:20 +08:00
from . dailymotion import DailymotionIE
2013-09-09 03:55:11 +08:00
2013-09-10 21:50:34 +08:00
class FranceTVBaseInfoExtractor ( InfoExtractor ) :
2017-05-10 00:07:01 +08:00
def _extract_video ( self , video_id , catalogue = None ) :
2014-09-17 00:47:59 +08:00
info = self . _download_json (
2017-05-10 00:07:01 +08:00
' https://sivideo.webservices.francetelevisions.fr/tools/getInfosOeuvre/v2/ ' ,
video_id , ' Downloading video JSON ' , query = {
' idDiffusion ' : video_id ,
' catalogue ' : catalogue or ' ' ,
} )
2014-09-17 00:47:59 +08:00
if info . get ( ' status ' ) == ' NOK ' :
raise ExtractorError (
' %s returned error: %s ' % ( self . IE_NAME , info [ ' message ' ] ) , expected = True )
2014-11-25 02:37:20 +08:00
allowed_countries = info [ ' videos ' ] [ 0 ] . get ( ' geoblocage ' )
if allowed_countries :
georestricted = True
geo_info = self . _download_json (
' http://geo.francetv.fr/ws/edgescape.json ' , video_id ,
' Downloading geo restriction info ' )
country = geo_info [ ' reponse ' ] [ ' geo_info ' ] [ ' country_code ' ]
if country not in allowed_countries :
raise ExtractorError (
' The video is not available from your location ' ,
expected = True )
else :
georestricted = False
2014-09-17 00:47:59 +08:00
formats = [ ]
for video in info [ ' videos ' ] :
if video [ ' statut ' ] != ' ONLINE ' :
continue
video_url = video [ ' url ' ]
if not video_url :
continue
format_id = video [ ' format ' ]
2015-04-05 01:02:04 +08:00
ext = determine_ext ( video_url )
if ext == ' f4m ' :
2014-11-25 02:37:20 +08:00
if georestricted :
# See https://github.com/rg3/youtube-dl/issues/3963
# m3u8 urls work fine
continue
2014-09-17 00:47:59 +08:00
f4m_url = self . _download_webpage (
2015-07-19 23:45:49 +08:00
' http://hdfauth.francetv.fr/esi/TA?url= %s ' % video_url ,
2014-09-17 00:47:59 +08:00
video_id , ' Downloading f4m manifest token ' , fatal = False )
if f4m_url :
2015-07-19 23:51:06 +08:00
formats . extend ( self . _extract_f4m_formats (
2016-03-20 15:00:46 +08:00
f4m_url + ' &hdcore=3.7.0&plugin=aasp-3.7.0.39.44 ' ,
video_id , f4m_id = format_id , fatal = False ) )
2015-04-05 01:02:04 +08:00
elif ext == ' m3u8 ' :
2016-03-20 15:00:46 +08:00
formats . extend ( self . _extract_m3u8_formats (
video_url , video_id , ' mp4 ' , entry_protocol = ' m3u8_native ' ,
m3u8_id = format_id , fatal = False ) )
2014-09-17 00:47:59 +08:00
elif video_url . startswith ( ' rtmp ' ) :
formats . append ( {
' url ' : video_url ,
' format_id ' : ' rtmp- %s ' % format_id ,
' ext ' : ' flv ' ,
} )
else :
2016-03-20 15:00:46 +08:00
if self . _is_valid_url ( video_url , video_id , format_id ) :
formats . append ( {
' url ' : video_url ,
' format_id ' : format_id ,
} )
2014-09-17 00:47:59 +08:00
self . _sort_formats ( formats )
2013-09-10 21:50:34 +08:00
2015-09-02 00:37:42 +08:00
title = info [ ' titre ' ]
subtitle = info . get ( ' sous_titre ' )
if subtitle :
title + = ' - %s ' % subtitle
2016-03-20 14:44:04 +08:00
title = title . strip ( )
2015-09-02 00:37:42 +08:00
2015-10-26 22:11:09 +08:00
subtitles = { }
2015-10-26 22:35:28 +08:00
subtitles_list = [ {
2015-10-28 15:22:04 +08:00
' url ' : subformat [ ' url ' ] ,
' ext ' : subformat . get ( ' format ' ) ,
} for subformat in info . get ( ' subtitles ' , [ ] ) if subformat . get ( ' url ' ) ]
2015-10-26 22:35:28 +08:00
if subtitles_list :
subtitles [ ' fr ' ] = subtitles_list
2015-10-26 22:11:09 +08:00
2014-07-28 20:37:13 +08:00
return {
' id ' : video_id ,
2015-09-02 00:37:42 +08:00
' title ' : title ,
2014-09-17 00:47:59 +08:00
' description ' : clean_html ( info [ ' synopsis ' ] ) ,
' thumbnail ' : compat_urlparse . urljoin ( ' http://pluzz.francetv.fr ' , info [ ' image ' ] ) ,
2015-07-19 00:43:42 +08:00
' duration ' : int_or_none ( info . get ( ' real_duration ' ) ) or parse_duration ( info [ ' duree ' ] ) ,
2014-09-17 00:47:59 +08:00
' timestamp ' : int_or_none ( info [ ' diffusion ' ] [ ' timestamp ' ] ) ,
2014-07-28 20:37:13 +08:00
' formats ' : formats ,
2015-10-26 22:11:09 +08:00
' subtitles ' : subtitles ,
2014-07-28 20:37:13 +08:00
}
2013-09-10 21:50:34 +08:00
2017-05-10 00:07:01 +08:00
class FranceTVIE ( FranceTVBaseInfoExtractor ) :
2017-06-01 01:15:15 +08:00
_VALID_URL = r ' https?://(?:(?:www \ .)?france \ .tv|mobile \ .france \ .tv)/(?:[^/]+/)*(?P<id>[^/]+) \ .html '
2013-09-09 03:55:11 +08:00
2017-05-10 00:07:01 +08:00
_TESTS = [ {
' url ' : ' https://www.france.tv/france-2/13h15-le-dimanche/140921-les-mysteres-de-jesus.html ' ,
' info_dict ' : {
' id ' : ' 157550144 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' 13h15, le dimanche... - Les mystères de Jésus ' ,
' description ' : ' md5:75efe8d4c0a8205e5904498ffe1e1a42 ' ,
' timestamp ' : 1494156300 ,
' upload_date ' : ' 20170507 ' ,
} ,
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
} ,
} , {
# france3
' url ' : ' https://www.france.tv/france-3/des-chiffres-et-des-lettres/139063-emission-du-mardi-9-mai-2017.html ' ,
' only_matching ' : True ,
} , {
# france4
' url ' : ' https://www.france.tv/france-4/hero-corp/saison-1/134151-apres-le-calme.html ' ,
' only_matching ' : True ,
} , {
# france5
' url ' : ' https://www.france.tv/france-5/c-a-dire/saison-10/137013-c-a-dire.html ' ,
' only_matching ' : True ,
} , {
# franceo
' url ' : ' https://www.france.tv/france-o/archipels/132249-mon-ancetre-l-esclave.html ' ,
' only_matching ' : True ,
} , {
# france2 live
' url ' : ' https://www.france.tv/france-2/direct.html ' ,
' only_matching ' : True ,
} , {
' url ' : ' https://www.france.tv/documentaires/histoire/136517-argentine-les-500-bebes-voles-de-la-dictature.html ' ,
' only_matching ' : True ,
} , {
' url ' : ' https://www.france.tv/jeux-et-divertissements/divertissements/133965-le-web-contre-attaque.html ' ,
' only_matching ' : True ,
2017-05-13 22:57:00 +08:00
} , {
' url ' : ' https://mobile.france.tv/france-5/c-dans-l-air/137347-emission-du-vendredi-12-mai-2017.html ' ,
' only_matching ' : True ,
2017-06-01 01:15:15 +08:00
} , {
' url ' : ' https://www.france.tv/142749-rouge-sang.html ' ,
' only_matching ' : True ,
2017-05-10 00:07:01 +08:00
} ]
2013-09-09 03:55:11 +08:00
def _real_extract ( self , url ) :
2015-10-27 23:43:29 +08:00
display_id = self . _match_id ( url )
webpage = self . _download_webpage ( url , display_id )
2017-05-10 00:07:01 +08:00
catalogue = None
video_id = self . _search_regex (
r ' data-main-video=([ " \' ])(?P<id>(?:(?! \ 1).)+) \ 1 ' ,
webpage , ' video id ' , default = None , group = ' id ' )
2015-10-27 23:43:29 +08:00
if not video_id :
2017-05-10 00:07:01 +08:00
video_id , catalogue = self . _html_search_regex (
r ' (?:href=|player \ .setVideo \ ( \ s*) " http://videos? \ .francetv \ .fr/video/([^@]+@[^ " ]+) " ' ,
webpage , ' video ID ' ) . split ( ' @ ' )
return self . _extract_video ( video_id , catalogue )
class FranceTVEmbedIE ( FranceTVBaseInfoExtractor ) :
_VALID_URL = r ' https?://embed \ .francetv \ .fr/* \ ?.*? \ bue=(?P<id>[^&]+) '
_TEST = {
' url ' : ' http://embed.francetv.fr/?ue=7fd581a2ccf59d2fc5719c5c13cf6961 ' ,
' info_dict ' : {
' id ' : ' NI_983319 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Le Pen Reims ' ,
' upload_date ' : ' 20170505 ' ,
' timestamp ' : 1493981780 ,
' duration ' : 16 ,
} ,
}
2015-10-27 23:43:29 +08:00
2017-05-10 00:07:01 +08:00
def _real_extract ( self , url ) :
video_id = self . _match_id ( url )
video = self . _download_json (
' http://api-embed.webservices.francetelevisions.fr/key/ %s ' % video_id ,
video_id )
2013-09-09 03:55:11 +08:00
2017-05-10 00:07:01 +08:00
return self . _extract_video ( video [ ' video_id ' ] , video . get ( ' catalog ' ) )
2013-09-09 03:55:11 +08:00
2017-05-10 00:07:01 +08:00
class FranceTVInfoIE ( FranceTVBaseInfoExtractor ) :
2014-01-30 13:13:57 +08:00
IE_NAME = ' francetvinfo.fr '
2016-08-11 22:42:55 +08:00
_VALID_URL = r ' https?://(?:www|mobile|france3-regions) \ .francetvinfo \ .fr/(?:[^/]+/)*(?P<title>[^/?#&.]+) '
2013-09-09 03:55:11 +08:00
2014-05-16 21:51:01 +08:00
_TESTS = [ {
2014-01-30 13:13:57 +08:00
' url ' : ' http://www.francetvinfo.fr/replay-jt/france-3/soir-3/jt-grand-soir-3-lundi-26-aout-2013_393427.html ' ,
' info_dict ' : {
2014-05-16 21:51:01 +08:00
' id ' : ' 84981923 ' ,
2016-03-20 15:00:46 +08:00
' ext ' : ' mp4 ' ,
2014-01-30 13:13:57 +08:00
' title ' : ' Soir 3 ' ,
2014-09-17 00:47:59 +08:00
' upload_date ' : ' 20130826 ' ,
' timestamp ' : 1377548400 ,
2015-10-26 22:35:45 +08:00
' subtitles ' : {
' fr ' : ' mincount:2 ' ,
} ,
2013-09-10 21:50:34 +08:00
} ,
2016-03-20 15:00:46 +08:00
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
} ,
2014-05-16 21:51:01 +08:00
} , {
' url ' : ' http://www.francetvinfo.fr/elections/europeennes/direct-europeennes-regardez-le-debat-entre-les-candidats-a-la-presidence-de-la-commission_600639.html ' ,
' info_dict ' : {
' id ' : ' EV_20019 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' Débat des candidats à la Commission européenne ' ,
' description ' : ' Débat des candidats à la Commission européenne ' ,
} ,
' params ' : {
' skip_download ' : ' HLS (reqires ffmpeg) '
2014-09-17 00:47:59 +08:00
} ,
' skip ' : ' Ce direct est terminé et sera disponible en rattrapage dans quelques minutes. ' ,
2015-06-21 21:31:33 +08:00
} , {
' url ' : ' http://www.francetvinfo.fr/economie/entreprises/les-entreprises-familiales-le-secret-de-la-reussite_933271.html ' ,
' md5 ' : ' f485bda6e185e7d15dbc69b72bae993e ' ,
' info_dict ' : {
2016-03-20 15:00:46 +08:00
' id ' : ' NI_173343 ' ,
2015-06-21 21:31:33 +08:00
' ext ' : ' mp4 ' ,
' title ' : ' Les entreprises familiales : le secret de la réussite ' ,
2017-01-02 20:08:07 +08:00
' thumbnail ' : r ' re:^https?://.* \ .jpe?g$ ' ,
2016-03-20 15:00:46 +08:00
' timestamp ' : 1433273139 ,
' upload_date ' : ' 20150602 ' ,
} ,
' params ' : {
# m3u8 downloads
' skip_download ' : True ,
} ,
2016-03-20 14:44:04 +08:00
} , {
' url ' : ' http://france3-regions.francetvinfo.fr/bretagne/cotes-d-armor/thalassa-echappee-breizh-ce-venredi-dans-les-cotes-d-armor-954961.html ' ,
' md5 ' : ' f485bda6e185e7d15dbc69b72bae993e ' ,
' info_dict ' : {
' id ' : ' NI_657393 ' ,
2016-03-20 15:00:46 +08:00
' ext ' : ' mp4 ' ,
2016-03-20 14:44:04 +08:00
' title ' : ' Olivier Monthus, réalisateur de " Bretagne, le choix de l’ Armor " ' ,
' description ' : ' md5:a3264114c9d29aeca11ced113c37b16c ' ,
2017-01-02 20:08:07 +08:00
' thumbnail ' : r ' re:^https?://.* \ .jpe?g$ ' ,
2016-03-20 14:44:04 +08:00
' timestamp ' : 1458300695 ,
' upload_date ' : ' 20160318 ' ,
} ,
' params ' : {
' skip_download ' : True ,
} ,
2016-07-06 23:37:54 +08:00
} , {
# Dailymotion embed
' url ' : ' http://www.francetvinfo.fr/politique/notre-dame-des-landes/video-sur-france-inter-cecile-duflot-denonce-le-regard-meprisant-de-patrick-cohen_1520091.html ' ,
' md5 ' : ' ee7f1828f25a648addc90cb2687b1f12 ' ,
' info_dict ' : {
' id ' : ' x4iiko0 ' ,
' ext ' : ' mp4 ' ,
' title ' : ' NDDL, référendum, Brexit : Cécile Duflot répond à Patrick Cohen ' ,
' description ' : ' Au lendemain de la victoire du " oui " au référendum sur l \' aéroport de Notre-Dame-des-Landes, l \' ancienne ministre écologiste est l \' invitée de Patrick Cohen. Plus d \' info : https://www.franceinter.fr/emissions/le-7-9/le-7-9-27-juin-2016 ' ,
' timestamp ' : 1467011958 ,
' upload_date ' : ' 20160627 ' ,
' uploader ' : ' France Inter ' ,
' uploader_id ' : ' x2q2ez ' ,
} ,
' add_ie ' : [ ' Dailymotion ' ] ,
2016-08-11 22:42:55 +08:00
} , {
' url ' : ' http://france3-regions.francetvinfo.fr/limousin/emissions/jt-1213-limousin ' ,
' only_matching ' : True ,
2014-05-16 21:51:01 +08:00
} ]
2013-09-10 21:50:34 +08:00
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
page_title = mobj . group ( ' title ' )
webpage = self . _download_webpage ( url , page_title )
2015-06-21 21:31:33 +08:00
2016-07-06 23:37:54 +08:00
dailymotion_urls = DailymotionIE . _extract_urls ( webpage )
if dailymotion_urls :
return self . playlist_result ( [
self . url_result ( dailymotion_url , DailymotionIE . ie_key ( ) )
for dailymotion_url in dailymotion_urls ] )
2015-06-21 21:31:33 +08:00
2014-09-17 00:47:59 +08:00
video_id , catalogue = self . _search_regex (
2016-03-20 14:44:04 +08:00
( r ' id-video=([^@]+@[^ " ]+) ' ,
r ' <a[^>]+href= " (?:https?:)?//videos \ .francetv \ .fr/video/([^@]+@[^ " ]+) " ' ) ,
webpage , ' video id ' ) . split ( ' @ ' )
2014-09-17 00:47:59 +08:00
return self . _extract_video ( video_id , catalogue )
2013-09-24 03:28:33 +08:00
2017-11-25 22:49:49 +08:00
class GenerationWhatIE ( InfoExtractor ) :
IE_NAME = ' france2.fr:generation-what '
_VALID_URL = r ' https?://generation-what \ .francetv \ .fr/[^/]+/video/(?P<id>[^/?#]+) '
2013-09-24 03:41:54 +08:00
2017-11-25 22:49:49 +08:00
_TESTS = [ {
' url ' : ' http://generation-what.francetv.fr/portrait/video/present-arms ' ,
2014-01-30 13:13:57 +08:00
' info_dict ' : {
2017-11-25 22:49:49 +08:00
' id ' : ' wtvKYUG45iw ' ,
2015-02-01 22:06:55 +08:00
' ext ' : ' mp4 ' ,
2017-11-25 22:49:49 +08:00
' title ' : ' Generation What - Garde à vous - FRA ' ,
' uploader ' : ' Generation What ' ,
' uploader_id ' : ' UCHH9p1eetWCgt4kXBYCb3_w ' ,
' upload_date ' : ' 20160411 ' ,
2013-09-24 03:41:54 +08:00
} ,
2017-11-25 22:49:49 +08:00
} , {
' url ' : ' http://generation-what.francetv.fr/europe/video/present-arms ' ,
' only_matching ' : True ,
} ]
2013-09-24 03:41:54 +08:00
def _real_extract ( self , url ) :
2015-02-01 22:06:55 +08:00
display_id = self . _match_id ( url )
2017-11-25 22:49:49 +08:00
webpage = self . _download_webpage ( url , display_id )
youtube_id = self . _search_regex (
r " window \ .videoURL \ s*= \ s* ' ([0-9A-Za-z_-] {11} ) ' ; " ,
webpage , ' youtube id ' )
return self . url_result ( youtube_id , ' Youtube ' , youtube_id )
2014-01-08 23:16:34 +08:00
class CultureboxIE ( FranceTVBaseInfoExtractor ) :
2014-01-30 13:13:57 +08:00
IE_NAME = ' culturebox.francetvinfo.fr '
2014-07-19 18:47:50 +08:00
_VALID_URL = r ' https?://(?:m \ .)?culturebox \ .francetvinfo \ .fr/(?P<name>.*?)( \ ?|$) '
2014-01-08 23:16:34 +08:00
_TEST = {
2015-04-05 00:50:13 +08:00
' url ' : ' http://culturebox.francetvinfo.fr/live/musique/musique-classique/le-livre-vermeil-de-montserrat-a-la-cathedrale-delne-214511 ' ,
2015-04-05 01:06:16 +08:00
' md5 ' : ' 9b88dc156781c4dbebd4c3e066e0b1d6 ' ,
2014-01-30 13:13:57 +08:00
' info_dict ' : {
2015-04-05 00:50:13 +08:00
' id ' : ' EV_50111 ' ,
2015-04-05 01:06:16 +08:00
' ext ' : ' flv ' ,
2015-04-05 00:50:13 +08:00
' title ' : " Le Livre Vermeil de Montserrat à la Cathédrale d ' Elne " ,
' description ' : ' md5:f8a4ad202e8fe533e2c493cc12e739d9 ' ,
' upload_date ' : ' 20150320 ' ,
' timestamp ' : 1426892400 ,
' duration ' : 2760.9 ,
} ,
2014-01-08 23:16:34 +08:00
}
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
name = mobj . group ( ' name ' )
2015-04-05 00:43:34 +08:00
2014-01-08 23:16:34 +08:00
webpage = self . _download_webpage ( url , name )
2015-04-05 00:43:34 +08:00
if " >Ce live n ' est plus disponible en replay< " in webpage :
raise ExtractorError ( ' Video %s is not available ' % name , expected = True )
2014-09-17 00:47:59 +08:00
video_id , catalogue = self . _search_regex (
2017-11-23 07:39:11 +08:00
r ' " https?://videos \ .francetv \ .fr/video/([^@]+@[^ " ]+) " ' , webpage , ' video id ' ) . split ( ' @ ' )
2014-09-17 00:47:59 +08:00
return self . _extract_video ( video_id , catalogue )