From 8a0905798e1716100ed3c39b397df3a4012a2f90 Mon Sep 17 00:00:00 2001 From: joshwd36 Date: Thu, 2 Jun 2022 17:33:02 +0100 Subject: [PATCH] Add MacProVideo extractor --- docs/supportedsites.md | 1 + youtube_dl/extractor/extractors.py | 4 ++ youtube_dl/extractor/macprovideo.py | 86 +++++++++++++++++++++++++++++ 3 files changed, 91 insertions(+) create mode 100644 youtube_dl/extractor/macprovideo.py diff --git a/docs/supportedsites.md b/docs/supportedsites.md index ae2a6b8b0..e9d8d416d 100644 --- a/docs/supportedsites.md +++ b/docs/supportedsites.md @@ -482,6 +482,7 @@ - **lynda**: lynda.com videos - **lynda:course**: lynda.com online courses - **m6** + - **MacProVideo.com** - **mailru**: Видео@Mail.Ru - **mailru:music**: Музыка@Mail.Ru - **mailru:music:search**: Музыка@Mail.Ru diff --git a/youtube_dl/extractor/extractors.py b/youtube_dl/extractor/extractors.py index 452caeade..ade0d31ea 100644 --- a/youtube_dl/extractor/extractors.py +++ b/youtube_dl/extractor/extractors.py @@ -635,6 +635,10 @@ from .lynda import ( LyndaCourseIE ) from .m6 import M6IE +from .macprovideo import ( + MacProVideoIE, + MacProVideoCourseIE +) from .mailru import ( MailRuIE, MailRuMusicIE, diff --git a/youtube_dl/extractor/macprovideo.py b/youtube_dl/extractor/macprovideo.py new file mode 100644 index 000000000..a3c84c09d --- /dev/null +++ b/youtube_dl/extractor/macprovideo.py @@ -0,0 +1,86 @@ +# coding: utf-8 +from __future__ import unicode_literals + +import re + +from youtube_dl.utils import try_get, str_or_none + +from .common import InfoExtractor + + +class MacProVideoBaseIE(InfoExtractor): + def _get_video_info(self, data): + meta = data.get('ned_programmanifest') + video_id = str_or_none(meta.get('fk_videoMediaID')) + + formats = self._extract_m3u8_formats( + data['video_medium']['mediaURL'], video_id, 'mp4', m3u8_id='hls', fatal=False) + self._sort_formats(formats) + + description = try_get(data, lambda x: x['ned_programmanifest_description']['description']) + thumbnail = try_get(data, lambda x: x['thumbnail_medium']['mediaURL']) + + return { + 'id': video_id, + 'title': meta.get('vid_title'), + 'description': description, + 'thumbnail': thumbnail, + 'duration': meta.get('vid_duration'), + 'formats': formats + } + + +class MacProVideoIE(MacProVideoBaseIE): + _VALID_URL = r'https?://(?:www\.)?macprovideo\.com/video/(?P[0-9a-zA-Z-]+)/(?P[0-9]+)-[0-9a-zA-Z-]+' + _TEST = { + 'url': 'https://www.macprovideo.com/video/apple-logic-pro-102-recording-and-editing-audio/1-1-transducers-and-converters', + 'md5': 'md5:608d7c8cf9f31202cbd269273df62aaf', + 'info_dict': { + 'id': '319676', + 'ext': 'mp4', + 'title': '1. Transducers and Converters', + 'thumbnail': r're:^https?://.*\.jpg$', + 'description': 'md5:c16dc8a42252e2be44225f129c591c3d', + }, + 'params': { + # m3u8 download + 'skip_download': True, + 'format': 'bestvideo' + }, + } + + def _real_extract(self, url): + course, num = re.match(self._VALID_URL, url).groups() + num = int(num) - 1 + data = self._download_json('https://www.macprovideo.com/ajax/videos/' + course, + None)['data']['videos'][num] + return self._get_video_info(data) + + +class MacProVideoCourseIE(MacProVideoBaseIE): + _VALID_URL = r'https?://(?:www\.)?macprovideo\.com/course/(?P[0-9a-zA-Z-]+)' + _TEST = { + 'url': 'https://www.macprovideo.com/course/apple-logic-pro-102-recording-and-editing-audio', + 'md5': 'md5:608d7c8cf9f31202cbd269273df62aaf', + 'info_dict': { + 'id': '319676', + 'ext': 'mp4', + 'title': '1. Transducers and Converters', + 'thumbnail': r're:^https?://.*\.jpg$', + 'description': 'md5:c16dc8a42252e2be44225f129c591c3d', + }, + 'params': { + # m3u8 download + 'skip_download': True, + 'format': 'bestvideo' + }, + } + + def _real_extract(self, url): + course_id = self._match_id(url) + videos = self._download_json('https://www.macprovideo.com/ajax/videos/' + course_id, None)['data']['videos'] + title = try_get(videos, lambda x: x[0]['tutorial']['tu_title']) + + return self.playlist_result( + [self._get_video_info(data) for data in videos], + playlist_id=course_id, playlist_title=title)