| 
									
										
										
										
											2014-01-17 04:06:18 +01:00
										 |  |  | from __future__ import unicode_literals | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  | 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-]+)' | 
					
						
							| 
									
										
										
										
											2014-01-17 04:06:18 +01:00
										 |  |  |     IE_NAME = 'mixcloud' | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |     _TEST = { | 
					
						
							| 
									
										
										
										
											2014-01-17 04:06:18 +01:00
										 |  |  |         'url': 'http://www.mixcloud.com/dholbach/cryptkeeper/', | 
					
						
							|  |  |  |         'file': 'dholbach-cryptkeeper.mp3', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'title': 'Cryptkeeper', | 
					
						
							|  |  |  |             'description': 'After quite a long silence from myself, finally another Drum\'n\'Bass mix with my favourite current dance floor bangers.', | 
					
						
							|  |  |  |             'uploader': 'Daniel Holbach', | 
					
						
							|  |  |  |             'uploader_id': 'dholbach', | 
					
						
							|  |  |  |             'upload_date': '20111115', | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |         }, | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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)) | 
					
						
							| 
									
										
										
										
											2014-01-17 04:05:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |         webpage = self._download_webpage(url, track_id) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-17 04:05:15 +01:00
										 |  |  |         api_url = 'http://api.mixcloud.com/%s/%s/' % (uploader, cloudcast_name) | 
					
						
							|  |  |  |         info = self._download_json( | 
					
						
							| 
									
										
										
										
											2014-01-17 04:06:18 +01:00
										 |  |  |             api_url, track_id, 'Downloading cloudcast info') | 
					
						
							| 
									
										
										
										
											2014-01-17 04:05:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         preview_url = self._search_regex( | 
					
						
							| 
									
										
										
										
											2014-01-17 04:06:18 +01:00
										 |  |  |             r'\s(?:data-preview-url|m-preview)="(.+?)"', webpage, 'preview url') | 
					
						
							| 
									
										
										
										
											2014-01-01 21:07:55 +01:00
										 |  |  |         song_url = preview_url.replace('/previews/', '/c/originals/') | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |         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'], | 
					
						
							|  |  |  |         } |