aboutsummaryrefslogtreecommitdiffstats
path: root/hypervideo_dl/extractor/afreecatv.py
diff options
context:
space:
mode:
Diffstat (limited to 'hypervideo_dl/extractor/afreecatv.py')
-rw-r--r--hypervideo_dl/extractor/afreecatv.py120
1 files changed, 111 insertions, 9 deletions
diff --git a/hypervideo_dl/extractor/afreecatv.py b/hypervideo_dl/extractor/afreecatv.py
index 063872b..77f0e3c 100644
--- a/hypervideo_dl/extractor/afreecatv.py
+++ b/hypervideo_dl/extractor/afreecatv.py
@@ -10,7 +10,11 @@ from ..utils import (
determine_ext,
ExtractorError,
int_or_none,
+ qualities,
+ traverse_obj,
unified_strdate,
+ unified_timestamp,
+ update_url_query,
url_or_none,
urlencode_postdata,
xpath_text,
@@ -28,7 +32,7 @@ class AfreecaTVIE(InfoExtractor):
/app/(?:index|read_ucc_bbs)\.cgi|
/player/[Pp]layer\.(?:swf|html)
)\?.*?\bnTitleNo=|
- vod\.afreecatv\.com/PLAYER/STATION/
+ vod\.afreecatv\.com/(PLAYER/STATION|player)/
)
(?P<id>\d+)
'''
@@ -166,6 +170,9 @@ class AfreecaTVIE(InfoExtractor):
}, {
'url': 'http://vod.afreecatv.com/PLAYER/STATION/15055030',
'only_matching': True,
+ }, {
+ 'url': 'http://vod.afreecatv.com/player/15055030',
+ 'only_matching': True,
}]
@staticmethod
@@ -177,14 +184,7 @@ class AfreecaTVIE(InfoExtractor):
video_key['part'] = int(m.group('part'))
return video_key
- def _real_initialize(self):
- self._login()
-
- def _login(self):
- username, password = self._get_login_info()
- if username is None:
- return
-
+ def _perform_login(self, username, password):
login_form = {
'szWork': 'login',
'szType': 'json',
@@ -380,3 +380,105 @@ class AfreecaTVIE(InfoExtractor):
})
return info
+
+
+class AfreecaTVLiveIE(AfreecaTVIE):
+
+ IE_NAME = 'afreecatv:live'
+ _VALID_URL = r'https?://play\.afreeca(?:tv)?\.com/(?P<id>[^/]+)(?:/(?P<bno>\d+))?'
+ _TESTS = [{
+ 'url': 'https://play.afreecatv.com/pyh3646/237852185',
+ 'info_dict': {
+ 'id': '237852185',
+ 'ext': 'mp4',
+ 'title': '【 우루과이 오늘은 무슨일이? 】',
+ 'uploader': '박진우[JINU]',
+ 'uploader_id': 'pyh3646',
+ 'timestamp': 1640661495,
+ 'is_live': True,
+ },
+ 'skip': 'Livestream has ended',
+ }, {
+ 'url': 'http://play.afreeca.com/pyh3646/237852185',
+ 'only_matching': True,
+ }, {
+ 'url': 'http://play.afreeca.com/pyh3646',
+ 'only_matching': True,
+ }]
+
+ _LIVE_API_URL = 'https://live.afreecatv.com/afreeca/player_live_api.php'
+
+ _QUALITIES = ('sd', 'hd', 'hd2k', 'original')
+
+ def _real_extract(self, url):
+ broadcaster_id, broadcast_no = self._match_valid_url(url).group('id', 'bno')
+ password = self.get_param('videopassword')
+
+ info = self._download_json(self._LIVE_API_URL, broadcaster_id, fatal=False,
+ data=urlencode_postdata({'bid': broadcaster_id})) or {}
+ channel_info = info.get('CHANNEL') or {}
+ broadcaster_id = channel_info.get('BJID') or broadcaster_id
+ broadcast_no = channel_info.get('BNO') or broadcast_no
+ password_protected = channel_info.get('BPWD')
+ if not broadcast_no:
+ raise ExtractorError(f'Unable to extract broadcast number ({broadcaster_id} may not be live)', expected=True)
+ if password_protected == 'Y' and password is None:
+ raise ExtractorError(
+ 'This livestream is protected by a password, use the --video-password option',
+ expected=True)
+
+ formats = []
+ quality_key = qualities(self._QUALITIES)
+ for quality_str in self._QUALITIES:
+ params = {
+ 'bno': broadcast_no,
+ 'stream_type': 'common',
+ 'type': 'aid',
+ 'quality': quality_str,
+ }
+ if password is not None:
+ params['pwd'] = password
+ aid_response = self._download_json(
+ self._LIVE_API_URL, broadcast_no, fatal=False,
+ data=urlencode_postdata(params),
+ note=f'Downloading access token for {quality_str} stream',
+ errnote=f'Unable to download access token for {quality_str} stream')
+ aid = traverse_obj(aid_response, ('CHANNEL', 'AID'))
+ if not aid:
+ continue
+
+ stream_base_url = channel_info.get('RMD') or 'https://livestream-manager.afreecatv.com'
+ stream_info = self._download_json(
+ f'{stream_base_url}/broad_stream_assign.html', broadcast_no, fatal=False,
+ query={
+ 'return_type': channel_info.get('CDN', 'gcp_cdn'),
+ 'broad_key': f'{broadcast_no}-common-{quality_str}-hls',
+ },
+ note=f'Downloading metadata for {quality_str} stream',
+ errnote=f'Unable to download metadata for {quality_str} stream') or {}
+
+ if stream_info.get('view_url'):
+ formats.append({
+ 'format_id': quality_str,
+ 'url': update_url_query(stream_info['view_url'], {'aid': aid}),
+ 'ext': 'mp4',
+ 'protocol': 'm3u8',
+ 'quality': quality_key(quality_str),
+ })
+
+ self._sort_formats(formats)
+
+ station_info = self._download_json(
+ 'https://st.afreecatv.com/api/get_station_status.php', broadcast_no,
+ query={'szBjId': broadcaster_id}, fatal=False,
+ note='Downloading channel metadata', errnote='Unable to download channel metadata') or {}
+
+ return {
+ 'id': broadcast_no,
+ 'title': channel_info.get('TITLE') or station_info.get('station_title'),
+ 'uploader': channel_info.get('BJNICK') or station_info.get('station_name'),
+ 'uploader_id': broadcaster_id,
+ 'timestamp': unified_timestamp(station_info.get('broad_start')),
+ 'formats': formats,
+ 'is_live': True,
+ }