from __future__ import annotations import functools import json import textwrap from .common import InfoExtractor from ..utils import ExtractorError, format_field, int_or_none, parse_iso8601 from ..utils.traversal import traverse_obj def _fmt_url(url): return functools.partial(format_field, template=url, default=None) class TelewebionIE(InfoExtractor): _VALID_URL = r'https?://(?:www\.)?telewebion\.com/episode/(?P<id>(?:0x[a-fA-F\d]+|\d+))' _TESTS = [{ 'url': 'http://www.telewebion.com/episode/0x1b3139c/', 'info_dict': { 'id': '0x1b3139c', 'ext': 'mp4', 'title': 'قرعهکشی لیگ قهرمانان اروپا', 'series': '+ فوتبال', 'series_id': '0x1b2505c', 'channel': 'شبکه 3', 'channel_id': '0x1b1a761', 'channel_url': 'https://telewebion.com/live/tv3', 'timestamp': 1425522414, 'upload_date': '20150305', 'release_timestamp': 1425517020, 'release_date': '20150305', 'duration': 420, 'view_count': int, 'tags': ['ورزشی', 'لیگ اروپا', 'اروپا'], 'thumbnail': 'https://static.telewebion.com/episodeImages/YjFhM2MxMDBkMDNiZTU0MjE5YjQ3ZDY0Mjk1ZDE0ZmUwZWU3OTE3OWRmMDAyODNhNzNkNjdmMWMzMWIyM2NmMA/default', }, 'skip_download': 'm3u8', }, { 'url': 'https://telewebion.com/episode/162175536', 'info_dict': { 'id': '0x9aa9a30', 'ext': 'mp4', 'title': 'کارما یعنی این !', 'series': 'پاورقی', 'series_id': '0x29a7426', 'channel': 'شبکه 2', 'channel_id': '0x1b1a719', 'channel_url': 'https://telewebion.com/live/tv2', 'timestamp': 1699979968, 'upload_date': '20231114', 'release_timestamp': 1699991638, 'release_date': '20231114', 'duration': 78, 'view_count': int, 'tags': ['کلیپ های منتخب', ' کلیپ طنز ', ' کلیپ سیاست ', 'پاورقی', 'ویژه فلسطین'], 'thumbnail': 'https://static.telewebion.com/episodeImages/871e9455-7567-49a5-9648-34c22c197f5f/default', }, 'skip_download': 'm3u8', }] def _call_graphql_api( self, operation, video_id, query, variables: dict[str, tuple[str, str]] | None = None, note='Downloading GraphQL JSON metadata', ): parameters = '' if variables: parameters = ', '.join(f'${name}: {type_}' for name, (type_, _) in variables.items()) parameters = f'({parameters})' result = self._download_json('https://graph.telewebion.com/graphql', video_id, note, data=json.dumps({ 'operationName': operation, 'query': f'query {operation}{parameters} @cacheControl(maxAge: 60) {{{query}\n}}\n', 'variables': {name: value for name, (_, value) in (variables or {}).items()} }, separators=(',', ':')).encode(), headers={ 'Content-Type': 'application/json', 'Accept': 'application/json', }) if not result or traverse_obj(result, 'errors'): message = ', '.join(traverse_obj(result, ('errors', ..., 'message', {str}))) raise ExtractorError(message or 'Unknown GraphQL API error') return result['data'] def _real_extract(self, url): video_id = self._match_id(url) if not video_id.startswith('0x'): video_id = hex(int(video_id)) episode_data = self._call_graphql_api('getEpisodeDetail', video_id, textwrap.dedent(''' queryEpisode(filter: {EpisodeID: $EpisodeId}, first: 1) { title program { ProgramID title } image view_count duration started_at created_at channel { ChannelID name descriptor } tags { name } } '''), {'EpisodeId': ('[ID!]', video_id)}) info_dict = traverse_obj(episode_data, ('queryEpisode', 0, { 'title': ('title', {str}), 'view_count': ('view_count', {int_or_none}), 'duration': ('duration', {int_or_none}), 'tags': ('tags', ..., 'name', {str}), 'release_timestamp': ('started_at', {parse_iso8601}), 'timestamp': ('created_at', {parse_iso8601}), 'series': ('program', 'title', {str}), 'series_id': ('program', 'ProgramID', {str}), 'channel': ('channel', 'name', {str}), 'channel_id': ('channel', 'ChannelID', {str}), 'channel_url': ('channel', 'descriptor', {_fmt_url('https://telewebion.com/live/%s')}), 'thumbnail': ('image', {_fmt_url('https://static.telewebion.com/episodeImages/%s/default')}), 'formats': ( 'channel', 'descriptor', {str}, {_fmt_url(f'https://cdna.telewebion.com/%s/episode/{video_id}/playlist.m3u8')}, {functools.partial(self._extract_m3u8_formats, video_id=video_id, ext='mp4', m3u8_id='hls')}), })) info_dict['id'] = video_id return info_dict