2013-12-09 11:08:58 -05:00
# encoding: utf-8
2014-01-16 21:29:41 -05:00
from __future__ import unicode_literals
2013-06-23 14:57:44 -04:00
import re
2013-09-14 15:41:49 -04:00
import itertools
2013-06-23 14:57:44 -04:00
from . common import InfoExtractor
from . . utils import (
compat_str ,
2013-08-21 11:06:37 -04:00
compat_urlparse ,
2013-09-14 15:41:49 -04:00
compat_urllib_parse ,
2013-06-23 14:57:44 -04:00
ExtractorError ,
2014-06-07 09:51:01 -04:00
int_or_none ,
2013-06-23 14:57:44 -04:00
unified_strdate ,
)
class SoundcloudIE ( InfoExtractor ) :
""" Information extractor for soundcloud.com
To access the media , the uid of the song and a stream token
must be extracted from the page source and the script must make
a request to media . soundcloud . com / crossdomain . xml . Then
the media can be grabbed by requesting from an url composed
of the stream token and uid
"""
2014-05-04 21:12:41 -04:00
_VALID_URL = r ''' (?x)^(?:https?://)?
2013-12-19 10:39:01 -05:00
( ? : ( ? : ( ? : www \. | m \. ) ? soundcloud \. com /
2013-12-09 13:57:00 -05:00
( ? P < uploader > [ \w \d - ] + ) /
2014-08-27 18:58:24 -04:00
( ? ! sets / | likes / ? ( ? : $ | [ ? #]))
( ? P < title > [ \w \d - ] + ) / ?
2013-12-09 11:08:58 -05:00
( ? P < token > [ ^ ? ] + ? ) ? ( ? : [ ? ] . * ) ? $ )
2013-07-24 08:39:21 -04:00
| ( ? : api \. soundcloud \. com / tracks / ( ? P < track_id > \d + ) )
2014-01-02 10:18:51 -05:00
| ( ? P < player > ( ? : w | player | p . ) \. soundcloud \. com / player / ? . * ? url = . * )
2013-07-24 08:39:21 -04:00
)
'''
2014-01-16 21:29:41 -05:00
IE_NAME = ' soundcloud '
2013-11-09 12:06:09 -05:00
_TESTS = [
{
2014-01-16 21:29:41 -05:00
' url ' : ' http://soundcloud.com/ethmusic/lostin-powers-she-so-heavy ' ,
' file ' : ' 62986583.mp3 ' ,
' md5 ' : ' ebef0a451b909710ed1d7787dddbf0d7 ' ,
' info_dict ' : {
" upload_date " : " 20121011 " ,
" description " : " No Downloads untill we record the finished version this weekend, i was too pumped n i had to post it , earl is prolly gonna b hella p.o ' d " ,
" uploader " : " E.T. ExTerrestrial Music " ,
2014-06-07 09:51:01 -04:00
" title " : " Lostin Powers - She so Heavy (SneakPreview) Adrian Ackers Blueprint 1 " ,
" duration " : 143 ,
2013-11-09 12:06:09 -05:00
}
} ,
# not streamable song
{
2014-01-16 21:29:41 -05:00
' url ' : ' https://soundcloud.com/the-concept-band/goldrushed-mastered?in=the-concept-band/sets/the-royal-concept-ep ' ,
' info_dict ' : {
' id ' : ' 47127627 ' ,
' ext ' : ' mp3 ' ,
' title ' : ' Goldrushed ' ,
2014-03-09 07:20:34 -04:00
' description ' : ' From Stockholm Sweden \r \n Povel / Magnus / Filip / David \r \n www.theroyalconcept.com ' ,
2014-01-16 21:29:41 -05:00
' uploader ' : ' The Royal Concept ' ,
' upload_date ' : ' 20120521 ' ,
2014-06-07 09:51:01 -04:00
' duration ' : 227 ,
2013-11-09 12:06:09 -05:00
} ,
2014-01-16 21:29:41 -05:00
' params ' : {
2013-11-09 12:06:09 -05:00
# rtmp
2014-01-16 21:29:41 -05:00
' skip_download ' : True ,
2013-11-09 12:06:09 -05:00
} ,
} ,
2013-12-09 11:08:58 -05:00
# private link
{
2014-01-16 21:29:41 -05:00
' url ' : ' https://soundcloud.com/jaimemf/youtube-dl-test-video-a-y-baw/s-8Pjrp ' ,
' md5 ' : ' aa0dd32bfea9b0c5ef4f02aacd080604 ' ,
' info_dict ' : {
' id ' : ' 123998367 ' ,
' ext ' : ' mp3 ' ,
' title ' : ' Youtube - Dl Test Video \' \' Ä↭ ' ,
' uploader ' : ' jaimeMF ' ,
' description ' : ' test chars: \" \' / \\ ä↭ ' ,
' upload_date ' : ' 20131209 ' ,
2014-06-07 09:51:01 -04:00
' duration ' : 9 ,
2013-12-09 11:08:58 -05:00
} ,
} ,
2013-12-10 07:04:21 -05:00
# downloadable song
{
2014-07-15 08:18:06 -04:00
' url ' : ' https://soundcloud.com/oddsamples/bus-brakes ' ,
2014-07-22 19:41:44 -04:00
' md5 ' : ' 7624f2351f8a3b2e7cd51522496e7631 ' ,
2014-01-16 21:29:41 -05:00
' info_dict ' : {
2014-07-15 08:18:06 -04:00
' id ' : ' 128590877 ' ,
2014-07-22 19:41:44 -04:00
' ext ' : ' mp3 ' ,
2014-07-15 08:18:06 -04:00
' title ' : ' Bus Brakes ' ,
' description ' : ' md5:0170be75dd395c96025d210d261c784e ' ,
' uploader ' : ' oddsamples ' ,
' upload_date ' : ' 20140109 ' ,
' duration ' : 17 ,
2013-12-10 07:04:21 -05:00
} ,
} ,
2013-11-09 12:06:09 -05:00
]
2013-06-23 14:57:44 -04:00
2013-07-24 08:05:14 -04:00
_CLIENT_ID = ' b45b1aa10f1ac2941910a7f0d10f8e28 '
2013-11-21 07:16:19 -05:00
_IPHONE_CLIENT_ID = ' 376f225bf427445fc4bfb6b99b72e0bf '
2013-07-24 08:05:14 -04:00
2013-06-23 14:57:44 -04:00
def report_resolve ( self , video_id ) :
""" Report information extraction. """
2014-03-23 21:15:31 -04:00
self . to_screen ( ' %s : Resolving id ' % video_id )
2013-06-23 14:57:44 -04:00
2013-07-24 08:05:14 -04:00
@classmethod
def _resolv_url ( cls , url ) :
return ' http://api.soundcloud.com/resolve.json?url= ' + url + ' &client_id= ' + cls . _CLIENT_ID
2013-12-09 11:08:58 -05:00
def _extract_info_dict ( self , info , full_title = None , quiet = False , secret_token = None ) :
2013-11-09 12:06:09 -05:00
track_id = compat_str ( info [ ' id ' ] )
name = full_title or track_id
2013-11-25 14:30:41 -05:00
if quiet :
2013-09-14 15:41:49 -04:00
self . report_extraction ( name )
2013-07-24 08:05:14 -04:00
thumbnail = info [ ' artwork_url ' ]
if thumbnail is not None :
thumbnail = thumbnail . replace ( ' -large ' , ' -t500x500 ' )
2014-01-16 21:29:41 -05:00
ext = ' mp3 '
2013-11-09 12:06:09 -05:00
result = {
2013-11-25 14:30:41 -05:00
' id ' : track_id ,
2013-07-24 08:05:14 -04:00
' uploader ' : info [ ' user ' ] [ ' username ' ] ,
' upload_date ' : unified_strdate ( info [ ' created_at ' ] ) ,
2013-11-25 14:30:41 -05:00
' title ' : info [ ' title ' ] ,
2013-07-24 08:05:14 -04:00
' description ' : info [ ' description ' ] ,
' thumbnail ' : thumbnail ,
2014-06-07 09:51:01 -04:00
' duration ' : int_or_none ( info . get ( ' duration ' ) , 1000 ) ,
2013-07-24 08:05:14 -04:00
}
2014-03-23 21:21:17 -04:00
formats = [ ]
2013-11-09 12:06:09 -05:00
if info . get ( ' downloadable ' , False ) :
2013-11-21 07:16:19 -05:00
# We can build a direct link to the song
2013-11-25 14:30:41 -05:00
format_url = (
2014-01-16 21:29:41 -05:00
' https://api.soundcloud.com/tracks/ {0} /download?client_id= {1} ' . format (
2013-11-25 14:30:41 -05:00
track_id , self . _CLIENT_ID ) )
2014-03-23 21:21:17 -04:00
formats . append ( {
2013-11-25 14:30:41 -05:00
' format_id ' : ' download ' ,
2014-01-16 21:29:41 -05:00
' ext ' : info . get ( ' original_format ' , ' mp3 ' ) ,
2013-11-25 14:30:41 -05:00
' url ' : format_url ,
2013-11-25 16:34:56 -05:00
' vcodec ' : ' none ' ,
2014-03-23 21:21:17 -04:00
' preference ' : 10 ,
} )
# We have to retrieve the url
streams_url = ( ' http://api.soundcloud.com/i1/tracks/ {0} /streams? '
' client_id= {1} &secret_token= {2} ' . format ( track_id , self . _IPHONE_CLIENT_ID , secret_token ) )
2014-05-04 21:12:41 -04:00
format_dict = self . _download_json (
2014-03-23 21:21:17 -04:00
streams_url ,
track_id , ' Downloading track url ' )
for key , stream_url in format_dict . items ( ) :
if key . startswith ( ' http ' ) :
formats . append ( {
' format_id ' : key ,
' ext ' : ext ,
' url ' : stream_url ,
' vcodec ' : ' none ' ,
} )
elif key . startswith ( ' rtmp ' ) :
# The url doesn't have an rtmp app, we have to extract the playpath
url , path = stream_url . split ( ' mp3: ' , 1 )
formats . append ( {
' format_id ' : key ,
' url ' : url ,
' play_path ' : ' mp3: ' + path ,
' ext ' : ext ,
' vcodec ' : ' none ' ,
} )
2013-11-25 14:30:41 -05:00
if not formats :
2013-11-21 07:16:19 -05:00
# We fallback to the stream_url in the original info, this
# cannot be always used, sometimes it can give an HTTP 404 error
2013-11-25 14:30:41 -05:00
formats . append ( {
2014-01-16 21:29:41 -05:00
' format_id ' : ' fallback ' ,
2013-11-25 14:30:41 -05:00
' url ' : info [ ' stream_url ' ] + ' ?client_id= ' + self . _CLIENT_ID ,
' ext ' : ext ,
2013-11-25 16:34:56 -05:00
' vcodec ' : ' none ' ,
2013-11-25 14:30:41 -05:00
} )
2014-01-16 21:29:41 -05:00
for f in formats :
2013-11-25 14:30:41 -05:00
if f [ ' format_id ' ] . startswith ( ' http ' ) :
2014-01-16 21:29:41 -05:00
f [ ' protocol ' ] = ' http '
2013-11-25 14:30:41 -05:00
if f [ ' format_id ' ] . startswith ( ' rtmp ' ) :
2014-01-16 21:29:41 -05:00
f [ ' protocol ' ] = ' rtmp '
2013-11-25 14:30:41 -05:00
2014-01-16 21:29:41 -05:00
self . _sort_formats ( formats )
2013-11-25 14:30:41 -05:00
result [ ' formats ' ] = formats
2013-11-21 07:16:19 -05:00
2013-11-09 12:06:09 -05:00
return result
2013-07-24 08:05:14 -04:00
2013-06-23 14:57:44 -04:00
def _real_extract ( self , url ) :
2013-07-24 08:39:21 -04:00
mobj = re . match ( self . _VALID_URL , url , flags = re . VERBOSE )
2013-06-23 14:57:44 -04:00
if mobj is None :
2014-03-23 21:15:31 -04:00
raise ExtractorError ( ' Invalid URL: %s ' % url )
2013-06-23 14:57:44 -04:00
2013-07-24 08:39:21 -04:00
track_id = mobj . group ( ' track_id ' )
2013-12-09 11:08:58 -05:00
token = None
2013-07-24 08:39:21 -04:00
if track_id is not None :
info_json_url = ' http://api.soundcloud.com/tracks/ ' + track_id + ' .json?client_id= ' + self . _CLIENT_ID
full_title = track_id
2014-01-02 10:18:51 -05:00
elif mobj . group ( ' player ' ) :
2013-08-21 11:06:37 -04:00
query = compat_urlparse . parse_qs ( compat_urlparse . urlparse ( url ) . query )
2014-05-04 21:12:41 -04:00
return self . url_result ( query [ ' url ' ] [ 0 ] )
2013-07-24 08:39:21 -04:00
else :
# extract uploader (which is in the url)
2013-12-09 11:08:58 -05:00
uploader = mobj . group ( ' uploader ' )
2013-07-24 08:39:21 -04:00
# extract simple title (uploader + slug of song title)
2013-12-09 11:08:58 -05:00
slug_title = mobj . group ( ' title ' )
token = mobj . group ( ' token ' )
full_title = resolve_title = ' %s / %s ' % ( uploader , slug_title )
if token :
resolve_title + = ' / %s ' % token
2013-07-24 08:39:21 -04:00
self . report_resolve ( full_title )
2013-12-09 11:08:58 -05:00
url = ' http://soundcloud.com/ %s ' % resolve_title
2013-07-24 08:39:21 -04:00
info_json_url = self . _resolv_url ( url )
2014-05-04 21:12:41 -04:00
info = self . _download_json ( info_json_url , full_title , ' Downloading info JSON ' )
2013-06-23 14:57:44 -04:00
2013-12-09 11:08:58 -05:00
return self . _extract_info_dict ( info , full_title , secret_token = token )
2013-06-23 14:57:44 -04:00
2014-05-04 21:12:41 -04:00
2013-07-24 08:05:14 -04:00
class SoundcloudSetIE ( SoundcloudIE ) :
2014-03-04 09:21:45 -05:00
_VALID_URL = r ' https?://(?:www \ .)?soundcloud \ .com/([ \ w \ d-]+)/sets/([ \ w \ d-]+) '
2014-01-16 21:29:41 -05:00
IE_NAME = ' soundcloud:set '
2014-08-27 18:58:24 -04:00
_TESTS = [ {
' url ' : ' https://soundcloud.com/the-concept-band/sets/the-royal-concept-ep ' ,
' info_dict ' : {
' title ' : ' The Royal Concept EP ' ,
} ,
' playlist_mincount ' : 6 ,
} ]
2013-06-23 14:57:44 -04:00
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
# extract uploader (which is in the url)
uploader = mobj . group ( 1 )
# extract simple title (uploader + slug of song title)
2014-05-04 21:12:41 -04:00
slug_title = mobj . group ( 2 )
2013-06-23 14:57:44 -04:00
full_title = ' %s /sets/ %s ' % ( uploader , slug_title )
self . report_resolve ( full_title )
url = ' http://soundcloud.com/ %s /sets/ %s ' % ( uploader , slug_title )
2013-07-24 08:05:14 -04:00
resolv_url = self . _resolv_url ( url )
2014-05-04 21:12:41 -04:00
info = self . _download_json ( resolv_url , full_title )
2013-06-23 14:57:44 -04:00
if ' errors ' in info :
for err in info [ ' errors ' ] :
2014-03-23 21:15:31 -04:00
self . _downloader . report_error ( ' unable to download video webpage: %s ' % compat_str ( err [ ' error_message ' ] ) )
2013-06-23 14:57:44 -04:00
return
2014-08-27 18:58:24 -04:00
return {
' _type ' : ' playlist ' ,
' entries ' : [ self . _extract_info_dict ( track ) for track in info [ ' tracks ' ] ] ,
' id ' : info [ ' id ' ] ,
' title ' : info [ ' title ' ] ,
}
2013-09-14 15:41:49 -04:00
class SoundcloudUserIE ( SoundcloudIE ) :
2014-07-07 14:21:02 -04:00
_VALID_URL = r ' https?://(www \ .)?soundcloud \ .com/(?P<user>[^/]+)/?((?P<rsrc>tracks|likes)/?)?( \ ?.*)?$ '
2014-01-16 21:29:41 -05:00
IE_NAME = ' soundcloud:user '
2014-08-27 18:58:24 -04:00
_TESTS = [ {
' url ' : ' https://soundcloud.com/the-concept-band ' ,
' info_dict ' : {
' id ' : ' 9615865 ' ,
' title ' : ' The Royal Concept ' ,
} ,
' playlist_mincount ' : 12
} , {
' url ' : ' https://soundcloud.com/the-concept-band/likes ' ,
' info_dict ' : {
' id ' : ' 9615865 ' ,
' title ' : ' The Royal Concept ' ,
} ,
' playlist_mincount ' : 1 ,
} ]
2013-09-14 15:41:49 -04:00
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
uploader = mobj . group ( ' user ' )
2014-07-07 14:21:02 -04:00
resource = mobj . group ( ' rsrc ' )
if resource is None :
resource = ' tracks '
elif resource == ' likes ' :
resource = ' favorites '
2013-09-14 15:41:49 -04:00
url = ' http://soundcloud.com/ %s / ' % uploader
resolv_url = self . _resolv_url ( url )
2014-05-04 21:12:41 -04:00
user = self . _download_json (
resolv_url , uploader , ' Downloading user info ' )
2014-07-07 14:21:02 -04:00
base_url = ' http://api.soundcloud.com/users/ %s / %s .json? ' % ( uploader , resource )
2013-09-14 15:41:49 -04:00
2014-05-04 21:12:41 -04:00
entries = [ ]
2013-09-14 15:41:49 -04:00
for i in itertools . count ( ) :
2014-05-04 21:12:41 -04:00
data = compat_urllib_parse . urlencode ( {
' offset ' : i * 50 ,
2014-07-07 14:21:02 -04:00
' limit ' : 50 ,
2014-05-04 21:12:41 -04:00
' client_id ' : self . _CLIENT_ID ,
} )
new_entries = self . _download_json (
base_url + data , uploader , ' Downloading track page %s ' % ( i + 1 ) )
2014-07-07 14:21:02 -04:00
if len ( new_entries ) == 0 :
self . to_screen ( ' %s : End page received ' % uploader )
2013-09-14 15:41:49 -04:00
break
2014-07-07 14:21:02 -04:00
entries . extend ( self . _extract_info_dict ( e , quiet = True ) for e in new_entries )
2013-09-14 15:41:49 -04:00
return {
' _type ' : ' playlist ' ,
' id ' : compat_str ( user [ ' id ' ] ) ,
' title ' : user [ ' username ' ] ,
2014-05-04 21:12:41 -04:00
' entries ' : entries ,
}
class SoundcloudPlaylistIE ( SoundcloudIE ) :
_VALID_URL = r ' https?://api \ .soundcloud \ .com/playlists/(?P<id>[0-9]+) '
IE_NAME = ' soundcloud:playlist '
2014-08-27 18:58:24 -04:00
_TESTS = [
2014-05-04 21:12:41 -04:00
2014-08-27 18:58:24 -04:00
{
' url ' : ' http://api.soundcloud.com/playlists/4110309 ' ,
' info_dict ' : {
' id ' : ' 4110309 ' ,
' title ' : ' TILT Brass - Bowery Poetry Club, August \' 03 [Non-Site SCR 02] ' ,
' description ' : ' re:.*?TILT Brass - Bowery Poetry Club ' ,
} ,
' playlist_count ' : 6 ,
}
]
2014-05-04 21:12:41 -04:00
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
playlist_id = mobj . group ( ' id ' )
base_url = ' %s //api.soundcloud.com/playlists/ %s .json? ' % ( self . http_scheme ( ) , playlist_id )
data = compat_urllib_parse . urlencode ( {
' client_id ' : self . _CLIENT_ID ,
} )
data = self . _download_json (
base_url + data , playlist_id , ' Downloading playlist ' )
entries = [
self . _extract_info_dict ( t , quiet = True ) for t in data [ ' tracks ' ] ]
return {
' _type ' : ' playlist ' ,
' id ' : playlist_id ,
' title ' : data . get ( ' title ' ) ,
' description ' : data . get ( ' description ' ) ,
' entries ' : entries ,
2013-09-14 15:41:49 -04:00
}