|
@ -3,10 +3,14 @@ from __future__ import unicode_literals |
|
|
import re |
|
|
import re |
|
|
|
|
|
|
|
|
from .common import InfoExtractor |
|
|
from .common import InfoExtractor |
|
|
from ..compat import compat_urlparse |
|
|
|
|
|
|
|
|
from ..compat import ( |
|
|
|
|
|
compat_urlparse, |
|
|
|
|
|
compat_str, |
|
|
|
|
|
) |
|
|
from ..utils import ( |
|
|
from ..utils import ( |
|
|
determine_ext, |
|
|
determine_ext, |
|
|
encode_dict, |
|
|
encode_dict, |
|
|
|
|
|
extract_attributes, |
|
|
ExtractorError, |
|
|
ExtractorError, |
|
|
sanitized_Request, |
|
|
sanitized_Request, |
|
|
urlencode_postdata, |
|
|
urlencode_postdata, |
|
@ -34,6 +38,10 @@ class AnimeOnDemandIE(InfoExtractor): |
|
|
# Episodes without titles |
|
|
# Episodes without titles |
|
|
'url': 'https://www.anime-on-demand.de/anime/162', |
|
|
'url': 'https://www.anime-on-demand.de/anime/162', |
|
|
'only_matching': True, |
|
|
'only_matching': True, |
|
|
|
|
|
}, { |
|
|
|
|
|
# ger/jap, Dub/OmU, account required |
|
|
|
|
|
'url': 'https://www.anime-on-demand.de/anime/169', |
|
|
|
|
|
'only_matching': True, |
|
|
}] |
|
|
}] |
|
|
|
|
|
|
|
|
def _login(self): |
|
|
def _login(self): |
|
@ -130,33 +138,70 @@ class AnimeOnDemandIE(InfoExtractor): |
|
|
|
|
|
|
|
|
formats = [] |
|
|
formats = [] |
|
|
|
|
|
|
|
|
playlist_url = self._search_regex( |
|
|
|
|
|
r'data-playlist=(["\'])(?P<url>.+?)\1', |
|
|
|
|
|
episode_html, 'data playlist', default=None, group='url') |
|
|
|
|
|
if playlist_url: |
|
|
|
|
|
request = sanitized_Request( |
|
|
|
|
|
compat_urlparse.urljoin(url, playlist_url), |
|
|
|
|
|
headers={ |
|
|
|
|
|
'X-Requested-With': 'XMLHttpRequest', |
|
|
|
|
|
'X-CSRF-Token': csrf_token, |
|
|
|
|
|
'Referer': url, |
|
|
|
|
|
'Accept': 'application/json, text/javascript, */*; q=0.01', |
|
|
|
|
|
}) |
|
|
|
|
|
|
|
|
|
|
|
playlist = self._download_json( |
|
|
|
|
|
request, video_id, 'Downloading playlist JSON', fatal=False) |
|
|
|
|
|
if playlist: |
|
|
|
|
|
playlist = playlist['playlist'][0] |
|
|
|
|
|
title = playlist['title'] |
|
|
|
|
|
|
|
|
for input_ in re.findall( |
|
|
|
|
|
r'<input[^>]+class=["\'].*?streamstarter_html5[^>]+>', episode_html): |
|
|
|
|
|
attributes = extract_attributes(input_) |
|
|
|
|
|
playlist_urls = [] |
|
|
|
|
|
for playlist_key in ('data-playlist', 'data-otherplaylist'): |
|
|
|
|
|
playlist_url = attributes.get(playlist_key) |
|
|
|
|
|
if isinstance(playlist_url, compat_str) and re.match( |
|
|
|
|
|
r'/?[\da-zA-Z]+', playlist_url): |
|
|
|
|
|
playlist_urls.append(attributes[playlist_key]) |
|
|
|
|
|
if not playlist_urls: |
|
|
|
|
|
continue |
|
|
|
|
|
|
|
|
|
|
|
lang = attributes.get('data-lang') |
|
|
|
|
|
lang_note = attributes.get('value') |
|
|
|
|
|
|
|
|
|
|
|
for playlist_url in playlist_urls: |
|
|
|
|
|
kind = self._search_regex( |
|
|
|
|
|
r'videomaterialurl/\d+/([^/]+)/', |
|
|
|
|
|
playlist_url, 'media kind', default=None) |
|
|
|
|
|
format_id_list = [] |
|
|
|
|
|
if lang: |
|
|
|
|
|
format_id_list.append(lang) |
|
|
|
|
|
if kind: |
|
|
|
|
|
format_id_list.append(kind) |
|
|
|
|
|
if not format_id_list: |
|
|
|
|
|
format_id_list.append('hls') |
|
|
|
|
|
format_id = '-'.join(format_id_list) |
|
|
|
|
|
format_note = ', '.join(filter(None, (kind, lang_note))) |
|
|
|
|
|
request = sanitized_Request( |
|
|
|
|
|
compat_urlparse.urljoin(url, playlist_url), |
|
|
|
|
|
headers={ |
|
|
|
|
|
'X-Requested-With': 'XMLHttpRequest', |
|
|
|
|
|
'X-CSRF-Token': csrf_token, |
|
|
|
|
|
'Referer': url, |
|
|
|
|
|
'Accept': 'application/json, text/javascript, */*; q=0.01', |
|
|
|
|
|
}) |
|
|
|
|
|
playlist = self._download_json( |
|
|
|
|
|
request, video_id, 'Downloading %s playlist JSON' % format_id, |
|
|
|
|
|
fatal=False) |
|
|
|
|
|
if not playlist: |
|
|
|
|
|
continue |
|
|
|
|
|
playlist = playlist.get('playlist') |
|
|
|
|
|
if not playlist or not isinstance(playlist, list): |
|
|
|
|
|
continue |
|
|
|
|
|
playlist = playlist[0] |
|
|
|
|
|
title = playlist.get('title') |
|
|
|
|
|
if not title: |
|
|
|
|
|
continue |
|
|
description = playlist.get('description') |
|
|
description = playlist.get('description') |
|
|
for source in playlist.get('sources', []): |
|
|
for source in playlist.get('sources', []): |
|
|
file_ = source.get('file') |
|
|
file_ = source.get('file') |
|
|
if file_ and determine_ext(file_) == 'm3u8': |
|
|
if file_ and determine_ext(file_) == 'm3u8': |
|
|
formats = self._extract_m3u8_formats( |
|
|
|
|
|
|
|
|
m3u8_formats = self._extract_m3u8_formats( |
|
|
file_, video_id, 'mp4', |
|
|
file_, video_id, 'mp4', |
|
|
entry_protocol='m3u8_native', m3u8_id='hls') |
|
|
|
|
|
|
|
|
entry_protocol='m3u8_native', m3u8_id=format_id) |
|
|
|
|
|
for f in m3u8_formats: |
|
|
|
|
|
f.update({ |
|
|
|
|
|
'language': lang, |
|
|
|
|
|
'format_note': format_note, |
|
|
|
|
|
}) |
|
|
|
|
|
formats.extend(m3u8_formats) |
|
|
|
|
|
|
|
|
if formats: |
|
|
if formats: |
|
|
|
|
|
self._sort_formats(formats) |
|
|
f = common_info.copy() |
|
|
f = common_info.copy() |
|
|
f.update({ |
|
|
f.update({ |
|
|
'title': title, |
|
|
'title': title, |
|
|