]>
Commit | Line | Data |
---|---|---|
8084951b PH |
1 | # coding: utf-8 |
2 | from __future__ import unicode_literals | |
3 | ||
8084951b | 4 | from .common import InfoExtractor |
514e8aef | 5 | from ..compat import compat_str |
dc6520aa | 6 | from ..utils import ( |
514e8aef | 7 | determine_ext, |
dc6520aa S |
8 | int_or_none, |
9 | try_get, | |
10 | unified_timestamp, | |
3052a30d | 11 | url_or_none, |
dc6520aa | 12 | ) |
8084951b PH |
13 | |
14 | ||
15 | class EggheadCourseIE(InfoExtractor): | |
16 | IE_DESC = 'egghead.io course' | |
17 | IE_NAME = 'egghead:course' | |
485cb375 | 18 | _VALID_URL = r'https://egghead\.io/courses/(?P<id>[^/?#&]+)' |
8084951b PH |
19 | _TEST = { |
20 | 'url': 'https://egghead.io/courses/professor-frisby-introduces-composable-functional-javascript', | |
21 | 'playlist_count': 29, | |
22 | 'info_dict': { | |
514e8aef | 23 | 'id': '72', |
8084951b PH |
24 | 'title': 'Professor Frisby Introduces Composable Functional JavaScript', |
25 | 'description': 're:(?s)^This course teaches the ubiquitous.*You\'ll start composing functionality before you know it.$', | |
26 | }, | |
27 | } | |
28 | ||
29 | def _real_extract(self, url): | |
30 | playlist_id = self._match_id(url) | |
8084951b | 31 | |
514e8aef S |
32 | lessons = self._download_json( |
33 | 'https://egghead.io/api/v1/series/%s/lessons' % playlist_id, | |
34 | playlist_id, 'Downloading course lessons JSON') | |
35 | ||
36 | entries = [] | |
37 | for lesson in lessons: | |
3052a30d S |
38 | lesson_url = url_or_none(lesson.get('http_url')) |
39 | if not lesson_url: | |
514e8aef S |
40 | continue |
41 | lesson_id = lesson.get('id') | |
42 | if lesson_id: | |
43 | lesson_id = compat_str(lesson_id) | |
44 | entries.append(self.url_result( | |
45 | lesson_url, ie=EggheadLessonIE.ie_key(), video_id=lesson_id)) | |
46 | ||
485cb375 | 47 | course = self._download_json( |
514e8aef S |
48 | 'https://egghead.io/api/v1/series/%s' % playlist_id, |
49 | playlist_id, 'Downloading course JSON', fatal=False) or {} | |
8084951b | 50 | |
514e8aef S |
51 | playlist_id = course.get('id') |
52 | if playlist_id: | |
53 | playlist_id = compat_str(playlist_id) | |
485cb375 S |
54 | |
55 | return self.playlist_result( | |
56 | entries, playlist_id, course.get('title'), | |
57 | course.get('description')) | |
dc6520aa S |
58 | |
59 | ||
60 | class EggheadLessonIE(InfoExtractor): | |
61 | IE_DESC = 'egghead.io lesson' | |
62 | IE_NAME = 'egghead:lesson' | |
514e8aef S |
63 | _VALID_URL = r'https://egghead\.io/(?:api/v1/)?lessons/(?P<id>[^/?#&]+)' |
64 | _TESTS = [{ | |
dc6520aa S |
65 | 'url': 'https://egghead.io/lessons/javascript-linear-data-flow-with-container-style-types-box', |
66 | 'info_dict': { | |
514e8aef S |
67 | 'id': '1196', |
68 | 'display_id': 'javascript-linear-data-flow-with-container-style-types-box', | |
dc6520aa S |
69 | 'ext': 'mp4', |
70 | 'title': 'Create linear data flow with container style types (Box)', | |
71 | 'description': 'md5:9aa2cdb6f9878ed4c39ec09e85a8150e', | |
72 | 'thumbnail': r're:^https?:.*\.jpg$', | |
73 | 'timestamp': 1481296768, | |
74 | 'upload_date': '20161209', | |
75 | 'duration': 304, | |
76 | 'view_count': 0, | |
77 | 'tags': ['javascript', 'free'], | |
78 | }, | |
79 | 'params': { | |
80 | 'skip_download': True, | |
514e8aef | 81 | 'format': 'bestvideo', |
dc6520aa | 82 | }, |
514e8aef S |
83 | }, { |
84 | 'url': 'https://egghead.io/api/v1/lessons/react-add-redux-to-a-react-application', | |
85 | 'only_matching': True, | |
86 | }] | |
dc6520aa S |
87 | |
88 | def _real_extract(self, url): | |
514e8aef | 89 | display_id = self._match_id(url) |
dc6520aa S |
90 | |
91 | lesson = self._download_json( | |
514e8aef S |
92 | 'https://egghead.io/api/v1/lessons/%s' % display_id, display_id) |
93 | ||
94 | lesson_id = compat_str(lesson['id']) | |
95 | title = lesson['title'] | |
96 | ||
97 | formats = [] | |
98 | for _, format_url in lesson['media_urls'].items(): | |
3052a30d S |
99 | format_url = url_or_none(format_url) |
100 | if not format_url: | |
514e8aef S |
101 | continue |
102 | ext = determine_ext(format_url) | |
103 | if ext == 'm3u8': | |
104 | formats.extend(self._extract_m3u8_formats( | |
105 | format_url, lesson_id, 'mp4', entry_protocol='m3u8', | |
106 | m3u8_id='hls', fatal=False)) | |
107 | elif ext == 'mpd': | |
108 | formats.extend(self._extract_mpd_formats( | |
109 | format_url, lesson_id, mpd_id='dash', fatal=False)) | |
110 | else: | |
111 | formats.append({ | |
112 | 'url': format_url, | |
113 | }) | |
114 | self._sort_formats(formats) | |
dc6520aa S |
115 | |
116 | return { | |
514e8aef S |
117 | 'id': lesson_id, |
118 | 'display_id': display_id, | |
119 | 'title': title, | |
dc6520aa S |
120 | 'description': lesson.get('summary'), |
121 | 'thumbnail': lesson.get('thumb_nail'), | |
122 | 'timestamp': unified_timestamp(lesson.get('published_at')), | |
123 | 'duration': int_or_none(lesson.get('duration')), | |
124 | 'view_count': int_or_none(lesson.get('plays_count')), | |
125 | 'tags': try_get(lesson, lambda x: x['tag_list'], list), | |
514e8aef S |
126 | 'series': try_get( |
127 | lesson, lambda x: x['series']['title'], compat_str), | |
128 | 'formats': formats, | |
dc6520aa | 129 | } |