Compare commits

..

3 Commits

Author SHA1 Message Date
sepro
616d3d08b9 Move m3u8 extraction in to _parse_video_data 2024-02-04 20:05:14 +01:00
sepro
43468b5933 Reduce playlist mincount to 1 2024-02-04 18:41:05 +01:00
sepro
b6f28c158b
Apply suggestions from code review
Co-authored-by: bashonly <88596187+bashonly@users.noreply.github.com>
2024-02-04 18:35:58 +01:00

View File

@ -5,6 +5,7 @@ from ..utils import (
ExtractorError, ExtractorError,
OnDemandPagedList, OnDemandPagedList,
UserNotLive, UserNotLive,
filter_dict,
int_or_none, int_or_none,
parse_iso8601, parse_iso8601,
str_or_none, str_or_none,
@ -14,7 +15,6 @@ from ..utils.traversal import traverse_obj
class NuumBaseIE(InfoExtractor): class NuumBaseIE(InfoExtractor):
def _call_api(self, path, video_id, description, query={}): def _call_api(self, path, video_id, description, query={}):
response = self._download_json( response = self._download_json(
f'https://nuum.ru/api/v2/{path}', video_id, query=query, f'https://nuum.ru/api/v2/{path}', video_id, query=query,
@ -22,7 +22,7 @@ class NuumBaseIE(InfoExtractor):
errnote=f'Unable to download {description} metadata') errnote=f'Unable to download {description} metadata')
if error := response.get('error'): if error := response.get('error'):
raise ExtractorError(f'API returned error: {error!r}') raise ExtractorError(f'API returned error: {error!r}')
return response.get('result') return response['result']
def _get_channel_info(self, channel_name): def _get_channel_info(self, channel_name):
return self._call_api( return self._call_api(
@ -33,15 +33,25 @@ class NuumBaseIE(InfoExtractor):
'with_deleted': 'true', 'with_deleted': 'true',
}) })
def _parse_video_data(self, container): def _parse_video_data(self, container, extract_formats=True):
stream = traverse_obj(container, ('media_container_streams', 0)) stream = traverse_obj(container, ('media_container_streams', 0, {dict})) or {}
media = traverse_obj(stream, ('stream_media', 0)) media = traverse_obj(stream, ('stream_media', 0, {dict})) or {}
media_url = traverse_obj(media, ( media_url = traverse_obj(media, (
'media_meta', ('media_archive_url', 'media_url'), {url_or_none}), get_all=False) 'media_meta', ('media_archive_url', 'media_url'), {url_or_none}), get_all=False)
return media_url, { video_id = str(container['media_container_id'])
'id': str(container['media_container_id']), is_live = media.get('media_status') == 'RUNNING'
'is_live': media.get('media_status') == 'RUNNING',
formats, subtitles = None, None
if extract_formats:
formats, subtitles = self._extract_m3u8_formats_and_subtitles(
media_url, video_id, 'mp4', live=is_live)
return filter_dict({
'id': video_id,
'is_live': is_live,
'formats': formats,
'subtitles': subtitles,
**traverse_obj(container, { **traverse_obj(container, {
'title': ('media_container_name', {str}), 'title': ('media_container_name', {str}),
'description': ('media_container_description', {str}), 'description': ('media_container_description', {str}),
@ -57,7 +67,7 @@ class NuumBaseIE(InfoExtractor):
'duration': ('media_duration', {int_or_none}), 'duration': ('media_duration', {int_or_none}),
'thumbnail': ('media_meta', ('media_preview_archive_url', 'media_preview_url'), {url_or_none}), 'thumbnail': ('media_meta', ('media_preview_archive_url', 'media_preview_url'), {url_or_none}),
}, get_all=False), }, get_all=False),
} })
class NuumMediaIE(NuumBaseIE): class NuumMediaIE(NuumBaseIE):
@ -105,15 +115,7 @@ class NuumMediaIE(NuumBaseIE):
video_id = self._match_id(url) video_id = self._match_id(url)
video_data = self._call_api(f'media-containers/{video_id}', video_id, 'media') video_data = self._call_api(f'media-containers/{video_id}', video_id, 'media')
m3u8_url, info = self._parse_video_data(video_data) return self._parse_video_data(video_data)
formats, subtitles = self._extract_m3u8_formats_and_subtitles(
m3u8_url, video_id, 'mp4', live=info['is_live'])
info.update({
'formats': formats,
'subtitles': subtitles,
})
return info
class NuumLiveIE(NuumBaseIE): class NuumLiveIE(NuumBaseIE):
@ -130,15 +132,12 @@ class NuumLiveIE(NuumBaseIE):
if traverse_obj(channel_info, ('channel', 'channel_is_live')) is False: if traverse_obj(channel_info, ('channel', 'channel_is_live')) is False:
raise UserNotLive(video_id=channel) raise UserNotLive(video_id=channel)
m3u8_url, metadata = self._parse_video_data(channel_info['media_container']) info = self._parse_video_data(channel_info['media_container'])
formats, subtitles = self._extract_m3u8_formats_and_subtitles(m3u8_url, channel, 'mp4', live=True)
return { return {
'formats': formats, 'webpage_url': f'https://nuum.ru/streams/{info["id"]}',
'subtitles': subtitles,
'webpage_url': f'https://nuum.ru/streams/{metadata["id"]}',
'extractor_key': NuumMediaIE.ie_key(), 'extractor_key': NuumMediaIE.ie_key(),
'extractor': NuumMediaIE.IE_NAME, 'extractor': NuumMediaIE.IE_NAME,
**metadata, **info,
} }
@ -165,7 +164,7 @@ class NuumTabIE(NuumBaseIE):
'id': 'dankon__streams', 'id': 'dankon__streams',
'title': 'Dankon_', 'title': 'Dankon_',
}, },
'playlist_mincount': 5, 'playlist_mincount': 1,
}] }]
_PAGE_SIZE = 50 _PAGE_SIZE = 50
@ -187,7 +186,7 @@ class NuumTabIE(NuumBaseIE):
'media_container_type': CONTAINER_TYPES[tab_type], 'media_container_type': CONTAINER_TYPES[tab_type],
}) })
for container in traverse_obj(media_containers, (..., {dict})): for container in traverse_obj(media_containers, (..., {dict})):
_, metadata = self._parse_video_data(container) metadata = self._parse_video_data(container, extract_formats=False)
yield self.url_result(f'https://nuum.ru/videos/{metadata["id"]}', NuumMediaIE, **metadata) yield self.url_result(f'https://nuum.ru/videos/{metadata["id"]}', NuumMediaIE, **metadata)
def _real_extract(self, url): def _real_extract(self, url):