]> jfr.im git - yt-dlp.git/blame - yt_dlp/extractor/zaiko.py
[extractor/yappy] YappyProfile: Add extractor (#7346)
[yt-dlp.git] / yt_dlp / extractor / zaiko.py
CommitLineData
345b4c0a 1from .common import InfoExtractor
2from ..utils import (
3 ExtractorError,
4 extract_attributes,
5 int_or_none,
6 str_or_none,
7 traverse_obj,
8 unescapeHTML,
9 url_or_none,
10)
11
12
13class ZaikoIE(InfoExtractor):
14 _VALID_URL = r'https?://(?:[\w-]+\.)?zaiko\.io/event/(?P<id>\d+)/stream(?:/\d+)+'
15 _TESTS = [{
16 'url': 'https://zaiko.io/event/324868/stream/20571/20571',
17 'info_dict': {
18 'id': '324868',
19 'ext': 'mp4',
20 'title': 'ZAIKO STREAMING TEST',
21 'alt_title': '[VOD] ZAIKO STREAMING TEST_20210603(Do Not Delete)',
22 'uploader_id': '454',
23 'uploader': 'ZAIKO ZERO',
24 'release_timestamp': 1583809200,
25 'thumbnail': r're:https://[a-z0-9]+.cloudfront.net/[a-z0-9_]+/[a-z0-9_]+',
26 'release_date': '20200310',
27 'categories': ['Tech House'],
28 'live_status': 'was_live',
29 },
30 'params': {'skip_download': 'm3u8'},
31 }]
32
33 def _parse_vue_element_attr(self, name, string, video_id):
34 page_elem = self._search_regex(rf'(<{name}[^>]+>)', string, name)
35 attrs = {}
36 for key, value in extract_attributes(page_elem).items():
37 if key.startswith(':'):
38 attrs[key[1:]] = self._parse_json(
39 value, video_id, transform_source=unescapeHTML, fatal=False)
40 return attrs
41
42 def _real_extract(self, url):
43 video_id = self._match_id(url)
44
45 webpage, urlh = self._download_webpage_handle(url, video_id)
46 final_url = urlh.geturl()
47 if 'zaiko.io/login' in final_url:
48 self.raise_login_required()
49 elif '/_buy/' in final_url:
50 raise ExtractorError('Your account does not have tickets to this event', expected=True)
51 stream_meta = self._parse_vue_element_attr('stream-page', webpage, video_id)
52
53 player_page = self._download_webpage(
54 stream_meta['stream-access']['video_source'], video_id,
55 'Downloading player page', headers={'referer': 'https://zaiko.io/'})
56 player_meta = self._parse_vue_element_attr('player', player_page, video_id)
57 status = traverse_obj(player_meta, ('initial_event_info', 'status', {str}))
58 live_status, msg, expected = {
59 'vod': ('was_live', 'No VOD stream URL was found', False),
60 'archiving': ('post_live', 'Event VOD is still being processed', True),
61 'deleting': ('post_live', 'This event has ended', True),
62 'deleted': ('post_live', 'This event has ended', True),
63 'error': ('post_live', 'This event has ended', True),
64 'disconnected': ('post_live', 'Stream has been disconnected', True),
65 'live_to_disconnected': ('post_live', 'Stream has been disconnected', True),
66 'live': ('is_live', 'No livestream URL found was found', False),
67 'waiting': ('is_upcoming', 'Live event has not yet started', True),
68 'cancelled': ('not_live', 'Event has been cancelled', True),
69 }.get(status) or ('not_live', f'Unknown event status "{status}"', False)
70
71 stream_url = traverse_obj(player_meta, ('initial_event_info', 'endpoint', {url_or_none}))
72 formats = self._extract_m3u8_formats(
73 stream_url, video_id, live=True, fatal=False) if stream_url else []
74 if not formats:
75 self.raise_no_formats(msg, expected=expected)
76
77 return {
78 'id': video_id,
79 'formats': formats,
80 'live_status': live_status,
81 **traverse_obj(stream_meta, {
82 'title': ('event', 'name', {str}),
83 'uploader': ('profile', 'name', {str}),
84 'uploader_id': ('profile', 'id', {str_or_none}),
85 'release_timestamp': ('stream', 'start', 'timestamp', {int_or_none}),
86 'categories': ('event', 'genres', ..., {lambda x: x or None}),
87 }),
88 **traverse_obj(player_meta, ('initial_event_info', {
89 'alt_title': ('title', {str}),
90 'thumbnail': ('poster_url', {url_or_none}),
91 })),
92 }