]>
jfr.im git - yt-dlp.git/blob - yt_dlp/extractor/plutotv.py
2 from __future__
import unicode_literals
7 from . common
import InfoExtractor
21 class PlutoTVIE ( InfoExtractor
):
23 https?://(?:www\.)?pluto\.tv(?:/[^/]+)?/on-demand
24 /(?P<video_type>movies|series)
25 /(?P<series_or_movie_slug>[^/]+)
27 (?:/seasons?/(?P<season_no>\d+))?
28 (?:/episode/(?P<episode_slug>[^/]+))?
32 _INFO_URL
= 'https://service-vod.clusters.pluto.tv/v3/vod/slugs/'
33 _INFO_QUERY_PARAMS
= {
36 'clientID' : compat_str ( uuid
. uuid1 ()),
37 'clientModelNumber' : 'na' ,
38 'serverSideAds' : 'false' ,
39 'deviceMake' : 'unknown' ,
42 'deviceVersion' : 'unknown' ,
43 'sid' : compat_str ( uuid
. uuid1 ()),
47 'url' : 'https://pluto.tv/on-demand/series/i-love-money/season/2/episode/its-in-the-cards-2009-2-3' ,
48 'md5' : 'ebcdd8ed89aaace9df37924f722fd9bd' ,
50 'id' : '5de6c598e9379ae4912df0a8' ,
52 'title' : 'It \' s In The Cards' ,
53 'episode' : 'It \' s In The Cards' ,
54 'description' : 'The teams face off against each other in a 3-on-2 soccer showdown. Strategy comes into play, though, as each team gets to select their opposing teams’ two defenders.' ,
55 'series' : 'I Love Money' ,
61 'url' : 'https://pluto.tv/on-demand/series/i-love-money/season/1/' ,
64 'id' : '5de6c582e9379ae4912dedbd' ,
65 'title' : 'I Love Money - Season 1' ,
68 'url' : 'https://pluto.tv/on-demand/series/i-love-money/' ,
71 'id' : '5de6c582e9379ae4912dedbd' ,
72 'title' : 'I Love Money' ,
75 'url' : 'https://pluto.tv/on-demand/movies/arrival-2015-1-1' ,
76 'md5' : '3cead001d317a018bf856a896dee1762' ,
78 'id' : '5e83ac701fa6a9001bb9df24' ,
81 'description' : 'When mysterious spacecraft touch down across the globe, an elite team - led by expert translator Louise Banks (Academy Award® nominee Amy Adams) – races against time to decipher their intent.' ,
85 'url' : 'https://pluto.tv/en/on-demand/series/manhunters-fugitive-task-force/seasons/1/episode/third-times-the-charm-1-1' ,
86 'only_matching' : True ,
88 'url' : 'https://pluto.tv/it/on-demand/series/csi-vegas/episode/legacy-2021-1-1' ,
89 'only_matching' : True ,
93 def _to_ad_free_formats ( self
, video_id
, formats
, subtitles
):
94 ad_free_formats
, ad_free_subtitles
, m3u8_urls
= [], {}, set ()
96 res
= self
._ download
_ webpage
(
97 fmt
. get ( 'url' ), video_id
, note
= 'Downloading m3u8 playlist' ,
101 first_segment_url
= re
. search (
102 r
'^(https?://.*/)0\-(end|[0-9]+)/[^/]+\.ts$' , res
,
104 if first_segment_url
:
106 compat_urlparse
. urljoin ( first_segment_url
. group ( 1 ), '0-end/master.m3u8' ))
108 first_segment_url
= re
. search (
109 r
'^(https?://.*/).+\-0+\.ts$' , res
,
111 if first_segment_url
:
113 compat_urlparse
. urljoin ( first_segment_url
. group ( 1 ), 'master.m3u8' ))
116 for m3u8_url
in m3u8_urls
:
117 fmts
, subs
= self
._ extract
_ m
3u8_ formats
_ and
_ subtitles
(
118 m3u8_url
, video_id
, 'mp4' , 'm3u8_native' , m3u8_id
= 'hls' , fatal
= False )
119 ad_free_formats
. extend ( fmts
)
120 ad_free_subtitles
= self
._ merge
_ subtitles
( ad_free_subtitles
, subs
)
122 formats
, subtitles
= ad_free_formats
, ad_free_subtitles
124 self
. report_warning ( 'Unable to find ad-free formats' )
125 return formats
, subtitles
127 def _get_video_info ( self
, video_json
, slug
, series_name
= None ):
128 video_id
= video_json
. get ( '_id' , slug
)
129 formats
, subtitles
= [], {}
130 for video_url
in try_get ( video_json
, lambda x
: x
[ 'stitched' ][ 'urls' ], list ) or []:
131 if video_url
. get ( 'type' ) != 'hls' :
133 url
= url_or_none ( video_url
. get ( 'url' ))
135 fmts
, subs
= self
._ extract
_ m
3u8_ formats
_ and
_ subtitles
(
136 url
, video_id
, 'mp4' , 'm3u8_native' , m3u8_id
= 'hls' , fatal
= False )
138 subtitles
= self
._ merge
_ subtitles
( subtitles
, subs
)
140 formats
, subtitles
= self
._ to
_ ad
_ free
_ formats
( video_id
, formats
, subtitles
)
141 self
._ sort
_ formats
( formats
)
146 'subtitles' : subtitles
,
147 'title' : video_json
. get ( 'name' ),
148 'description' : video_json
. get ( 'description' ),
149 'duration' : float_or_none ( video_json
. get ( 'duration' ), scale
= 1000 ),
153 'series' : series_name
,
154 'episode' : video_json
. get ( 'name' ),
155 'season_number' : int_or_none ( video_json
. get ( 'season' )),
156 'episode_number' : int_or_none ( video_json
. get ( 'number' )),
160 def _real_extract ( self
, url
):
161 mobj
= self
._ match
_ valid
_u rl
( url
). groupdict ()
162 info_slug
= mobj
[ 'series_or_movie_slug' ]
163 video_json
= self
._ download
_ json
( self
._ INFO
_U RL
+ info_slug
, info_slug
, query
= self
._ INFO
_ QUERY
_ PARAMS
)
165 if mobj
[ 'video_type' ] == 'series' :
166 series_name
= video_json
. get ( 'name' , info_slug
)
167 season_number
, episode_slug
= mobj
. get ( 'season_number' ), mobj
. get ( 'episode_slug' )
170 for season
in video_json
[ 'seasons' ]:
171 if season_number
is not None and season_number
!= int_or_none ( season
. get ( 'number' )):
173 for episode
in season
[ 'episodes' ]:
174 if episode_slug
is not None and episode_slug
!= episode
. get ( 'slug' ):
176 videos
. append ( self
._ get
_ video
_ info
( episode
, episode_slug
, series_name
))
178 raise ExtractorError ( 'Failed to find any videos to extract' )
179 if episode_slug
is not None and len ( videos
) == 1 :
181 playlist_title
= series_name
182 if season_number
is not None :
183 playlist_title
+= ' - Season %d ' % season_number
184 return self
. playlist_result ( videos
,
185 playlist_id
= video_json
. get ( '_id' , info_slug
),
186 playlist_title
= playlist_title
)
187 return self
._ get
_ video
_ info
( video_json
, info_slug
)