| 
									
										
										
										
											2016-10-02 13:39:18 +02:00
										 |  |  | # coding: utf-8 | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  | from __future__ import unicode_literals | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  | import itertools | 
					
						
							| 
									
										
										
										
											2018-04-02 22:19:42 +07:00
										 |  |  | import json | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 23:06:13 +07:00
										 |  |  | from .common import InfoExtractor | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  | from ..compat import ( | 
					
						
							| 
									
										
										
										
											2018-04-02 22:19:42 +07:00
										 |  |  |     compat_HTTPError, | 
					
						
							| 
									
										
										
										
											2018-03-29 23:06:13 +07:00
										 |  |  |     compat_str, | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  |     compat_urlparse, | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | from ..utils import ( | 
					
						
							|  |  |  |     clean_html, | 
					
						
							| 
									
										
										
										
											2018-03-29 23:06:13 +07:00
										 |  |  |     ExtractorError, | 
					
						
							| 
									
										
										
										
											2016-01-05 01:17:33 +06:00
										 |  |  |     int_or_none, | 
					
						
							| 
									
										
										
										
											2018-03-29 23:06:13 +07:00
										 |  |  |     parse_age_limit, | 
					
						
							|  |  |  |     parse_duration, | 
					
						
							|  |  |  |     unified_timestamp, | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 23:06:13 +07:00
										 |  |  | class DramaFeverBaseIE(InfoExtractor): | 
					
						
							| 
									
										
										
										
											2015-06-19 21:57:31 +06:00
										 |  |  |     _NETRC_MACHINE = 'dramafever' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-14 01:35:44 +06:00
										 |  |  |     _CONSUMER_SECRET = 'DA59dtVXYLxajktV' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _consumer_secret = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_consumer_secret(self): | 
					
						
							|  |  |  |         mainjs = self._download_webpage( | 
					
						
							|  |  |  |             'http://www.dramafever.com/static/51afe95/df2014/scripts/main.js', | 
					
						
							|  |  |  |             None, 'Downloading main.js', fatal=False) | 
					
						
							|  |  |  |         if not mainjs: | 
					
						
							|  |  |  |             return self._CONSUMER_SECRET | 
					
						
							|  |  |  |         return self._search_regex( | 
					
						
							|  |  |  |             r"var\s+cs\s*=\s*'([^']+)'", mainjs, | 
					
						
							|  |  |  |             'consumer secret', default=self._CONSUMER_SECRET) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-19 21:57:31 +06:00
										 |  |  |     def _real_initialize(self): | 
					
						
							| 
									
										
										
										
											2015-07-14 01:35:44 +06:00
										 |  |  |         self._consumer_secret = self._get_consumer_secret() | 
					
						
							| 
									
										
										
										
											2018-04-02 22:19:42 +07:00
										 |  |  |         self._login() | 
					
						
							| 
									
										
										
										
											2015-06-19 21:57:31 +06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _login(self): | 
					
						
							| 
									
										
										
										
											2018-05-26 16:12:44 +01:00
										 |  |  |         username, password = self._get_login_info() | 
					
						
							| 
									
										
										
										
											2015-06-19 21:57:31 +06:00
										 |  |  |         if username is None: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         login_form = { | 
					
						
							|  |  |  |             'username': username, | 
					
						
							|  |  |  |             'password': password, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-02 22:19:42 +07:00
										 |  |  |         try: | 
					
						
							|  |  |  |             response = self._download_json( | 
					
						
							|  |  |  |                 'https://www.dramafever.com/api/users/login', None, 'Logging in', | 
					
						
							|  |  |  |                 data=json.dumps(login_form).encode('utf-8'), headers={ | 
					
						
							|  |  |  |                     'x-consumer-key': self._consumer_secret, | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  |         except ExtractorError as e: | 
					
						
							|  |  |  |             if isinstance(e.cause, compat_HTTPError) and e.cause.code in (403, 404): | 
					
						
							|  |  |  |                 response = self._parse_json( | 
					
						
							|  |  |  |                     e.cause.read().decode('utf-8'), None) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Successful login | 
					
						
							|  |  |  |         if response.get('result') or response.get('guid') or response.get('user_guid'): | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2015-06-19 21:57:31 +06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-02 22:19:42 +07:00
										 |  |  |         errors = response.get('errors') | 
					
						
							|  |  |  |         if errors and isinstance(errors, list): | 
					
						
							|  |  |  |             error = errors[0] | 
					
						
							|  |  |  |             message = error.get('message') or error['reason'] | 
					
						
							|  |  |  |             raise ExtractorError('Unable to login: %s' % message, expected=True) | 
					
						
							|  |  |  |         raise ExtractorError('Unable to log in') | 
					
						
							| 
									
										
										
										
											2015-06-19 21:57:31 +06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class DramaFeverIE(DramaFeverBaseIE): | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  |     IE_NAME = 'dramafever' | 
					
						
							| 
									
										
										
										
											2017-01-14 18:27:22 +07:00
										 |  |  |     _VALID_URL = r'https?://(?:www\.)?dramafever\.com/(?:[^/]+/)?drama/(?P<id>[0-9]+/[0-9]+)(?:/|$)' | 
					
						
							| 
									
										
										
										
											2016-01-05 01:41:18 +06:00
										 |  |  |     _TESTS = [{ | 
					
						
							| 
									
										
										
										
											2018-03-29 23:06:13 +07:00
										 |  |  |         'url': 'https://www.dramafever.com/drama/4274/1/Heirs/', | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  |         'info_dict': { | 
					
						
							| 
									
										
										
										
											2018-03-29 23:06:13 +07:00
										 |  |  |             'id': '4274.1', | 
					
						
							|  |  |  |             'ext': 'wvm', | 
					
						
							|  |  |  |             'title': 'Heirs - Episode 1', | 
					
						
							|  |  |  |             'description': 'md5:362a24ba18209f6276e032a651c50bc2', | 
					
						
							|  |  |  |             'thumbnail': r're:^https?://.*\.jpg', | 
					
						
							|  |  |  |             'duration': 3783, | 
					
						
							|  |  |  |             'timestamp': 1381354993, | 
					
						
							|  |  |  |             'upload_date': '20131009', | 
					
						
							|  |  |  |             'series': 'Heirs', | 
					
						
							|  |  |  |             'season_number': 1, | 
					
						
							| 
									
										
										
										
											2016-01-05 01:41:18 +06:00
										 |  |  |             'episode': 'Episode 1', | 
					
						
							|  |  |  |             'episode_number': 1, | 
					
						
							| 
									
										
										
										
											2015-11-07 16:54:35 +01:00
										 |  |  |         }, | 
					
						
							|  |  |  |         'params': { | 
					
						
							|  |  |  |             # m3u8 download | 
					
						
							|  |  |  |             'skip_download': True, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-01-05 01:41:18 +06:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'http://www.dramafever.com/drama/4826/4/Mnet_Asian_Music_Awards_2015/?ap=1', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': '4826.4', | 
					
						
							| 
									
										
										
										
											2017-07-22 11:41:40 +07:00
										 |  |  |             'ext': 'flv', | 
					
						
							| 
									
										
										
										
											2017-07-22 11:40:46 +07:00
										 |  |  |             'title': 'Mnet Asian Music Awards 2015', | 
					
						
							| 
									
										
										
										
											2016-01-05 01:41:18 +06:00
										 |  |  |             'description': 'md5:3ff2ee8fedaef86e076791c909cf2e91', | 
					
						
							|  |  |  |             'episode': 'Mnet Asian Music Awards 2015 - Part 3', | 
					
						
							|  |  |  |             'episode_number': 4, | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |             'thumbnail': r're:^https?://.*\.jpg', | 
					
						
							| 
									
										
										
										
											2016-01-05 01:41:18 +06:00
										 |  |  |             'timestamp': 1450213200, | 
					
						
							|  |  |  |             'upload_date': '20151215', | 
					
						
							| 
									
										
										
										
											2017-07-22 11:41:40 +07:00
										 |  |  |             'duration': 5359, | 
					
						
							| 
									
										
										
										
											2016-01-05 01:41:18 +06:00
										 |  |  |         }, | 
					
						
							|  |  |  |         'params': { | 
					
						
							|  |  |  |             # m3u8 download | 
					
						
							|  |  |  |             'skip_download': True, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2017-01-14 18:27:22 +07:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'https://www.dramafever.com/zh-cn/drama/4972/15/Doctor_Romantic/', | 
					
						
							|  |  |  |         'only_matching': True, | 
					
						
							| 
									
										
										
										
											2016-01-05 01:41:18 +06:00
										 |  |  |     }] | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 23:06:13 +07:00
										 |  |  |     def _call_api(self, path, video_id, note, fatal=False): | 
					
						
							|  |  |  |         return self._download_json( | 
					
						
							|  |  |  |             'https://www.dramafever.com/api/5/' + path, | 
					
						
							|  |  |  |             video_id, note=note, headers={ | 
					
						
							|  |  |  |                 'x-consumer-key': self._consumer_secret, | 
					
						
							|  |  |  |             }, fatal=fatal) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_subtitles(self, video_id): | 
					
						
							|  |  |  |         subtitles = {} | 
					
						
							|  |  |  |         subs = self._call_api( | 
					
						
							|  |  |  |             'video/%s/subtitles/webvtt/' % video_id, video_id, | 
					
						
							|  |  |  |             'Downloading subtitles JSON', fatal=False) | 
					
						
							|  |  |  |         if not subs or not isinstance(subs, list): | 
					
						
							|  |  |  |             return subtitles | 
					
						
							|  |  |  |         for sub in subs: | 
					
						
							|  |  |  |             if not isinstance(sub, dict): | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             sub_url = sub.get('url') | 
					
						
							|  |  |  |             if not sub_url or not isinstance(sub_url, compat_str): | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             subtitles.setdefault( | 
					
						
							|  |  |  |                 sub.get('code') or sub.get('language') or 'en', []).append({ | 
					
						
							|  |  |  |                     'url': sub_url | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  |         return subtitles | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  |     def _real_extract(self, url): | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  |         video_id = self._match_id(url).replace('/', '.') | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-14 01:35:44 +06:00
										 |  |  |         series_id, episode_number = video_id.split('.') | 
					
						
							| 
									
										
										
										
											2018-03-29 23:06:13 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         video = self._call_api( | 
					
						
							|  |  |  |             'series/%s/episodes/%s/' % (series_id, episode_number), video_id, | 
					
						
							|  |  |  |             'Downloading video JSON') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         formats = [] | 
					
						
							|  |  |  |         download_assets = video.get('download_assets') | 
					
						
							|  |  |  |         if download_assets and isinstance(download_assets, dict): | 
					
						
							|  |  |  |             for format_id, format_dict in download_assets.items(): | 
					
						
							|  |  |  |                 if not isinstance(format_dict, dict): | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 format_url = format_dict.get('url') | 
					
						
							|  |  |  |                 if not format_url or not isinstance(format_url, compat_str): | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 formats.append({ | 
					
						
							|  |  |  |                     'url': format_url, | 
					
						
							|  |  |  |                     'format_id': format_id, | 
					
						
							|  |  |  |                     'filesize': int_or_none(video.get('filesize')), | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         stream = self._call_api( | 
					
						
							|  |  |  |             'video/%s/stream/' % video_id, video_id, 'Downloading stream JSON', | 
					
						
							|  |  |  |             fatal=False) | 
					
						
							|  |  |  |         if stream: | 
					
						
							|  |  |  |             stream_url = stream.get('stream_url') | 
					
						
							|  |  |  |             if stream_url: | 
					
						
							|  |  |  |                 formats.extend(self._extract_m3u8_formats( | 
					
						
							|  |  |  |                     stream_url, video_id, 'mp4', entry_protocol='m3u8_native', | 
					
						
							|  |  |  |                     m3u8_id='hls', fatal=False)) | 
					
						
							|  |  |  |         self._sort_formats(formats) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         title = video.get('title') or 'Episode %s' % episode_number | 
					
						
							|  |  |  |         description = video.get('description') | 
					
						
							|  |  |  |         thumbnail = video.get('thumbnail') | 
					
						
							|  |  |  |         timestamp = unified_timestamp(video.get('release_date')) | 
					
						
							|  |  |  |         duration = parse_duration(video.get('duration')) | 
					
						
							|  |  |  |         age_limit = parse_age_limit(video.get('tv_rating')) | 
					
						
							|  |  |  |         series = video.get('series_title') | 
					
						
							|  |  |  |         season_number = int_or_none(video.get('season')) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if series: | 
					
						
							|  |  |  |             title = '%s - %s' % (series, title) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         subtitles = self.extract_subtitles(video_id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             'id': video_id, | 
					
						
							|  |  |  |             'title': title, | 
					
						
							|  |  |  |             'description': description, | 
					
						
							|  |  |  |             'thumbnail': thumbnail, | 
					
						
							|  |  |  |             'duration': duration, | 
					
						
							|  |  |  |             'timestamp': timestamp, | 
					
						
							|  |  |  |             'age_limit': age_limit, | 
					
						
							|  |  |  |             'series': series, | 
					
						
							|  |  |  |             'season_number': season_number, | 
					
						
							|  |  |  |             'episode_number': int_or_none(episode_number), | 
					
						
							|  |  |  |             'formats': formats, | 
					
						
							|  |  |  |             'subtitles': subtitles, | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-19 21:57:31 +06:00
										 |  |  | class DramaFeverSeriesIE(DramaFeverBaseIE): | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  |     IE_NAME = 'dramafever:series' | 
					
						
							| 
									
										
										
										
											2017-01-14 18:27:22 +07:00
										 |  |  |     _VALID_URL = r'https?://(?:www\.)?dramafever\.com/(?:[^/]+/)?drama/(?P<id>[0-9]+)(?:/(?:(?!\d+(?:/|$)).+)?)?$' | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  |     _TESTS = [{ | 
					
						
							|  |  |  |         'url': 'http://www.dramafever.com/drama/4512/Cooking_with_Shin/', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': '4512', | 
					
						
							|  |  |  |             'title': 'Cooking with Shin', | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  |             'description': 'md5:84a3f26e3cdc3fb7f500211b3593b5c1', | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  |         }, | 
					
						
							|  |  |  |         'playlist_count': 4, | 
					
						
							|  |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'http://www.dramafever.com/drama/124/IRIS/', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': '124', | 
					
						
							|  |  |  |             'title': 'IRIS', | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  |             'description': 'md5:b3a30e587cf20c59bd1c01ec0ee1b862', | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  |         }, | 
					
						
							|  |  |  |         'playlist_count': 20, | 
					
						
							|  |  |  |     }] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-14 09:51:07 +06:00
										 |  |  |     _PAGE_SIZE = 60  # max is 60 (see http://api.drama9.com/#get--api-4-episode-series-) | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  |     def _real_extract(self, url): | 
					
						
							|  |  |  |         series_id = self._match_id(url) | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         series = self._download_json( | 
					
						
							|  |  |  |             'http://www.dramafever.com/api/4/series/query/?cs=%s&series_id=%s' | 
					
						
							| 
									
										
										
										
											2015-07-14 01:35:44 +06:00
										 |  |  |             % (self._consumer_secret, series_id), | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  |             series_id, 'Downloading series JSON')['series'][series_id] | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  |         title = clean_html(series['name']) | 
					
						
							|  |  |  |         description = clean_html(series.get('description') or series.get('description_short')) | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         entries = [] | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  |         for page_num in itertools.count(1): | 
					
						
							|  |  |  |             episodes = self._download_json( | 
					
						
							|  |  |  |                 'http://www.dramafever.com/api/4/episode/series/?cs=%s&series_id=%s&page_size=%d&page_number=%d' | 
					
						
							| 
									
										
										
										
											2015-07-14 01:35:44 +06:00
										 |  |  |                 % (self._consumer_secret, series_id, self._PAGE_SIZE, page_num), | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  |                 series_id, 'Downloading episodes JSON page #%d' % page_num) | 
					
						
							|  |  |  |             for episode in episodes.get('value', []): | 
					
						
							| 
									
										
										
										
											2015-06-19 22:02:07 +06:00
										 |  |  |                 episode_url = episode.get('episode_url') | 
					
						
							|  |  |  |                 if not episode_url: | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  |                 entries.append(self.url_result( | 
					
						
							| 
									
										
										
										
											2015-06-19 22:02:07 +06:00
										 |  |  |                     compat_urlparse.urljoin(url, episode_url), | 
					
						
							| 
									
										
										
										
											2015-06-14 07:43:14 +06:00
										 |  |  |                     'DramaFever', episode.get('guid'))) | 
					
						
							|  |  |  |             if page_num == episodes['num_pages']: | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-20 13:51:43 +08:00
										 |  |  |         return self.playlist_result(entries, series_id, title, description) |