mirror of
https://github.com/yt-dlp/yt-dlp.git
synced 2025-08-15 00:48:28 +00:00
Merge ea26e74296
into e651a53a2f
This commit is contained in:
commit
27dd885e34
@ -1,9 +1,11 @@
|
|||||||
import base64
|
import base64
|
||||||
|
import datetime
|
||||||
import random
|
import random
|
||||||
import re
|
import re
|
||||||
import urllib.parse
|
import urllib.parse
|
||||||
|
|
||||||
from .common import InfoExtractor
|
from .common import InfoExtractor
|
||||||
|
from ..networking.exceptions import HTTPError
|
||||||
from ..utils import (
|
from ..utils import (
|
||||||
ExtractorError,
|
ExtractorError,
|
||||||
clean_html,
|
clean_html,
|
||||||
@ -12,12 +14,14 @@
|
|||||||
try_call,
|
try_call,
|
||||||
unified_timestamp,
|
unified_timestamp,
|
||||||
update_url_query,
|
update_url_query,
|
||||||
|
urlencode_postdata,
|
||||||
)
|
)
|
||||||
from ..utils.traversal import traverse_obj
|
from ..utils.traversal import traverse_obj
|
||||||
|
|
||||||
|
|
||||||
class RadikoBaseIE(InfoExtractor):
|
class RadikoBaseIE(InfoExtractor):
|
||||||
_GEO_BYPASS = False
|
_GEO_BYPASS = False
|
||||||
|
_NETRC_MACHINE = 'radiko'
|
||||||
_FULL_KEY = None
|
_FULL_KEY = None
|
||||||
_HOSTS_FOR_TIME_FREE_FFMPEG_UNSUPPORTED = (
|
_HOSTS_FOR_TIME_FREE_FFMPEG_UNSUPPORTED = (
|
||||||
'https://c-rpaa.smartstream.ne.jp',
|
'https://c-rpaa.smartstream.ne.jp',
|
||||||
@ -26,6 +30,7 @@ class RadikoBaseIE(InfoExtractor):
|
|||||||
'https://tf-c-rpaa-radiko.smartstream.ne.jp',
|
'https://tf-c-rpaa-radiko.smartstream.ne.jp',
|
||||||
'https://si-f-radiko.smartstream.ne.jp',
|
'https://si-f-radiko.smartstream.ne.jp',
|
||||||
'https://rpaa.smartstream.ne.jp',
|
'https://rpaa.smartstream.ne.jp',
|
||||||
|
'https://alliance-stream-radiko.smartstream.ne.jp',
|
||||||
)
|
)
|
||||||
_HOSTS_FOR_TIME_FREE_FFMPEG_SUPPORTED = (
|
_HOSTS_FOR_TIME_FREE_FFMPEG_SUPPORTED = (
|
||||||
'https://rd-wowza-radiko.radiko-cf.com',
|
'https://rd-wowza-radiko.radiko-cf.com',
|
||||||
@ -37,6 +42,41 @@ class RadikoBaseIE(InfoExtractor):
|
|||||||
'https://c-radiko.smartstream.ne.jp',
|
'https://c-radiko.smartstream.ne.jp',
|
||||||
)
|
)
|
||||||
|
|
||||||
|
_JST = datetime.timezone(datetime.timedelta(hours=9))
|
||||||
|
_account_privileges = None
|
||||||
|
|
||||||
|
def _perform_login(self, username, password):
|
||||||
|
try:
|
||||||
|
login_info = self._download_json('https://radiko.jp/ap/member/webapi/member/login', None, note='Logging in',
|
||||||
|
data=urlencode_postdata({'mail': username, 'pass': password}))
|
||||||
|
privileges_list = login_info.get('privileges', [])
|
||||||
|
self._account_privileges = {
|
||||||
|
'areafree': '1' in privileges_list,
|
||||||
|
'tf30': '2' in privileges_list,
|
||||||
|
} # areafree = 1, timefree30 = 2, double plan = both
|
||||||
|
except ExtractorError as error:
|
||||||
|
if isinstance(error.cause, HTTPError) and error.cause.status == 401:
|
||||||
|
raise ExtractorError('Invalid username and/or password', expected=True)
|
||||||
|
raise
|
||||||
|
|
||||||
|
def _check_privileges(self):
|
||||||
|
if self._account_privileges is not None:
|
||||||
|
# if already checked via perform_login
|
||||||
|
return self._account_privileges
|
||||||
|
|
||||||
|
if self._get_cookies('https://radiko.jp').get('radiko_session') is None:
|
||||||
|
# if no account at all
|
||||||
|
return {'areafree': False, 'tf30': False}
|
||||||
|
|
||||||
|
# if passed cookies
|
||||||
|
account_info = self._download_json('https://radiko.jp/ap/member/webapi/v2/member/login/check',
|
||||||
|
None, note='Checking account status from cookies', expected_status=400) or {}
|
||||||
|
self._account_privileges = {
|
||||||
|
'areafree': account_info.get('areafree') == '1',
|
||||||
|
'tf30': account_info.get('timefreeplus') == '1',
|
||||||
|
}
|
||||||
|
return self._account_privileges
|
||||||
|
|
||||||
def _negotiate_token(self):
|
def _negotiate_token(self):
|
||||||
_, auth1_handle = self._download_webpage_handle(
|
_, auth1_handle = self._download_webpage_handle(
|
||||||
'https://radiko.jp/v2/api/auth1', None, 'Downloading authentication page',
|
'https://radiko.jp/v2/api/auth1', None, 'Downloading authentication page',
|
||||||
@ -99,17 +139,39 @@ def _extract_full_key(self):
|
|||||||
self._FULL_KEY = full_key
|
self._FULL_KEY = full_key
|
||||||
return full_key
|
return full_key
|
||||||
|
|
||||||
|
def _get_broadcast_day(self, timestring):
|
||||||
|
dt = datetime.datetime.strptime(timestring, '%Y%m%d%H%M%S')
|
||||||
|
if dt.hour < 5:
|
||||||
|
dt -= datetime.timedelta(days=1)
|
||||||
|
return dt
|
||||||
|
|
||||||
|
def _get_broadcast_day_end(self, dt):
|
||||||
|
dt += datetime.timedelta(days=1)
|
||||||
|
return datetime.datetime(dt.year, dt.month, dt.day, 5, 0, 0, tzinfo=self._JST)
|
||||||
|
|
||||||
def _find_program(self, video_id, station, cursor):
|
def _find_program(self, video_id, station, cursor):
|
||||||
|
broadcast_day = self._get_broadcast_day(cursor)
|
||||||
|
broadcast_day_str = broadcast_day.strftime('%Y%m%d')
|
||||||
|
|
||||||
|
broadcast_day_end = self._get_broadcast_day_end(broadcast_day)
|
||||||
|
now = datetime.datetime.now(tz=self._JST)
|
||||||
|
|
||||||
|
if broadcast_day_end + datetime.timedelta(days=30) < now:
|
||||||
|
self.raise_no_formats('Programme is no longer available.', video_id=video_id, expected=True)
|
||||||
|
elif broadcast_day_end + datetime.timedelta(days=7) < now and not self._check_privileges()['tf30']:
|
||||||
|
self.raise_login_required('Programme is only available with a Timefree 30 subscription',
|
||||||
|
metadata_available=True)
|
||||||
|
|
||||||
station_program = self._download_xml(
|
station_program = self._download_xml(
|
||||||
f'https://radiko.jp/v3/program/station/weekly/{station}.xml', video_id,
|
f'https://api.radiko.jp/program/v3/date/{broadcast_day_str}/station/{station}.xml', station,
|
||||||
note=f'Downloading radio program for {station} station')
|
note=f'Downloading programme information for {broadcast_day_str}')
|
||||||
|
|
||||||
prog = None
|
prog = None
|
||||||
for p in station_program.findall('.//prog'):
|
for p in station_program.findall('.//prog'):
|
||||||
ft_str, to_str = p.attrib['ft'], p.attrib['to']
|
ft_str, to_str = p.attrib['ft'], p.attrib['to']
|
||||||
ft = unified_timestamp(ft_str, False)
|
ft = unified_timestamp(ft_str, False)
|
||||||
to = unified_timestamp(to_str, False)
|
to = unified_timestamp(to_str, False)
|
||||||
if ft <= cursor and cursor < to:
|
if ft_str <= cursor and cursor < to_str:
|
||||||
prog = p
|
prog = p
|
||||||
break
|
break
|
||||||
if not prog:
|
if not prog:
|
||||||
@ -122,10 +184,19 @@ def _extract_formats(self, video_id, station, is_onair, ft, cursor, auth_token,
|
|||||||
f'https://radiko.jp/v3/station/stream/pc_html5/{station}.xml', video_id,
|
f'https://radiko.jp/v3/station/stream/pc_html5/{station}.xml', video_id,
|
||||||
note='Downloading stream information')
|
note='Downloading stream information')
|
||||||
|
|
||||||
|
station_info = self._download_json(
|
||||||
|
f'https://radiko.jp/api/stations/batchGetStations?stationId={station}', video_id,
|
||||||
|
note='Checking station broadcast areas')
|
||||||
|
station_areas = traverse_obj(station_info, ('stationList', ..., 'prefecturesList'), get_all=False)
|
||||||
|
|
||||||
formats = []
|
formats = []
|
||||||
found = set()
|
found = set()
|
||||||
|
|
||||||
timefree_int = 0 if is_onair else 1
|
timefree_int = 0 if is_onair else 1
|
||||||
|
stream_type = 'b' if area_id in station_areas else 'c'
|
||||||
|
|
||||||
|
if stream_type == 'c' and not self._check_privileges()['areafree']:
|
||||||
|
self.raise_login_required('Programme is only available with an Areafree subscription')
|
||||||
|
|
||||||
for element in m3u8_playlist_data.findall(f'.//url[@timefree="{timefree_int}"]/playlist_create_url'):
|
for element in m3u8_playlist_data.findall(f'.//url[@timefree="{timefree_int}"]/playlist_create_url'):
|
||||||
pcu = element.text
|
pcu = element.text
|
||||||
@ -137,7 +208,7 @@ def _extract_formats(self, video_id, station, is_onair, ft, cursor, auth_token,
|
|||||||
**query,
|
**query,
|
||||||
'l': '15',
|
'l': '15',
|
||||||
'lsid': ''.join(random.choices('0123456789abcdef', k=32)),
|
'lsid': ''.join(random.choices('0123456789abcdef', k=32)),
|
||||||
'type': 'b',
|
'type': stream_type,
|
||||||
})
|
})
|
||||||
|
|
||||||
time_to_skip = None if is_onair else cursor - ft
|
time_to_skip = None if is_onair else cursor - ft
|
||||||
@ -187,7 +258,7 @@ def _real_extract(self, url):
|
|||||||
station, timestring = self._match_valid_url(url).group('station', 'timestring')
|
station, timestring = self._match_valid_url(url).group('station', 'timestring')
|
||||||
video_id = join_nonempty(station, timestring)
|
video_id = join_nonempty(station, timestring)
|
||||||
vid_int = unified_timestamp(timestring, False)
|
vid_int = unified_timestamp(timestring, False)
|
||||||
prog, station_program, ft, radio_begin, radio_end = self._find_program(video_id, station, vid_int)
|
prog, station_program, ft, radio_begin, radio_end = self._find_program(video_id, station, timestring)
|
||||||
|
|
||||||
auth_token, area_id = self._auth_client()
|
auth_token, area_id = self._auth_client()
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user