| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2014-12-13 12:24:42 +01:00
										 |  |  | from ..compat import ( | 
					
						
							| 
									
										
										
										
											2014-02-27 18:58:09 +01:00
										 |  |  |     compat_urllib_parse, | 
					
						
							| 
									
										
										
										
											2014-12-13 12:24:42 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | from ..utils import ( | 
					
						
							| 
									
										
										
										
											2013-12-08 22:24:55 +01:00
										 |  |  |     ExtractorError, | 
					
						
							| 
									
										
										
										
											2014-08-26 14:55:15 +02:00
										 |  |  |     HEADRequest, | 
					
						
							| 
									
										
										
										
											2015-01-30 23:21:44 +06:00
										 |  |  |     str_to_int, | 
					
						
							| 
									
										
										
										
											2014-05-13 09:42:38 +02:00
										 |  |  |     parse_iso8601, | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MixcloudIE(InfoExtractor): | 
					
						
							| 
									
										
										
										
											2014-02-26 00:04:03 +01:00
										 |  |  |     _VALID_URL = r'^(?:https?://)?(?:www\.)?mixcloud\.com/([^/]+)/([^/]+)' | 
					
						
							| 
									
										
										
										
											2014-01-17 04:06:18 +01:00
										 |  |  |     IE_NAME = 'mixcloud' | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-04 19:47:55 +06:00
										 |  |  |     _TESTS = [{ | 
					
						
							| 
									
										
										
										
											2014-01-17 04:06:18 +01:00
										 |  |  |         'url': 'http://www.mixcloud.com/dholbach/cryptkeeper/', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							| 
									
										
										
										
											2014-02-27 18:58:09 +01:00
										 |  |  |             'id': 'dholbach-cryptkeeper', | 
					
						
							|  |  |  |             'ext': 'mp3', | 
					
						
							| 
									
										
										
										
											2014-01-17 04:06:18 +01:00
										 |  |  |             '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', | 
					
						
							| 
									
										
										
										
											2014-05-13 09:42:38 +02:00
										 |  |  |             'timestamp': 1321359578, | 
					
						
							|  |  |  |             'thumbnail': 're:https?://.*\.jpg', | 
					
						
							|  |  |  |             'view_count': int, | 
					
						
							|  |  |  |             'like_count': int, | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |         }, | 
					
						
							| 
									
										
										
										
											2015-02-04 19:47:55 +06:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'http://www.mixcloud.com/gillespeterson/caribou-7-inch-vinyl-mix-chat/', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': 'gillespeterson-caribou-7-inch-vinyl-mix-chat', | 
					
						
							|  |  |  |             'ext': 'm4a', | 
					
						
							|  |  |  |             'title': 'Electric Relaxation vol. 3', | 
					
						
							|  |  |  |             'description': 'md5:2b8aec6adce69f9d41724647c65875e8', | 
					
						
							|  |  |  |             'uploader': 'Daniel Drumz', | 
					
						
							|  |  |  |             'uploader_id': 'gillespeterson', | 
					
						
							|  |  |  |             'thumbnail': 're:https?://.*\.jpg', | 
					
						
							|  |  |  |             'view_count': int, | 
					
						
							|  |  |  |             'like_count': int, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |     }] | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-15 00:53:54 +02:00
										 |  |  |     def _get_url(self, track_id, template_url): | 
					
						
							|  |  |  |         server_count = 30 | 
					
						
							|  |  |  |         for i in range(server_count): | 
					
						
							|  |  |  |             url = template_url % i | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2013-12-08 22:24:55 +01:00
										 |  |  |                 # We only want to know if the request succeed | 
					
						
							|  |  |  |                 # don't download the whole file | 
					
						
							| 
									
										
										
										
											2014-10-15 00:53:54 +02:00
										 |  |  |                 self._request_webpage( | 
					
						
							|  |  |  |                     HEADRequest(url), track_id, | 
					
						
							|  |  |  |                     'Checking URL %d/%d ...' % (i + 1, server_count + 1)) | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  |                 return url | 
					
						
							| 
									
										
										
										
											2013-12-08 22:24:55 +01:00
										 |  |  |             except ExtractorError: | 
					
						
							| 
									
										
										
										
											2014-10-15 00:53:54 +02:00
										 |  |  |                 pass | 
					
						
							| 
									
										
										
										
											2013-06-23 21:59:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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) | 
					
						
							| 
									
										
										
										
											2014-02-27 18:58:09 +01:00
										 |  |  |         track_id = compat_urllib_parse.unquote('-'.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
										 |  |  |         preview_url = self._search_regex( | 
					
						
							| 
									
										
										
										
											2015-02-04 19:47:55 +06: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) | 
					
						
							| 
									
										
										
										
											2014-10-15 00:53:54 +02:00
										 |  |  |         final_song_url = self._get_url(track_id, template_url) | 
					
						
							| 
									
										
										
										
											2013-12-10 13:42:41 +01:00
										 |  |  |         if final_song_url is None: | 
					
						
							|  |  |  |             self.to_screen('Trying with m4a extension') | 
					
						
							|  |  |  |             template_url = template_url.replace('.mp3', '.m4a').replace('originals/', 'm4a/64/') | 
					
						
							| 
									
										
										
										
											2014-10-15 00:53:54 +02:00
										 |  |  |             final_song_url = self._get_url(track_id, template_url) | 
					
						
							| 
									
										
										
										
											2013-12-10 13:42:41 +01:00
										 |  |  |         if final_song_url is None: | 
					
						
							| 
									
										
										
										
											2014-05-13 09:42:38 +02:00
										 |  |  |             raise ExtractorError('Unable to extract track url') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         PREFIX = ( | 
					
						
							| 
									
										
										
										
											2014-12-11 23:16:40 +01:00
										 |  |  |             r'<span class="play-button[^"]*?"' | 
					
						
							| 
									
										
										
										
											2014-05-13 09:42:38 +02:00
										 |  |  |             r'(?:\s+[a-zA-Z0-9-]+(?:="[^"]+")?)*?\s+') | 
					
						
							|  |  |  |         title = self._html_search_regex( | 
					
						
							|  |  |  |             PREFIX + r'm-title="([^"]+)"', webpage, 'title') | 
					
						
							|  |  |  |         thumbnail = self._proto_relative_url(self._html_search_regex( | 
					
						
							|  |  |  |             PREFIX + r'm-thumbnail-url="([^"]+)"', webpage, 'thumbnail', | 
					
						
							|  |  |  |             fatal=False)) | 
					
						
							|  |  |  |         uploader = self._html_search_regex( | 
					
						
							|  |  |  |             PREFIX + r'm-owner-name="([^"]+)"', | 
					
						
							|  |  |  |             webpage, 'uploader', fatal=False) | 
					
						
							|  |  |  |         uploader_id = self._search_regex( | 
					
						
							|  |  |  |             r'\s+"profile": "([^"]+)",', webpage, 'uploader id', fatal=False) | 
					
						
							|  |  |  |         description = self._og_search_description(webpage) | 
					
						
							| 
									
										
										
										
											2015-01-30 23:21:44 +06:00
										 |  |  |         like_count = str_to_int(self._search_regex( | 
					
						
							|  |  |  |             [r'<meta itemprop="interactionCount" content="UserLikes:([0-9]+)"', | 
					
						
							|  |  |  |              r'/favorites/?">([0-9]+)<'], | 
					
						
							| 
									
										
										
										
											2014-05-13 09:42:38 +02:00
										 |  |  |             webpage, 'like count', fatal=False)) | 
					
						
							| 
									
										
										
										
											2015-01-30 23:21:44 +06:00
										 |  |  |         view_count = str_to_int(self._search_regex( | 
					
						
							|  |  |  |             [r'<meta itemprop="interactionCount" content="UserPlays:([0-9]+)"', | 
					
						
							|  |  |  |              r'/listeners/?">([0-9,.]+)</a>'], | 
					
						
							| 
									
										
										
										
											2014-05-13 09:42:38 +02:00
										 |  |  |             webpage, 'play count', fatal=False)) | 
					
						
							|  |  |  |         timestamp = parse_iso8601(self._search_regex( | 
					
						
							|  |  |  |             r'<time itemprop="dateCreated" datetime="([^"]+)">', | 
					
						
							| 
									
										
										
										
											2015-01-30 23:21:44 +06:00
										 |  |  |             webpage, 'upload date', default=None)) | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             'id': track_id, | 
					
						
							| 
									
										
										
										
											2014-05-13 09:42:38 +02:00
										 |  |  |             'title': title, | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |             'url': final_song_url, | 
					
						
							| 
									
										
										
										
											2014-05-13 09:42:38 +02:00
										 |  |  |             'description': description, | 
					
						
							|  |  |  |             'thumbnail': thumbnail, | 
					
						
							|  |  |  |             'uploader': uploader, | 
					
						
							|  |  |  |             'uploader_id': uploader_id, | 
					
						
							|  |  |  |             'timestamp': timestamp, | 
					
						
							|  |  |  |             'view_count': view_count, | 
					
						
							|  |  |  |             'like_count': like_count, | 
					
						
							| 
									
										
										
										
											2013-09-14 14:26:42 +02:00
										 |  |  |         } |