9 from .common
import InfoExtractor
19 from ..utils
.traversal
import traverse_obj
22 class JioCinemaBaseIE(InfoExtractor
):
23 _NETRC_MACHINE
= 'jiocinema'
30 _API_HEADERS
= {'Origin': 'https://www.jiocinema.com', 'Referer': 'https://www.jiocinema.com/'}
31 _APP_NAME
= {'appName': 'RJIL_JioCinema'}
32 _APP_VERSION
= {'appVersion': '5.0.0'}
33 _API_SIGNATURES
= 'o668nxgzwff'
34 _METADATA_API_BASE
= 'https://content-jiovoot.voot.com/psapi'
35 _ACCESS_HINT
= 'the `accessToken` from your browser local storage'
37 'Log in with "-u phone -p <PHONE_NUMBER>" to authenticate with OTP, '
38 f
'or use "-u token -p <ACCESS_TOKEN>" to log in with {_ACCESS_HINT}. '
39 'If you have previously logged in with yt-dlp and your session '
40 'has been cached, you can use "-u device -p <DEVICE_ID>"')
42 def _cache_token(self
, token_type
):
43 assert token_type
in ('access', 'refresh', 'all')
44 if token_type
in ('access', 'all'):
46 JioCinemaBaseIE
._NETRC
_MACHINE
, f
'{JioCinemaBaseIE._DEVICE_ID}-access', JioCinemaBaseIE
._ACCESS
_TOKEN
)
47 if token_type
in ('refresh', 'all'):
49 JioCinemaBaseIE
._NETRC
_MACHINE
, f
'{JioCinemaBaseIE._DEVICE_ID}-refresh', JioCinemaBaseIE
._REFRESH
_TOKEN
)
51 def _call_api(self
, url
, video_id
, note
='Downloading API JSON', headers
={}, data
={}):
52 return self
._download
_json
(
53 url
, video_id
, note
, data
=json
.dumps(data
, separators
=(',', ':')).encode(), headers
={
54 'Content-Type': 'application/json',
55 'Accept': 'application/json',
58 }, expected_status
=(400, 403, 474))
60 def _call_auth_api(self
, service
, endpoint
, note
, headers
={}, data
={}):
61 return self
._call
_api
(
62 f
'https://auth-jiocinema.voot.com/{service}service/apis/v4/{endpoint}',
63 None, note
=note
, headers
=headers
, data
=data
)
65 def _refresh_token(self
):
66 if not JioCinemaBaseIE
._REFRESH
_TOKEN
or not JioCinemaBaseIE
._DEVICE
_ID
:
67 raise ExtractorError('User token has expired', expected
=True)
68 response
= self
._call
_auth
_api
(
69 'token', 'refreshtoken', 'Refreshing token',
70 headers
={'accesstoken': self
._ACCESS
_TOKEN
}, data
={
72 'deviceId': self
._DEVICE
_ID
,
73 'refreshToken': self
._REFRESH
_TOKEN
,
76 refresh_token
= response
.get('refreshTokenId')
77 if refresh_token
and refresh_token
!= JioCinemaBaseIE
._REFRESH
_TOKEN
:
78 JioCinemaBaseIE
._REFRESH
_TOKEN
= refresh_token
79 self
._cache
_token
('refresh')
80 JioCinemaBaseIE
._ACCESS
_TOKEN
= response
['authToken']
81 self
._cache
_token
('access')
83 def _fetch_guest_token(self
):
84 JioCinemaBaseIE
._DEVICE
_ID
= ''.join(random
.choices(string
.digits
, k
=10))
85 guest_token
= self
._call
_auth
_api
(
86 'token', 'guest', 'Downloading guest token', data
={
88 'deviceType': 'phone',
90 'deviceId': self
._DEVICE
_ID
,
92 'adId': self
._DEVICE
_ID
,
95 self
._GUEST
_TOKEN
= guest_token
['authToken']
96 self
._USER
_ID
= guest_token
['userId']
98 def _call_login_api(self
, endpoint
, guest_token
, data
, note
):
99 return self
._call
_auth
_api
(
100 'user', f
'loginotp/{endpoint}', note
, headers
={
101 **self
.geo_verification_headers(),
102 'accesstoken': self
._GUEST
_TOKEN
,
104 **traverse_obj(guest_token
, 'data', {
105 'deviceType': ('deviceType', {str}
),
109 def _is_token_expired(self
, token
):
110 return (try_call(lambda: jwt_decode_hs256(token
)['exp']) or 0) <= int(time
.time() - 180)
112 def _perform_login(self
, username
, password
):
113 if self
._ACCESS
_TOKEN
and not self
._is
_token
_expired
(self
._ACCESS
_TOKEN
):
116 UUID_RE
= r
'[\da-f]{8}-(?:[\da-f]{4}-){3}[\da-f]{12}'
118 if username
.lower() == 'token':
119 if try_call(lambda: jwt_decode_hs256(password
)):
120 JioCinemaBaseIE
._ACCESS
_TOKEN
= password
121 refresh_hint
= 'the `refreshToken` UUID from your browser local storage'
122 refresh_token
= self
._configuration
_arg
('refresh_token', [''], ie_key
=JioCinemaIE
)[0]
123 if not refresh_token
:
125 'To extend the life of your login session, in addition to your access token, '
126 'you can pass --extractor-args "jiocinema:refresh_token=REFRESH_TOKEN" '
127 f
'where REFRESH_TOKEN is {refresh_hint}')
128 elif re
.fullmatch(UUID_RE
, refresh_token
):
129 JioCinemaBaseIE
._REFRESH
_TOKEN
= refresh_token
131 self
.report_warning(f
'Invalid refresh_token value. Use {refresh_hint}')
133 raise ExtractorError(
134 f
'The password given could not be decoded as a token; use {self._ACCESS_HINT}', expected
=True)
136 elif username
.lower() == 'device' and re
.fullmatch(rf
'(?:{UUID_RE}|\d+)', password
):
137 JioCinemaBaseIE
._REFRESH
_TOKEN
= self
.cache
.load(JioCinemaBaseIE
._NETRC
_MACHINE
, f
'{password}-refresh')
138 JioCinemaBaseIE
._ACCESS
_TOKEN
= self
.cache
.load(JioCinemaBaseIE
._NETRC
_MACHINE
, f
'{password}-access')
139 if not JioCinemaBaseIE
._REFRESH
_TOKEN
or not JioCinemaBaseIE
._ACCESS
_TOKEN
:
140 raise ExtractorError(f
'Failed to load cached tokens for device ID "{password}"', expected
=True)
142 elif username
.lower() == 'phone' and re
.fullmatch(r
'\+?\d+', password
):
143 self
._fetch
_guest
_token
()
144 guest_token
= jwt_decode_hs256(self
._GUEST
_TOKEN
)
146 'number': base64
.b64encode(password
.encode()).decode(),
149 response
= self
._call
_login
_api
('send', guest_token
, initial_data
, 'Requesting OTP')
150 if not traverse_obj(response
, ('OTPInfo', {dict}
)):
151 raise ExtractorError('There was a problem with the phone number login attempt')
153 is_iphone
= guest_token
.get('os') == 'ios'
154 response
= self
._call
_login
_api
('verify', guest_token
, {
156 'consumptionDeviceName': 'iPhone' if is_iphone
else 'Android',
158 'platform': {'name': 'iPhone OS' if is_iphone
else 'Android'},
159 'androidId': self
._DEVICE
_ID
,
160 'type': 'iOS' if is_iphone
else 'Android',
164 'otp': self
._get
_tfa
_info
('the one-time password sent to your phone'),
166 if traverse_obj(response
, 'code') == 1043:
167 raise ExtractorError('Wrong OTP', expected
=True)
168 JioCinemaBaseIE
._REFRESH
_TOKEN
= response
['refreshToken']
169 JioCinemaBaseIE
._ACCESS
_TOKEN
= response
['authToken']
172 raise ExtractorError(self
._LOGIN
_HINT
, expected
=True)
174 user_token
= jwt_decode_hs256(JioCinemaBaseIE
._ACCESS
_TOKEN
)['data']
175 JioCinemaBaseIE
._USER
_ID
= user_token
['userId']
176 JioCinemaBaseIE
._DEVICE
_ID
= user_token
['deviceId']
177 if JioCinemaBaseIE
._REFRESH
_TOKEN
and username
!= 'device':
178 self
._cache
_token
('all')
179 if self
.get_param('cachedir') is not False:
181 f
'NOTE: For subsequent logins you can use "-u device -p {JioCinemaBaseIE._DEVICE_ID}"')
182 elif not JioCinemaBaseIE
._REFRESH
_TOKEN
:
183 JioCinemaBaseIE
._REFRESH
_TOKEN
= self
.cache
.load(
184 JioCinemaBaseIE
._NETRC
_MACHINE
, f
'{JioCinemaBaseIE._DEVICE_ID}-refresh')
185 if JioCinemaBaseIE
._REFRESH
_TOKEN
:
186 self
._cache
_token
('access')
187 self
.to_screen(f
'Logging in as device ID "{JioCinemaBaseIE._DEVICE_ID}"')
188 if self
._is
_token
_expired
(JioCinemaBaseIE
._ACCESS
_TOKEN
):
189 self
._refresh
_token
()
192 class JioCinemaIE(JioCinemaBaseIE
):
193 IE_NAME
= 'jiocinema'
194 _VALID_URL
= r
'https?://(?:www\.)?jiocinema\.com/?(?:movies?/[^/?#]+/|tv-shows/(?:[^/?#]+/){3})(?P<id>\d{3,})'
196 'url': 'https://www.jiocinema.com/tv-shows/agnisakshi-ek-samjhauta/1/pradeep-to-stop-the-wedding/3759931',
200 'title': 'Pradeep to stop the wedding?',
201 'description': 'md5:75f72d1d1a66976633345a3de6d672b1',
202 'episode': 'Pradeep to stop the wedding?',
203 'episode_number': 89,
204 'season': 'Agnisakshi…Ek Samjhauta-S1',
206 'series': 'Agnisakshi Ek Samjhauta',
208 'thumbnail': r
're:https?://.+\.jpg',
210 'season_id': '3698031',
211 'upload_date': '20230606',
212 'timestamp': 1686009600,
213 'release_date': '20230607',
216 'params': {'skip_download': 'm3u8'},
218 'url': 'https://www.jiocinema.com/movies/bhediya/3754021/watch',
223 'description': 'md5:a6bf2900371ac2fc3f1447401a9f7bb0',
224 'episode': 'Bhediya',
226 'thumbnail': r
're:https?://.+\.jpg',
228 'upload_date': '20230525',
229 'timestamp': 1685026200,
230 'release_date': '20230524',
231 'genres': ['Comedy'],
233 'params': {'skip_download': 'm3u8'},
236 def _extract_formats_and_subtitles(self
, playback
, video_id
):
237 m3u8_url
= traverse_obj(playback
, (
238 'data', 'playbackUrls', lambda _
, v
: v
['streamtype'] == 'hls', 'url', {url_or_none}
, any
))
239 if not m3u8_url
: # DRM-only content only serves dash urls
240 self
.report_drm(video_id
)
241 formats
, subtitles
= self
._extract
_m
3u8_formats
_and
_subtitles
(m3u8_url
, video_id
, m3u8_id
='hls')
242 self
._remove
_duplicate
_formats
(formats
)
245 # '/_definst_/smil:vod/' m3u8 manifests claim to have 720p+ formats but max out at 480p
246 'formats': traverse_obj(formats
, (
247 lambda _
, v
: '/_definst_/smil:vod/' not in v
['url'] or v
['height'] <= 480)),
248 'subtitles': subtitles
,
251 def _real_extract(self
, url
):
252 video_id
= self
._match
_id
(url
)
253 if not self
._ACCESS
_TOKEN
and self
._is
_token
_expired
(self
._GUEST
_TOKEN
):
254 self
._fetch
_guest
_token
()
255 elif self
._ACCESS
_TOKEN
and self
._is
_token
_expired
(self
._ACCESS
_TOKEN
):
256 self
._refresh
_token
()
258 playback
= self
._call
_api
(
259 f
'https://apis-jiovoot.voot.com/playbackjv/v3/{video_id}', video_id
,
260 'Downloading playback JSON', headers
={
261 **self
.geo_verification_headers(),
262 'accesstoken': self
._ACCESS
_TOKEN
or self
._GUEST
_TOKEN
,
264 'deviceid': self
._DEVICE
_ID
,
265 'uniqueid': self
._USER
_ID
,
266 'x-apisignatures': self
._API
_SIGNATURES
,
267 'x-platform': 'androidweb',
268 'x-platform-token': 'web',
272 'appVersion': '3.4.0',
273 'bitrateProfile': 'xhdpi',
277 'fairPlayDrmSupport': 'none',
278 'playreadyDrmSupport': 'none',
279 'widevineDRMSupport': 'none',
281 'frameRateCapability': [{
282 'frameRateSupport': '30fps',
283 'videoQuality': '1440p',
286 'continueWatchingRequired': False,
288 'downloadRequest': False,
291 'manufacturer': 'Windows',
293 'multiAudioRequired': True,
295 'parentalPinValid': True,
296 'x-apisignatures': self
._API
_SIGNATURES
,
299 status_code
= traverse_obj(playback
, ('code', {int}
))
300 if status_code
== 474:
301 self
.raise_geo_restricted(countries
=['IN'])
302 elif status_code
== 1008:
303 error_msg
= 'This content is only available for premium users'
304 if self
._ACCESS
_TOKEN
:
305 raise ExtractorError(error_msg
, expected
=True)
306 self
.raise_login_required(f
'{error_msg}. {self._LOGIN_HINT}', method
=None)
307 elif status_code
== 400:
308 raise ExtractorError('The requested content is not available', expected
=True)
309 elif status_code
is not None and status_code
!= 200:
310 raise ExtractorError(
311 f
'JioCinema says: {traverse_obj(playback, ("message", {str})) or status_code}')
313 metadata
= self
._download
_json
(
314 f
'{self._METADATA_API_BASE}/voot/v1/voot-web/content/query/asset-details',
315 video_id
, fatal
=False, query
={
316 'ids': f
'include:{video_id}',
317 'responseType': 'common',
318 'devicePlatformType': 'desktop',
323 'http_headers': self
._API
_HEADERS
,
324 **self
._extract
_formats
_and
_subtitles
(playback
, video_id
),
325 **traverse_obj(playback
, ('data', {
327 'title': ('name', {str}
),
328 'description': ('fullSynopsis', {str}
),
329 'series': ('show', 'name', {str}
, {lambda x
: x
or None}),
330 'season': ('tournamentName', {str}
, {lambda x
: x
if x
!= 'Season 0' else None}),
331 'season_number': ('episode', 'season', {int_or_none}
, {lambda x
: x
or None}),
332 'episode': ('fullTitle', {str}
),
333 'episode_number': ('episode', 'episodeNo', {int_or_none}
, {lambda x
: x
or None}),
334 'age_limit': ('ageNemonic', {parse_age_limit}
),
335 'duration': ('totalDuration', {float_or_none}
),
336 'thumbnail': ('images', {url_or_none}
),
338 **traverse_obj(metadata
, ('result', 0, {
339 'title': ('fullTitle', {str}
),
340 'description': ('fullSynopsis', {str}
),
341 'series': ('showName', {str}
, {lambda x
: x
or None}),
342 'season': ('seasonName', {str}
, {lambda x
: x
or None}),
343 'season_number': ('season', {int_or_none}
),
344 'season_id': ('seasonId', {str}
, {lambda x
: x
or None}),
345 'episode': ('fullTitle', {str}
),
346 'episode_number': ('episode', {int_or_none}
),
347 'timestamp': ('uploadTime', {int_or_none}
),
348 'release_date': ('telecastDate', {str}
),
349 'age_limit': ('ageNemonic', {parse_age_limit}
),
350 'duration': ('duration', {float_or_none}
),
351 'genres': ('genres', ..., {str}
),
352 'thumbnail': ('seo', 'ogImage', {url_or_none}
),
357 class JioCinemaSeriesIE(JioCinemaBaseIE
):
358 IE_NAME
= 'jiocinema:series'
359 _VALID_URL
= r
'https?://(?:www\.)?jiocinema\.com/tv-shows/(?P<slug>[\w-]+)/(?P<id>\d{3,})'
361 'url': 'https://www.jiocinema.com/tv-shows/naagin/3499917',
366 'playlist_mincount': 120,
368 'url': 'https://www.jiocinema.com/tv-shows/mtv-splitsvilla-x5/3499820',
371 'title': 'mtv-splitsvilla-x5',
373 'playlist_mincount': 310,
376 def _entries(self
, series_id
):
377 seasons
= traverse_obj(self
._download
_json
(
378 f
'{self._METADATA_API_BASE}/voot/v1/voot-web/view/show/{series_id}', series_id
,
379 'Downloading series metadata JSON', query
={'responseType': 'common'}), (
380 'trays', lambda _
, v
: v
['trayId'] == 'season-by-show-multifilter',
381 'trayTabs', lambda _
, v
: v
['id']))
383 for season_num
, season
in enumerate(seasons
, start
=1):
384 season_id
= season
['id']
385 label
= season
.get('label') or season_num
386 for page_num
in itertools
.count(1):
387 episodes
= traverse_obj(self
._download
_json
(
388 f
'{self._METADATA_API_BASE}/voot/v1/voot-web/content/generic/series-wise-episode',
389 season_id
, f
'Downloading season {label} page {page_num} JSON', query
={
390 'sort': 'episode:asc',
392 'responseType': 'common',
394 }), ('result', lambda _
, v
: v
['id'] and url_or_none(v
['slug'])))
397 for episode
in episodes
:
398 yield self
.url_result(
399 episode
['slug'], JioCinemaIE
, **traverse_obj(episode
, {
401 'video_title': ('fullTitle', {str}
),
402 'season_number': ('season', {int_or_none}
),
403 'episode_number': ('episode', {int_or_none}
),
406 def _real_extract(self
, url
):
407 slug
, series_id
= self
._match
_valid
_url
(url
).group('slug', 'id')
408 return self
.playlist_result(self
._entries
(series_id
), series_id
, slug
)