|
|
@ -2,26 +2,61 @@
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
|
|
|
|
|
|
|
import random
|
|
|
|
import random
|
|
|
|
|
|
|
|
import re
|
|
|
|
import string
|
|
|
|
import string
|
|
|
|
|
|
|
|
|
|
|
|
from .common import InfoExtractor
|
|
|
|
from .common import InfoExtractor
|
|
|
|
from ..compat import compat_HTTPError
|
|
|
|
from ..compat import compat_HTTPError
|
|
|
|
from ..utils import (
|
|
|
|
from ..utils import (
|
|
|
|
determine_ext,
|
|
|
|
determine_ext,
|
|
|
|
dict_get,
|
|
|
|
|
|
|
|
int_or_none,
|
|
|
|
int_or_none,
|
|
|
|
js_to_json,
|
|
|
|
js_to_json,
|
|
|
|
|
|
|
|
orderedSet,
|
|
|
|
|
|
|
|
qualities,
|
|
|
|
str_or_none,
|
|
|
|
str_or_none,
|
|
|
|
|
|
|
|
traverse_obj,
|
|
|
|
try_get,
|
|
|
|
try_get,
|
|
|
|
qualities,
|
|
|
|
|
|
|
|
urlencode_postdata,
|
|
|
|
urlencode_postdata,
|
|
|
|
ExtractorError,
|
|
|
|
ExtractorError,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class FunimationPageIE(InfoExtractor):
|
|
|
|
class FunimationBaseIE(InfoExtractor):
|
|
|
|
|
|
|
|
_NETRC_MACHINE = 'funimation'
|
|
|
|
|
|
|
|
_REGION = None
|
|
|
|
|
|
|
|
_TOKEN = None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _get_region(self):
|
|
|
|
|
|
|
|
region_cookie = self._get_cookies('https://www.funimation.com').get('region')
|
|
|
|
|
|
|
|
region = region_cookie.value if region_cookie else self.get_param('geo_bypass_country')
|
|
|
|
|
|
|
|
return region or traverse_obj(
|
|
|
|
|
|
|
|
self._download_json(
|
|
|
|
|
|
|
|
'https://geo-service.prd.funimationsvc.com/geo/v1/region/check', None, fatal=False,
|
|
|
|
|
|
|
|
note='Checking geo-location', errnote='Unable to fetch geo-location information'),
|
|
|
|
|
|
|
|
'region') or 'US'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _login(self):
|
|
|
|
|
|
|
|
username, password = self._get_login_info()
|
|
|
|
|
|
|
|
if username is None:
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
|
|
data = self._download_json(
|
|
|
|
|
|
|
|
'https://prod-api-funimationnow.dadcdigital.com/api/auth/login/',
|
|
|
|
|
|
|
|
None, 'Logging in', data=urlencode_postdata({
|
|
|
|
|
|
|
|
'username': username,
|
|
|
|
|
|
|
|
'password': password,
|
|
|
|
|
|
|
|
}))
|
|
|
|
|
|
|
|
return data['token']
|
|
|
|
|
|
|
|
except ExtractorError as e:
|
|
|
|
|
|
|
|
if isinstance(e.cause, compat_HTTPError) and e.cause.code == 401:
|
|
|
|
|
|
|
|
error = self._parse_json(e.cause.read().decode(), None)['error']
|
|
|
|
|
|
|
|
raise ExtractorError(error, expected=True)
|
|
|
|
|
|
|
|
raise
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class FunimationPageIE(FunimationBaseIE):
|
|
|
|
IE_NAME = 'funimation:page'
|
|
|
|
IE_NAME = 'funimation:page'
|
|
|
|
_VALID_URL = r'(?P<origin>https?://(?:www\.)?funimation(?:\.com|now\.uk))/(?P<lang>[^/]+/)?(?P<path>shows/(?P<id>[^/]+/[^/?#&]+).*$)'
|
|
|
|
_VALID_URL = r'https?://(?:www\.)?funimation(?:\.com|now\.uk)/(?:(?P<lang>[^/]+)/)?(?:shows|v)/(?P<show>[^/]+)/(?P<episode>[^/?#&]+)'
|
|
|
|
|
|
|
|
|
|
|
|
_TESTS = [{
|
|
|
|
_TESTS = [{
|
|
|
|
'url': 'https://www.funimation.com/shows/attack-on-titan-junior-high/broadcast-dub-preview/',
|
|
|
|
'url': 'https://www.funimation.com/shows/attack-on-titan-junior-high/broadcast-dub-preview/',
|
|
|
@ -46,38 +81,34 @@ class FunimationPageIE(InfoExtractor):
|
|
|
|
}, {
|
|
|
|
}, {
|
|
|
|
'url': 'https://www.funimationnow.uk/shows/puzzle-dragons-x/drop-impact/simulcast/',
|
|
|
|
'url': 'https://www.funimationnow.uk/shows/puzzle-dragons-x/drop-impact/simulcast/',
|
|
|
|
'only_matching': True,
|
|
|
|
'only_matching': True,
|
|
|
|
|
|
|
|
}, {
|
|
|
|
|
|
|
|
'url': 'https://www.funimation.com/v/a-certain-scientific-railgun/super-powered-level-5',
|
|
|
|
|
|
|
|
'only_matching': True,
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _real_initialize(self):
|
|
|
|
|
|
|
|
if not self._REGION:
|
|
|
|
|
|
|
|
FunimationBaseIE._REGION = self._get_region()
|
|
|
|
|
|
|
|
if not self._TOKEN:
|
|
|
|
|
|
|
|
FunimationBaseIE._TOKEN = self._login()
|
|
|
|
|
|
|
|
|
|
|
|
def _real_extract(self, url):
|
|
|
|
def _real_extract(self, url):
|
|
|
|
mobj = self._match_valid_url(url)
|
|
|
|
locale, show, episode = self._match_valid_url(url).group('lang', 'show', 'episode')
|
|
|
|
display_id = mobj.group('id').replace('/', '_')
|
|
|
|
|
|
|
|
if not mobj.group('lang'):
|
|
|
|
video_id = traverse_obj(self._download_json(
|
|
|
|
url = '%s/en/%s' % (mobj.group('origin'), mobj.group('path'))
|
|
|
|
f'https://title-api.prd.funimationsvc.com/v1/shows/{show}/episodes/{episode}',
|
|
|
|
|
|
|
|
f'{show}_{episode}', query={
|
|
|
|
webpage = self._download_webpage(url, display_id)
|
|
|
|
'deviceType': 'web',
|
|
|
|
title_data = self._parse_json(self._search_regex(
|
|
|
|
'region': self._REGION,
|
|
|
|
r'TITLE_DATA\s*=\s*({[^}]+})',
|
|
|
|
'locale': locale or 'en'
|
|
|
|
webpage, 'title data', default=''),
|
|
|
|
}), ('videoList', ..., 'id'), get_all=False)
|
|
|
|
display_id, js_to_json, fatal=False) or {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
video_id = (
|
|
|
|
|
|
|
|
title_data.get('id')
|
|
|
|
|
|
|
|
or self._search_regex(
|
|
|
|
|
|
|
|
(r"KANE_customdimensions.videoID\s*=\s*'(\d+)';", r'<iframe[^>]+src="/player/(\d+)'),
|
|
|
|
|
|
|
|
webpage, 'video_id', default=None)
|
|
|
|
|
|
|
|
or self._search_regex(
|
|
|
|
|
|
|
|
r'/player/(\d+)',
|
|
|
|
|
|
|
|
self._html_search_meta(['al:web:url', 'og:video:url', 'og:video:secure_url'], webpage, fatal=True),
|
|
|
|
|
|
|
|
'video id'))
|
|
|
|
|
|
|
|
return self.url_result(f'https://www.funimation.com/player/{video_id}', FunimationIE.ie_key(), video_id)
|
|
|
|
return self.url_result(f'https://www.funimation.com/player/{video_id}', FunimationIE.ie_key(), video_id)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class FunimationIE(InfoExtractor):
|
|
|
|
class FunimationIE(FunimationBaseIE):
|
|
|
|
_VALID_URL = r'https?://(?:www\.)?funimation\.com/player/(?P<id>\d+)'
|
|
|
|
_VALID_URL = r'https?://(?:www\.)?funimation\.com/player/(?P<id>\d+)'
|
|
|
|
|
|
|
|
|
|
|
|
_NETRC_MACHINE = 'funimation'
|
|
|
|
|
|
|
|
_TOKEN = None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
_TESTS = [{
|
|
|
|
_TESTS = [{
|
|
|
|
'url': 'https://www.funimation.com/player/210051',
|
|
|
|
'url': 'https://www.funimation.com/player/210051',
|
|
|
|
'info_dict': {
|
|
|
|
'info_dict': {
|
|
|
@ -93,7 +124,7 @@ class FunimationIE(InfoExtractor):
|
|
|
|
'season_number': 99,
|
|
|
|
'season_number': 99,
|
|
|
|
'series': 'Attack on Titan: Junior High',
|
|
|
|
'series': 'Attack on Titan: Junior High',
|
|
|
|
'description': '',
|
|
|
|
'description': '',
|
|
|
|
'duration': 154,
|
|
|
|
'duration': 155,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
'params': {
|
|
|
|
'params': {
|
|
|
|
'skip_download': 'm3u8',
|
|
|
|
'skip_download': 'm3u8',
|
|
|
@ -114,7 +145,7 @@ class FunimationIE(InfoExtractor):
|
|
|
|
'season_number': 99,
|
|
|
|
'season_number': 99,
|
|
|
|
'series': 'Attack on Titan: Junior High',
|
|
|
|
'series': 'Attack on Titan: Junior High',
|
|
|
|
'description': '',
|
|
|
|
'description': '',
|
|
|
|
'duration': 154,
|
|
|
|
'duration': 155,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
'params': {
|
|
|
|
'params': {
|
|
|
|
'skip_download': 'm3u8',
|
|
|
|
'skip_download': 'm3u8',
|
|
|
@ -122,26 +153,9 @@ class FunimationIE(InfoExtractor):
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
|
|
|
|
|
|
|
|
def _login(self):
|
|
|
|
|
|
|
|
username, password = self._get_login_info()
|
|
|
|
|
|
|
|
if username is None:
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
|
|
data = self._download_json(
|
|
|
|
|
|
|
|
'https://prod-api-funimationnow.dadcdigital.com/api/auth/login/',
|
|
|
|
|
|
|
|
None, 'Logging in', data=urlencode_postdata({
|
|
|
|
|
|
|
|
'username': username,
|
|
|
|
|
|
|
|
'password': password,
|
|
|
|
|
|
|
|
}))
|
|
|
|
|
|
|
|
self._TOKEN = data['token']
|
|
|
|
|
|
|
|
except ExtractorError as e:
|
|
|
|
|
|
|
|
if isinstance(e.cause, compat_HTTPError) and e.cause.code == 401:
|
|
|
|
|
|
|
|
error = self._parse_json(e.cause.read().decode(), None)['error']
|
|
|
|
|
|
|
|
raise ExtractorError(error, expected=True)
|
|
|
|
|
|
|
|
raise
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _real_initialize(self):
|
|
|
|
def _real_initialize(self):
|
|
|
|
self._login()
|
|
|
|
if not self._TOKEN:
|
|
|
|
|
|
|
|
FunimationBaseIE._TOKEN = self._login()
|
|
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
@staticmethod
|
|
|
|
def _get_experiences(episode):
|
|
|
|
def _get_experiences(episode):
|
|
|
@ -283,7 +297,7 @@ class FunimationIE(InfoExtractor):
|
|
|
|
return subtitles
|
|
|
|
return subtitles
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class FunimationShowIE(FunimationIE):
|
|
|
|
class FunimationShowIE(FunimationBaseIE):
|
|
|
|
IE_NAME = 'funimation:show'
|
|
|
|
IE_NAME = 'funimation:show'
|
|
|
|
_VALID_URL = r'(?P<url>https?://(?:www\.)?funimation(?:\.com|now\.uk)/(?P<locale>[^/]+)?/?shows/(?P<id>[^/?#&]+))/?(?:[?#]|$)'
|
|
|
|
_VALID_URL = r'(?P<url>https?://(?:www\.)?funimation(?:\.com|now\.uk)/(?P<locale>[^/]+)?/?shows/(?P<id>[^/?#&]+))/?(?:[?#]|$)'
|
|
|
|
|
|
|
|
|
|
|
@ -311,31 +325,28 @@ class FunimationShowIE(FunimationIE):
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
|
|
|
|
|
|
|
|
def _real_initialize(self):
|
|
|
|
def _real_initialize(self):
|
|
|
|
region = self._get_cookies('https://www.funimation.com').get('region')
|
|
|
|
if not self._REGION:
|
|
|
|
self._region = region.value if region else try_get(
|
|
|
|
FunimationBaseIE._REGION = self._get_region()
|
|
|
|
self._download_json(
|
|
|
|
|
|
|
|
'https://geo-service.prd.funimationsvc.com/geo/v1/region/check', None, fatal=False,
|
|
|
|
|
|
|
|
note='Checking geo-location', errnote='Unable to fetch geo-location information'),
|
|
|
|
|
|
|
|
lambda x: x['region']) or 'US'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _real_extract(self, url):
|
|
|
|
def _real_extract(self, url):
|
|
|
|
base_url, locale, display_id = self._match_valid_url(url).groups()
|
|
|
|
base_url, locale, display_id = self._match_valid_url(url).groups()
|
|
|
|
|
|
|
|
|
|
|
|
show_info = self._download_json(
|
|
|
|
show_info = self._download_json(
|
|
|
|
'https://title-api.prd.funimationsvc.com/v2/shows/%s?region=%s&deviceType=web&locale=%s'
|
|
|
|
'https://title-api.prd.funimationsvc.com/v2/shows/%s?region=%s&deviceType=web&locale=%s'
|
|
|
|
% (display_id, self._region, locale or 'en'), display_id)
|
|
|
|
% (display_id, self._REGION, locale or 'en'), display_id)
|
|
|
|
items = self._download_json(
|
|
|
|
items_info = self._download_json(
|
|
|
|
'https://prod-api-funimationnow.dadcdigital.com/api/funimation/episodes/?limit=99999&title_id=%s'
|
|
|
|
'https://prod-api-funimationnow.dadcdigital.com/api/funimation/episodes/?limit=99999&title_id=%s'
|
|
|
|
% show_info.get('id'), display_id).get('items')
|
|
|
|
% show_info.get('id'), display_id)
|
|
|
|
vod_items = map(lambda k: dict_get(k, ('mostRecentSvod', 'mostRecentAvod')).get('item'), items)
|
|
|
|
|
|
|
|
|
|
|
|
vod_items = traverse_obj(items_info, ('items', ..., re.compile('(?i)mostRecent[AS]vod').match, 'item'))
|
|
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
return {
|
|
|
|
'_type': 'playlist',
|
|
|
|
'_type': 'playlist',
|
|
|
|
'id': show_info['id'],
|
|
|
|
'id': show_info['id'],
|
|
|
|
'title': show_info['name'],
|
|
|
|
'title': show_info['name'],
|
|
|
|
'entries': [
|
|
|
|
'entries': orderedSet(
|
|
|
|
self.url_result(
|
|
|
|
self.url_result(
|
|
|
|
'%s/%s' % (base_url, vod_item.get('episodeSlug')), FunimationPageIE.ie_key(),
|
|
|
|
'%s/%s' % (base_url, vod_item.get('episodeSlug')), FunimationPageIE.ie_key(),
|
|
|
|
vod_item.get('episodeId'), vod_item.get('episodeName'))
|
|
|
|
vod_item.get('episodeId'), vod_item.get('episodeName'))
|
|
|
|
for vod_item in sorted(vod_items, key=lambda x: x.get('episodeOrder'))],
|
|
|
|
for vod_item in sorted(vod_items, key=lambda x: x.get('episodeOrder', -1))),
|
|
|
|
}
|
|
|
|
}
|
|
|
|