From 125728b038a17f53397a85584b4e9205ca3a062c Mon Sep 17 00:00:00 2001 From: Mevious Date: Thu, 24 Jun 2021 17:15:23 -0700 Subject: [PATCH] [funimation] Add `FunimationShowIE` (#442) Closes #436 Authored by: Mevious --- yt_dlp/extractor/extractors.py | 5 +++- yt_dlp/extractor/funimation.py | 52 ++++++++++++++++++++++++++++++++++ 2 files changed, 56 insertions(+), 1 deletion(-) diff --git a/yt_dlp/extractor/extractors.py b/yt_dlp/extractor/extractors.py index 8af15ec7b..a084b4b2a 100644 --- a/yt_dlp/extractor/extractors.py +++ b/yt_dlp/extractor/extractors.py @@ -455,7 +455,10 @@ from .frontendmasters import ( FrontendMastersCourseIE ) from .fujitv import FujiTVFODPlus7IE -from .funimation import FunimationIE +from .funimation import ( + FunimationIE, + FunimationShowIE +) from .funk import FunkIE from .fusion import FusionIE from .gaia import GaiaIE diff --git a/yt_dlp/extractor/funimation.py b/yt_dlp/extractor/funimation.py index 9ec1627fa..310986574 100644 --- a/yt_dlp/extractor/funimation.py +++ b/yt_dlp/extractor/funimation.py @@ -2,12 +2,14 @@ from __future__ import unicode_literals import random +import re import string from .common import InfoExtractor from ..compat import compat_HTTPError from ..utils import ( determine_ext, + dict_get, int_or_none, js_to_json, urlencode_postdata, @@ -176,3 +178,53 @@ class FunimationIE(InfoExtractor): language += '_CC' subtitles.setdefault(language, []).append(url_element) return subtitles + + +class FunimationShowIE(FunimationIE): + IE_NAME = 'funimation:show' + _VALID_URL = r'(?Phttps?://(?:www\.)?funimation(?:\.com|now\.uk)/(?P[^/]+)?/?shows/(?P[^/?#&]+))/?(?:[?#]|$)' + + _TESTS = [{ + 'url': 'https://www.funimation.com/en/shows/sk8-the-infinity', + 'info_dict': { + 'id': 1315000, + 'title': 'SK8 the Infinity' + }, + 'playlist_count': 13, + 'params': { + 'skip_download': True, + }, + }, { + # without lang code + 'url': 'https://www.funimation.com/shows/ouran-high-school-host-club/', + 'info_dict': { + 'id': 39643, + 'title': 'Ouran High School Host Club' + }, + 'playlist_count': 26, + 'params': { + 'skip_download': True, + }, + }] + + def _real_extract(self, url): + base_url, locale, display_id = re.match(self._VALID_URL, url).groups() + + show_info = self._download_json( + 'https://title-api.prd.funimationsvc.com/v2/shows/%s?region=US&deviceType=web&locale=%s' + % (display_id, locale or 'en'), display_id) + items = self._download_json( + 'https://prod-api-funimationnow.dadcdigital.com/api/funimation/episodes/?limit=99999&title_id=%s' + % show_info.get('id'), display_id).get('items') + vod_items = map(lambda k: dict_get(k, ('mostRecentSvod', 'mostRecentAvod')).get('item'), items) + + return { + '_type': 'playlist', + 'id': show_info['id'], + 'title': show_info['name'], + 'entries': [ + self.url_result( + '%s/%s' % (base_url, vod_item.get('episodeSlug')), FunimationIE.ie_key(), + vod_item.get('episodeId'), vod_item.get('episodeName')) + for vod_item in sorted(vod_items, key=lambda x: x.get('episodeOrder'))], + }