| 
									
										
										
										
											2015-08-06 19:37:45 +01:00
										 |  |  | # coding: utf-8 | 
					
						
							|  |  |  | from __future__ import unicode_literals | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-03 00:43:33 +01:00
										 |  |  | import re | 
					
						
							|  |  |  | import json | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-18 22:10:57 +01:00
										 |  |  | from .common import InfoExtractor | 
					
						
							| 
									
										
										
										
											2016-11-03 00:43:33 +01:00
										 |  |  | from ..compat import compat_HTTPError | 
					
						
							| 
									
										
										
										
											2015-07-18 22:55:40 +01:00
										 |  |  | from ..utils import ( | 
					
						
							|  |  |  |     ExtractorError, | 
					
						
							| 
									
										
										
										
											2015-08-15 18:13:37 +06:00
										 |  |  |     int_or_none, | 
					
						
							|  |  |  |     parse_iso8601, | 
					
						
							| 
									
										
										
										
											2016-07-12 23:15:38 +01:00
										 |  |  |     str_or_none, | 
					
						
							| 
									
										
										
										
											2016-11-03 00:43:33 +01:00
										 |  |  |     urlencode_postdata, | 
					
						
							|  |  |  |     clean_html, | 
					
						
							| 
									
										
										
										
											2015-07-18 23:10:20 +01:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2015-07-18 22:10:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-06 19:37:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-18 22:10:57 +01:00
										 |  |  | class ShahidIE(InfoExtractor): | 
					
						
							| 
									
										
										
										
											2016-11-03 00:43:33 +01:00
										 |  |  |     _NETRC_MACHINE = 'shahid' | 
					
						
							|  |  |  |     _VALID_URL = r'https?://shahid\.mbc\.net/ar/(?P<type>episode|movie)/(?P<id>\d+)' | 
					
						
							| 
									
										
										
										
											2015-08-15 18:13:37 +06:00
										 |  |  |     _TESTS = [{ | 
					
						
							|  |  |  |         'url': 'https://shahid.mbc.net/ar/episode/90574/%D8%A7%D9%84%D9%85%D9%84%D9%83-%D8%B9%D8%A8%D8%AF%D8%A7%D9%84%D9%84%D9%87-%D8%A7%D9%84%D8%A5%D9%86%D8%B3%D8%A7%D9%86-%D8%A7%D9%84%D9%85%D9%88%D8%B3%D9%85-1-%D9%83%D9%84%D9%8A%D8%A8-3.html', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': '90574', | 
					
						
							| 
									
										
										
										
											2015-09-18 22:10:36 +01:00
										 |  |  |             'ext': 'mp4', | 
					
						
							| 
									
										
										
										
											2015-08-15 18:13:37 +06:00
										 |  |  |             'title': 'الملك عبدالله الإنسان الموسم 1 كليب 3', | 
					
						
							|  |  |  |             'description': 'الفيلم الوثائقي - الملك عبد الله الإنسان', | 
					
						
							|  |  |  |             'duration': 2972, | 
					
						
							|  |  |  |             'timestamp': 1422057420, | 
					
						
							|  |  |  |             'upload_date': '20150123', | 
					
						
							| 
									
										
										
										
											2015-07-18 22:10:57 +01:00
										 |  |  |         }, | 
					
						
							| 
									
										
										
										
											2015-08-15 18:13:37 +06:00
										 |  |  |         'params': { | 
					
						
							|  |  |  |             # m3u8 download | 
					
						
							|  |  |  |             'skip_download': True, | 
					
						
							| 
									
										
										
										
											2015-07-18 22:10:57 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-11-03 00:43:33 +01:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'https://shahid.mbc.net/ar/movie/151746/%D8%A7%D9%84%D9%82%D9%86%D8%A7%D8%B5%D8%A9.html', | 
					
						
							|  |  |  |         'only_matching': True | 
					
						
							| 
									
										
										
										
											2015-08-15 18:13:37 +06:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         # shahid plus subscriber only | 
					
						
							|  |  |  |         'url': 'https://shahid.mbc.net/ar/episode/90511/%D9%85%D8%B1%D8%A7%D9%8A%D8%A7-2011-%D8%A7%D9%84%D9%85%D9%88%D8%B3%D9%85-1-%D8%A7%D9%84%D8%AD%D9%84%D9%82%D8%A9-1.html', | 
					
						
							|  |  |  |         'only_matching': True | 
					
						
							|  |  |  |     }] | 
					
						
							| 
									
										
										
										
											2015-07-18 22:10:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-03 00:43:33 +01:00
										 |  |  |     def _real_initialize(self): | 
					
						
							|  |  |  |         email, password = self._get_login_info() | 
					
						
							|  |  |  |         if email is None: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             user_data = self._download_json( | 
					
						
							|  |  |  |                 'https://shahid.mbc.net/wd/service/users/login', | 
					
						
							|  |  |  |                 None, 'Logging in', data=json.dumps({ | 
					
						
							|  |  |  |                     'email': email, | 
					
						
							|  |  |  |                     'password': password, | 
					
						
							|  |  |  |                     'basic': 'false', | 
					
						
							|  |  |  |                 }).encode('utf-8'), headers={ | 
					
						
							|  |  |  |                     'Content-Type': 'application/json; charset=UTF-8', | 
					
						
							|  |  |  |                 })['user'] | 
					
						
							|  |  |  |         except ExtractorError as e: | 
					
						
							|  |  |  |             if isinstance(e.cause, compat_HTTPError): | 
					
						
							|  |  |  |                 fail_data = self._parse_json( | 
					
						
							|  |  |  |                     e.cause.read().decode('utf-8'), None, fatal=False) | 
					
						
							|  |  |  |                 if fail_data: | 
					
						
							|  |  |  |                     faults = fail_data.get('faults', []) | 
					
						
							|  |  |  |                     faults_message = ', '.join([clean_html(fault['userMessage']) for fault in faults if fault.get('userMessage')]) | 
					
						
							|  |  |  |                     if faults_message: | 
					
						
							|  |  |  |                         raise ExtractorError(faults_message, expected=True) | 
					
						
							|  |  |  |             raise | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._download_webpage( | 
					
						
							|  |  |  |             'https://shahid.mbc.net/populateContext', | 
					
						
							|  |  |  |             None, 'Populate Context', data=urlencode_postdata({ | 
					
						
							|  |  |  |                 'firstName': user_data['firstName'], | 
					
						
							|  |  |  |                 'lastName': user_data['lastName'], | 
					
						
							|  |  |  |                 'userName': user_data['email'], | 
					
						
							|  |  |  |                 'csg_user_name': user_data['email'], | 
					
						
							|  |  |  |                 'subscriberId': user_data['id'], | 
					
						
							|  |  |  |                 'sessionId': user_data['sessionId'], | 
					
						
							|  |  |  |             })) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_api_data(self, response): | 
					
						
							|  |  |  |         data = response.get('data', {}) | 
					
						
							| 
									
										
										
										
											2016-07-12 23:15:38 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         error = data.get('error') | 
					
						
							| 
									
										
										
										
											2015-08-15 18:13:37 +06:00
										 |  |  |         if error: | 
					
						
							|  |  |  |             raise ExtractorError( | 
					
						
							|  |  |  |                 '%s returned error: %s' % (self.IE_NAME, '\n'.join(error.values())), | 
					
						
							|  |  |  |                 expected=True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-12 23:15:38 +01:00
										 |  |  |         return data | 
					
						
							| 
									
										
										
										
											2015-08-08 12:59:10 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-18 22:10:57 +01:00
										 |  |  |     def _real_extract(self, url): | 
					
						
							| 
									
										
										
										
											2016-11-03 00:43:33 +01:00
										 |  |  |         page_type, video_id = re.match(self._VALID_URL, url).groups() | 
					
						
							| 
									
										
										
										
											2015-08-15 18:13:37 +06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-03 00:43:33 +01:00
										 |  |  |         player = self._get_api_data(self._download_json( | 
					
						
							|  |  |  |             'https://shahid.mbc.net/arContent/getPlayerContent-param-.id-%s.type-player.html' % video_id, | 
					
						
							|  |  |  |             video_id, 'Downloading player JSON')) | 
					
						
							| 
									
										
										
										
											2015-07-25 18:13:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-10 07:55:58 +01:00
										 |  |  |         if player.get('drm'): | 
					
						
							|  |  |  |             raise ExtractorError('This video is DRM protected.', expected=True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-15 18:13:37 +06:00
										 |  |  |         formats = self._extract_m3u8_formats(player['url'], video_id, 'mp4') | 
					
						
							| 
									
										
										
										
											2016-03-27 07:03:08 +06:00
										 |  |  |         self._sort_formats(formats) | 
					
						
							| 
									
										
										
										
											2015-08-06 19:37:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-03 00:43:33 +01:00
										 |  |  |         video = self._get_api_data(self._download_json( | 
					
						
							|  |  |  |             'http://api.shahid.net/api/v1_1/%s/%s' % (page_type, video_id), | 
					
						
							|  |  |  |             video_id, 'Downloading video JSON', query={ | 
					
						
							|  |  |  |                 'apiKey': 'sh@hid0nlin3', | 
					
						
							|  |  |  |                 'hash': 'b2wMCTHpSmyxGqQjJFOycRmLSex+BpTK/ooxy6vHaqs=', | 
					
						
							|  |  |  |             }))[page_type] | 
					
						
							| 
									
										
										
										
											2015-08-15 18:13:37 +06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         title = video['title'] | 
					
						
							|  |  |  |         categories = [ | 
					
						
							|  |  |  |             category['name'] | 
					
						
							|  |  |  |             for category in video.get('genres', []) if 'name' in category] | 
					
						
							| 
									
										
										
										
											2015-07-25 18:13:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-18 22:10:57 +01:00
										 |  |  |         return { | 
					
						
							|  |  |  |             'id': video_id, | 
					
						
							|  |  |  |             'title': title, | 
					
						
							| 
									
										
										
										
											2016-07-12 23:15:38 +01:00
										 |  |  |             'description': video.get('description'), | 
					
						
							|  |  |  |             'thumbnail': video.get('thumbnailUrl'), | 
					
						
							|  |  |  |             'duration': int_or_none(video.get('duration')), | 
					
						
							|  |  |  |             'timestamp': parse_iso8601(video.get('referenceDate')), | 
					
						
							| 
									
										
										
										
											2015-08-15 18:13:37 +06:00
										 |  |  |             'categories': categories, | 
					
						
							| 
									
										
										
										
											2016-07-12 23:15:38 +01:00
										 |  |  |             'series': video.get('showTitle') or video.get('showName'), | 
					
						
							|  |  |  |             'season': video.get('seasonTitle'), | 
					
						
							|  |  |  |             'season_number': int_or_none(video.get('seasonNumber')), | 
					
						
							|  |  |  |             'season_id': str_or_none(video.get('seasonId')), | 
					
						
							|  |  |  |             'episode_number': int_or_none(video.get('number')), | 
					
						
							|  |  |  |             'episode_id': video_id, | 
					
						
							| 
									
										
										
										
											2015-07-18 22:10:57 +01:00
										 |  |  |             'formats': formats, | 
					
						
							|  |  |  |         } |