aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--lisp/emms-lastfm-client.el743
-rw-r--r--lisp/emms-lastfm.el697
2 files changed, 743 insertions, 697 deletions
diff --git a/lisp/emms-lastfm-client.el b/lisp/emms-lastfm-client.el
new file mode 100644
index 0000000..20ec018
--- /dev/null
+++ b/lisp/emms-lastfm-client.el
@@ -0,0 +1,743 @@
+;;; -*- show-trailing-whitespace: t -*-
+;;; emms-lastfm-client.el --- Last.FM Music API
+
+;; Copyright (C) 2009 Free Software Foundation, Inc.
+
+;; Author: Yoni Rabkin <yonirabkin@member.fsf.org>
+
+;; Keywords: emms, lastfm
+
+;; EMMS is free software; you can redistribute it and/or modify it
+;; under the terms of the GNU General Public License as published by
+;; the Free Software Foundation; either version 3, or (at your option)
+;; any later version.
+;;
+;; EMMS is distributed in the hope that it will be useful, but WITHOUT
+;; ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
+;; or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
+;; License for more details.
+;;
+;; You should have received a copy of the GNU General Public License
+;; along with EMMS; see the file COPYING. If not, write to the Free
+;; Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
+;; MA 02110-1301, USA.
+
+;;; Commentary:
+;;
+;; There are restrictions on the use of this service. Quoting from
+;; [http://www.last.fm/api/radio]: "Who can I stream radio to? Any API
+;; account can only stream radio to Last.fm's paid subscribers".
+;;
+;; We've spoken to representatives from Last.fm and arrived at the
+;; following agreement: In order to be able to use the service while
+;; preserving the essential freedoms of the GPL each client must apply
+;; for their own API key from Last.fm.
+
+;;; Installation:
+;;
+;; Here is how to get authorization from Last.fm to stream
+;; music. Thankfully this only needs to be done _once_:
+;;
+;; 1. Complete steps 1 and 2 from
+;; [http://www.last.fm/api/authentication] to get an API key and a
+;; secret key. Set `emms-lastfm-client-api-key' and
+;; `emms-lastfm-client-api-secret-key' accordingly.
+;;
+;; 2. M-x emms-lastfm-client-user-authorization
+;;
+;; If this completes successfully a browser window will open asking
+;; for confirmation to allow this application to access the Last.fm
+;; account. Confirm and close the browser.
+;;
+;; 3. M-x emms-lastfm-client-get-session
+;;
+;; After this last step the permanent session key will be stored in
+;; `emms-lastfm-client-session-key-file'. As long as this key value
+;; is accessible the authentication process need never be repeated.
+
+;;; Use:
+;;
+;; M-x emms-lastfm-client-play-similar-artists
+;;
+;; Call the ...-play- family of interactive functions, such the above,
+;; to start streaming music.
+;;
+;; To show the currently playing track: M-x emms-lastfm-client-show
+;;
+;; To skip to the next track: M-x emms-lastfm-client-track-advance
+;;
+;; ...more Last.fm functionality to come, so stay tuned.
+
+;;; Code:
+
+(require 'md5)
+(require 'xml)
+
+(defvar emms-lastfm-client-api-key nil
+ "Key for the Last.fm API.")
+
+(defvar emms-lastfm-client-api-secret-key nil
+ "Secret key for the Last.fm API.")
+
+(defvar emms-lastfm-client-api-session-key nil
+ "Session key for the Last.fm API.")
+
+(defvar emms-lastfm-client-token nil
+ "Authorization token for API.")
+
+(defvar emms-lastfm-client-api-base-url
+ "http://ws.audioscrobbler.com/2.0/"
+ "URL for API calls.")
+
+(defvar emms-lastfm-client-session-key-file
+ (concat (file-name-as-directory emms-directory)
+ "emms-lastfm-client-sessionkey")
+ "File for storing the Last.fm API session key.")
+
+(defvar emms-lastfm-client-playlist-valid nil
+ "True if the playlist hasn't expired.")
+
+(defvar emms-lastfm-client-playlist-timer nil
+ "Playlist timer object.")
+
+(defvar emms-lastfm-client-playlist nil
+ "Latest Last.fm playlist.")
+
+(defvar emms-lastfm-client-track nil
+ "Latest Last.fm track.")
+
+(defvar emms-lastfm-client-original-next-function nil
+ "Original `-next-function' to be restored.")
+
+(defvar emms-lastfm-client-playlist-buffer-name "*Emms Last.fm*"
+ "Name for non-interactive Emms Last.fm buffer.")
+
+(defvar emms-lastfm-client-playlist-buffer nil
+ "Non-interactive Emms Last.fm buffer.")
+
+(defvar emms-lastfm-client-api-method-dict
+ '((auth-get-token . ("auth.gettoken"
+ emms-lastfm-client-auth-get-token-ok
+ emms-lastfm-client-auth-get-token-failed))
+ (auth-get-session . ("auth.getsession"
+ emms-lastfm-client-auth-get-session-ok
+ emms-lastfm-client-auth-get-session-failed))
+ (radio-tune . ("radio.tune"
+ emms-lastfm-client-radio-tune-ok
+ emms-lastfm-client-radio-tune-failed))
+ (radio-getplaylist . ("radio.getplaylist"
+ emms-lastfm-client-radio-getplaylist-ok
+ emms-lastfm-client-radio-getplaylist-failed)))
+ "Mapping symbols to method calls. This is a list of cons pairs
+ where the CAR is the symbol name of the method and the CDR is a
+ list whose CAR is the method call string, CADR is the function
+ to call on a success and CADDR is the function to call on
+ failure.")
+
+;;; ------------------------------------------------------------------
+;;; API method call
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-get-method (method)
+ "Return the associated method cons for the symbol METHOD."
+ (let ((m (cdr (assoc method emms-lastfm-client-api-method-dict))))
+ (if (not m)
+ (error "method not in dictionary: %s" method)
+ m)))
+
+(defun emms-lastfm-client-get-method-name (method)
+ "Return the associated method string for the symbol METHOD."
+ (let ((this (nth 0 (emms-lastfm-client-get-method method))))
+ (if (not this)
+ (error "no name string registered for method: %s" method)
+ this)))
+
+(defun emms-lastfm-client-get-method-ok (method)
+ "Return the associated OK function for METHOD.
+
+This function is called when the method call returns
+successfully."
+ (let ((this (nth 1 (emms-lastfm-client-get-method method))))
+ (if (not this)
+ (error "no OK function registered for method: %s" method)
+ this)))
+
+(defun emms-lastfm-client-get-method-fail (method)
+ "Return the associated fail function for METHOD.
+
+This function is called when the method call returns a failure
+status message."
+ (let ((this (nth 2 (emms-lastfm-client-get-method method))))
+ (if (not this)
+ (error "no fail function registered for method: %s" method)
+ this)))
+
+(defun emms-lastfm-client-encode-arguments (arguments)
+ "Encode ARGUMENTS in UTF-8 for the Last.fm API."
+ (let ((result nil))
+ (while arguments
+ (setq result
+ (append result
+ (list
+ (cons
+ (encode-coding-string (caar arguments) 'utf-8)
+ (encode-coding-string (cdar arguments) 'utf-8)))))
+ (setq arguments (cdr arguments)))
+ result))
+
+(defun emms-lastfm-client-construct-arguments (str arguments)
+ "Return a concatenation of arguments for the URL."
+ (cond ((not arguments) str)
+ (t (emms-lastfm-client-construct-arguments
+ (concat str "&" (caar arguments) "=" (url-hexify-string (cdar arguments)))
+ (cdr arguments)))))
+
+(defun emms-lastfm-client-construct-method-call (method arguments)
+ "Return a complete URL method call for METHOD with ARGUMENTS.
+
+This function includes the cryptographic signature."
+ (concat emms-lastfm-client-api-base-url "?"
+ "method=" (emms-lastfm-client-get-method-name method)
+ (emms-lastfm-client-construct-arguments
+ "" arguments)
+ "&api_sig="
+ (emms-lastfm-client-construct-signature method arguments)))
+
+(defun emms-lastfm-client-construct-write-method-call (method arguments)
+ "Return a complete POST body method call for METHOD with ARGUMENTS.
+
+This function includes the cryptographic signature."
+ (concat "method=" (emms-lastfm-client-get-method-name method)
+ (emms-lastfm-client-construct-arguments
+ "" arguments)
+ "&api_sig="
+ (emms-lastfm-client-construct-signature method arguments)))
+
+;;; ------------------------------------------------------------------
+;;; Response handler
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-handle-response (method xml-response)
+ "Dispatch the handler functions of METHOD for XML-RESPONSE."
+ (let ((status (cdr (assoc 'status (nth 1 (car xml-response)))))
+ (data (cddar xml-response)))
+ (when (not status)
+ (error "error parsing status from: %s" xml-response))
+ (cond ((string= status "failed")
+ (funcall (emms-lastfm-client-get-method-fail method) data))
+ ((string= status "ok")
+ (funcall (emms-lastfm-client-get-method-ok method) data))
+ (t (error "unknown response status %s" status)))))
+
+;;; ------------------------------------------------------------------
+;;; Unathorized request token for an API account
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-construct-urt ()
+ "Return a request for an Unauthorized Request Token."
+ (let ((arguments
+ (emms-lastfm-client-encode-arguments
+ `(("api_key" . ,emms-lastfm-client-api-key)))))
+ (emms-lastfm-client-construct-method-call
+ 'auth-get-token arguments)))
+
+(defun emms-lastfm-client-make-call-urt ()
+ "Make method call for Unauthorized Request Token."
+ (let* ((url-request-method "POST"))
+ (let ((response
+ (url-retrieve-synchronously
+ (emms-lastfm-client-construct-urt))))
+ (emms-lastfm-client-handle-response
+ 'auth-get-token
+ (with-current-buffer response
+ (xml-parse-region (point-min) (point-max)))))))
+
+;; example response: ((lfm ((status . \"ok\")) \"\" (token nil
+;; \"31cab3398a9b46cf7231ef84d73169cf\")))
+
+;;; ------------------------------------------------------------------
+;;; Signatures
+;;; ------------------------------------------------------------------
+;;
+;; From [http://www.last.fm/api/desktopauth]:
+;;
+;; Construct your api method signatures by first ordering all the
+;; parameters sent in your call alphabetically by parameter name and
+;; concatenating them into one string using a <name><value>
+;; scheme. So for a call to auth.getSession you may have:
+;;
+;; api_keyxxxxxxxxmethodauth.getSessiontokenxxxxxxx
+;;
+;; Ensure your parameters are utf8 encoded. Now append your secret
+;; to this string. Finally, generate an md5 hash of the resulting
+;; string. For example, for an account with a secret equal to
+;; 'mysecret', your api signature will be:
+;;
+;; api signature = md5("api_keyxxxxxxxxmethodauth.getSessiontokenxxxxxxxmysecret")
+;;
+;; Where md5() is an md5 hashing operation and its argument is the
+;; string to be hashed. The hashing operation should return a
+;; 32-character hexadecimal md5 hash.
+
+(defun emms-lastfm-client-construct-lexi (arguments)
+ "Return ARGUMENTS sorted in lexicographic order."
+ (let ((lexi (sort arguments
+ '(lambda (a b) (string< (car a) (car b)))))
+ (out ""))
+ (while lexi
+ (setq out (concat out (caar lexi) (cdar lexi)))
+ (setq lexi (cdr lexi)))
+ out))
+
+(defun emms-lastfm-client-construct-signature (method arguments)
+ "Return request signature for METHOD and ARGUMENTS."
+ (let ((complete-arguments
+ (append arguments
+ `(("method" .
+ ,(emms-lastfm-client-get-method-name method))))))
+ (md5
+ (concat (emms-lastfm-client-construct-lexi complete-arguments)
+ emms-lastfm-client-api-secret-key))))
+
+;;; ------------------------------------------------------------------
+;;; General error handling
+;;; ------------------------------------------------------------------
+
+;; Each method call provides its own error codes, but if we don't want
+;; to code a handler for a method we call this instead:
+(defun emms-lastfm-client-default-error-handler (data)
+ "Default method failure handler."
+ (let ((errorcode (cdr (assoc 'code (nth 1 (cadr data)))))
+ (message (nth 2 (cadr data))))
+ (when (not (and errorcode message))
+ (error "failed to read errorcode or message: %s %s"
+ errorcode message))
+ (error "method call failed with code %s: %s"
+ errorcode message)))
+
+;;; ------------------------------------------------------------------
+;;; Request authorization from the user
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-ask-for-auth ()
+ "Open a Web browser for authorizing the application."
+ (when (not (and emms-lastfm-client-api-key
+ emms-lastfm-client-token))
+ (error "API key and authorization token needed."))
+ (browse-url
+ (format "http://www.last.fm/api/auth/?api_key=%s&token=%s"
+ emms-lastfm-client-api-key
+ emms-lastfm-client-token)))
+
+;;; ------------------------------------------------------------------
+;;; Parse XSPF
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-xspf-header (data)
+ "Return an alist representing the XSPF header of DATA."
+ (let (out
+ (orig data))
+ (setq data (cadr data))
+ (while data
+ (when (and (car data)
+ (listp (car data))
+ (= (length (car data)) 3))
+ (setq out (append out (list (cons (nth 0 (car data))
+ (nth 2 (car data)))))))
+ (setq data (cdr data)))
+ (if (not out)
+ (error "failed to parse XSPF header from: %s" orig)
+ out)))
+
+(defun emms-lastfm-client-xspf-tracklist (data)
+ "Return the start of the track-list in DATE."
+ (nthcdr 3 (nth 11 (cadr data))))
+
+(defun emms-lastfm-client-xspf-header-date (header-alist)
+ "Return the date parameter from HEADER-ALIST."
+ (let ((out (cdr (assoc 'date header-alist))))
+ (if (not out)
+ (error "could not read date from header alist: %s"
+ header-alist)
+ out)))
+
+(defun emms-lastfm-client-xspf-header-expiry (header-alist)
+ "Return the expiry parameter from HEADER-ALIST."
+ (let ((out (cdr (assoc 'link header-alist))))
+ (if (not out)
+ (error "could not read expiry from header alist: %s"
+ header-alist)
+ out)))
+
+(defun emms-lastfm-client-xspf-header-creator (header-alist)
+ "Return the creator parameter from HEADER-ALIST."
+ (let ((out (cdr (assoc 'creator header-alist))))
+ (if (not out)
+ (error "could not read creator from header alist: %s"
+ header-alist)
+ out)))
+
+(defun emms-lastfm-client-xspf-playlist (data)
+ "Return the playlist from the XSPF DATA."
+ (let ((playlist (car (nthcdr 11 data))))
+ (if (not playlist)
+ (error "could not read playlist from: %s" data)
+ playlist)))
+
+(defun emms-lastfm-client-xspf-get (node track)
+ "Return data associated with NODE in TRACK."
+ (let ((result nil))
+ (while track
+ (when (consp track)
+ (let ((this (car track)))
+ (when (and (consp this)
+ (= (length this) 3)
+ (symbolp (nth 0 this))
+ (stringp (nth 2 this))
+ (equal (nth 0 this) node))
+ (setq result (nth 2 this)))))
+ (setq track (cdr track)))
+ (if (not result)
+ nil
+ result)))
+
+;;; ------------------------------------------------------------------
+;;; Timers
+;;; ------------------------------------------------------------------
+
+;; timed playlist invalidation is a part of the Last.fm API
+(defun emms-lastfm-client-set-timer (header)
+ "Start timer countdown to playlist invalidation"
+ (when (not header)
+ (error "can't set timer with no header data"))
+ (let ((expiry (parse-integer
+ (emms-lastfm-client-xspf-header-expiry header))))
+ (setq emms-lastfm-client-playlist-valid t)
+ (setq emms-lastfm-client-playlist-timer
+ (run-at-time
+ expiry nil
+ '(lambda () (setq emms-lastfm-client-playlist-valid
+ nil))))))
+
+;;; ------------------------------------------------------------------
+;;; Player
+;;; ------------------------------------------------------------------
+
+;; this should return `nil' to the track-manager when the playlist has
+;; been exhausted
+(defun emms-lastfm-client-consume-next-track ()
+ "Pop and return the next track from the playlist or nil."
+ (when emms-lastfm-client-playlist
+ (if emms-lastfm-client-playlist-valid
+ (let ((track (car emms-lastfm-client-playlist)))
+ ;; we can only request each track once so we pop it off the
+ ;; playlist
+ (setq emms-lastfm-client-playlist
+ (if (stringp (cdr emms-lastfm-client-playlist))
+ (cddr emms-lastfm-client-playlist)
+ (cdr emms-lastfm-client-playlist)))
+ track)
+ (error "playlist invalid"))))
+
+(defun emms-lastfm-client-set-lastfm-playlist-buffer ()
+ (when (not (buffer-live-p emms-lastfm-client-playlist-buffer))
+ (setq emms-lastfm-client-playlist-buffer
+ (emms-playlist-new
+ emms-lastfm-client-playlist-buffer-name))
+ (setq emms-playlist-buffer emms-lastfm-client-playlist-buffer)))
+
+(defun emms-lastfm-client-load-next-track ()
+ (with-current-buffer emms-lastfm-client-playlist-buffer
+ (emms-playlist-clear)
+ (if emms-lastfm-client-playlist
+ (let ((track (emms-lastfm-client-consume-next-track)))
+ (setq emms-lastfm-client-track track)
+ (when emms-player-playing-p
+ (emms-stop))
+ (emms-play-url
+ (emms-lastfm-client-xspf-get 'location track)))
+ (emms-lastfm-client-make-call-radio-getplaylist)
+ (emms-lastfm-client-load-next-track))))
+
+;; call this `-track-advance' to avoid confusion with Emms'
+;; `-next-track-' mechanism
+(defun emms-lastfm-client-track-advance ()
+ (interactive)
+ (when (equal emms-playlist-buffer
+ emms-lastfm-client-playlist-buffer)
+ (emms-lastfm-client-load-next-track)))
+
+(defun emms-lastfm-client-play-playlist ()
+ "Entry point to play tracks from Last.fm."
+ (emms-lastfm-client-set-lastfm-playlist-buffer)
+ (add-hook 'emms-player-finished-hook
+ 'emms-lastfm-client-track-advance)
+ (emms-lastfm-client-track-advance))
+
+;; stolen from Tassilo Horn's original emms-lastfm.el
+(defun emms-lastfm-client-read-artist ()
+ "Read an artist name from the user."
+ (let ((artists nil))
+ (when (boundp 'emms-cache-db)
+ (maphash
+ #'(lambda (file track)
+ (let ((artist (emms-track-get track 'info-artist)))
+ (when artist
+ (add-to-list 'artists artist))))
+ emms-cache-db))
+ (if artists
+ (emms-completing-read "Artist: " artists)
+ (read-string "Artist: "))))
+
+(defun emms-lastfm-client-play-similar-artists (artist)
+ "Play a Last.fm station with music similar to ARTIST."
+ (interactive (list (emms-lastfm-client-read-artist)))
+ (when (not (stringp artist))
+ (error "not a string: %s" artist))
+ (emms-lastfm-client-check-session-key)
+ (emms-lastfm-client-make-call-radio-tune
+ (format "lastfm://artist/%s/similarartists" artist))
+ (emms-lastfm-client-make-call-radio-getplaylist)
+ (emms-lastfm-client-play-playlist))
+
+;;; ------------------------------------------------------------------
+;;; Information
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-convert-track (track)
+ "Convert a Last.fm track to an Emms track."
+ (let ((emms-track (emms-dictionary '*track*)))
+ (emms-track-set emms-track 'name
+ (emms-lastfm-client-xspf-get 'location track))
+ (emms-track-set emms-track 'info-artist
+ (emms-lastfm-client-xspf-get 'creator track))
+ (emms-track-set emms-track 'info-title
+ (emms-lastfm-client-xspf-get 'title track))
+ (emms-track-set emms-track 'info-album
+ (emms-lastfm-client-xspf-get 'album track))
+ (emms-track-set emms-track 'info-playing-time
+ (/
+ (parse-integer
+ (emms-lastfm-client-xspf-get 'duration track))
+ 1000))
+ emms-track))
+
+(defun emms-lastfm-client-show-track (track)
+ "Return description of TRACK."
+ (decode-coding-string
+ (format emms-show-format
+ (emms-track-description
+ (emms-lastfm-client-convert-track track)))
+ 'utf-8))
+
+(defun emms-lastfm-client-show ()
+ "Display a description of the current track."
+ (interactive)
+ (if emms-player-playing-p
+ (message
+ (emms-lastfm-client-show-track emms-lastfm-client-track))
+ nil))
+
+;;; ------------------------------------------------------------------
+;;; Desktop application authorization [http://www.last.fm/api/desktopauth]
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-user-authorization ()
+ "Ask user to authorize the application."
+ (interactive)
+ (emms-lastfm-client-make-call-urt)
+ (emms-lastfm-client-ask-for-auth))
+
+(defun emms-lastfm-client-get-session ()
+ "Retrieve and store session key."
+ (interactive)
+ (emms-lastfm-client-make-call-get-session)
+ (emms-lastfm-client-save-session-key
+ emms-lastfm-client-api-session-key))
+
+;;; ------------------------------------------------------------------
+;;; method: auth.getToken [http://www.last.fm/api/show?service=265]
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-auth-get-token-ok (data)
+ "Function called when auth.getToken succeeds."
+ (setq emms-lastfm-client-token
+ (nth 2 (cadr data)))
+ (if (or (not emms-lastfm-client-token)
+ (not (= (length emms-lastfm-client-token) 32)))
+ (error "could not read token from response %s" data)
+ (message "Emms Last.FM auth.getToken method call success.")))
+
+(defun emms-lastfm-client-auth-get-token-failed (data)
+ "Function called when auth.getToken fails."
+ (emms-lastfm-client-default-error-handler data))
+
+;;; ------------------------------------------------------------------
+;;; method: auth.getSession [http://www.last.fm/api/show?service=125]
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-construct-get-session ()
+ "Return an auth.getSession request string."
+ (let ((arguments
+ (emms-lastfm-client-encode-arguments
+ `(("token" . ,emms-lastfm-client-token)
+ ("api_key" . ,emms-lastfm-client-api-key)))))
+ (emms-lastfm-client-construct-method-call
+ 'auth-get-session arguments)))
+
+(defun emms-lastfm-client-make-call-get-session ()
+ "Make auth.getSession call."
+ (let* ((url-request-method "POST"))
+ (let ((response
+ (url-retrieve-synchronously
+ (emms-lastfm-client-construct-get-session))))
+ (emms-lastfm-client-handle-response
+ 'auth-get-session
+ (with-current-buffer response
+ (xml-parse-region (point-min) (point-max)))))))
+
+(defun emms-lastfm-client-save-session-key (key)
+ "Store KEY."
+ (let ((buffer (find-file-noselect
+ emms-lastfm-client-session-key-file)))
+ (set-buffer buffer)
+ (erase-buffer)
+ (insert key)
+ (save-buffer)
+ (kill-buffer buffer)))
+
+(defun emms-lastfm-client-load-session-key ()
+ "Return stored session key."
+ (let ((file (expand-file-name emms-lastfm-client-session-key-file)))
+ (setq emms-lastfm-client-api-session-key
+ (if (file-readable-p file)
+ (with-temp-buffer
+ (emms-insert-file-contents file)
+ (goto-char (point-min))
+ (buffer-substring-no-properties
+ (point) (point-at-eol)))
+ nil))))
+
+(defun emms-lastfm-client-check-session-key ()
+ "Signal an error condition if there is no session key."
+ (if emms-lastfm-client-api-session-key
+ emms-lastfm-client-api-session-key
+ (if (emms-lastfm-client-load-session-key)
+ emms-lastfm-client-api-session-key
+ (error "no session key for API access"))))
+
+(defun emms-lastfm-client-auth-get-session-ok (data)
+ "Function called on DATA if auth.getSession succeeds."
+ (let ((session-key (nth 2 (nth 5 (cadr data)))))
+ (cond (session-key
+ (setq emms-lastfm-client-api-session-key
+ session-key)
+ (message "Emms Last.fm session key retrieval successful"
+ session-key))
+ (t (error "failed to parse session key data %s" data)))))
+
+(defun emms-lastfm-client-auth-get-session-failed (data)
+ "Function called on DATA if auth.getSession fails."
+ (emms-lastfm-client-default-error-handler data))
+
+;;; ------------------------------------------------------------------
+;;; method: radio.tune [http://www.last.fm/api/show?service=160]
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-construct-radio-tune (station)
+ "Return a request to tune to STATION."
+ (let ((arguments
+ (emms-lastfm-client-encode-arguments
+ `(("sk" . ,emms-lastfm-client-api-session-key)
+ ("station" . ,station)
+ ("api_key" . ,emms-lastfm-client-api-key)))))
+ (emms-lastfm-client-construct-write-method-call
+ 'radio-tune arguments)))
+
+(defun emms-lastfm-client-make-call-radio-tune (station)
+ "Make call to tune to STATION."
+ (let ((url-request-method "POST")
+ (url-request-extra-headers
+ `(("Content-type" . "application/x-www-form-urlencoded")))
+ (url-request-data
+ (emms-lastfm-client-construct-radio-tune station)))
+ (let ((response
+ (url-retrieve-synchronously
+ emms-lastfm-client-api-base-url)))
+ (emms-lastfm-client-handle-response
+ 'radio-tune
+ (with-current-buffer response
+ (xml-parse-region (point-min) (point-max)))))))
+
+(defun emms-lastfm-client-radio-tune-failed (data)
+ "Function called on DATA when tuning fails."
+ (emms-lastfm-client-default-error-handler data))
+
+(defun emms-lastfm-client-radio-tune-ok (data)
+ "Set the current radio station according to DATA."
+ (let ((response (cdr (cadr data)))
+ data)
+ (while response
+ (when (and (listp (car response))
+ (car response)
+ (= (length (car response)) 3))
+ (add-to-list 'data (cons (caar response)
+ (caddr (car response)))))
+ (setq response (cdr response)))
+ (when (not data)
+ (error "could not parse station information %s" data))
+ (setq emms-lastfm-client-tuned-station-alist data)))
+
+;;; ------------------------------------------------------------------
+;;; method: radio.getPlaylist [http://www.last.fm/api/show?service=256]
+;;; ------------------------------------------------------------------
+
+(defun emms-lastfm-client-construct-radio-getplaylist ()
+ "Return a request for a playlist from the tuned station."
+ (let ((arguments
+ (emms-lastfm-client-encode-arguments
+ `(("sk" . ,emms-lastfm-client-api-session-key)
+ ("api_key" . ,emms-lastfm-client-api-key)))))
+ (emms-lastfm-client-construct-write-method-call
+ 'radio-getplaylist arguments)))
+
+(defun emms-lastfm-client-make-call-radio-getplaylist ()
+ "Make call for playlist from the tuned station."
+ (let ((url-request-method "POST")
+ (url-request-extra-headers
+ `(("Content-type" . "application/x-www-form-urlencoded")))
+ (url-request-data
+ (emms-lastfm-client-construct-radio-getplaylist)))
+ (let ((response
+ (url-retrieve-synchronously
+ emms-lastfm-client-api-base-url)))
+ (emms-lastfm-client-handle-response
+ 'radio-getplaylist
+ (with-current-buffer response
+ (xml-parse-region (point-min) (point-max)))))))
+
+(defun emms-lastfm-client-radio-getplaylist-failed (data)
+ "Function called on DATA when retrieving a playlist fails."
+ 'stub-needs-to-handle-playlist-issues
+ (emms-lastfm-client-default-error-handler data))
+
+(defun emms-lastfm-client-list-filter (l)
+ "Remove strings from the roots of list L."
+ (let (acc)
+ (while l
+ (when (listp (car l))
+ (push (car l) acc))
+ (setq l (cdr l)))
+ (reverse acc)))
+
+(defun emms-lastfm-client-radio-getplaylist-ok (data)
+ "Function called on DATA when retrieving a playlist succeeds."
+ (let ((header (emms-lastfm-client-xspf-header data))
+ (tracklist (emms-lastfm-client-xspf-tracklist data)))
+ (emms-lastfm-client-set-timer header)
+ (setq emms-lastfm-client-playlist
+ (emms-lastfm-client-list-filter tracklist))))
+
+(provide 'emms-lastfm-client)
+
+;;; emms-lastfm-client.el ends here
diff --git a/lisp/emms-lastfm.el b/lisp/emms-lastfm.el
deleted file mode 100644
index a58e13a..0000000
--- a/lisp/emms-lastfm.el
+++ /dev/null
@@ -1,697 +0,0 @@
-;;; emms-lastfm.el --- add your listened songs to your profile at last.fm
-
-;; Copyright (C) 2006, 2007, 2008, 2009 Free Software Foundation, Inc.
-
-;; Author: Tassilo Horn <tassilo@member.fsf.org>
-
-;; Keywords: emms, mp3, mpeg, multimedia
-
-;; This file is part of EMMS.
-
-;; EMMS is free software; you can redistribute it and/or modify it under the
-;; terms of the GNU General Public License as published by the Free Software
-;; Foundation; either version 3, or (at your option) any later version.
-
-;; EMMS is distributed in the hope that it will be useful, but WITHOUT ANY
-;; WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
-;; FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
-;; details.
-
-;; You should have received a copy of the GNU General Public License along with
-;; EMMS; see the file COPYING. If not, write to the Free Software Foundation,
-;; Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
-
-;;; Commentary:
-
-;; This code sends information about what music you are playing to last.fm.
-;; See <URL:http://www.last.fm> and
-;; <URL:http://www.audioscrobbler.net/wiki/Protocol1.1>.
-
-;;; Sample configuration:
-
-;; (setq emms-lastfm-username "my-user-name"
-;; emms-lastfm-password "very-secret!")
-
-;;; Usage:
-
-;; To activate the last.fm emms plugin, run:
-;; `M-x emms-lastfm-enable'
-
-;; Now all music you listen to will be submitted to Last.fm to enhance your
-;; profile.
-
-;; To deactivate the last.fm emms plugin, run:
-;; `M-x emms-lastfm-disable'
-
-;; Beside submitting the tracks you listen to, you can also listen to Last.fm
-;; radio. Simply copy the lastfm:// URL and run & paste:
-;; `M-x emms-lastfm-radio RET lastfm://artist/Britney Spears/fans'
-;; (Of course you don't need to use _this_ URL. :-))
-
-;; You can also insert Last.fm streams into playlists (or use
-;; emms-streams.el to listen to them) by activating the player as
-;; follows.
-;; (add-to-list 'emms-player-list 'emms-player-lastfm-radio)
-;; To insert a Last.fm stream into a playlist, do
-;; (emms-insert-lastfm "lastfm://rest-of-url")
-
-;; There are some functions for conveniently playing the Similar
-;; Artists, Fan Radio, and the Global Tag Radio. Here you only need to
-;; enter the band's name (for the first two) or the tag.
-;; `M-x emms-play-lastfm-similar-artists RET Britney Spears'
-;; `M-x emms-play-lastfm-artist-fan RET Modest Mouse'
-;; `M-x emms-play-lastfm-global-tag RET pop'
-
-;; When you're listening to a Last.fm radio station you have the possibility to
-;; give feedback to them. If you like the current song, type
-;; `M-x emms-lastfm-radio-love'.
-;; If it's not that good, or it just happens to not fit to your actual mood,
-;; type
-;; `M-x emms-lastfm-radio-skip'
-;; and this song will be skipped.
-;; If you really hate that song and you never want to hear it again, ban it by
-;; typing
-;; `M-x emms-lastfm-radio-ban'.
-
-;;; TODO
-;;
-;; - Get the last.fm radio stuff right again. Currently the rating stuff seems
-;; to be broken. There seems to be no official API, so one needs to look
-;; into the sources of the official client which can be found at
-;; http://www.audioscrobbler.net/development/client/.
-
-;; -----------------------------------------------------------------------
-
-(require 'url)
-(require 'emms)
-(require 'emms-mode-line)
-(require 'emms-playing-time)
-(require 'emms-source-file)
-(require 'emms-url)
-
-;;; Variables
-
-(defgroup emms-lastfm nil
- "Interaction with the services offered by http://www.last.fm."
- :prefix "emms-lastfm-"
- :group 'emms)
-
-(defcustom emms-lastfm-username ""
- "Your last.fm username"
- :type 'string
- :group 'emms-lastfm)
-
-(defcustom emms-lastfm-password ""
- "Your last.fm password"
- :type 'string
- :group 'emms-lastfm)
-
-(defcustom emms-lastfm-submission-verbose-p nil
- "If non-nil, display a message every time we submit a track to Last.fm."
- :type 'boolean
- :group 'emms-lastfm)
-
-(defcustom emms-lastfm-submit-track-types '(file)
- "Specify what types of tracks to submit to Last.fm.
-The default is to only submit files.
-
-To submit every track to Last.fm, set this to t.
-
-Note that it is not very meaningful to submit playlists,
-streamlists, or Last.fm streams to Last.fm."
- :type '(choice (const :tag "All" t)
- (set :tag "Types"
- (const :tag "Files" file)
- (const :tag "URLs" url)
- (const :tag "Playlists" playlist)
- (const :tag "Streamlists" streamlist)
- (const :tag "Last.fm streams" lastfm)))
- :group 'emms-lastfm)
-
-(defconst emms-lastfm-server "http://post.audioscrobbler.com/"
- "The last.fm server responsible for the handshaking
-procedure. Only for internal use.")
-(defconst emms-lastfm-client-id "ems"
- "The client ID of EMMS. Don't change it!")
-(defconst emms-lastfm-client-version 0.2
- "The version registered at last.fm. Don't change it!")
-(defconst emms-lastfm-protocol-version 1.2
- "The version of the supported last.fm protocol. Don't change it.")
-
-;; used internally
-(defvar emms-lastfm-process nil "-- only used internally --")
-(defvar emms-lastfm-session-id nil "-- only used internally --")
-(defvar emms-lastfm-now-playing-url nil "-- only used internally --")
-(defvar emms-lastfm-submit-url nil "-- only used internally --")
-(defvar emms-lastfm-current-track nil "-- only used internally --")
-(defvar emms-lastfm-timer nil "-- only used internally --")
-(defvar emms-lastfm-current-track-starting-time-string nil "-- only used internally --")
-
-;;; Scrobbling
-
-(defun emms-lastfm-new-track-function ()
- "This function should run whenever a new track starts (or a
-paused track resumes) and sets the track submission timer."
- (setq emms-lastfm-current-track
- (emms-playlist-current-selected-track))
- (setq emms-lastfm-current-track-starting-time-string
- (emms-lastfm-current-unix-time-string))
- ;; Tracks should be submitted, if they played 240 secs or half of their
- ;; length, whichever comes first.
- (let ((secs (emms-track-get emms-lastfm-current-track 'info-playing-time))
- (type (emms-track-type emms-lastfm-current-track)))
- (when (and secs
- (or (eq emms-lastfm-submit-track-types t)
- (and (listp emms-lastfm-submit-track-types)
- (memq type emms-lastfm-submit-track-types))))
- (when (> secs 240)
- (setq secs 240))
- (unless (< secs 30) ;; Skip titles shorter than 30 seconds
- (setq secs (- (/ secs 2) emms-playing-time))
- (unless (< secs 0)
- (setq emms-lastfm-timer
- (run-with-timer secs nil 'emms-lastfm-submit-track))))))
- ;; Update the now playing info displayed on the user's last.fm page. This
- ;; doesn't affect the user's profile, so it can be done even for tracks that
- ;; should not be submitted.
- (emms-lastfm-submit-now-playing))
-
-(defun emms-lastfm-http-POST (url string sentinel &optional sentinel-args)
- "Perform a HTTP POST request to URL using STRING as data.
-STRING will be encoded to utf8 before the request. Call SENTINEL
-with the result buffer."
- (let ((url-http-attempt-keepalives nil)
- (url-show-status emms-lastfm-submission-verbose-p)
- (url-request-method "POST")
- (url-request-extra-headers
- '(("Content-type"
- . "application/x-www-form-urlencoded; charset=utf-8")))
- (url-request-data (encode-coding-string string 'utf-8)))
- (url-retrieve url sentinel sentinel-args)))
-
-(defun emms-lastfm-http-GET (url sentinel &optional sentinel-args)
- "Perform a HTTP GET request to URL.
-Call SENTINEL with SENTINEL-ARGS and the result buffer."
- (let ((url-show-status emms-lastfm-submission-verbose-p)
- (url-request-method "GET"))
- (url-retrieve url sentinel sentinel-args)))
-
-(defun emms-lastfm-submit-now-playing ()
- "Submit now-playing infos to last.fm.
-These will be displayed on the user's last.fm page."
- (let* ((artist (emms-track-get emms-lastfm-current-track 'info-artist))
- (title (emms-track-get emms-lastfm-current-track 'info-title))
- (album (emms-track-get emms-lastfm-current-track 'info-album))
- (track-number (emms-track-get emms-lastfm-current-track
- 'info-tracknumber))
- (musicbrainz-id "")
- (track-length (number-to-string
- (or (emms-track-get emms-lastfm-current-track
- 'info-playing-time)
- 0))))
- ;; wait up to 5 seconds to submit np infos in order to finish handshaking.
- (dotimes (i 5)
- (when (not (and emms-lastfm-session-id
- emms-lastfm-now-playing-url))
- (sit-for 1)))
- (when (and emms-lastfm-session-id
- emms-lastfm-now-playing-url)
- (emms-lastfm-http-POST emms-lastfm-now-playing-url
- (concat "&s=" emms-lastfm-session-id
- "&a[0]=" (emms-url-quote artist)
- "&t[0]=" (emms-url-quote title)
- "&b[0]=" (emms-url-quote album)
- "&l[0]=" track-length
- "&n[0]=" track-number
- "&m[0]=" musicbrainz-id)
- 'emms-lastfm-submit-now-playing-sentinel))))
-
-(defun emms-lastfm-submit-now-playing-sentinel (&rest args)
- "Parses the server reponse and inform the user if all worked
-well or if an error occured."
- (let ((buffer (current-buffer)))
- (emms-http-decode-buffer buffer)
- (goto-char (point-min))
- ;; skip to the first empty line and go one line further. There the last.fm
- ;; response starts.
- (re-search-forward "^$" nil t)
- (forward-line)
- (if (re-search-forward "^OK$" nil t)
- (progn
- (when emms-lastfm-submission-verbose-p
- (message "EMMS: Now playing infos submitted to last.fm"))
- (kill-buffer buffer))
- (message "EMMS: Now playing infos couldn't be submitted to last.fm: %s"
- (emms-read-line)))))
-
-(defun emms-lastfm-cancel-timer ()
- "Cancels `emms-lastfm-timer' if it is running."
- (emms-cancel-timer emms-lastfm-timer)
- (setq emms-lastfm-timer nil))
-
-(defun emms-lastfm-pause ()
- "Handles things to be done when the player is paused or
-resumed."
- (if emms-player-paused-p
- ;; the player paused
- (emms-lastfm-cancel-timer)
- ;; The player resumed
- (emms-lastfm-new-track-function)))
-
-(defun emms-lastfm (&optional ARG)
- "Start submitting the tracks you listened to to
-http://www.last.fm, if ARG is positive. If ARG is negative or
-zero submission of the tracks will be stopped. This applies to
-the current track, too."
- (interactive "p")
- (cond
- ((not (and emms-lastfm-username emms-lastfm-password))
- (message "%s"
- (concat "EMMS: In order to activate the last.fm plugin you "
- "first have to set both `emms-lastfm-username' and "
- "`emms-lastfm-password'")))
- ((not emms-playing-time-p)
- (message "%s"
- (concat "EMMS: The last.fm plugin needs the functionality "
- "provided by `emms-playing-time'. It seems that you "
- "disabled it explicitly in your init file using code "
- "like this: `(emms-playing-time -1)'. Delete that "
- "line and have a look at `emms-playing-time's doc "
- "string")))
- (t
- (if (and ARG (> ARG 0))
- (progn
- ;; Append it. Else the playing time could be started a bit too late.
- (add-hook 'emms-player-started-hook
- 'emms-lastfm-handshake-if-needed t)
- ;; Has to be appended, because it has to run after
- ;; `emms-playing-time-start'
- (add-hook 'emms-player-started-hook
- 'emms-lastfm-new-track-function t)
- (add-hook 'emms-player-stopped-hook
- 'emms-lastfm-cancel-timer)
- (add-hook 'emms-player-paused-hook
- 'emms-lastfm-pause)
- ;; Clean up after EMMS radio
- (remove-hook 'emms-player-started-hook
- 'emms-lastfm-cancel-timer-after-stop)
- (message "EMMS Last.fm plugin activated"))
- (remove-hook 'emms-player-started-hook
- 'emms-lastfm-handshake-if-needed)
- (remove-hook 'emms-player-started-hook
- 'emms-lastfm-new-track-function)
- (remove-hook 'emms-player-stopped-hook
- 'emms-lastfm-cancel-timer)
- (remove-hook 'emms-player-paused-hook
- 'emms-lastfm-pause)
- (when emms-lastfm-timer (emms-cancel-timer emms-lastfm-timer))
- (setq emms-lastfm-session-id nil
- emms-lastfm-submit-url nil
- emms-lastfm-process nil
- emms-lastfm-current-track nil)
- (message "EMMS Last.fm plugin deactivated")))))
-
-(defalias 'emms-lastfm-activate 'emms-lastfm)
-(emms-make-obsolete 'emms-lastfm-activate 'emms-lastfm "EMMS 2.2")
-
-(defun emms-lastfm-enable ()
- "Enable the emms last.fm plugin."
- (interactive)
- (emms-lastfm 1))
-
-(defun emms-lastfm-disable ()
- "Disable the emms last.fm plugin."
- (interactive)
- (emms-lastfm -1))
-
-(defun emms-lastfm-restart ()
- "Disable and reenable the last.fm plugin. This will cause a new
-handshake."
- (emms-lastfm-disable)
- (emms-lastfm-enable))
-
-(defun emms-lastfm-handshake-if-needed ()
- (when (not (and emms-lastfm-session-id
- emms-lastfm-submit-url
- emms-lastfm-now-playing-url))
- (emms-lastfm-handshake)))
-
-(defun emms-lastfm-current-unix-time-string ()
- (replace-regexp-in-string "\\..*" "" (number-to-string (float-time))))
-
-(defun emms-lastfm-handshake ()
- "Handshakes with the last.fm server."
- (let ((timestamp (emms-lastfm-current-unix-time-string)))
- (emms-lastfm-http-GET
- (concat emms-lastfm-server
- "?hs=true"
- "&p=" (number-to-string emms-lastfm-protocol-version)
- "&c=" emms-lastfm-client-id
- "&v=" (number-to-string emms-lastfm-client-version)
- "&u=" (emms-url-quote emms-lastfm-username)
- "&t=" timestamp
- "&a=" (md5 (concat (md5 emms-lastfm-password) timestamp)))
- 'emms-lastfm-handshake-sentinel)))
-
-(defun emms-lastfm-handshake-sentinel (&rest args)
- "Parses the server reponse and inform the user if all worked
-well or if an error occured."
- (let ((buffer (current-buffer)))
- (emms-http-decode-buffer buffer)
- (goto-char (point-min))
- ;; skip to the first empty line and go one line further. There the last.fm
- ;; response starts.
- (re-search-forward "^$" nil t)
- (forward-line)
- (let ((response (emms-read-line)))
- (if (not (string-match (rx (or "OK")) response))
- (message "EMMS: Handshake failed: %s" response)
- (forward-line)
- (setq emms-lastfm-session-id (emms-read-line))
- (forward-line)
- (setq emms-lastfm-now-playing-url (emms-read-line))
- (forward-line)
- (setq emms-lastfm-submit-url (emms-read-line))
- (message "EMMS: Handshaking with server done")
- (kill-buffer buffer)))))
-
-(defun emms-lastfm-submit-track ()
- "Submits the current track (`emms-lastfm-current-track') to
-last.fm."
- (let* ((artist (emms-track-get emms-lastfm-current-track 'info-artist))
- (title (emms-track-get emms-lastfm-current-track 'info-title))
- (album (emms-track-get emms-lastfm-current-track 'info-album))
- (track-number (emms-track-get emms-lastfm-current-track 'info-tracknumber))
- (musicbrainz-id "")
- (track-length (number-to-string
- (emms-track-get emms-lastfm-current-track
- 'info-playing-time))))
- (emms-lastfm-http-POST
- emms-lastfm-submit-url
- (concat "&s=" emms-lastfm-session-id
- "&a[0]=" (emms-url-quote artist)
- "&t[0]=" (emms-url-quote title)
- "&i[0]=" emms-lastfm-current-track-starting-time-string
- "&o[0]=P" ;; TODO: Maybe support others. See the API.
- "&r[0]=" ;; The rating. Empty if not applicable (for P it's not)
- "&l[0]=" track-length
- "&b[0]=" (emms-url-quote album)
- "&n[0]=" track-number
- "&m[0]=" musicbrainz-id)
- 'emms-lastfm-submission-sentinel)))
-
-(defun emms-lastfm-submission-sentinel (&rest args)
- "Parses the server reponse and inform the user if all worked
-well or if an error occured."
- (let ((buffer (current-buffer)))
- (emms-http-decode-buffer buffer)
- (goto-char (point-min))
- ;; skip to the first empty line and go one line further. There the last.fm
- ;; response starts.
- (re-search-forward "^$" nil t)
- (forward-line)
- (if (re-search-forward "^OK$" nil t)
- (progn
- (when emms-lastfm-submission-verbose-p
- (message "EMMS: \"%s\" submitted to last.fm"
- (emms-track-description emms-lastfm-current-track)))
- (kill-buffer buffer))
- (message "EMMS: Song couldn't be submitted to last.fm: %s"
- (emms-read-line)))))
-
-;;; Playback of lastfm:// streams
-
-(defgroup emms-player-lastfm-radio nil
- "EMMS player for Last.fm streams."
- :group 'emms-player
- :prefix "emms-player-lastfm-")
-
-(defcustom emms-player-lastfm-radio (emms-player 'emms-lastfm-radio-start
- 'ignore ; no need to stop
- 'emms-lastfm-radio-playable-p)
- "*Parameters for the Last.fm radio player."
- :type '(cons symbol alist)
- :group 'emms-player-lastfm-radio)
-
-(defconst emms-lastfm-radio-base-url "http://ws.audioscrobbler.com/radio/"
- "The base URL for playing lastfm:// stream.
--- only used internally --")
-
-(defvar emms-lastfm-radio-session nil "-- only used internally --")
-(defvar emms-lastfm-radio-stream-url nil "-- only used internally --")
-
-(defun emms-lastfm-radio-get-handshake-url ()
- (concat emms-lastfm-radio-base-url
- "handshake.php?version=" (number-to-string
- emms-lastfm-client-version)
- "&platform=" emms-lastfm-client-id
- "&username=" (emms-url-quote emms-lastfm-username)
- "&passwordmd5=" (md5 emms-lastfm-password)
- "&debug=" (number-to-string 9)))
-
-(defun emms-lastfm-radio-handshake (fn radio-url)
- "Handshakes with the last.fm server.
-Calls FN when done with RADIO-URL as its only argument."
- (emms-lastfm-http-GET (emms-lastfm-radio-get-handshake-url)
- 'emms-lastfm-radio-handshake-sentinel
- (list fn radio-url)))
-
-(defun emms-lastfm-radio-handshake-sentinel (status fn radio-url)
- (let ((buffer (current-buffer)))
- (emms-http-decode-buffer buffer)
- (setq emms-lastfm-radio-session (emms-key-value "session"))
- (setq emms-lastfm-radio-stream-url (emms-key-value "stream_url"))
- (kill-buffer buffer)
- (if (and emms-lastfm-radio-session emms-lastfm-radio-stream-url)
- (progn
- (message "EMMS: Handshaking for Last.fm playback successful")
- (funcall fn radio-url))
- (message "EMMS: Failed handshaking for Last.fm playback"))))
-
-(defun emms-lastfm-radio-1 (lastfm-url)
- "Internal function used by `emms-lastfm-radio'."
- (if (and emms-lastfm-radio-session
- emms-lastfm-radio-stream-url)
- (progn
- (emms-lastfm-http-GET
- (concat emms-lastfm-radio-base-url
- "adjust.php?"
- "session=" emms-lastfm-radio-session
- "&url=" (emms-url-quote lastfm-url)
- "&debug=" (number-to-string 0))
- 'emms-lastfm-radio-sentinel))
- (message "EMMS: Cannot play Last.fm stream")))
-
-(defun emms-lastfm-radio (lastfm-url)
- "Plays the stream associated with the given Last.fm URL. (A
-Last.fm URL has the form lastfm://foo/bar/baz, e.g.
-
- lastfm://artist/Manowar/similarartists
-
-or
-
- lastfm://globaltags/metal."
- (interactive "sLast.fm URL: ")
- ;; Streamed songs must not be added to the lastfm profile
- (emms-lastfm-disable)
- (if (not (and emms-lastfm-radio-session
- emms-lastfm-radio-stream-url))
- (emms-lastfm-radio-handshake #'emms-lastfm-radio-1 lastfm-url)
- (emms-lastfm-radio-1 lastfm-url)))
-
-(defun emms-lastfm-radio-playable-p (track)
- "Determine whether the Last.fm player can play this track."
- (let ((name (emms-track-get track 'name))
- (type (emms-track-get track 'type)))
- (and (eq type 'lastfm)
- (string-match "^lastfm://" name))))
-
-(defun emms-lastfm-radio-start (track)
- "Start playing TRACK."
- (when (emms-lastfm-radio-playable-p track)
- (let ((name (emms-track-get track 'name)))
- (emms-lastfm-radio name))))
-
-(defcustom emms-lastfm-radio-metadata-period 15
- "When listening to Last.fm Radio every how many seconds should
-emms-lastfm poll for metadata? If set to nil, there won't be any
-polling at all.
-
-The default is 15: That means that the mode line will display the
-wrong (last) track's data for a maximum of 15 seconds. If your
-network connection has a big latency this value may be too
-high. (But then streaming a 128KHz mp3 won't be fun anyway.)"
- :type '(choice integer
- (const :tag "Disable" nil))
- :group 'emms-lastfm)
-
-(defun emms-lastfm-cancel-timer-after-stop ()
- (add-hook 'emms-player-stopped-hook
- 'emms-lastfm-cancel-timer))
-
-(defun emms-lastfm-radio-sentinel (&rest args)
- (let ((buffer (current-buffer)))
- (emms-http-decode-buffer buffer)
- (if (string= (emms-key-value "response" buffer) "OK")
- (progn
- (kill-buffer buffer)
- (add-hook 'emms-player-started-hook
- 'emms-lastfm-cancel-timer-after-stop)
- (emms-play-url emms-lastfm-radio-stream-url)
- (when emms-lastfm-radio-metadata-period
- (when emms-lastfm-timer
- (emms-lastfm-cancel-timer))
- (setq emms-lastfm-timer
- (run-with-timer 0 emms-lastfm-radio-metadata-period
- 'emms-lastfm-radio-request-metadata)))
- (message "EMMS: Playing Last.fm stream"))
- (kill-buffer buffer)
- (message "EMMS: Bad response from Last.fm"))))
-
-(defun emms-lastfm-np (&optional insertp callback)
- "Show the currently-playing lastfm radio tune.
-
-If INSERTP is non-nil, insert the description into the current
-buffer instead.
-
-If CALLBACK is a function, call it with the current buffer and
-description as arguments instead of displaying the description or
-inserting it."
- (interactive "P")
- (emms-lastfm-radio-request-metadata
- (lambda (status insertp buffer callback)
- (let ((response-buf (current-buffer))
- artist title)
- (emms-http-decode-buffer response-buf)
- (setq artist (emms-key-value "artist" response-buf)
- title (emms-key-value "track" response-buf))
- (kill-buffer response-buf)
- (let ((msg (if (and title artist)
- (format emms-show-format
- (format "%s - %s" artist title))
- "Nothing playing right now")))
- (cond ((functionp callback)
- (when (and title artist)
- (funcall callback buffer msg)))
- ((and insertp title artist)
- (with-current-buffer buffer
- (insert msg)))
- (t (message msg))))))
- (list insertp (current-buffer) callback)))
-
-(defun emms-lastfm-read-artist ()
- "Read an artist name from the user."
- (let ((artists nil))
- (when (boundp 'emms-cache-db)
- (maphash
- #'(lambda (file track)
- (let ((artist (emms-track-get track 'info-artist)))
- (when artist
- (add-to-list 'artists artist))))
- emms-cache-db))
- (if artists
- (emms-completing-read "Artist: " artists)
- (read-string "Artist: "))))
-
-(defun emms-play-lastfm-similar-artists (artist)
- "Plays the similar artist radio of ARTIST."
- (interactive (list (emms-lastfm-read-artist)))
- (emms-lastfm-radio (concat "lastfm://artist/"
- artist
- "/similarartists")))
-
-(defun emms-play-lastfm-global-tag (tag)
- "Plays the global tag radio of TAG."
- (interactive "sGlobal Tag: ")
- (emms-lastfm-radio (concat "lastfm://globaltags/" tag)))
-
-(defun emms-play-lastfm-artist-fan (artist)
- "Plays the artist fan radio of ARTIST."
- (interactive (list (emms-lastfm-read-artist)))
- (emms-lastfm-radio (concat "lastfm://artist/" artist "/fans")))
-
-(defun emms-lastfm-radio-love ()
- "Inform Last.fm that you love the currently playing song."
- (interactive)
- (emms-lastfm-radio-rating "love"))
-
-(defun emms-lastfm-radio-skip ()
- "Inform Last.fm that you want to skip the currently playing
-song."
- (interactive)
- (emms-lastfm-radio-rating "skip"))
-
-(defun emms-lastfm-radio-ban ()
- "Inform Last.fm that you want to ban the currently playing
-song."
- (interactive)
- (emms-lastfm-radio-rating "ban"))
-
-(defun emms-lastfm-radio-rating (command)
- (emms-lastfm-http-GET
- (concat emms-lastfm-radio-base-url
- "control.php?"
- "session=" emms-lastfm-radio-session
- "&command=" command
- "&debug=" (number-to-string 0))
- 'emms-lastfm-radio-rating-sentinel))
-
-(defun emms-lastfm-radio-rating-sentinel (&rest args)
- (let ((buffer (current-buffer)))
- (emms-http-decode-buffer buffer)
- (if (string= (emms-key-value "response" buffer) "OK")
- (message "EMMS: Rated current track")
- (message "EMMS: Rating failed"))
- (kill-buffer buffer)))
-
-(defun emms-lastfm-radio-request-metadata (&optional fn data)
- "Request the metadata of the current song and display it.
-
-If FN is given, call it instead of
-`emms-lastfm-radio-request-metadata-sentinel', with DATA as its
-first parameter.
-
-If DATA is given, it should be a list."
- (interactive)
- (emms-lastfm-http-GET
- (concat emms-lastfm-radio-base-url
- "np.php?"
- "session=" emms-lastfm-radio-session
- "&debug=" (number-to-string 0))
- (or fn 'emms-lastfm-radio-request-metadata-sentinel)
- data))
-
-(defun emms-lastfm-radio-request-metadata-sentinel (&rest args)
- (let ((buffer (current-buffer)))
- (emms-http-decode-buffer buffer)
- (let ((artist (emms-key-value "artist" buffer))
- (title (emms-key-value "track" buffer))
- (track (emms-playlist-current-selected-track)))
- (kill-buffer buffer)
- (emms-track-set track 'info-artist artist)
- (emms-track-set track 'info-title title)
- (emms-track-updated track))))
-
-
-;;; Utility functions
-
-(defun emms-read-line ()
- (buffer-substring-no-properties (line-beginning-position)
- (line-end-position)))
-
-(defun emms-key-value (key &optional buffer)
- "Returns the value of KEY from BUFFER.
-If BUFFER is nil, use the current buffer.
-
-BUFFER has to contain a key-value list like:
-
-foo=bar
-x=17"
- (unless (and buffer (not (buffer-live-p buffer)))
- (with-current-buffer (or buffer (current-buffer))
- (goto-char (point-min))
- (when (re-search-forward (concat "^" key "=") nil t)
- (buffer-substring-no-properties (point) (line-end-position))))))
-
-(provide 'emms-lastfm)
-;;; emms-lastfm.el ends here