3 from .common
import InfoExtractor
4 from ..utils
import ExtractorError
, int_or_none
, traverse_obj
, try_get
7 class AmazonMiniTVBaseIE(InfoExtractor
):
8 def _real_initialize(self
):
9 self
._download
_webpage
(
10 'https://www.amazon.in/minitv', None,
11 note
='Fetching guest session cookies')
12 AmazonMiniTVBaseIE
.session_id
= self
._get
_cookies
('https://www.amazon.in')['session-id'].value
14 def _call_api(self
, asin
, data
=None, note
=None):
15 device
= {'clientId': 'ATVIN', 'deviceLocale': 'en_GB'}
17 data
['variables'].update({
19 'sessionIdToken': self
.session_id
,
23 resp
= self
._download
_json
(
24 f
'https://www.amazon.in/minitv/api/web/{"graphql" if data else "prs"}',
25 asin
, note
=note
, headers
={'Content-Type': 'application/json'}
,
26 data
=json
.dumps(data
).encode() if data
else None,
27 query
=None if data
else {
28 'deviceType': 'A1WMMUXPCUJL4N',
33 if resp
.get('errors'):
34 raise ExtractorError(f
'MiniTV said: {resp["errors"][0]["message"]}')
37 return resp
['data'][data
['operationName']]
40 class AmazonMiniTVIE(AmazonMiniTVBaseIE
):
41 _VALID_URL
= r
'(?:https?://(?:www\.)?amazon\.in/minitv/tp/|amazonminitv:(?:amzn1\.dv\.gti\.)?)(?P<id>[a-f0-9-]+)'
43 'url': 'https://www.amazon.in/minitv/tp/75fe3a75-b8fe-4499-8100-5c9424344840?referrer=https%3A%2F%2Fwww.amazon.in%2Fminitv',
45 'id': 'amzn1.dv.gti.75fe3a75-b8fe-4499-8100-5c9424344840',
47 'title': 'May I Kiss You?',
49 'thumbnail': r
're:^https?://.*\.jpg$',
50 'description': 'md5:a549bfc747973e04feb707833474e59d',
51 'release_timestamp': 1644710400,
52 'release_date': '20220213',
54 'chapters': 'count:2',
55 'series': 'Couple Goals',
56 'series_id': 'amzn1.dv.gti.56521d46-b040-4fd5-872e-3e70476a04b0',
59 'season_id': 'amzn1.dv.gti.20331016-d9b9-4968-b991-c89fa4927a36',
60 'episode': 'May I Kiss You?',
62 'episode_id': 'amzn1.dv.gti.75fe3a75-b8fe-4499-8100-5c9424344840',
65 'url': 'https://www.amazon.in/minitv/tp/280d2564-584f-452f-9c98-7baf906e01ab?referrer=https%3A%2F%2Fwww.amazon.in%2Fminitv',
67 'id': 'amzn1.dv.gti.280d2564-584f-452f-9c98-7baf906e01ab',
71 'thumbnail': r
're:^https?://.*\.jpg',
72 'description': 'md5:05eb765a77bf703f322f120ec6867339',
73 'release_timestamp': 1647475200,
74 'release_date': '20220317',
79 'url': 'https://www.amazon.in/minitv/tp/280d2564-584f-452f-9c98-7baf906e01ab',
80 'only_matching': True,
82 'url': 'amazonminitv:amzn1.dv.gti.280d2564-584f-452f-9c98-7baf906e01ab',
83 'only_matching': True,
85 'url': 'amazonminitv:280d2564-584f-452f-9c98-7baf906e01ab',
86 'only_matching': True,
89 _GRAPHQL_QUERY_CONTENT
= '''
90 query content($sessionIdToken: String!, $deviceLocale: String, $contentId: ID!, $contentType: ContentType!, $clientId: String) {
92 applicationContextInput: {deviceLocale: $deviceLocale, sessionIdToken: $sessionIdToken, clientId: $clientId}
94 contentType: $contentType
105 contentLengthInSeconds
118 ... on MovieContent {
124 contentLengthInSeconds
133 def _real_extract(self
, url
):
134 asin
= f
'amzn1.dv.gti.{self._match_id(url)}'
135 prs
= self
._call
_api
(asin
, note
='Downloading playback info')
137 formats
, subtitles
= [], {}
138 for type_
, asset
in prs
['playbackAssets'].items():
139 if not traverse_obj(asset
, 'manifestUrl'):
142 m3u8_fmts
, m3u8_subs
= self
._extract
_m
3u8_formats
_and
_subtitles
(
143 asset
['manifestUrl'], asin
, ext
='mp4', entry_protocol
='m3u8_native',
144 m3u8_id
=type_
, fatal
=False)
145 formats
.extend(m3u8_fmts
)
146 subtitles
= self
._merge
_subtitles
(subtitles
, m3u8_subs
)
147 elif type_
== 'dash':
148 mpd_fmts
, mpd_subs
= self
._extract
_mpd
_formats
_and
_subtitles
(
149 asset
['manifestUrl'], asin
, mpd_id
=type_
, fatal
=False)
150 formats
.extend(mpd_fmts
)
151 subtitles
= self
._merge
_subtitles
(subtitles
, mpd_subs
)
153 self
.report_warning(f
'Unknown asset type: {type_}')
155 title_info
= self
._call
_api
(
156 asin
, note
='Downloading title info', data
={
157 'operationName': 'content',
158 'variables': {'contentId': asin}
,
159 'query': self
._GRAPHQL
_QUERY
_CONTENT
,
161 credits_time
= try_get(title_info
, lambda x
: x
['timecode']['endCreditsTime'] / 1000)
162 is_episode
= title_info
.get('vodType') == 'EPISODE'
166 'title': title_info
.get('name'),
168 'subtitles': subtitles
,
169 'language': traverse_obj(title_info
, ('audioTracks', 0)),
173 } for type_
, url
in (title_info
.get('images') or {}).items()],
174 'description': traverse_obj(title_info
, ('description', 'synopsis')),
175 'release_timestamp': int_or_none(try_get(title_info
, lambda x
: x
['publicReleaseDateUTC'] / 1000)),
176 'duration': traverse_obj(title_info
, ('description', 'contentLengthInSeconds')),
178 'start_time': credits_time
,
179 'title': 'End Credits',
180 }] if credits_time
else [],
181 'series': title_info
.get('seriesName'),
182 'series_id': title_info
.get('seriesId'),
183 'season_number': title_info
.get('seasonNumber'),
184 'season_id': title_info
.get('seasonId'),
185 'episode': title_info
.get('name') if is_episode
else None,
186 'episode_number': title_info
.get('episodeNumber'),
187 'episode_id': asin
if is_episode
else None,
191 class AmazonMiniTVSeasonIE(AmazonMiniTVBaseIE
):
192 IE_NAME
= 'amazonminitv:season'
193 _VALID_URL
= r
'amazonminitv:season:(?:amzn1\.dv\.gti\.)?(?P<id>[a-f0-9-]+)'
194 IE_DESC
= 'Amazon MiniTV Series, "minitv:season:" prefix'
196 'url': 'amazonminitv:season:amzn1.dv.gti.0aa996eb-6a1b-4886-a342-387fbd2f1db0',
197 'playlist_mincount': 6,
199 'id': 'amzn1.dv.gti.0aa996eb-6a1b-4886-a342-387fbd2f1db0',
202 'url': 'amazonminitv:season:0aa996eb-6a1b-4886-a342-387fbd2f1db0',
203 'only_matching': True,
207 query getEpisodes($sessionIdToken: String!, $clientId: String, $episodeOrSeasonId: ID!, $deviceLocale: String) {
209 applicationContextInput: {sessionIdToken: $sessionIdToken, deviceLocale: $deviceLocale, clientId: $clientId}
210 episodeOrSeasonId: $episodeOrSeasonId
224 contentLengthInSeconds
233 def _entries(self
, asin
):
234 season_info
= self
._call
_api
(
235 asin
, note
='Downloading season info', data
={
236 'operationName': 'getEpisodes',
237 'variables': {'episodeOrSeasonId': asin}
,
238 'query': self
._GRAPHQL
_QUERY
,
241 for episode
in season_info
['episodes']:
242 yield self
.url_result(
243 f
'amazonminitv:{episode["contentId"]}', AmazonMiniTVIE
, episode
['contentId'])
245 def _real_extract(self
, url
):
246 asin
= f
'amzn1.dv.gti.{self._match_id(url)}'
247 return self
.playlist_result(self
._entries
(asin
), asin
)
250 class AmazonMiniTVSeriesIE(AmazonMiniTVBaseIE
):
251 IE_NAME
= 'amazonminitv:series'
252 _VALID_URL
= r
'amazonminitv:series:(?:amzn1\.dv\.gti\.)?(?P<id>[a-f0-9-]+)'
254 'url': 'amazonminitv:series:amzn1.dv.gti.56521d46-b040-4fd5-872e-3e70476a04b0',
255 'playlist_mincount': 3,
257 'id': 'amzn1.dv.gti.56521d46-b040-4fd5-872e-3e70476a04b0',
260 'url': 'amazonminitv:series:56521d46-b040-4fd5-872e-3e70476a04b0',
261 'only_matching': True,
265 query getSeasons($sessionIdToken: String!, $deviceLocale: String, $episodeOrSeasonOrSeriesId: ID!, $clientId: String) {
267 applicationContextInput: {deviceLocale: $deviceLocale, sessionIdToken: $sessionIdToken, clientId: $clientId}
268 episodeOrSeasonOrSeriesId: $episodeOrSeasonOrSeriesId
277 def _entries(self
, asin
):
278 season_info
= self
._call
_api
(
279 asin
, note
='Downloading series info', data
={
280 'operationName': 'getSeasons',
281 'variables': {'episodeOrSeasonOrSeriesId': asin}
,
282 'query': self
._GRAPHQL
_QUERY
,
285 for season
in season_info
['seasons']:
286 yield self
.url_result(f
'amazonminitv:season:{season["seasonId"]}', AmazonMiniTVSeasonIE
, season
['seasonId'])
288 def _real_extract(self
, url
):
289 asin
= f
'amzn1.dv.gti.{self._match_id(url)}'
290 return self
.playlist_result(self
._entries
(asin
), asin
)