diff --git a/youtube_dl/extractor/extractors.py b/youtube_dl/extractor/extractors.py index 108d7ca695..95c03522e0 100644 --- a/youtube_dl/extractor/extractors.py +++ b/youtube_dl/extractor/extractors.py @@ -408,6 +408,7 @@ from .ivi import ( from .ivideon import IvideonIE from .iwara import IwaraIE from .izlesene import IzleseneIE +from .jamendo import JamendoIE, JamendoAlbumIE from .jeuxvideo import JeuxVideoIE from .jove import JoveIE from .jwplatform import JWPlatformIE diff --git a/youtube_dl/extractor/jamendo.py b/youtube_dl/extractor/jamendo.py new file mode 100644 index 0000000000..4aacd10f44 --- /dev/null +++ b/youtube_dl/extractor/jamendo.py @@ -0,0 +1,117 @@ +# coding: utf-8 +from __future__ import unicode_literals + +import re +from collections import namedtuple + +from ..compat import compat_urlparse +from .common import InfoExtractor + +FormatData = namedtuple('FormatData', [ + 'format_id', 'sub_domain', 'ext', 'quality']) + + +class JamendoIE(InfoExtractor): + _VALID_URL = r'https?://(?:www\.)?jamendo\.com/track/(?P[0-9]+)/(?P[\w-]+)' + _TEST = { + 'url': 'https://www.jamendo.com/track/196219/stories-from-emona-i', + 'md5': '6e9e82ed6db98678f171c25a8ed09ffd', + 'info_dict': { + 'id': '196219', + 'display_id': 'stories-from-emona-i', + 'ext': 'flac', + 'title': 'Stories from Emona I', + 'thumbnail': 're:^https?://.*\.jpg' + } + } + + def _real_extract(self, url): + url_data = self._VALID_URL_RE.match(url) + track_id = url_data.group('id') + display_id = url_data.group('display_id') + webpage = self._download_webpage(url, display_id) + + thumbnail = self._html_search_meta( + 'image', webpage, 'thumbnail', fatal=False) + title = self._html_search_meta('name', webpage, 'title') + + url_template = 'https://%s.jamendo.com/?trackid=%s&format=%s&from=app-97dab294' + format_data = [ + FormatData( + format_id='mp31', sub_domain='mp3l', ext='mp3', quality=0), + FormatData( + format_id='mp32', sub_domain='mp3d', ext='mp3', quality=1), + FormatData( + format_id='ogg1', sub_domain='ogg', ext='ogg', quality=2), + FormatData( + format_id='flac', sub_domain='flac', ext='flac', quality=3), + ] + formats = [ + { + 'format_id': fd.format_id, + 'url': url_template % (fd.sub_domain, track_id, fd.format_id), + 'ext': fd.ext, + 'quality': fd.quality + } + for fd in format_data + ] + self._check_formats(formats, video_id=display_id) + return { + 'id': track_id, + 'display_id': display_id, + 'thumbnail': thumbnail, + 'title': title, + 'formats': formats + } + + +class JamendoAlbumIE(InfoExtractor): + _VALID_URL = r'https?://(?:www\.)?jamendo\.com/album/(?P[0-9]+)/(?P[\w-]+)' + _TEST = { + 'url': 'https://www.jamendo.com/album/121486/duck-on-cover', + 'info_dict': { + 'id': '121486', + 'title': 'Duck On Cover' + }, + 'playlist_mincount': 2, + 'playlist': [ + { + 'md5': 'e1a2fcb42bda30dfac990212924149a8', + 'info_dict': { + 'id': '1032333', + 'ext': 'flac', + 'title': 'Warmachine' + } + }, + { + 'md5': '1f358d7b2f98edfe90fd55dac0799d50', + 'info_dict': { + 'id': '1032330', + 'ext': 'flac', + 'title': 'Without Your Ghost' + } + } + ], + 'params': { + 'playlistend': 2 + } + } + + def _real_extract(self, url): + url_data = self._VALID_URL_RE.match(url) + album_id = url_data.group('id') + webpage = self._download_webpage(url, url_data.group('display_id')) + + title = self._html_search_meta('name', webpage, 'title') + + track_paths = re.findall(r'