haruhi-dl/haruhi_dl/extractor/tvn24.py
2021-04-26 14:40:15 +02:00

285 lines
11 KiB
Python

# coding: utf-8
from __future__ import unicode_literals
import re
from urllib.parse import (
parse_qs,
urlparse,
)
from .common import InfoExtractor
from ..utils import (
clean_html,
extract_attributes,
ExtractorError,
int_or_none,
NO_DEFAULT,
unescapeHTML,
)
class TVNBaseIE(InfoExtractor):
def _parse_nuvi_data(self, data, display_id):
video = data['movie']['video']
info = data['movie']['info']
if video.get('protections'):
raise ExtractorError(
'This video is protected by %s DRM protection' % '/'.join(video['protections'].keys()),
expected=True)
formats = []
for fmt_id, fmt_data in video['sources'].items():
if fmt_id == 'hls':
formats.extend(self._extract_m3u8_formats(fmt_data['url'], display_id, ext='mp4'))
elif fmt_id == 'dash':
formats.extend(self._extract_mpd_formats(fmt_data['url'], display_id))
elif fmt_id == 'mp4':
for quality, mp4_url in fmt_data.items():
formats.append({
'url': mp4_url,
'ext': 'mp4',
'height': int_or_none(quality),
})
self._sort_formats(formats)
return {
'id': display_id,
'formats': formats,
'title': unescapeHTML(info.get('episode_title')),
'description': unescapeHTML(info.get('description')),
'duration': int_or_none(info.get('total_time')),
'age_limit': int_or_none(data['movie']['options'].get('parental_rating', {}).get('rating')),
'is_live': video.get('is_live'),
}
class TVN24IE(TVNBaseIE):
_VALID_URL = r'https?://(?:www\.)?(?P<domain>(?:(?:[^/]+)\.)?tvn(?:24)?\.pl)/(?:[^/]+/)*[^/?#\s]+[,-](?P<id>\d+)(?:\.html)?'
_TESTS = [{
'url': 'https://tvn24.pl/polska/edyta-gorniak-napisala-o-statystach-w-szpitalach-udajacych-chorych-na-covid-19-jerzy-polaczek-i-marek-posobkiewicz-odpowiadaja-zapraszamy-4747899',
'info_dict': {
'id': '4747899',
'title': '"Nie wszyscy \'statyści\' wychodzą na własnych nogach". Chorzy na COVID-19 odpowiadają Edycie Górniak',
},
'playlist_count': 5,
}, {
# different layout
'url': 'https://tvn24.pl/tvnmeteo/magazyny/nowa-maja-w-ogrodzie,13/odcinki-online,1,4,1,0/pnacza-ptaki-i-iglaki-odc-691-hgtv-odc-29,1771763.html',
'info_dict': {
'id': '1771763',
'ext': 'mp4',
'title': 'Pnącza, ptaki i iglaki (odc. 691 /HGTV odc. 29)',
'thumbnail': 're:https?://.*',
},
'params': {
'skip_download': True,
},
}, {
'url': 'https://tvn24.pl/magazyn-tvn24/zrobimy-z-ciebie-mezczyzne,242,4189',
'info_dict': {
'id': '4189',
'title': 'Zrobimy z ciebie mężczyznę',
'description': 'Milo nie miało myśli erotycznych. Tak boleśnie myślało o własnym ciele, że nie potrafiło myśleć o nim w towarzystwie innych ciał. Ale żeby zmienić ciało, m...',
},
'playlist_count': 2,
}, {
'url': 'https://fakty.tvn24.pl/ogladaj-online,60/akcja-media-bez-wyboru-i-misja-telewizji-publicznej,1048910.html',
'info_dict': {
'id': '1048910',
'ext': 'mp4',
'title': '11.02.2021 | Misja telewizji publicznej i reakcja na protest "Media bez wyboru"',
'description': 'md5:684d2e09f57c7ed03a277bc5ce295d63',
},
}, {
# no data-qualities, just data-src
'url': 'https://uwaga.tvn.pl/reportaze,2671,n/po-wyroku-trybunalu-kobiety-nie-moga-poddac-sie-aborcji,337993.html',
'info_dict': {
'id': '337993',
'ext': 'mp4',
'title': 'Wady letalne, czyli śmiertelne. "Boję się następnej ciąży"',
'description': 'md5:4f5efe579b7f801d5a8d7a75c0809260',
},
}, {
'url': 'http://fakty.tvn24.pl/ogladaj-online,60/53-konferencja-bezpieczenstwa-w-monachium,716431.html',
'only_matching': True,
}, {
'url': 'http://sport.tvn24.pl/pilka-nozna,105/ligue-1-kamil-glik-rozcial-glowe-monaco-tylko-remisuje-z-bastia,716522.html',
'only_matching': True,
}, {
'url': 'https://www.tvn24.pl/magazyn-tvn24/angie-w-jednej-czwartej-polka-od-szarej-myszki-do-cesarzowej-europy,119,2158',
'only_matching': True,
}]
def _real_extract(self, url):
mobj = re.match(self._VALID_URL, url)
domain, display_id = mobj.group('domain', 'id')
webpage = self._download_webpage(url, display_id)
if domain == 'tvn24.pl':
if '<script id="__NEXT_DATA__"' in webpage:
return self._handle_nextjs_frontend(url, display_id, webpage)
if '/magazyn-tvn24/' in url:
return self._handle_magazine_frontend(url, display_id, webpage)
if 'window.VideoManager.initVideo(' in webpage:
return self._handle_fakty_frontend(url, display_id, webpage)
return self._handle_old_frontend(url, display_id, webpage)
def _handle_old_frontend(self, url, display_id, webpage):
title = self._og_search_title(
webpage, default=None) or self._search_regex(
r'<h\d+[^>]+class=["\']magazineItemHeader[^>]+>(.+?)</h',
webpage, 'title')
def extract_json(attr, name, default=NO_DEFAULT, fatal=True):
return self._parse_json(
self._search_regex(
r'\b%s=(["\'])(?P<json>(?!\1).+?)\1' % attr, webpage,
name, group='json', default=default, fatal=fatal) or '{}',
display_id, transform_source=unescapeHTML, fatal=fatal)
quality_data = extract_json('data-quality', 'formats', default=None, fatal=False)
formats = []
if quality_data:
for format_id, url in quality_data.items():
formats.append({
'url': url,
'format_id': format_id,
'height': int_or_none(format_id.rstrip('p')),
})
self._sort_formats(formats)
else:
formats.append({
'url': self._search_regex(
r'\bdata-src=(["\'])(?P<url>(?!\1).+?)\1',
webpage, 'video url', group='url'),
})
description = self._og_search_description(webpage, default=None)
thumbnail = self._og_search_thumbnail(
webpage, default=None) or self._html_search_regex(
r'\bdata-poster=(["\'])(?P<url>(?!\1).+?)\1', webpage,
'thumbnail', group='url')
video_id = None
share_params = extract_json(
'data-share-params', 'share params', default=None)
if isinstance(share_params, dict):
video_id = share_params.get('id')
if not video_id:
video_id = self._search_regex(
r'data-vid-id=["\'](\d+)', webpage, 'video id',
default=None) or self._search_regex(
r',(\d+)\.html', url, 'video id', default=display_id)
return {
'id': video_id,
'title': title,
'description': description,
'thumbnail': thumbnail,
'formats': formats,
}
def _handle_magazine_frontend(self, url, display_id, webpage):
entries = []
for vid_el in re.finditer(r'(?P<video><div\b[^>]+\bdata-src=[^>]+>)\s*(?:</[^>]+>\s*)*?(?:<figcaption>(?P<title>(?:.|\s)+?)</figcaption>)?', webpage):
vid = extract_attributes(vid_el.group('video'))
formats = []
qualities = vid.get('data-quality')
if qualities:
for fmt_name, fmt_url in self._parse_json(unescapeHTML(qualities), display_id).items():
formats.append({
'format_id': fmt_name,
'height': int_or_none(fmt_name[:-1]),
'url': fmt_url,
})
else:
formats = [{
'url': vid['data-src'],
}]
self._sort_formats(formats)
entries.append({
'id': vid['data-video-id'],
'title': clean_html(vid_el.group('title')),
'formats': formats,
'thumbnail': vid.get('data-poster'),
})
if not entries:
raise ExtractorError('No videos found')
return {
'_type': 'playlist',
'id': display_id,
'entries': entries,
'title': self._og_search_title(webpage),
'description': self._og_search_description(webpage),
}
def _handle_nextjs_frontend(self, url, display_id, webpage):
next_data = self._search_nextjs_data(webpage, display_id)
context = next_data['props']['initialProps']['pageProps']['context']
videos = []
for element in context['mainVideo']:
if element['content']['type'] == 'Video':
videos.append(element['content'])
for element in context['elements']:
if element.get('type') == 'video' \
and 'relation' in element \
and element['relation']['type'] == 'Video' \
and not any(video['id'] == element['relation']['id'] for video in videos):
videos.append(element['relation'])
route_name = next_data['props']['initialState']['resolution']['routeName']
entries = []
for video in videos:
fields = video['fields']
plst_url = re.sub(r'[?#].+', '', url)
plst_url += '/nuviArticle?playlist&id=%s&r=%s' % (video['id'], route_name)
entries.append({
'_type': 'url_transparent',
'url': plst_url,
'ie_key': 'TVN24Nuvi',
'title': fields['title'],
'description': fields['description'],
})
return {
'_type': 'playlist',
'entries': entries,
'id': display_id,
'title': context['title'],
'alt_title': context['fields']['title'],
}
def _handle_fakty_frontend(self, url, display_id, webpage):
data = self._parse_json(
self._search_regex(
r"window\.VideoManager\.initVideo\('[^']+',\s*({.+?})\s*,\s*{.+?}\s*\);",
webpage, 'video metadata'), display_id)
return self._parse_nuvi_data(data, display_id)
class TVN24NuviIE(TVNBaseIE):
# handles getting specific videos from the list of nuvi urls
_VALID_URL = r'https?://(?:www\.)?(?P<domain>(?:(?:[^/]+)\.)?tvn24\.pl)/(?:[^/]+/)*[^/?#\s]+[,-](?P<id>\d+)(?:\.html)?/nuviArticle'
IE_NAME = 'tvn24:nuvi'
IE_DESC = False # do not list
def _real_extract(self, url):
qs = parse_qs(urlparse(url).query)
video_id = qs['id'][0]
plst = self._download_json(url, video_id)
return self._parse_nuvi_data(plst, video_id)