]>
Commit | Line | Data |
---|---|---|
8428fdcc S |
1 | # coding: utf-8 |
2 | from __future__ import unicode_literals | |
3 | ||
dc9de9cb A |
4 | import itertools |
5 | import re | |
6 | ||
8428fdcc S |
7 | from .common import InfoExtractor |
8 | from ..utils import ( | |
33b529fa | 9 | determine_ext, |
a109acbf | 10 | extract_attributes, |
8428fdcc | 11 | int_or_none, |
29f7c58a | 12 | try_get, |
8428fdcc | 13 | url_or_none, |
e26f9cc1 | 14 | lowercase_escape, |
8428fdcc S |
15 | ) |
16 | ||
17 | ||
18 | class YandexVideoIE(InfoExtractor): | |
19 | _VALID_URL = r'''(?x) | |
20 | https?:// | |
21 | (?: | |
29f7c58a | 22 | yandex\.ru(?:/(?:portal/(?:video|efir)|efir))?/?\?.*?stream_id=| |
8428fdcc S |
23 | frontend\.vh\.yandex\.ru/player/ |
24 | ) | |
29f7c58a | 25 | (?P<id>(?:[\da-f]{32}|[\w-]{12})) |
8428fdcc S |
26 | ''' |
27 | _TESTS = [{ | |
29f7c58a | 28 | 'url': 'https://yandex.ru/portal/video?stream_id=4dbb36ec4e0526d58f9f2dc8f0ecf374', |
29 | 'md5': 'e02a05bfaf0d9615ef07ae3a10f4faf4', | |
8428fdcc | 30 | 'info_dict': { |
29f7c58a | 31 | 'id': '4dbb36ec4e0526d58f9f2dc8f0ecf374', |
8428fdcc | 32 | 'ext': 'mp4', |
29f7c58a | 33 | 'title': 'Русский Вудсток - главный рок-фест в истории СССР / вДудь', |
34 | 'description': 'md5:7d6b8d4bc4a3b9a56499916c1ea5b5fa', | |
35 | 'thumbnail': r're:^https?://', | |
36 | 'timestamp': 1549972939, | |
37 | 'duration': 5575, | |
8428fdcc | 38 | 'age_limit': 18, |
29f7c58a | 39 | 'upload_date': '20190212', |
40 | 'view_count': int, | |
41 | 'like_count': int, | |
42 | 'dislike_count': int, | |
8428fdcc S |
43 | }, |
44 | }, { | |
29f7c58a | 45 | 'url': 'https://yandex.ru/portal/efir?stream_id=4dbb262b4fe5cf15a215de4f34eee34d&from=morda', |
8428fdcc S |
46 | 'only_matching': True, |
47 | }, { | |
48 | 'url': 'https://yandex.ru/?stream_id=4dbb262b4fe5cf15a215de4f34eee34d', | |
49 | 'only_matching': True, | |
50 | }, { | |
51 | 'url': 'https://frontend.vh.yandex.ru/player/4dbb262b4fe5cf15a215de4f34eee34d?from=morda', | |
52 | 'only_matching': True, | |
53 | }, { | |
54 | # vod-episode, series episode | |
55 | 'url': 'https://yandex.ru/portal/video?stream_id=45b11db6e4b68797919c93751a938cee', | |
56 | 'only_matching': True, | |
57 | }, { | |
58 | # episode, sports | |
59 | 'url': 'https://yandex.ru/?stream_channel=1538487871&stream_id=4132a07f71fb0396be93d74b3477131d', | |
60 | 'only_matching': True, | |
33b529fa S |
61 | }, { |
62 | # DASH with DRM | |
63 | 'url': 'https://yandex.ru/portal/video?from=morda&stream_id=485a92d94518d73a9d0ff778e13505f8', | |
64 | 'only_matching': True, | |
29f7c58a | 65 | }, { |
66 | 'url': 'https://yandex.ru/efir?stream_active=watching&stream_id=v7a2dZ-v5mSI&from_block=efir_newtab', | |
67 | 'only_matching': True, | |
8428fdcc S |
68 | }] |
69 | ||
70 | def _real_extract(self, url): | |
71 | video_id = self._match_id(url) | |
72 | ||
29f7c58a | 73 | player = try_get((self._download_json( |
74 | 'https://frontend.vh.yandex.ru/graphql', video_id, data=('''{ | |
75 | player(content_id: "%s") { | |
76 | computed_title | |
77 | content_url | |
78 | description | |
79 | dislikes | |
80 | duration | |
81 | likes | |
82 | program_title | |
83 | release_date | |
84 | release_date_ut | |
85 | release_year | |
86 | restriction_age | |
87 | season | |
88 | start_time | |
89 | streams | |
90 | thumbnail | |
91 | title | |
92 | views_count | |
93 | } | |
94 | }''' % video_id).encode(), fatal=False)), lambda x: x['player']['content']) | |
95 | if not player or player.get('error'): | |
96 | player = self._download_json( | |
97 | 'https://frontend.vh.yandex.ru/v23/player/%s.json' % video_id, | |
98 | video_id, query={ | |
99 | 'stream_options': 'hires', | |
100 | 'disable_trackings': 1, | |
101 | }) | |
102 | content = player['content'] | |
8428fdcc | 103 | |
29f7c58a | 104 | title = content.get('title') or content['computed_title'] |
33b529fa | 105 | |
29f7c58a | 106 | formats = [] |
107 | streams = content.get('streams') or [] | |
108 | streams.append({'url': content.get('content_url')}) | |
109 | for stream in streams: | |
110 | content_url = url_or_none(stream.get('url')) | |
111 | if not content_url: | |
112 | continue | |
113 | ext = determine_ext(content_url) | |
114 | if ext == 'ismc': | |
115 | continue | |
116 | elif ext == 'm3u8': | |
117 | formats.extend(self._extract_m3u8_formats( | |
118 | content_url, video_id, 'mp4', | |
119 | 'm3u8_native', m3u8_id='hls', fatal=False)) | |
120 | elif ext == 'mpd': | |
121 | formats.extend(self._extract_mpd_formats( | |
122 | content_url, video_id, mpd_id='dash', fatal=False)) | |
123 | else: | |
124 | formats.append({'url': content_url}) | |
33b529fa | 125 | |
8428fdcc S |
126 | self._sort_formats(formats) |
127 | ||
3089bc74 S |
128 | timestamp = (int_or_none(content.get('release_date')) |
129 | or int_or_none(content.get('release_date_ut')) | |
130 | or int_or_none(content.get('start_time'))) | |
29f7c58a | 131 | season = content.get('season') or {} |
8428fdcc S |
132 | |
133 | return { | |
134 | 'id': video_id, | |
135 | 'title': title, | |
29f7c58a | 136 | 'description': content.get('description'), |
137 | 'thumbnail': content.get('thumbnail'), | |
8428fdcc | 138 | 'timestamp': timestamp, |
29f7c58a | 139 | 'duration': int_or_none(content.get('duration')), |
140 | 'series': content.get('program_title'), | |
141 | 'age_limit': int_or_none(content.get('restriction_age')), | |
142 | 'view_count': int_or_none(content.get('views_count')), | |
143 | 'like_count': int_or_none(content.get('likes')), | |
144 | 'dislike_count': int_or_none(content.get('dislikes')), | |
145 | 'season_number': int_or_none(season.get('season_number')), | |
146 | 'season_id': season.get('id'), | |
147 | 'release_year': int_or_none(content.get('release_year')), | |
8428fdcc S |
148 | 'formats': formats, |
149 | } | |
dc9de9cb A |
150 | |
151 | ||
e26f9cc1 K |
152 | class YandexVideoPreviewIE(InfoExtractor): |
153 | _VALID_URL = r'https?://(?:www\.)?yandex\.ru/video/preview(?:/?\?.*?filmId=|/)(?P<id>\d+)' | |
154 | _TESTS = [{ # Odnoklassniki | |
155 | 'url': 'https://yandex.ru/video/preview/?filmId=10682852472978372885&text=summer', | |
156 | 'info_dict': { | |
157 | 'id': '1352565459459', | |
158 | 'ext': 'mp4', | |
159 | 'like_count': int, | |
160 | 'upload_date': '20191202', | |
161 | 'age_limit': 0, | |
162 | 'duration': 196, | |
163 | 'thumbnail': 'https://i.mycdn.me/videoPreview?id=544866765315&type=37&idx=13&tkn=TY5qjLYZHxpmcnK8U2LgzYkgmaU&fn=external_8', | |
164 | 'uploader_id': '481054701571', | |
165 | 'title': 'LOFT - summer, summer, summer HD', | |
e26f9cc1 K |
166 | 'uploader': 'АРТЁМ КУДРОВ', |
167 | }, | |
168 | }, { # youtube | |
169 | 'url': 'https://yandex.ru/video/preview/?filmId=4479424425337895262&source=main_redirect&text=видео&utm_source=main_stripe_big', | |
170 | 'only_matching': True, | |
171 | }, { # YandexVideo | |
172 | 'url': 'https://yandex.ru/video/preview/5275069442094787341', | |
173 | 'only_matching': True, | |
174 | }, { # youtube | |
175 | 'url': 'https://yandex.ru/video/preview/?filmId=16658118429797832897&from=tabbar&p=1&text=%D0%BF%D1%80%D0%BE%D1%81%D0%BC%D0%BE%D1%82%D1%80+%D1%84%D1%80%D0%B0%D0%B3%D0%BC%D0%B5%D0%BD%D1%82%D0%B0+%D0%BC%D0%B0%D0%BB%D0%B5%D0%BD%D1%8C%D0%BA%D0%B8%D0%B9+%D0%BF%D1%80%D0%B8%D0%BD%D1%86+%D0%BC%D1%8B+%D0%B2+%D0%BE%D1%82%D0%B2%D0%B5%D1%82%D0%B5+%D0%B7%D0%B0+%D1%82%D0%B5%D1%85+%D0%BA%D0%BE%D0%B3%D0%BE+%D0%BF%D1%80%D0%B8%D1%80%D1%83%D1%87%D0%B8%D0%BB%D0%B8', | |
176 | 'only_matching': True, | |
177 | }, { # Odnoklassniki | |
178 | 'url': 'https://yandex.ru/video/preview/?text=Francis%20Lai%20-%20Le%20Bon%20Et%20Les%20MC)chants&path=wizard&parent-reqid=1643208087979310-1481782809207673478-sas3-0931-2f9-sas-l7-balancer-8080-BAL-9380&wiz_type=vital&filmId=12508152936505397283', | |
179 | 'only_matching': True, | |
180 | }] | |
181 | ||
182 | def _real_extract(self, url): | |
183 | id = self._match_id(url) | |
184 | webpage = self._download_webpage(url, id) | |
185 | data_raw = self._search_regex(r'window.Ya.__inline_params__\s*=\s*JSON.parse\(\'([^"]+?\\u0022video\\u0022:[^"]+?})\'\);', webpage, 'data_raw') | |
186 | data_json = self._parse_json(data_raw, id, transform_source=lowercase_escape) | |
187 | return self.url_result(data_json['video']['url']) | |
188 | ||
189 | ||
dc9de9cb | 190 | class ZenYandexIE(InfoExtractor): |
a109acbf | 191 | _VALID_URL = r'https?://zen\.yandex\.ru(?:/video)?/(media|watch)/(?:(?:id/[^/]+/|[^/]+/)(?:[a-z0-9-]+)-)?(?P<id>[a-z0-9-]+)' |
dc9de9cb A |
192 | _TESTS = [{ |
193 | 'url': 'https://zen.yandex.ru/media/popmech/izverjenie-vulkana-iz-spichek-zreliscnyi-opyt-6002240ff8b1af50bb2da5e3', | |
194 | 'info_dict': { | |
195 | 'id': '6002240ff8b1af50bb2da5e3', | |
196 | 'ext': 'mp4', | |
197 | 'title': 'Извержение вулкана из спичек: зрелищный опыт', | |
198 | 'description': 'md5:053ad3c61b5596d510c9a199dc8ee633', | |
a109acbf | 199 | 'thumbnail': 're:^https://avatars.mds.yandex.net/', |
dc9de9cb A |
200 | 'uploader': 'Популярная механика', |
201 | }, | |
a109acbf | 202 | 'params': { |
203 | 'skip_download': 'm3u8', | |
204 | }, | |
dc9de9cb A |
205 | }, { |
206 | 'url': 'https://zen.yandex.ru/media/id/606fd806cc13cb3c58c05cf5/vot-eto-focus-dedy-morozy-na-gidrociklah-60c7c443da18892ebfe85ed7', | |
207 | 'info_dict': { | |
208 | 'id': '60c7c443da18892ebfe85ed7', | |
209 | 'ext': 'mp4', | |
210 | 'title': 'ВОТ ЭТО Focus. Деды Морозы на гидроциклах', | |
a109acbf | 211 | 'description': 'md5:f3db3d995763b9bbb7b56d4ccdedea89', |
212 | 'thumbnail': 're:^https://avatars.mds.yandex.net/', | |
dc9de9cb A |
213 | 'uploader': 'AcademeG DailyStream' |
214 | }, | |
a109acbf | 215 | 'params': { |
216 | 'skip_download': 'm3u8', | |
217 | 'format': 'bestvideo', | |
218 | }, | |
219 | }, { | |
220 | 'url': 'https://zen.yandex.ru/video/watch/6002240ff8b1af50bb2da5e3', | |
221 | 'info_dict': { | |
222 | 'id': '6002240ff8b1af50bb2da5e3', | |
223 | 'ext': 'mp4', | |
224 | 'title': 'Извержение вулкана из спичек: зрелищный опыт', | |
225 | 'description': 'md5:053ad3c61b5596d510c9a199dc8ee633', | |
226 | 'uploader': 'Популярная механика', | |
227 | }, | |
228 | 'params': { | |
229 | 'skip_download': 'm3u8', | |
230 | }, | |
dc9de9cb A |
231 | }, { |
232 | 'url': 'https://zen.yandex.ru/media/id/606fd806cc13cb3c58c05cf5/novyi-samsung-fold-3-moskvich-barahlit-612f93b7f8d48e7e945792a2?from=channel&rid=2286618386.482.1630817595976.42360', | |
233 | 'only_matching': True, | |
234 | }] | |
235 | ||
236 | def _real_extract(self, url): | |
237 | id = self._match_id(url) | |
238 | webpage = self._download_webpage(url, id) | |
a109acbf | 239 | data_json = self._parse_json( |
240 | self._search_regex(r'data\s*=\s*({["\']_*serverState_*video.+?});', webpage, 'metadata'), id) | |
241 | serverstate = self._search_regex(r'(_+serverState_+video-site_[^_]+_+)', | |
242 | webpage, 'server state').replace('State', 'Settings') | |
243 | uploader = self._search_regex(r'(<a\s*class=["\']card-channel-link[^"\']+["\'][^>]+>)', | |
244 | webpage, 'uploader', default='<a>') | |
245 | uploader_name = extract_attributes(uploader).get('aria-label') | |
246 | video_json = try_get(data_json, lambda x: x[serverstate]['exportData']['video'], dict) | |
247 | stream_urls = try_get(video_json, lambda x: x['video']['streams']) | |
248 | formats = [] | |
249 | for s_url in stream_urls: | |
250 | ext = determine_ext(s_url) | |
251 | if ext == 'mpd': | |
252 | formats.extend(self._extract_mpd_formats(s_url, id, mpd_id='dash')) | |
253 | elif ext == 'm3u8': | |
254 | formats.extend(self._extract_m3u8_formats(s_url, id, 'mp4')) | |
dc9de9cb A |
255 | self._sort_formats(formats) |
256 | return { | |
257 | 'id': id, | |
a109acbf | 258 | 'title': video_json.get('title') or self._og_search_title(webpage), |
dc9de9cb | 259 | 'formats': formats, |
a109acbf | 260 | 'duration': int_or_none(video_json.get('duration')), |
261 | 'view_count': int_or_none(video_json.get('views')), | |
262 | 'uploader': uploader_name or data_json.get('authorName') or try_get(data_json, lambda x: x['publisher']['name']), | |
263 | 'description': self._og_search_description(webpage) or try_get(data_json, lambda x: x['og']['description']), | |
264 | 'thumbnail': self._og_search_thumbnail(webpage) or try_get(data_json, lambda x: x['og']['imageUrl']), | |
dc9de9cb A |
265 | } |
266 | ||
267 | ||
268 | class ZenYandexChannelIE(InfoExtractor): | |
a109acbf | 269 | _VALID_URL = r'https?://zen\.yandex\.ru/(?!media|video)(?:id/)?(?P<id>[a-z0-9-_]+)' |
dc9de9cb A |
270 | _TESTS = [{ |
271 | 'url': 'https://zen.yandex.ru/tok_media', | |
272 | 'info_dict': { | |
273 | 'id': 'tok_media', | |
274 | }, | |
275 | 'playlist_mincount': 169, | |
276 | }, { | |
277 | 'url': 'https://zen.yandex.ru/id/606fd806cc13cb3c58c05cf5', | |
278 | 'info_dict': { | |
279 | 'id': '606fd806cc13cb3c58c05cf5', | |
280 | }, | |
281 | 'playlist_mincount': 657, | |
282 | }] | |
283 | ||
284 | def _entries(self, id, url): | |
285 | webpage = self._download_webpage(url, id) | |
286 | data_json = self._parse_json(re.findall(r'var\s?data\s?=\s?({.+?})\s?;', webpage)[-1], id) | |
287 | for key in data_json.keys(): | |
288 | if key.startswith('__serverState__'): | |
289 | data_json = data_json[key] | |
290 | items = list(try_get(data_json, lambda x: x['feed']['items'], dict).values()) | |
291 | more = try_get(data_json, lambda x: x['links']['more']) or None | |
292 | for page in itertools.count(1): | |
293 | for item in items: | |
294 | video_id = item.get('publication_id') or item.get('publicationId') | |
295 | video_url = item.get('link') | |
296 | yield self.url_result(video_url, ie=ZenYandexIE.ie_key(), video_id=video_id.split(':')[-1]) | |
297 | if not more: | |
298 | break | |
299 | data_json = self._download_json(more, id, note='Downloading Page %d' % page) | |
300 | items = data_json.get('items', []) | |
301 | more = try_get(data_json, lambda x: x['more']['link']) or None | |
302 | ||
303 | def _real_extract(self, url): | |
304 | id = self._match_id(url) | |
305 | return self.playlist_result(self._entries(id, url), playlist_id=id) |