2023-09-18 23:08:40 +02:00
import itertools
2013-12-16 21:34:41 +01:00
import re
2023-09-18 23:08:40 +02:00
import urllib . parse
2013-12-16 21:34:41 +01:00
from . common import InfoExtractor
2023-09-18 23:08:40 +02:00
from . . utils import (
int_or_none ,
join_nonempty ,
js_to_json ,
parse_duration ,
strftime_or_none ,
traverse_obj ,
unified_strdate ,
urljoin ,
)
2013-12-16 21:34:41 +01:00
class RadioFranceIE ( InfoExtractor ) :
_VALID_URL = r ' ^https?://maison \ .radiofrance \ .fr/radiovisions/(?P<id>[^?#]+) '
2014-03-23 17:43:33 +01:00
IE_NAME = ' radiofrance '
2013-12-16 21:34:41 +01:00
_TEST = {
2014-03-23 17:43:33 +01:00
' url ' : ' http://maison.radiofrance.fr/radiovisions/one-one ' ,
' md5 ' : ' bdbb28ace95ed0e04faab32ba3160daf ' ,
' info_dict ' : {
' id ' : ' one-one ' ,
' ext ' : ' ogg ' ,
2016-02-14 10:37:17 +01:00
' title ' : ' One to one ' ,
' description ' : " Plutôt que d ' imaginer la radio de demain comme technologie ou comme création de contenu, je veux montrer que quelles que soient ses évolutions, j ' ai l ' intime conviction que la radio continuera d ' être un grand média de proximité pour les auditeurs. " ,
' uploader ' : ' Thomas Hercouët ' ,
2013-12-16 21:34:41 +01:00
} ,
}
def _real_extract ( self , url ) :
2021-08-19 03:41:24 +02:00
m = self . _match_valid_url ( url )
2013-12-16 21:34:41 +01:00
video_id = m . group ( ' id ' )
webpage = self . _download_webpage ( url , video_id )
2014-03-23 17:43:33 +01:00
title = self . _html_search_regex ( r ' <h1>(.*?)</h1> ' , webpage , ' title ' )
2013-12-16 21:34:41 +01:00
description = self . _html_search_regex (
r ' <div class= " bloc_page_wrapper " ><div class= " text " >(.*?)</div> ' ,
2014-03-23 17:43:33 +01:00
webpage , ' description ' , fatal = False )
2013-12-16 21:34:41 +01:00
uploader = self . _html_search_regex (
r ' <div class= " credit " > © (.*?)</div> ' ,
2014-03-23 17:43:33 +01:00
webpage , ' uploader ' , fatal = False )
2013-12-16 21:34:41 +01:00
formats_str = self . _html_search_regex (
r ' class= " jp-jplayer[^ " ]* " data-source= " ([^ " ]+) " > ' ,
2014-03-23 17:43:33 +01:00
webpage , ' audio URLs ' )
2013-12-16 21:34:41 +01:00
formats = [
{
2013-12-17 12:35:16 +01:00
' format_id ' : fm [ 0 ] ,
' url ' : fm [ 1 ] ,
2013-12-16 21:34:41 +01:00
' vcodec ' : ' none ' ,
2021-02-18 23:03:16 +01:00
' quality ' : i ,
2013-12-16 21:34:41 +01:00
}
2014-03-23 17:43:33 +01:00
for i , fm in
enumerate ( re . findall ( r " ([a-z0-9]+) \ s*: \ s* ' ([^ ' ]+) ' " , formats_str ) )
2013-12-16 21:34:41 +01:00
]
return {
' id ' : video_id ,
' title ' : title ,
' formats ' : formats ,
' description ' : description ,
' uploader ' : uploader ,
}
2022-06-10 21:03:54 +02:00
2023-09-18 23:08:40 +02:00
class RadioFranceBaseIE ( InfoExtractor ) :
_VALID_URL_BASE = r ' https?://(?:www \ .)?radiofrance \ .fr '
_STATIONS_RE = ' | ' . join ( map ( re . escape , (
' franceculture ' ,
' franceinfo ' ,
' franceinter ' ,
' francemusique ' ,
' fip ' ,
' mouv ' ,
) ) )
def _extract_data_from_webpage ( self , webpage , display_id , key ) :
return traverse_obj ( self . _search_json (
r ' \ bconst \ s+data \ s*= ' , webpage , key , display_id ,
2023-09-23 22:00:31 +02:00
contains_pattern = r ' \ [ \ { (?s:.+) \ } \ ] ' , transform_source = js_to_json ) ,
2023-09-18 23:08:40 +02:00
( . . . , ' data ' , key , { dict } ) , get_all = False ) or { }
class FranceCultureIE ( RadioFranceBaseIE ) :
_VALID_URL = rf ''' (?x)
{ RadioFranceBaseIE . _VALID_URL_BASE }
/ ( ? : { RadioFranceBaseIE . _STATIONS_RE } )
/ podcasts / ( ? : [ ^ ? #]+/)?(?P<display_id>[^?#]+)-(?P<id>\d{{6,}})(?:$|[?#])
'''
2022-06-10 21:03:54 +02:00
_TESTS = [
{
' url ' : ' https://www.radiofrance.fr/franceculture/podcasts/science-en-questions/la-physique-d-einstein-aiderait-elle-a-comprendre-le-cerveau-8440487 ' ,
' info_dict ' : {
' id ' : ' 8440487 ' ,
' display_id ' : ' la-physique-d-einstein-aiderait-elle-a-comprendre-le-cerveau ' ,
' ext ' : ' mp3 ' ,
' title ' : ' La physique d’ Einstein aiderait-elle à comprendre le cerveau ? ' ,
' description ' : ' Existerait-il un pont conceptuel entre la physique de l’ espace-temps et les neurosciences ? ' ,
2023-09-18 23:08:40 +02:00
' thumbnail ' : r ' re:^https?://.* \ .(?:jpg|png) ' ,
2022-06-10 21:03:54 +02:00
' upload_date ' : ' 20220514 ' ,
' duration ' : 2750 ,
} ,
} ,
2023-09-18 23:08:40 +02:00
{
' url ' : ' https://www.radiofrance.fr/franceinter/podcasts/le-7-9-30/le-7-9-30-du-vendredi-10-mars-2023-2107675 ' ,
' info_dict ' : {
' id ' : ' 2107675 ' ,
' display_id ' : ' le-7-9-30-du-vendredi-10-mars-2023 ' ,
' title ' : ' Inflation alimentaire : comment en sortir ? - Régis Debray et Claude Grange - Cybèle Idelot ' ,
' description ' : ' md5:36ee74351ede77a314fdebb94026b916 ' ,
' thumbnail ' : r ' re:^https?://.* \ .(?:jpg|png) ' ,
' upload_date ' : ' 20230310 ' ,
' duration ' : 8977 ,
' ext ' : ' mp3 ' ,
} ,
} ,
2022-06-19 03:36:14 +02:00
{
' url ' : ' https://www.radiofrance.fr/franceinter/podcasts/la-rafle-du-vel-d-hiv-une-affaire-d-etat/les-racines-du-crime-episode-1-3715507 ' ,
' only_matching ' : True ,
2023-09-18 23:08:40 +02:00
} , {
' url ' : ' https://www.radiofrance.fr/franceinfo/podcasts/le-billet-sciences/sante-bientot-un-vaccin-contre-l-asthme-allergique-3057200 ' ,
' only_matching ' : True ,
2022-06-19 03:36:14 +02:00
}
2022-06-10 21:03:54 +02:00
]
def _real_extract ( self , url ) :
video_id , display_id = self . _match_valid_url ( url ) . group ( ' id ' , ' display_id ' )
webpage = self . _download_webpage ( url , display_id )
# _search_json_ld doesn't correctly handle this. See https://github.com/yt-dlp/yt-dlp/pull/3874#discussion_r891903846
2022-10-03 13:20:27 +02:00
video_data = self . _search_json ( ' ' , webpage , ' audio data ' , display_id , contains_pattern = r ' { \ s* " @type " \ s*: \ s* " AudioObject " .+} ' )
2022-06-10 21:03:54 +02:00
return {
' id ' : video_id ,
' display_id ' : display_id ,
' url ' : video_data [ ' contentUrl ' ] ,
' vcodec ' : ' none ' if video_data . get ( ' encodingFormat ' ) == ' mp3 ' else None ,
' duration ' : parse_duration ( video_data . get ( ' duration ' ) ) ,
' title ' : self . _html_search_regex ( r ' (?s)<h1[^>]*itemprop= " [^ " ]*name[^ " ]* " [^>]*>(.+?)</h1> ' ,
webpage , ' title ' , default = self . _og_search_title ( webpage ) ) ,
' description ' : self . _html_search_regex (
r ' (?s)<meta name= " description " \ s*content= " ([^ " ]+) ' , webpage , ' description ' , default = None ) ,
' thumbnail ' : self . _og_search_thumbnail ( webpage ) ,
' uploader ' : self . _html_search_regex (
r ' (?s)<span class= " author " >(.*?)</span> ' , webpage , ' uploader ' , default = None ) ,
' upload_date ' : unified_strdate ( self . _search_regex (
r ' " datePublished " \ s*: \ s* " ([^ " ]+) ' , webpage , ' timestamp ' , fatal = False ) )
}
2023-09-18 23:08:40 +02:00
class RadioFranceLiveIE ( RadioFranceBaseIE ) :
_VALID_URL = rf ''' (?x)
https ? : / / ( ? : www \. ) ? radiofrance \. fr
/ ( ? P < id > { RadioFranceBaseIE . _STATIONS_RE } )
/ ? ( ? P < substation_id > radio - [ \w - ] + ) ? ( ? : [ #?]|$)
'''
_TESTS = [ {
' url ' : ' https://www.radiofrance.fr/franceinter/ ' ,
' info_dict ' : {
' id ' : ' franceinter ' ,
' title ' : str ,
' live_status ' : ' is_live ' ,
' ext ' : ' aac ' ,
} ,
' params ' : {
' skip_download ' : ' Livestream ' ,
} ,
} , {
' url ' : ' https://www.radiofrance.fr/franceculture ' ,
' info_dict ' : {
' id ' : ' franceculture ' ,
' title ' : str ,
' live_status ' : ' is_live ' ,
' ext ' : ' aac ' ,
} ,
' params ' : {
' skip_download ' : ' Livestream ' ,
} ,
} , {
' url ' : ' https://www.radiofrance.fr/mouv/radio-musique-kids-family ' ,
' info_dict ' : {
' id ' : ' mouv-radio-musique-kids-family ' ,
' title ' : str ,
' live_status ' : ' is_live ' ,
' ext ' : ' aac ' ,
} ,
' params ' : {
' skip_download ' : ' Livestream ' ,
} ,
} , {
' url ' : ' https://www.radiofrance.fr/mouv/radio-rnb-soul ' ,
' info_dict ' : {
' id ' : ' mouv-radio-rnb-soul ' ,
' title ' : str ,
' live_status ' : ' is_live ' ,
' ext ' : ' aac ' ,
} ,
' params ' : {
' skip_download ' : ' Livestream ' ,
} ,
} , {
' url ' : ' https://www.radiofrance.fr/mouv/radio-musique-mix ' ,
' info_dict ' : {
' id ' : ' mouv-radio-musique-mix ' ,
' title ' : str ,
' live_status ' : ' is_live ' ,
' ext ' : ' aac ' ,
} ,
' params ' : {
' skip_download ' : ' Livestream ' ,
} ,
} , {
' url ' : ' https://www.radiofrance.fr/fip/radio-rock ' ,
' info_dict ' : {
' id ' : ' fip-radio-rock ' ,
' title ' : str ,
' live_status ' : ' is_live ' ,
' ext ' : ' aac ' ,
} ,
' params ' : {
' skip_download ' : ' Livestream ' ,
} ,
} , {
' url ' : ' https://www.radiofrance.fr/mouv ' ,
' only_matching ' : True ,
} ]
def _real_extract ( self , url ) :
station_id , substation_id = self . _match_valid_url ( url ) . group ( ' id ' , ' substation_id ' )
if substation_id :
webpage = self . _download_webpage ( url , station_id )
api_response = self . _extract_data_from_webpage ( webpage , station_id , ' webRadioData ' )
else :
api_response = self . _download_json (
f ' https://www.radiofrance.fr/ { station_id } /api/live ' , station_id )
formats , subtitles = [ ] , { }
for media_source in traverse_obj ( api_response , ( ( ' now ' , None ) , ' media ' , ' sources ' , lambda _ , v : v [ ' url ' ] ) ) :
if media_source . get ( ' format ' ) == ' hls ' :
fmts , subs = self . _extract_m3u8_formats_and_subtitles ( media_source [ ' url ' ] , station_id , fatal = False )
formats . extend ( fmts )
self . _merge_subtitles ( subs , target = subtitles )
else :
formats . append ( {
' url ' : media_source [ ' url ' ] ,
' abr ' : media_source . get ( ' bitrate ' ) ,
} )
return {
' id ' : join_nonempty ( station_id , substation_id ) ,
' title ' : traverse_obj ( api_response , ( ' visual ' , ' legend ' ) ) or join_nonempty (
( ' now ' , ' firstLine ' , ' title ' ) , ( ' now ' , ' secondLine ' , ' title ' ) , from_dict = api_response , delim = ' - ' ) ,
' formats ' : formats ,
' subtitles ' : subtitles ,
' is_live ' : True ,
}
2023-12-30 22:27:36 +01:00
class RadioFrancePlaylistBaseIE ( RadioFranceBaseIE ) :
2023-09-18 23:08:40 +02:00
""" Subclasses must set _METADATA_KEY """
2024-05-15 17:39:56 +02:00
def _call_api ( self , station , content_id , cursor ) :
2023-09-18 23:08:40 +02:00
raise NotImplementedError ( ' This method must be implemented by subclasses ' )
2024-05-15 17:39:56 +02:00
def _generate_playlist_entries ( self , station , content_id , content_response ) :
2023-09-18 23:08:40 +02:00
for page_num in itertools . count ( 2 ) :
for entry in content_response [ ' items ' ] :
yield self . url_result (
f ' https://www.radiofrance.fr/ { entry [ " path " ] } ' , url_transparent = True , * * traverse_obj ( entry , {
' title ' : ' title ' ,
' description ' : ' standFirst ' ,
' timestamp ' : ( ' publishedDate ' , { int_or_none } ) ,
' thumbnail ' : ( ' visual ' , ' src ' ) ,
} ) )
2024-05-15 17:39:56 +02:00
if not content_response [ " next " ] :
2023-09-18 23:08:40 +02:00
break
2024-05-15 17:39:56 +02:00
content_response = self . _call_api ( station , content_id , content_response [ " next " ] )
2023-09-18 23:08:40 +02:00
def _real_extract ( self , url ) :
2024-05-15 17:39:56 +02:00
playlist_id = self . _match_id ( url )
# If it is a podcast playlist, get the name of the station it is on
# profile page playlists are not attached to a station currently
station = self . _match_valid_url ( url ) . group ( ' station ' ) if isinstance ( self , RadioFrancePodcastIE ) else None
# Check if user started request from a page other than 1
startpage = 1
parsedurl = urllib . parse . urlparse ( url )
if parsedurl . query :
startpagequery = urllib . parse . parse_qs ( parsedurl . query )
if ' p ' in startpagequery :
startpage = int ( startpagequery [ ' p ' ] [ 0 ] )
# Get data for the first page, and the uuid for the playlist
metadata = self . _call_api ( station , playlist_id , startpage )
uuid = traverse_obj ( metadata , ( ' metadata ' , ' id ' ) )
# This method should return the final playlist metadata which yt-dlp can then use to download everything
2023-09-18 23:08:40 +02:00
return self . playlist_result (
2024-05-15 17:39:56 +02:00
self . _generate_playlist_entries ( station , playlist_id , metadata ) ,
uuid ,
display_id = playlist_id ,
* * { * * traverse_obj ( metadata [ ' metadata ' ] , {
2023-09-18 23:08:40 +02:00
' title ' : ' title ' ,
' description ' : ' standFirst ' ,
' thumbnail ' : ( ' visual ' , ' src ' ) ,
2024-05-15 17:39:56 +02:00
} ) , * * traverse_obj ( metadata [ ' metadata ' ] , {
2023-09-18 23:08:40 +02:00
' title ' : ' name ' ,
' description ' : ' role ' ,
} ) } )
2023-12-30 22:27:36 +01:00
class RadioFrancePodcastIE ( RadioFrancePlaylistBaseIE ) :
2023-09-18 23:08:40 +02:00
_VALID_URL = rf ''' (?x)
{ RadioFranceBaseIE . _VALID_URL_BASE }
2024-05-15 17:39:56 +02:00
/ ( ? P < station > { RadioFranceBaseIE . _STATIONS_RE } )
2023-09-18 23:08:40 +02:00
/ podcasts / ( ? P < id > [ \w - ] + ) / ? ( ? : [ ? #]|$)
'''
_TESTS = [ {
' url ' : ' https://www.radiofrance.fr/franceinfo/podcasts/le-billet-vert ' ,
' info_dict ' : {
' id ' : ' eaf6ef81-a980-4f1c-a7d1-8a75ecd54b17 ' ,
' display_id ' : ' le-billet-vert ' ,
' title ' : ' Le billet sciences ' ,
' description ' : ' md5:eb1007b34b0c0a680daaa71525bbd4c1 ' ,
' thumbnail ' : r ' re:^https?://.* \ .(?:jpg|png) ' ,
} ,
' playlist_mincount ' : 11 ,
} , {
' url ' : ' https://www.radiofrance.fr/franceinter/podcasts/jean-marie-le-pen-l-obsession-nationale ' ,
' info_dict ' : {
' id ' : ' 566fd524-3074-4fbc-ac69-8696f2152a54 ' ,
' display_id ' : ' jean-marie-le-pen-l-obsession-nationale ' ,
' title ' : ' Jean-Marie Le Pen, l \' obsession nationale ' ,
' description ' : ' md5:a07c0cfb894f6d07a62d0ad12c4b7d73 ' ,
' thumbnail ' : r ' re:^https?://.* \ .(?:jpg|png) ' ,
} ,
' playlist_count ' : 7 ,
} , {
' url ' : ' https://www.radiofrance.fr/franceculture/podcasts/serie-thomas-grjebine ' ,
' info_dict ' : {
' id ' : ' 63c1ddc9-9f15-457a-98b2-411bac63f48d ' ,
' display_id ' : ' serie-thomas-grjebine ' ,
' title ' : ' Thomas Grjebine ' ,
} ,
' playlist_count ' : 1 ,
} , {
' url ' : ' https://www.radiofrance.fr/fip/podcasts/certains-l-aiment-fip ' ,
' info_dict ' : {
' id ' : ' 143dff38-e956-4a5d-8576-1c0b7242b99e ' ,
' display_id ' : ' certains-l-aiment-fip ' ,
' title ' : ' Certains l’ aiment Fip ' ,
' description ' : ' md5:ff974672ba00d4fd5be80fb001c5b27e ' ,
' thumbnail ' : r ' re:^https?://.* \ .(?:jpg|png) ' ,
} ,
' playlist_mincount ' : 321 ,
} , {
' url ' : ' https://www.radiofrance.fr/franceinter/podcasts/le-7-9 ' ,
' only_matching ' : True ,
} , {
' url ' : ' https://www.radiofrance.fr/mouv/podcasts/dirty-mix ' ,
' only_matching ' : True ,
} ]
_METADATA_KEY = ' expressions '
2024-05-15 17:39:56 +02:00
def _call_api ( self , station , podcast_id , cursor ) :
# The data is stored in the last <script> tag on a page
url = ' https://www.radiofrance.fr/ ' + station + ' /podcasts/ ' + podcast_id + ' ?p= ' + str ( cursor )
webpage = self . _download_webpage ( url , podcast_id , note = f ' Downloading { podcast_id } page { cursor } ' )
resp = dict ( )
# _search_json cannot parse the data as it contains javascript
# Therefore, parse the episodes objects array separately
resp [ ' items ' ] = self . _search_json ( r ' a.items \ s*= \ s* ' , webpage , podcast_id , podcast_id ,
contains_pattern = r ' \ [.+ \ ] ' , transform_source = js_to_json )
# the pagination data is stored in a javascript object 'a'
lastPage = int ( re . search ( r ' a \ .lastPage \ s*= \ s*( \ d+); ' , webpage ) . group ( 1 ) )
hasMorePages = cursor < lastPage
resp [ ' next ' ] = cursor + 1 if hasMorePages else None
resp [ ' metadata ' ] = self . _search_json ( r ' content: \ s* ' , webpage , podcast_id , podcast_id ,
transform_source = js_to_json )
return resp
2023-09-18 23:08:40 +02:00
2023-12-30 22:27:36 +01:00
class RadioFranceProfileIE ( RadioFrancePlaylistBaseIE ) :
2023-09-18 23:08:40 +02:00
_VALID_URL = rf ' { RadioFranceBaseIE . _VALID_URL_BASE } /personnes/(?P<id>[ \ w-]+) '
_TESTS = [ {
' url ' : ' https://www.radiofrance.fr/personnes/thomas-pesquet?p=3 ' ,
' info_dict ' : {
' id ' : ' 86c62790-e481-11e2-9f7b-782bcb6744eb ' ,
' display_id ' : ' thomas-pesquet ' ,
' title ' : ' Thomas Pesquet ' ,
' description ' : ' Astronaute à l \' agence spatiale européenne ' ,
} ,
2024-05-15 17:39:56 +02:00
' playlist_mincount ' : 158 ,
2023-09-18 23:08:40 +02:00
} , {
' url ' : ' https://www.radiofrance.fr/personnes/eugenie-bastie ' ,
' info_dict ' : {
' id ' : ' 9593050b-0183-4972-a0b5-d8f699079e02 ' ,
' display_id ' : ' eugenie-bastie ' ,
' title ' : ' Eugénie Bastié ' ,
' description ' : ' Journaliste et essayiste ' ,
' thumbnail ' : r ' re:^https?://.* \ .(?:jpg|png) ' ,
} ,
' playlist_mincount ' : 39 ,
} , {
' url ' : ' https://www.radiofrance.fr/personnes/lea-salame ' ,
' only_matching ' : True ,
} ]
_METADATA_KEY = ' documents '
2024-05-15 17:39:56 +02:00
def _call_api ( self , station , profile_id , cursor ) :
url = ' https://www.radiofrance.fr/personnes/ ' + profile_id + ' ?p= ' + str ( cursor )
webpage = self . _download_webpage ( url , profile_id , note = f ' Downloading { profile_id } page { cursor } ' )
resp = dict ( )
# On profile pages, the data is stored in a javascript array in the final <script>
# Each episode is stored as
# a[0] = { id: ... }; a[1] = [ id: ... ]; etc.
# Annoyingly, sometimes it is delivered using 'b', with 'a' holding metadata
resp [ ' items ' ] = [ ]
podcastindex = 0
nextmatch = True
while nextmatch :
nextmatch = self . _search_json ( r ' \ w+ \ [ ' + str ( podcastindex ) + r ' \ ] \ s*= \ s* ' , webpage , profile_id ,
profile_id , transform_source = js_to_json , fatal = False , default = None )
podcastindex + = 1
if nextmatch is not None :
resp [ ' items ' ] . append ( nextmatch )
# There is more than one pagination key in the final <script>
# We should use pick the pagination object which is within a documents object
pagedata = self . _search_json ( r ' documents \ s*: \ s* ' , webpage , profile_id , profile_id ,
transform_source = js_to_json )
lastPage = traverse_obj ( pagedata , ( ' pagination ' , ' lastPage ' ) )
hasMorePages = cursor < lastPage
resp [ ' next ' ] = cursor + 1 if hasMorePages else None
resp [ ' metadata ' ] = self . _search_json ( r ' content: \ s* ' , webpage , profile_id , profile_id ,
transform_source = js_to_json )
2023-09-18 23:08:40 +02:00
return resp
class RadioFranceProgramScheduleIE ( RadioFranceBaseIE ) :
_VALID_URL = rf ''' (?x)
{ RadioFranceBaseIE . _VALID_URL_BASE }
/ ( ? P < station > { RadioFranceBaseIE . _STATIONS_RE } )
/ grille - programmes ( ? : \? date = ( ? P < date > [ \d - ] + ) ) ?
'''
_TESTS = [ {
' url ' : ' https://www.radiofrance.fr/franceinter/grille-programmes?date=17-02-2023 ' ,
' info_dict ' : {
' id ' : ' franceinter-program-20230217 ' ,
' upload_date ' : ' 20230217 ' ,
} ,
' playlist_count ' : 25 ,
} , {
' url ' : ' https://www.radiofrance.fr/franceculture/grille-programmes?date=01-02-2023 ' ,
' info_dict ' : {
' id ' : ' franceculture-program-20230201 ' ,
' upload_date ' : ' 20230201 ' ,
} ,
' playlist_count ' : 25 ,
} , {
' url ' : ' https://www.radiofrance.fr/mouv/grille-programmes?date=19-03-2023 ' ,
' info_dict ' : {
' id ' : ' mouv-program-20230319 ' ,
' upload_date ' : ' 20230319 ' ,
} ,
' playlist_count ' : 3 ,
} , {
' url ' : ' https://www.radiofrance.fr/francemusique/grille-programmes?date=18-03-2023 ' ,
' info_dict ' : {
' id ' : ' francemusique-program-20230318 ' ,
' upload_date ' : ' 20230318 ' ,
} ,
' playlist_count ' : 15 ,
} , {
' url ' : ' https://www.radiofrance.fr/franceculture/grille-programmes ' ,
' only_matching ' : True ,
} ]
def _generate_playlist_entries ( self , webpage_url , api_response ) :
for entry in traverse_obj ( api_response , ( ' steps ' , lambda _ , v : v [ ' expression ' ] [ ' path ' ] ) ) :
yield self . url_result (
urljoin ( webpage_url , f ' / { entry [ " expression " ] [ " path " ] } ' ) , ie = FranceCultureIE ,
url_transparent = True , * * traverse_obj ( entry , {
' title ' : ( ' expression ' , ' title ' ) ,
' thumbnail ' : ( ' expression ' , ' visual ' , ' src ' ) ,
' timestamp ' : ( ' startTime ' , { int_or_none } ) ,
' series_id ' : ( ' concept ' , ' id ' ) ,
' series ' : ( ' concept ' , ' title ' ) ,
} ) )
def _real_extract ( self , url ) :
station , date = self . _match_valid_url ( url ) . group ( ' station ' , ' date ' )
webpage = self . _download_webpage ( url , station )
grid_data = self . _extract_data_from_webpage ( webpage , station , ' grid ' )
upload_date = strftime_or_none ( grid_data . get ( ' date ' ) , ' % Y % m %d ' )
return self . playlist_result (
self . _generate_playlist_entries ( url , grid_data ) ,
join_nonempty ( station , ' program ' , upload_date ) , upload_date = upload_date )