2 from __future__
import unicode_literals
7 from .common
import InfoExtractor
8 from ..compat
import compat_str
18 class PicartoIE(InfoExtractor
):
19 _VALID_URL
= r
'https?://(?:www.)?picarto\.tv/(?P<id>[a-zA-Z0-9]+)(?:/(?P<token>[a-zA-Z0-9]+))?'
21 'url': 'https://picarto.tv/Setz',
25 'title': 're:^Setz [0-9]{4}-[0-9]{2}-[0-9]{2} [0-9]{2}:[0-9]{2}$',
29 'skip': 'Stream is offline',
33 def suitable(cls
, url
):
34 return False if PicartoVodIE
.suitable(url
) else super(PicartoIE
, cls
).suitable(url
)
36 def _real_extract(self
, url
):
37 mobj
= re
.match(self
._VALID
_URL
, url
)
38 channel_id
= mobj
.group('id')
40 metadata
= self
._download
_json
(
41 'https://api.picarto.tv/v1/channel/name/' + channel_id
,
44 if metadata
.get('online') is False:
45 raise ExtractorError('Stream is offline', expected
=True)
47 cdn_data
= self
._download
_json
(
48 'https://picarto.tv/process/channel', channel_id
,
49 data
=urlencode_postdata({'loadbalancinginfo': channel_id
}),
50 note
='Downloading load balancing info')
52 token
= mobj
.group('token') or 'public'
54 'con': int(time
.time() * 1000),
58 prefered_edge
= cdn_data
.get('preferedEdge')
61 for edge
in cdn_data
['edges']:
62 edge_ep
= edge
.get('ep')
63 if not edge_ep
or not isinstance(edge_ep
, compat_str
):
65 edge_id
= edge
.get('id')
66 for tech
in cdn_data
['techs']:
67 tech_label
= tech
.get('label')
68 tech_type
= tech
.get('type')
70 if edge_id
== prefered_edge
:
74 format_id
.append(edge_id
)
75 if tech_type
== 'application/x-mpegurl' or tech_label
== 'HLS':
76 format_id
.append('hls')
77 formats
.extend(self
._extract
_m
3u8_formats
(
79 'https://%s/hls/%s/index.m3u8'
80 % (edge_ep
, channel_id
), params
),
81 channel_id
, 'mp4', preference
=preference
,
82 m3u8_id
='-'.join(format_id
), fatal
=False))
84 elif tech_type
== 'video/mp4' or tech_label
== 'MP4':
85 format_id
.append('mp4')
87 'url': update_url_query(
88 'https://%s/mp4/%s.mp4' % (edge_ep
, channel_id
),
90 'format_id': '-'.join(format_id
),
91 'preference': preference
,
94 # rtmp format does not seem to work
96 self
._sort
_formats
(formats
)
98 mature
= metadata
.get('adult')
102 age_limit
= 18 if mature
is True else 0
106 'title': self
._live
_title
(metadata
.get('title') or channel_id
),
108 'thumbnail': try_get(metadata
, lambda x
: x
['thumbnails']['web']),
109 'channel': channel_id
,
110 'channel_url': 'https://picarto.tv/%s' % channel_id
,
111 'age_limit': age_limit
,
116 class PicartoVodIE(InfoExtractor
):
117 _VALID_URL
= r
'https?://(?:www.)?picarto\.tv/videopopout/(?P<id>[^/?#&]+)'
119 'url': 'https://picarto.tv/videopopout/ArtofZod_2017.12.12.00.13.23.flv',
120 'md5': '3ab45ba4352c52ee841a28fb73f2d9ca',
122 'id': 'ArtofZod_2017.12.12.00.13.23.flv',
124 'title': 'ArtofZod_2017.12.12.00.13.23.flv',
125 'thumbnail': r
're:^https?://.*\.jpg'
128 'url': 'https://picarto.tv/videopopout/Plague',
129 'only_matching': True,
132 def _real_extract(self
, url
):
133 video_id
= self
._match
_id
(url
)
135 webpage
= self
._download
_webpage
(url
, video_id
)
137 vod_info
= self
._parse
_json
(
139 r
'(?s)#vod-player["\']\s
*,\s
*(\
{.+?\
})\s
*\
)', webpage,
141 video_id, transform_source=js_to_json)
143 formats = self._extract_m3u8_formats(
144 vod_info['vod
'], video_id, 'mp4
', entry_protocol='m3u8_native
',
146 self._sort_formats(formats)
151 'thumbnail
': vod_info.get('vodThumb
'),