mirror of
				https://github.com/yt-dlp/yt-dlp.git
				synced 2025-10-31 14:45:14 +00:00 
			
		
		
		
	 65de7d204c
			
		
	
	65de7d204c
	
	
	
		
			
			- [utils] Make restricted filenames ignore some Unicode categories (by dirkf) - [ie/telewebion] Fix extraction (by Grub4K) - [ie/imgur] Overhaul extractor (by bashonly, Grub4K) - [ie/EpidemicSound] Add extractor (by Grub4K) Authored by: bashonly, dirkf, Grub4K Co-authored-by: bashonly <bashonly@protonmail.com>
		
			
				
	
	
		
			134 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			134 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from __future__ import annotations
 | |
| 
 | |
| import json
 | |
| from functools import partial
 | |
| from textwrap import dedent
 | |
| 
 | |
| 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 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, 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')},
 | |
|                 {partial(self._extract_m3u8_formats, video_id=video_id, ext='mp4', m3u8_id='hls')}),
 | |
|         }))
 | |
|         info_dict['id'] = video_id
 | |
|         return info_dict
 |