]>
jfr.im git - yt-dlp.git/blob - yt_dlp/extractor/plutotv.py
4 from . common
import InfoExtractor
18 class PlutoTVIE ( InfoExtractor
):
20 https?://(?:www\.)?pluto\.tv(?:/[^/]+)?/on-demand
21 /(?P<video_type>movies|series)
22 /(?P<series_or_movie_slug>[^/]+)
24 (?:/seasons?/(?P<season_no>\d+))?
25 (?:/episode/(?P<episode_slug>[^/]+))?
29 _INFO_URL
= 'https://service-vod.clusters.pluto.tv/v3/vod/slugs/'
30 _INFO_QUERY_PARAMS
= {
33 'clientID' : compat_str ( uuid
. uuid1 ()),
34 'clientModelNumber' : 'na' ,
35 'serverSideAds' : 'false' ,
36 'deviceMake' : 'unknown' ,
39 'deviceVersion' : 'unknown' ,
40 'sid' : compat_str ( uuid
. uuid1 ()),
44 'url' : 'https://pluto.tv/on-demand/series/i-love-money/season/2/episode/its-in-the-cards-2009-2-3' ,
45 'md5' : 'ebcdd8ed89aaace9df37924f722fd9bd' ,
47 'id' : '5de6c598e9379ae4912df0a8' ,
49 'title' : 'It \' s In The Cards' ,
50 'episode' : 'It \' s In The Cards' ,
51 '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.' ,
52 'series' : 'I Love Money' ,
58 'url' : 'https://pluto.tv/on-demand/series/i-love-money/season/1/' ,
61 'id' : '5de6c582e9379ae4912dedbd' ,
62 'title' : 'I Love Money - Season 1' ,
65 'url' : 'https://pluto.tv/on-demand/series/i-love-money/' ,
68 'id' : '5de6c582e9379ae4912dedbd' ,
69 'title' : 'I Love Money' ,
72 'url' : 'https://pluto.tv/on-demand/movies/arrival-2015-1-1' ,
73 'md5' : '3cead001d317a018bf856a896dee1762' ,
75 'id' : '5e83ac701fa6a9001bb9df24' ,
78 '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.' ,
82 'url' : 'https://pluto.tv/en/on-demand/series/manhunters-fugitive-task-force/seasons/1/episode/third-times-the-charm-1-1' ,
83 'only_matching' : True ,
85 'url' : 'https://pluto.tv/it/on-demand/series/csi-vegas/episode/legacy-2021-1-1' ,
86 'only_matching' : True ,
89 'url' : 'https://pluto.tv/en/on-demand/movies/attack-of-the-killer-tomatoes-1977-1-1-ptv1' ,
90 'md5' : '7db56369c0da626a32d505ec6eb3f89f' ,
92 'id' : '5b190c7bb0875c36c90c29c4' ,
94 'title' : 'Attack of the Killer Tomatoes' ,
95 'description' : 'A group of scientists band together to save the world from mutated tomatoes that KILL! (1978)' ,
101 def _to_ad_free_formats ( self
, video_id
, formats
, subtitles
):
102 ad_free_formats
, ad_free_subtitles
, m3u8_urls
= [], {}, set ()
104 res
= self
._ download
_ webpage
(
105 fmt
. get ( 'url' ), video_id
, note
= 'Downloading m3u8 playlist' ,
109 first_segment_url
= re
. search (
110 r
'^(https?://.*/)0\-(end|[0-9]+)/[^/]+\.ts$' , res
,
112 if first_segment_url
:
114 compat_urlparse
. urljoin ( first_segment_url
. group ( 1 ), '0-end/master.m3u8' ))
116 first_segment_url
= re
. search (
117 r
'^(https?://.*/).+\-0+[0-1]0\.ts$' , res
,
119 if first_segment_url
:
121 compat_urlparse
. urljoin ( first_segment_url
. group ( 1 ), 'master.m3u8' ))
124 for m3u8_url
in m3u8_urls
:
125 fmts
, subs
= self
._ extract
_ m
3u8_ formats
_ and
_ subtitles
(
126 m3u8_url
, video_id
, 'mp4' , 'm3u8_native' , m3u8_id
= 'hls' , fatal
= False )
127 ad_free_formats
. extend ( fmts
)
128 ad_free_subtitles
= self
._ merge
_ subtitles
( ad_free_subtitles
, subs
)
130 formats
, subtitles
= ad_free_formats
, ad_free_subtitles
132 self
. report_warning ( 'Unable to find ad-free formats' )
133 return formats
, subtitles
135 def _get_video_info ( self
, video_json
, slug
, series_name
= None ):
136 video_id
= video_json
. get ( '_id' , slug
)
137 formats
, subtitles
= [], {}
138 for video_url
in try_get ( video_json
, lambda x
: x
[ 'stitched' ][ 'urls' ], list ) or []:
139 if video_url
. get ( 'type' ) != 'hls' :
141 url
= url_or_none ( video_url
. get ( 'url' ))
143 fmts
, subs
= self
._ extract
_ m
3u8_ formats
_ and
_ subtitles
(
144 url
, video_id
, 'mp4' , 'm3u8_native' , m3u8_id
= 'hls' , fatal
= False )
146 subtitles
= self
._ merge
_ subtitles
( subtitles
, subs
)
148 formats
, subtitles
= self
._ to
_ ad
_ free
_ formats
( video_id
, formats
, subtitles
)
153 'subtitles' : subtitles
,
154 'title' : video_json
. get ( 'name' ),
155 'description' : video_json
. get ( 'description' ),
156 'duration' : float_or_none ( video_json
. get ( 'duration' ), scale
= 1000 ),
160 'series' : series_name
,
161 'episode' : video_json
. get ( 'name' ),
162 'season_number' : int_or_none ( video_json
. get ( 'season' )),
163 'episode_number' : int_or_none ( video_json
. get ( 'number' )),
167 def _real_extract ( self
, url
):
168 mobj
= self
._ match
_ valid
_u rl
( url
). groupdict ()
169 info_slug
= mobj
[ 'series_or_movie_slug' ]
170 video_json
= self
._ download
_ json
( self
._ INFO
_U RL
+ info_slug
, info_slug
, query
= self
._ INFO
_ QUERY
_ PARAMS
)
172 if mobj
[ 'video_type' ] == 'series' :
173 series_name
= video_json
. get ( 'name' , info_slug
)
174 season_number
, episode_slug
= mobj
. get ( 'season_number' ), mobj
. get ( 'episode_slug' )
177 for season
in video_json
[ 'seasons' ]:
178 if season_number
is not None and season_number
!= int_or_none ( season
. get ( 'number' )):
180 for episode
in season
[ 'episodes' ]:
181 if episode_slug
is not None and episode_slug
!= episode
. get ( 'slug' ):
183 videos
. append ( self
._ get
_ video
_ info
( episode
, episode_slug
, series_name
))
185 raise ExtractorError ( 'Failed to find any videos to extract' )
186 if episode_slug
is not None and len ( videos
) == 1 :
188 playlist_title
= series_name
189 if season_number
is not None :
190 playlist_title
+= ' - Season %d ' % season_number
191 return self
. playlist_result ( videos
,
192 playlist_id
= video_json
. get ( '_id' , info_slug
),
193 playlist_title
= playlist_title
)
194 return self
._ get
_ video
_ info
( video_json
, info_slug
)