| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  | import json | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from .common import InfoExtractor | 
					
						
							|  |  |  | from ..utils import ( | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |     unified_strdate, | 
					
						
							| 
									
										
										
										
											2013-12-08 22:24:55 +01:00
										 |  |  |     ExtractorError, | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MixcloudIE(InfoExtractor): | 
					
						
							|  |  |  |     _VALID_URL = r'^(?:https?://)?(?:www\.)?mixcloud\.com/([\w\d-]+)/([\w\d-]+)' | 
					
						
							|  |  |  |     IE_NAME = u'mixcloud' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |     _TEST = { | 
					
						
							|  |  |  |         u'url': u'http://www.mixcloud.com/dholbach/cryptkeeper/', | 
					
						
							|  |  |  |         u'file': u'dholbach-cryptkeeper.mp3', | 
					
						
							|  |  |  |         u'info_dict': { | 
					
						
							|  |  |  |             u'title': u'Cryptkeeper', | 
					
						
							|  |  |  |             u'description': u'After quite a long silence from myself, finally another Drum\'n\'Bass mix with my favourite current dance floor bangers.', | 
					
						
							|  |  |  |             u'uploader': u'Daniel Holbach', | 
					
						
							|  |  |  |             u'uploader_id': u'dholbach', | 
					
						
							|  |  |  |             u'upload_date': u'20111115', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def check_urls(self, url_list): | 
					
						
							|  |  |  |         """Returns 1st active url from list""" | 
					
						
							|  |  |  |         for url in url_list: | 
					
						
							|  |  |  |             try: | 
					
						
							| 
									
										
										
										
											2013-12-08 22:24:55 +01:00
										 |  |  |                 # We only want to know if the request succeed | 
					
						
							|  |  |  |                 # don't download the whole file | 
					
						
							|  |  |  |                 self._request_webpage(url, None, False) | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  |                 return url | 
					
						
							| 
									
										
										
										
											2013-12-08 22:24:55 +01:00
										 |  |  |             except ExtractorError: | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  |                 url = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-10 13:42:41 +01:00
										 |  |  |     def _get_url(self, template_url): | 
					
						
							|  |  |  |         return self.check_urls(template_url % i for i in range(30)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  |     def _real_extract(self, url): | 
					
						
							|  |  |  |         mobj = re.match(self._VALID_URL, url) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |         uploader = mobj.group(1) | 
					
						
							|  |  |  |         cloudcast_name = mobj.group(2) | 
					
						
							|  |  |  |         track_id = '-'.join((uploader, cloudcast_name)) | 
					
						
							|  |  |  |         api_url = 'http://api.mixcloud.com/%s/%s/' % (uploader, cloudcast_name) | 
					
						
							|  |  |  |         webpage = self._download_webpage(url, track_id) | 
					
						
							|  |  |  |         json_data = self._download_webpage(api_url, track_id, | 
					
						
							|  |  |  |             u'Downloading cloudcast info') | 
					
						
							|  |  |  |         info = json.loads(json_data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         preview_url = self._search_regex(r'data-preview-url="(.+?)"', webpage, u'preview url') | 
					
						
							|  |  |  |         song_url = preview_url.replace('/previews/', '/cloudcasts/originals/') | 
					
						
							|  |  |  |         template_url = re.sub(r'(stream\d*)', 'stream%d', song_url) | 
					
						
							| 
									
										
										
										
											2013-12-10 13:42:41 +01:00
										 |  |  |         final_song_url = self._get_url(template_url) | 
					
						
							|  |  |  |         if final_song_url is None: | 
					
						
							|  |  |  |             self.to_screen('Trying with m4a extension') | 
					
						
							|  |  |  |             template_url = template_url.replace('.mp3', '.m4a').replace('originals/', 'm4a/64/') | 
					
						
							|  |  |  |             final_song_url = self._get_url(template_url) | 
					
						
							|  |  |  |         if final_song_url is None: | 
					
						
							|  |  |  |             raise ExtractorError(u'Unable to extract track url') | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             'id': track_id, | 
					
						
							|  |  |  |             'title': info['name'], | 
					
						
							|  |  |  |             'url': final_song_url, | 
					
						
							| 
									
										
										
										
											2013-11-24 11:28:44 +01:00
										 |  |  |             'description': info.get('description'), | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |             'thumbnail': info['pictures'].get('extra_large'), | 
					
						
							|  |  |  |             'uploader': info['user']['name'], | 
					
						
							|  |  |  |             'uploader_id': info['user']['username'], | 
					
						
							|  |  |  |             'upload_date': unified_strdate(info['created_time']), | 
					
						
							|  |  |  |             'view_count': info['play_count'], | 
					
						
							|  |  |  |         } |