module Grooveshark class Client attr_accessor :session, :comm_token attr_reader :user, :comm_token_ttl, :country def initialize(params = {}) @ttl = params[:ttl] || 120 # 2 minutes @session, @country = get_session_and_country @uuid = UUID.new.generate.upcase get_comm_token end # Authenticate user def login(user, password) data = request('authenticateUser', {:username => user, :password => password}, true) @user = User.new(self, data) raise InvalidAuthentication, 'Wrong username or password!' if @user.id == 0 return @user end # Find user by ID def get_user_by_id(id) resp = request('getUserByID', {:userID => id})['user'] resp['username'].empty? ? nil : User.new(self, resp) end # Find user by username def get_user_by_username(name) resp = request('getUserByUsername', {:username => name})['user'] resp['username'].empty? ? nil : User.new(self, resp) end # Get recently active users def recent_users request('getRecentlyActiveUsers', {})['users'].map { |u| User.new(self, u) } end # Get popular songs # type => daily, monthly def popular_songs(type='daily') raise ArgumentError, 'Invalid type' unless ['daily', 'monthly'].include?(type) request('popularGetSongs', {:type => type})['songs'].map { |s| Song.new(s) } end # Perform search request for query def search(type, query) results = request('getResultsFromSearch', {:type => type, :query => query})['result'] results.map { |song| Song.new song } end # Perform songs search request for query def search_songs(query) search('Songs', query) end # Return raw response for songs search request def search_songs_pure(query) request('getSearchResultsEx', {:type => 'Songs', :query => query}) end # Get stream authentication by song ID def get_stream_auth_by_songid(song_id) result = request('getStreamKeyFromSongIDEx', { 'type' => 0, 'prefetch' => false, 'songID' => song_id, 'country' => @country, 'mobile' => false, }) if result == [] then raise GeneralError, "No data for this song. Maybe Grooveshark banned your IP." end result end # Get stream authentication for song object def get_stream_auth(song) get_stream_auth_by_songid(song.id) end # Get song stream url by ID def get_song_url_by_id(id) resp = get_stream_auth_by_songid(id) "http://#{resp['ip']}/stream.php?streamKey=#{resp['stream_key']}" end # Get song stream def get_song_url(song) get_song_url_by_id(song.id) end def get_session_and_country response = RestClient.get('http://grooveshark.com') session = response.headers[:set_cookie].to_s.scan(/PHPSESSID=([a-z\d]{32});/i).flatten.first preload_regex = /gsPreloadAjax\(\{url: '\/preload.php\?(.*)&hash=' \+ clientPage\}\)/ preload_id = response.to_s.scan(preload_regex).flatten.first preload_url = "http://grooveshark.com/preload.php?#{preload_id}&getCommunicationToken=1&hash=%2F" preload_response = RestClient.get(preload_url) config_json = preload_response.to_s.scan(/window.tokenData = (.*);/).flatten.first raise GeneralError, "gsConfig not found" if not config_json config = JSON.parse(config_json)['getGSConfig'] [session, config['country']] end # Get communication token def get_comm_token @comm_token = nil # request() uses it @comm_token = request('getCommunicationToken', {:secretKey => Digest::MD5.hexdigest(@session)}, true) @comm_token_ttl = Time.now.to_i end # Sign method def create_token(method) rnd = get_random_hex_chars(6) salt = 'gooeyFlubber' plain = [method, @comm_token, salt, rnd].join(':') hash = Digest::SHA1.hexdigest(plain) "#{rnd}#{hash}" end def get_random_hex_chars(length) chars = ('a'..'f').to_a | (0..9).to_a (0...length).map { chars[rand(chars.length)] }.join end # Perform API request def request(method, params={}, secure=false) refresh_token if @comm_token url = "#{secure ? 'https' : 'http'}://grooveshark.com/more.php?#{method}" body = { 'header' => { 'client' => 'mobileshark', 'clientRevision' => '20120830', 'country' => @country, 'privacy' => 0, 'session' => @session, 'uuid' => @uuid }, 'method' => method, 'parameters' => params } body['header']['token'] = create_token(method) if @comm_token begin data = RestClient.post(url, body.to_json, {'Content-Type' => 'application/json'}) rescue Exception => ex raise GeneralError, ex.message end data = JSON.parse(data) data = data.normalize if data.kind_of?(Hash) if data.key?('fault') raise ApiError.new(data['fault']) else data['result'] end end # Refresh communications token on ttl def refresh_token get_comm_token if Time.now.to_i - @comm_token_ttl > @ttl end end end