/* * Copyright (C) 2016 Espen Jürgensen * * Stiched together from libspotify examples * * This program 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 2 of the License, or * (at your option) any later version. * * This program 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 this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #ifdef HAVE_CONFIG_H # include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef HAVE_PTHREAD_NP_H # include #endif #include #include #include #include "spotify.h" #include "logger.h" #include "misc.h" #include "http.h" #include "conffile.h" #include "filescanner.h" #include "cache.h" #include "commands.h" /* TODO for the web api: * - UI should be prettier * - map "added_at" to time_added * - what to do about the lack of push? * - use the web api more, implement proper init */ /* A few words on our reloading sequence of saved tracks * * 1. libspotify will not tell us about the user's saved tracks when loading * so we keep track of them with the special playlist spotify:savedtracks. * 2. spotify_login will copy all paths in spotify:savedtracks to a temporary * spotify_reload_list before all Spotify items in the database get purged. * 3. when the connection to Spotify is established after login, we register * all the paths with libspotify, and we also add them back to the * spotify:savedtracks playlist - however, that's just for the * playlistsitems table. Adding the items to the files table is done when * libspotify calls back with metadata - see spotify_pending_process(). * 4. if the user reloads saved tracks, we first clear all items in the * playlist, then add those back that are returned from the web api, and * then use our normal cleanup of stray files to tidy db and cache. */ // How long to wait for audio (in sec) before giving up #define SPOTIFY_TIMEOUT 20 // How long to wait for artwork (in sec) before giving up #define SPOTIFY_ARTWORK_TIMEOUT 3 // An upper limit on sequential requests to Spotify's web api // - each request will return 50 objects (tracks) #define SPOTIFY_WEB_REQUESTS_MAX 20 /* --- Types --- */ typedef struct audio_fifo_data { TAILQ_ENTRY(audio_fifo_data) link; int nsamples; int16_t samples[0]; } audio_fifo_data_t; typedef struct audio_fifo { TAILQ_HEAD(, audio_fifo_data) q; int qlen; int fullcount; pthread_mutex_t mutex; pthread_cond_t cond; } audio_fifo_t; enum spotify_state { SPOTIFY_STATE_INACTIVE, SPOTIFY_STATE_WAIT, SPOTIFY_STATE_PLAYING, SPOTIFY_STATE_PAUSED, SPOTIFY_STATE_STOPPING, SPOTIFY_STATE_STOPPED, }; struct audio_get_param { struct evbuffer *evbuf; int wanted; }; struct artwork_get_param { struct evbuffer *evbuf; char *path; int max_w; int max_h; sp_image *image; pthread_mutex_t mutex; pthread_cond_t cond; int is_loaded; }; struct pending_metadata { sp_link *link; sp_track *track; struct pending_metadata *next; }; struct reload_list { char *uri; struct reload_list *next; }; /* --- Globals --- */ // Spotify thread static pthread_t tid_spotify; // Used to make sure no login is attempted before the logout cb from Spotify static pthread_mutex_t login_lck; static pthread_cond_t login_cond; // Event base, pipes and events struct event_base *evbase_spotify; static int g_notify_pipe[2]; static struct event *g_notifyev; static struct commands_base *cmdbase; // The session handle static sp_session *g_sess; // The library handle static void *g_libhandle; // The state telling us what the thread is currently doing static enum spotify_state g_state; // The base playlist id for all Spotify playlists in the db static int spotify_base_plid; // The base playlist id for Spotify saved tracks in the db static int spotify_saved_plid; // Linked list of tracks where we are waiting for metadata static struct pending_metadata *spotify_pending_metadata; // Linked list of saved tracks which we want to reload at startup static struct reload_list *spotify_reload_list; // Audio fifo static audio_fifo_t *g_audio_fifo; /** * The application key is specific to forked-daapd, and allows Spotify * to produce statistics on how their service is used. */ const uint8_t g_appkey[] = { 0x01, 0xC6, 0x9D, 0x18, 0xA4, 0xF7, 0x79, 0x12, 0x43, 0x55, 0x0F, 0xAD, 0xBF, 0x23, 0x23, 0x10, 0x2E, 0x51, 0x46, 0x8F, 0x06, 0x3D, 0xEE, 0xC3, 0xF0, 0x2A, 0x5D, 0x8E, 0x72, 0x35, 0xD1, 0x21, 0x44, 0xE3, 0x19, 0x80, 0xED, 0xD5, 0xAD, 0xE6, 0xE1, 0xDD, 0xBE, 0xCB, 0xA9, 0x84, 0xBD, 0xC2, 0xAF, 0xB1, 0xF2, 0xD5, 0x87, 0xFC, 0x35, 0xD6, 0x1C, 0x5F, 0x5B, 0x76, 0x38, 0x1D, 0x6E, 0x49, 0x6D, 0x85, 0x15, 0xCD, 0x38, 0x14, 0xD6, 0xB8, 0xFE, 0x05, 0x0A, 0xAC, 0x9B, 0x31, 0xD1, 0xC0, 0xAF, 0x16, 0x78, 0x48, 0x49, 0x27, 0x41, 0xCA, 0xAF, 0x07, 0xEC, 0x10, 0x5D, 0x19, 0x43, 0x2E, 0x84, 0xEB, 0x43, 0x5D, 0x4B, 0xBF, 0xD0, 0x5C, 0xDF, 0x3D, 0x12, 0x6D, 0x1C, 0x76, 0x4E, 0x9F, 0xBF, 0x14, 0xC9, 0x46, 0x95, 0x99, 0x32, 0x6A, 0xC2, 0xF1, 0x89, 0xA4, 0xB3, 0xF3, 0xA0, 0xEB, 0xDA, 0x84, 0x67, 0x27, 0x07, 0x1F, 0xF6, 0x19, 0xAC, 0xF1, 0xB8, 0xB6, 0xCF, 0xAB, 0xF8, 0x0A, 0xEE, 0x4D, 0xAC, 0xC2, 0x39, 0x63, 0x50, 0x13, 0x7B, 0x51, 0x3A, 0x50, 0xE0, 0x03, 0x6E, 0xB7, 0x17, 0xEE, 0x58, 0xCE, 0xF8, 0x15, 0x3C, 0x70, 0xDE, 0xE6, 0xEB, 0xE6, 0xD4, 0x2C, 0x27, 0xB9, 0xCA, 0x15, 0xCE, 0x2E, 0x31, 0x54, 0xF5, 0x0A, 0x98, 0x8D, 0x78, 0xE5, 0xB6, 0xF8, 0xE4, 0x62, 0x43, 0xAA, 0x37, 0x93, 0xFF, 0xE3, 0xAB, 0x17, 0xC5, 0x81, 0x4F, 0xFD, 0xF1, 0x84, 0xE1, 0x8A, 0x99, 0xB0, 0x1D, 0x85, 0x80, 0xA2, 0x49, 0x35, 0x8D, 0xDD, 0xBC, 0x74, 0x0B, 0xBA, 0x33, 0x5B, 0xD5, 0x7A, 0xB9, 0x2F, 0x9B, 0x24, 0xA5, 0xAB, 0xF6, 0x1E, 0xE3, 0xA3, 0xA8, 0x0D, 0x1E, 0x48, 0xF7, 0xDB, 0xE2, 0x54, 0x65, 0x43, 0xA6, 0xD3, 0x3F, 0x2C, 0x9B, 0x13, 0x9A, 0xBE, 0x0F, 0x4D, 0x51, 0xC3, 0x73, 0xA5, 0xFE, 0xFC, 0x93, 0x12, 0xEF, 0x9C, 0x4D, 0x68, 0xE3, 0xDA, 0x52, 0x67, 0x28, 0x41, 0x17, 0x22, 0x3E, 0x33, 0xB0, 0x3A, 0xFB, 0x44, 0xB0, 0x2E, 0xA6, 0xD2, 0x95, 0xC0, 0x9A, 0xBA, 0x32, 0xA3, 0xC5, 0xFE, 0x86, 0x5D, 0xC8, 0xBB, 0xB5, 0xDE, 0x92, 0x8C, 0x7D, 0xE4, 0x03, 0xD4, 0xF9, 0xAE, 0x41, 0xE3, 0xBD, 0x35, 0x4B, 0x94, 0x27, 0xE0, 0x12, 0x21, 0x46, 0xE9, 0x09, }; // Endpoints and credentials for the web api static char *spotify_access_token; static char *spotify_refresh_token; static const char *spotify_client_id = "0e684a5422384114a8ae7ac020f01789"; static const char *spotify_client_secret = "232af95f39014c9ba218285a5c11a239"; static const char *spotify_auth_uri = "https://accounts.spotify.com/authorize"; static const char *spotify_token_uri = "https://accounts.spotify.com/api/token"; static const char *spotify_tracks_uri = "https://api.spotify.com/v1/me/tracks?limit=50"; // This section defines and assigns function pointers to the libspotify functions // The arguments and return values must be in sync with the spotify api // Please scroll through the ugliness which follows typedef const char* (*fptr_sp_error_message_t)(sp_error error); typedef sp_error (*fptr_sp_session_create_t)(const sp_session_config *config, sp_session **sess); typedef sp_error (*fptr_sp_session_release_t)(sp_session *sess); typedef sp_error (*fptr_sp_session_login_t)(sp_session *session, const char *username, const char *password, bool remember_me, const char *blob); typedef sp_error (*fptr_sp_session_relogin_t)(sp_session *session); typedef sp_error (*fptr_sp_session_logout_t)(sp_session *session); typedef sp_error (*fptr_sp_session_process_events_t)(sp_session *session, int *next_timeout); typedef sp_playlist* (*fptr_sp_session_starred_create_t)(sp_session *session); typedef sp_playlistcontainer* (*fptr_sp_session_playlistcontainer_t)(sp_session *session); typedef sp_error (*fptr_sp_session_player_load_t)(sp_session *session, sp_track *track); typedef sp_error (*fptr_sp_session_player_unload_t)(sp_session *session); typedef sp_error (*fptr_sp_session_player_play_t)(sp_session *session, bool play); typedef sp_error (*fptr_sp_session_player_seek_t)(sp_session *session, int offset); typedef sp_connectionstate (*fptr_sp_session_connectionstate_t)(sp_session *session); typedef sp_error (*fptr_sp_session_preferred_bitrate_t)(sp_session *session, sp_bitrate bitrate); typedef sp_error (*fptr_sp_playlistcontainer_add_callbacks_t)(sp_playlistcontainer *pc, sp_playlistcontainer_callbacks *callbacks, void *userdata); typedef int (*fptr_sp_playlistcontainer_num_playlists_t)(sp_playlistcontainer *pc); typedef sp_playlist* (*fptr_sp_playlistcontainer_playlist_t)(sp_playlistcontainer *pc, int index); typedef sp_error (*fptr_sp_playlist_add_callbacks_t)(sp_playlist *playlist, sp_playlist_callbacks *callbacks, void *userdata); typedef const char* (*fptr_sp_playlist_name_t)(sp_playlist *playlist); typedef sp_error (*fptr_sp_playlist_remove_callbacks_t)(sp_playlist *playlist, sp_playlist_callbacks *callbacks, void *userdata); typedef int (*fptr_sp_playlist_num_tracks_t)(sp_playlist *playlist); typedef sp_track* (*fptr_sp_playlist_track_t)(sp_playlist *playlist, int index); typedef bool (*fptr_sp_playlist_is_loaded_t)(sp_playlist *playlist); typedef int (*fptr_sp_playlist_track_create_time_t)(sp_playlist *playlist, int index); typedef sp_user* (*fptr_sp_playlist_owner_t)(sp_playlist *playlist); typedef sp_error (*fptr_sp_track_error_t)(sp_track *track); typedef bool (*fptr_sp_track_is_loaded_t)(sp_track *track); typedef const char* (*fptr_sp_track_name_t)(sp_track *track); typedef int (*fptr_sp_track_duration_t)(sp_track *track); typedef int (*fptr_sp_track_index_t)(sp_track *track); typedef int (*fptr_sp_track_disc_t)(sp_track *track); typedef sp_album* (*fptr_sp_track_album_t)(sp_track *track); typedef sp_track_availability (*fptr_sp_track_get_availability_t)(sp_session *session, sp_track *track); typedef bool (*fptr_sp_track_is_starred_t)(sp_session *session, sp_track *track); typedef sp_link* (*fptr_sp_link_create_from_playlist_t)(sp_playlist *playlist); typedef sp_link* (*fptr_sp_link_create_from_track_t)(sp_track *track, int offset); typedef sp_link* (*fptr_sp_link_create_from_string_t)(const char *link); typedef int (*fptr_sp_link_as_string_t)(sp_link *link, char *buffer, int buffer_size); typedef sp_track* (*fptr_sp_link_as_track_t)(sp_link *link); typedef sp_error (*fptr_sp_link_release_t)(sp_link *link); typedef const char* (*fptr_sp_album_name_t)(sp_album *album); typedef sp_artist* (*fptr_sp_album_artist_t)(sp_album *album); typedef int (*fptr_sp_album_year_t)(sp_album *album); typedef sp_albumtype (*fptr_sp_album_type_t)(sp_album *album); typedef const byte* (*fptr_sp_album_cover_t)(sp_album *album, sp_image_size size); typedef const char* (*fptr_sp_artist_name_t)(sp_artist *artist); typedef sp_image* (*fptr_sp_image_create_t)(sp_session *session, const byte image_id[20]); typedef bool (*fptr_sp_image_is_loaded_t)(sp_image *image); typedef sp_error (*fptr_sp_image_error_t)(sp_image *image); typedef sp_imageformat (*fptr_sp_image_format_t)(sp_image *image); typedef const void* (*fptr_sp_image_data_t)(sp_image *image, size_t *data_size); typedef sp_error (*fptr_sp_image_release_t)(sp_image *image); typedef sp_error (*fptr_sp_image_add_load_callback_t)(sp_image *image, image_loaded_cb *callback, void *userdata); typedef sp_error (*fptr_sp_image_remove_load_callback_t)(sp_image *image, image_loaded_cb *callback, void *userdata); typedef const char* (*fptr_sp_user_display_name_t)(sp_user *user); typedef const char* (*fptr_sp_user_canonical_name_t)(sp_user *user); /* Define actual function pointers */ fptr_sp_error_message_t fptr_sp_error_message; fptr_sp_session_create_t fptr_sp_session_create; fptr_sp_session_release_t fptr_sp_session_release; fptr_sp_session_login_t fptr_sp_session_login; fptr_sp_session_relogin_t fptr_sp_session_relogin; fptr_sp_session_logout_t fptr_sp_session_logout; fptr_sp_session_starred_create_t fptr_sp_session_starred_create; fptr_sp_session_playlistcontainer_t fptr_sp_session_playlistcontainer; fptr_sp_session_process_events_t fptr_sp_session_process_events; fptr_sp_session_player_load_t fptr_sp_session_player_load; fptr_sp_session_player_unload_t fptr_sp_session_player_unload; fptr_sp_session_player_play_t fptr_sp_session_player_play; fptr_sp_session_player_seek_t fptr_sp_session_player_seek; fptr_sp_session_connectionstate_t fptr_sp_session_connectionstate; fptr_sp_session_preferred_bitrate_t fptr_sp_session_preferred_bitrate; fptr_sp_playlistcontainer_add_callbacks_t fptr_sp_playlistcontainer_add_callbacks; fptr_sp_playlistcontainer_num_playlists_t fptr_sp_playlistcontainer_num_playlists; fptr_sp_playlistcontainer_playlist_t fptr_sp_playlistcontainer_playlist; fptr_sp_playlist_add_callbacks_t fptr_sp_playlist_add_callbacks; fptr_sp_playlist_name_t fptr_sp_playlist_name; fptr_sp_playlist_remove_callbacks_t fptr_sp_playlist_remove_callbacks; fptr_sp_playlist_num_tracks_t fptr_sp_playlist_num_tracks; fptr_sp_playlist_track_t fptr_sp_playlist_track; fptr_sp_playlist_is_loaded_t fptr_sp_playlist_is_loaded; fptr_sp_playlist_track_create_time_t fptr_sp_playlist_track_create_time; fptr_sp_playlist_owner_t fptr_sp_playlist_owner; fptr_sp_track_error_t fptr_sp_track_error; fptr_sp_track_is_loaded_t fptr_sp_track_is_loaded; fptr_sp_track_name_t fptr_sp_track_name; fptr_sp_track_duration_t fptr_sp_track_duration; fptr_sp_track_index_t fptr_sp_track_index; fptr_sp_track_disc_t fptr_sp_track_disc; fptr_sp_track_album_t fptr_sp_track_album; fptr_sp_track_get_availability_t fptr_sp_track_get_availability; fptr_sp_track_is_starred_t fptr_sp_track_is_starred; fptr_sp_link_create_from_playlist_t fptr_sp_link_create_from_playlist; fptr_sp_link_create_from_track_t fptr_sp_link_create_from_track; fptr_sp_link_create_from_string_t fptr_sp_link_create_from_string; fptr_sp_link_as_string_t fptr_sp_link_as_string; fptr_sp_link_as_track_t fptr_sp_link_as_track; fptr_sp_link_release_t fptr_sp_link_release; fptr_sp_album_name_t fptr_sp_album_name; fptr_sp_album_artist_t fptr_sp_album_artist; fptr_sp_album_year_t fptr_sp_album_year; fptr_sp_album_type_t fptr_sp_album_type; fptr_sp_album_cover_t fptr_sp_album_cover; fptr_sp_artist_name_t fptr_sp_artist_name; fptr_sp_image_create_t fptr_sp_image_create; fptr_sp_image_is_loaded_t fptr_sp_image_is_loaded; fptr_sp_image_error_t fptr_sp_image_error; fptr_sp_image_format_t fptr_sp_image_format; fptr_sp_image_data_t fptr_sp_image_data; fptr_sp_image_release_t fptr_sp_image_release; fptr_sp_image_add_load_callback_t fptr_sp_image_add_load_callback; fptr_sp_image_remove_load_callback_t fptr_sp_image_remove_load_callback; fptr_sp_user_display_name_t fptr_sp_user_display_name; fptr_sp_user_canonical_name_t fptr_sp_user_canonical_name; /* Assign function pointers to libspotify symbol */ static int fptr_assign_all() { void *h; char *err; int ret; h = g_libhandle; // The following is non-ISO compliant ret = (fptr_sp_error_message = dlsym(h, "sp_error_message")) && (fptr_sp_session_create = dlsym(h, "sp_session_create")) && (fptr_sp_session_release = dlsym(h, "sp_session_release")) && (fptr_sp_session_login = dlsym(h, "sp_session_login")) && (fptr_sp_session_relogin = dlsym(h, "sp_session_relogin")) && (fptr_sp_session_logout = dlsym(h, "sp_session_logout")) && (fptr_sp_session_playlistcontainer = dlsym(h, "sp_session_playlistcontainer")) && (fptr_sp_session_process_events = dlsym(h, "sp_session_process_events")) && (fptr_sp_session_player_load = dlsym(h, "sp_session_player_load")) && (fptr_sp_session_player_unload = dlsym(h, "sp_session_player_unload")) && (fptr_sp_session_player_play = dlsym(h, "sp_session_player_play")) && (fptr_sp_session_player_seek = dlsym(h, "sp_session_player_seek")) && (fptr_sp_session_connectionstate = dlsym(h, "sp_session_connectionstate")) && (fptr_sp_session_preferred_bitrate = dlsym(h, "sp_session_preferred_bitrate")) && (fptr_sp_playlistcontainer_add_callbacks = dlsym(h, "sp_playlistcontainer_add_callbacks")) && (fptr_sp_playlistcontainer_num_playlists = dlsym(h, "sp_playlistcontainer_num_playlists")) && (fptr_sp_session_starred_create = dlsym(h, "sp_session_starred_create")) && (fptr_sp_playlistcontainer_playlist = dlsym(h, "sp_playlistcontainer_playlist")) && (fptr_sp_playlist_add_callbacks = dlsym(h, "sp_playlist_add_callbacks")) && (fptr_sp_playlist_name = dlsym(h, "sp_playlist_name")) && (fptr_sp_playlist_remove_callbacks = dlsym(h, "sp_playlist_remove_callbacks")) && (fptr_sp_playlist_num_tracks = dlsym(h, "sp_playlist_num_tracks")) && (fptr_sp_playlist_track = dlsym(h, "sp_playlist_track")) && (fptr_sp_playlist_is_loaded = dlsym(h, "sp_playlist_is_loaded")) && (fptr_sp_playlist_track_create_time = dlsym(h, "sp_playlist_track_create_time")) && (fptr_sp_playlist_owner = dlsym(h, "sp_playlist_owner")) && (fptr_sp_track_error = dlsym(h, "sp_track_error")) && (fptr_sp_track_is_loaded = dlsym(h, "sp_track_is_loaded")) && (fptr_sp_track_name = dlsym(h, "sp_track_name")) && (fptr_sp_track_duration = dlsym(h, "sp_track_duration")) && (fptr_sp_track_index = dlsym(h, "sp_track_index")) && (fptr_sp_track_disc = dlsym(h, "sp_track_disc")) && (fptr_sp_track_album = dlsym(h, "sp_track_album")) && (fptr_sp_track_get_availability = dlsym(h, "sp_track_get_availability")) && (fptr_sp_track_is_starred = dlsym(h, "sp_track_is_starred")) && (fptr_sp_link_create_from_playlist = dlsym(h, "sp_link_create_from_playlist")) && (fptr_sp_link_create_from_track = dlsym(h, "sp_link_create_from_track")) && (fptr_sp_link_create_from_string = dlsym(h, "sp_link_create_from_string")) && (fptr_sp_link_as_string = dlsym(h, "sp_link_as_string")) && (fptr_sp_link_as_track = dlsym(h, "sp_link_as_track")) && (fptr_sp_link_release = dlsym(h, "sp_link_release")) && (fptr_sp_album_name = dlsym(h, "sp_album_name")) && (fptr_sp_album_artist = dlsym(h, "sp_album_artist")) && (fptr_sp_album_year = dlsym(h, "sp_album_year")) && (fptr_sp_album_type = dlsym(h, "sp_album_type")) && (fptr_sp_album_cover = dlsym(h, "sp_album_cover")) && (fptr_sp_artist_name = dlsym(h, "sp_artist_name")) && (fptr_sp_image_create = dlsym(h, "sp_image_create")) && (fptr_sp_image_is_loaded = dlsym(h, "sp_image_is_loaded")) && (fptr_sp_image_error = dlsym(h, "sp_image_error")) && (fptr_sp_image_format = dlsym(h, "sp_image_format")) && (fptr_sp_image_data = dlsym(h, "sp_image_data")) && (fptr_sp_image_release = dlsym(h, "sp_image_release")) && (fptr_sp_image_add_load_callback = dlsym(h, "sp_image_add_load_callback")) && (fptr_sp_image_remove_load_callback = dlsym(h, "sp_image_remove_load_callback")) && (fptr_sp_user_display_name = dlsym(h, "sp_user_display_name")) && (fptr_sp_user_canonical_name = dlsym(h, "sp_user_canonical_name")) ; err = dlerror(); if (ret && !err) return ret; else if (err) DPRINTF(E_LOG, L_SPOTIFY, "Assignment error (%d): %s\n", ret, err); else DPRINTF(E_LOG, L_SPOTIFY, "Unknown assignment error (%d)\n", ret); return -1; } // End of ugly part /* ------------------------------- MISC HELPERS ---------------------------- */ static int spotify_file_read(char *path, char **username, char **password) { FILE *fp; char *u; char *p; char buf[256]; int len; fp = fopen(path, "rb"); if (!fp) { DPRINTF(E_LOG, L_SPOTIFY, "Could not open Spotify credentials file %s: %s\n", path, strerror(errno)); return -1; } u = fgets(buf, sizeof(buf), fp); if (!u) { DPRINTF(E_LOG, L_SPOTIFY, "Empty Spotify credentials file %s\n", path); fclose(fp); return -1; } len = strlen(u); if (buf[len - 1] != '\n') { DPRINTF(E_LOG, L_SPOTIFY, "Invalid Spotify credentials file %s: username name too long or missing password\n", path); fclose(fp); return -1; } while (len) { if ((buf[len - 1] == '\r') || (buf[len - 1] == '\n')) { buf[len - 1] = '\0'; len--; } else break; } if (!len) { DPRINTF(E_LOG, L_SPOTIFY, "Invalid Spotify credentials file %s: empty line where username expected\n", path); fclose(fp); return -1; } u = strdup(buf); if (!u) { DPRINTF(E_LOG, L_SPOTIFY, "Out of memory for username while reading %s\n", path); fclose(fp); return -1; } p = fgets(buf, sizeof(buf), fp); fclose(fp); if (!p) { DPRINTF(E_LOG, L_SPOTIFY, "Invalid Spotify credentials file %s: no password\n", path); free(u); return -1; } len = strlen(p); while (len) { if ((buf[len - 1] == '\r') || (buf[len - 1] == '\n')) { buf[len - 1] = '\0'; len--; } else break; } p = strdup(buf); if (!p) { DPRINTF(E_LOG, L_SPOTIFY, "Out of memory for password while reading %s\n", path); free(u); return -1; } DPRINTF(E_LOG, L_SPOTIFY, "Spotify credentials file OK, logging in with username %s\n", u); *username = u; *password = p; return 0; } /* -------------------------- PLAYLIST HELPERS ------------------------- */ /* Should only be called from within the spotify thread */ static int spotify_metadata_get(sp_track *track, struct media_file_info *mfi, const char *pltitle, int time_added) { cfg_t *spotify_cfg; bool artist_override; bool album_override; sp_album *album; sp_artist *artist; sp_albumtype albumtype; bool starred; int compilation; char *albumname; spotify_cfg = cfg_getsec(cfg, "spotify"); artist_override = cfg_getbool(spotify_cfg, "artist_override"); album_override = cfg_getbool(spotify_cfg, "album_override"); album = fptr_sp_track_album(track); if (!album) return -1; artist = fptr_sp_album_artist(album); if (!artist) return -1; albumtype = fptr_sp_album_type(album); starred = fptr_sp_track_is_starred(g_sess, track); /* * Treat album as compilation if one of the following conditions is true: * - spotfy album type is compilation * - artist_override in config is set to true and track is not part of the starred playlist * - starred_artist_override in config is set to true and track is part of the starred playlist */ compilation = ((albumtype == SP_ALBUMTYPE_COMPILATION) || artist_override); if (album_override && pltitle) albumname = strdup(pltitle); else albumname = strdup(fptr_sp_album_name(album)); mfi->title = strdup(fptr_sp_track_name(track)); mfi->album = albumname; mfi->artist = strdup(fptr_sp_artist_name(artist)); mfi->year = fptr_sp_album_year(album); mfi->song_length = fptr_sp_track_duration(track); mfi->track = fptr_sp_track_index(track); mfi->disc = fptr_sp_track_disc(track); mfi->compilation = compilation; mfi->artwork = ARTWORK_SPOTIFY; mfi->type = strdup("spotify"); mfi->codectype = strdup("wav"); mfi->description = strdup("Spotify audio"); mfi->time_added = time_added; DPRINTF(E_SPAM, L_SPOTIFY, "Metadata for track:\n" "Title: %s\n" "Album: %s\n" "Artist: %s\n" "Year: %u\n" "Track: %u\n" "Disc: %u\n" "Compilation: %d\n" "Starred: %d\n", mfi->title, mfi->album, mfi->artist, mfi->year, mfi->track, mfi->disc, mfi->compilation, starred); return 0; } static int spotify_track_save(int plid, sp_track *track, const char *pltitle, int time_added) { struct media_file_info mfi; sp_link *link; char url[1024]; int ret; int dir_id; char virtual_path[PATH_MAX]; if (!fptr_sp_track_is_loaded(track)) { DPRINTF(E_LOG, L_SPOTIFY, "Track appears to no longer have the proper status\n"); return -1; } if (fptr_sp_track_get_availability(g_sess, track) != SP_TRACK_AVAILABILITY_AVAILABLE) { DPRINTF(E_LOG, L_SPOTIFY, "Track not available for playback: '%s'\n", fptr_sp_track_name(track)); return 0; } link = fptr_sp_link_create_from_track(track, 0); if (!link) { DPRINTF(E_LOG, L_SPOTIFY, "Could not create link for track: '%s'\n", fptr_sp_track_name(track)); return -1; } ret = fptr_sp_link_as_string(link, url, sizeof(url)); if (ret == sizeof(url)) { DPRINTF(E_DBG, L_SPOTIFY, "Spotify link truncated: '%s'\n", url); } fptr_sp_link_release(link); /* Add to playlistitems table */ if (plid) { ret = db_pl_add_item_bypath(plid, url); if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Could not save playlist item: '%s'\n", url); return -1; } } memset(&mfi, 0, sizeof(struct media_file_info)); ret = spotify_metadata_get(track, &mfi, pltitle, time_added); if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Metadata missing (but track should be loaded?): '%s'\n", fptr_sp_track_name(track)); free_mfi(&mfi, 1); return -1; } ret = snprintf(virtual_path, sizeof(virtual_path), "/spotify:/%s", mfi.artist); if ((ret < 0) || (ret >= sizeof(virtual_path))) { DPRINTF(E_LOG, L_SPOTIFY, "Virtual path exceeds PATH_MAX (/spotify:/%s)\n", mfi.artist); free_mfi(&mfi, 1); return -1; } dir_id = db_directory_addorupdate(virtual_path, 0, DIR_SPOTIFY); if (dir_id <= 0) { DPRINTF(E_LOG, L_SPOTIFY, "Could not add or update directory '%s'\n", virtual_path); free_mfi(&mfi, 1); return -1; } ret = snprintf(virtual_path, sizeof(virtual_path), "/spotify:/%s/%s", mfi.artist, mfi.album); if ((ret < 0) || (ret >= sizeof(virtual_path))) { DPRINTF(E_LOG, L_SPOTIFY, "Virtual path exceeds PATH_MAX (/spotify:/%s/%s)\n", mfi.artist, mfi.album); free_mfi(&mfi, 1); return -1; } dir_id = db_directory_addorupdate(virtual_path, 0, dir_id); if (dir_id <= 0) { DPRINTF(E_LOG, L_SPOTIFY, "Could not add or update directory '%s'\n", virtual_path); free_mfi(&mfi, 1); return -1; } // DPRINTF(E_DBG, L_SPOTIFY, "Saving track '%s': '%s' by %s (%s)\n", url, mfi.title, mfi.artist, mfi.album); filescanner_process_media(url, time(NULL), 0, F_SCAN_TYPE_SPOTIFY, &mfi, dir_id); free_mfi(&mfi, 1); return 0; } static int spotify_cleanup_files(void) { struct query_params qp; char *path; int ret; memset(&qp, 0, sizeof(struct query_params)); qp.type = Q_BROWSE_PATH; qp.sort = S_NONE; qp.filter = "f.path LIKE 'spotify:%%' AND NOT f.path IN (SELECT filepath FROM playlistitems)"; ret = db_query_start(&qp); if (ret < 0) { db_query_end(&qp); return -1; } while (((ret = db_query_fetch_string(&qp, &path)) == 0) && (path)) { cache_artwork_delete_by_path(path); } db_query_end(&qp); db_spotify_files_delete(); return 0; } static int spotify_playlist_save(sp_playlist *pl) { struct playlist_info *pli; sp_track *track; sp_link *link; sp_user *owner; char url[1024]; const char *name; const char *ownername; int plid; int num_tracks; char virtual_path[PATH_MAX]; int created; int ret; int i; if (!fptr_sp_playlist_is_loaded(pl)) { DPRINTF(E_DBG, L_SPOTIFY, "Playlist still not loaded - will wait for next callback\n"); return 0; } name = fptr_sp_playlist_name(pl); num_tracks = fptr_sp_playlist_num_tracks(pl); // The starred playlist has an empty name, set it manually to "Starred" if (*name == '\0') name = "Starred"; for (i = 0; i < num_tracks; i++) { track = fptr_sp_playlist_track(pl, i); if (track && !fptr_sp_track_is_loaded(track)) { DPRINTF(E_DBG, L_SPOTIFY, "All playlist tracks not loaded (will wait for next callback): %s\n", name); return 0; } } DPRINTF(E_LOG, L_SPOTIFY, "Saving playlist (%d tracks): '%s'\n", num_tracks, name); // Save playlist (playlists table) link = fptr_sp_link_create_from_playlist(pl); if (!link) { DPRINTF(E_LOG, L_SPOTIFY, "Could not create link for playlist (wait): '%s'\n", name); return -1; } ret = fptr_sp_link_as_string(link, url, sizeof(url)); if (ret == sizeof(url)) { DPRINTF(E_DBG, L_SPOTIFY, "Spotify link truncated: %s\n", url); } fptr_sp_link_release(link); owner = fptr_sp_playlist_owner(pl); if (owner) { DPRINTF(E_DBG, L_SPOTIFY, "Playlist '%s' owner: '%s' (canonical) / '%s' (display)\n", name, fptr_sp_user_canonical_name(owner), fptr_sp_user_display_name(owner)); ownername = fptr_sp_user_canonical_name(owner); snprintf(virtual_path, PATH_MAX, "/spotify:/%s (%s)", name, ownername); } else { snprintf(virtual_path, PATH_MAX, "/spotify:/%s", name); } pli = db_pl_fetch_bypath(url); if (pli) { DPRINTF(E_DBG, L_SPOTIFY, "Playlist found ('%s', link %s), updating\n", name, url); plid = pli->id; free(pli->title); pli->title = strdup(name); free(pli->virtual_path); pli->virtual_path = strdup(virtual_path); pli->directory_id = DIR_SPOTIFY; ret = db_pl_update(pli); if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Error updating playlist ('%s', link %s)\n", name, url); free_pli(pli, 0); return -1; } db_pl_clear_items(plid); } else { DPRINTF(E_DBG, L_SPOTIFY, "Adding playlist ('%s', link %s)\n", name, url); pli = (struct playlist_info *)malloc(sizeof(struct playlist_info)); if (!pli) { DPRINTF(E_LOG, L_SCAN, "Out of memory\n"); return -1; } memset(pli, 0, sizeof(struct playlist_info)); pli->type = PL_PLAIN; pli->title = strdup(name); pli->path = strdup(url); pli->virtual_path = strdup(virtual_path); pli->parent_id = spotify_base_plid; pli->directory_id = DIR_SPOTIFY; ret = db_pl_add(pli, &plid); if ((ret < 0) || (plid < 1)) { DPRINTF(E_LOG, L_SPOTIFY, "Error adding playlist ('%s', link %s, ret %d, plid %d)\n", name, url, ret, plid); free_pli(pli, 0); return -1; } } free_pli(pli, 0); // Save tracks and playlistitems (files and playlistitems table) db_transaction_begin(); for (i = 0; i < num_tracks; i++) { track = fptr_sp_playlist_track(pl, i); if (!track) { DPRINTF(E_LOG, L_SPOTIFY, "Track %d in playlist '%s' (id %d) is invalid\n", i, name, plid); continue; } created = fptr_sp_playlist_track_create_time(pl, i); ret = spotify_track_save(plid, track, name, created); if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Error saving track %d to playlist '%s' (id %d)\n", i, name, plid); continue; } } spotify_cleanup_files(); db_transaction_end(); return plid; } // Registers a track with libspotify, which will make it start loading the track // metadata. When that is done metadata_updated() is called (but we won't be // told which track it was...). Note that this function will result in a ref // count on the sp_link, which the caller must decrease with sp_link_release. static enum command_state spotify_uri_register(void *arg, int *retval) { struct playlist_info pli; struct pending_metadata *pm; sp_link *link; sp_track *track; int ret; char *uri = arg; if (SP_CONNECTION_STATE_LOGGED_IN != fptr_sp_session_connectionstate(g_sess)) { DPRINTF(E_LOG, L_SPOTIFY, "Can't register music, not connected and logged in to Spotify\n"); *retval = -1; return COMMAND_END; } // Must have playlist for these items, otherwise spotify_cleanup_files will delete them again if (!spotify_saved_plid) { memset(&pli, 0, sizeof(struct playlist_info)); pli.title = "Spotify Saved"; pli.type = PL_PLAIN; pli.path = "spotify:savedtracks"; pli.virtual_path = "/spotify:/Spotify Saved"; ret = db_pl_add(&pli, &spotify_saved_plid); if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Error adding playlist for saved tracks\n"); *retval = -1; return COMMAND_END; } } ret = db_pl_add_item_bypath(spotify_saved_plid, uri); if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Could not add '%s' to spotify:savedtracks\n", uri); *retval = -1; return COMMAND_END; } link = fptr_sp_link_create_from_string(uri); if (!link) { DPRINTF(E_LOG, L_SPOTIFY, "Invalid Spotify link: '%s'\n", uri); *retval = -1; return COMMAND_END; } track = fptr_sp_link_as_track(link); if (!track) { DPRINTF(E_LOG, L_SPOTIFY, "Invalid Spotify track: '%s'\n", uri); *retval = -1; return COMMAND_END; } // Maybe we already had the track if (fptr_sp_track_is_loaded(track)) { db_file_ping_bymatch(uri, 0); fptr_sp_link_release(link); *retval = 0; return COMMAND_END; } pm = malloc(sizeof(struct pending_metadata)); if (!pm) { DPRINTF(E_LOG, L_SPOTIFY, "Out of memory\n"); *retval = -1; return COMMAND_END; } pm->link = link; pm->track = track; pm->next = spotify_pending_metadata; spotify_pending_metadata = pm; *retval = 0; return COMMAND_END; } // TODO Maybe use the commands bh instead? static enum command_state spotify_pending_process(void *arg, int *retval) { struct pending_metadata *pm; int i; *retval = 0; if (!spotify_pending_metadata) return COMMAND_END; // Too early i = 0; for (pm = spotify_pending_metadata; pm; pm = pm->next) { i++; if (!fptr_sp_track_is_loaded(pm->track)) return COMMAND_END; } DPRINTF(E_DBG, L_SPOTIFY, "All %d tracks loaded, now saving\n", i); while ((pm = spotify_pending_metadata)) { spotify_track_save(0, pm->track, NULL, time(NULL)); // Not sure if we should release link here? We are done with it, but maybe // libspotify will unload the track if we release, and we don't want that //fptr_sp_link_release(pm->link); spotify_pending_metadata = pm->next; free(pm); } return COMMAND_END; } static enum command_state spotify_saved_pl_clear_items(void *arg, int *retval) { if (spotify_saved_plid) db_pl_clear_items(spotify_saved_plid); *retval = 0; return COMMAND_END; } static enum command_state spotify_cleanup_wrapper(void *arg, int *retval) { *retval = spotify_cleanup_files(); return COMMAND_END; } /*--------------------- HELPERS FOR SPOTIFY WEB API -------------------------*/ /* All the below is in the httpd thread */ static char * jparse_str_from_obj(json_object *haystack, const char *key) { json_object *needle; if (json_object_object_get_ex(haystack, key, &needle) && json_object_get_type(needle) == json_type_string) return strdup(json_object_get_string(needle)); else return NULL; } static char * jparse_str_from_str(const char *s, const char *key) { json_object *haystack; char *val; haystack = json_tokener_parse(s); if (!haystack) { DPRINTF(E_LOG, L_SPOTIFY, "JSON parser returned an error\n"); return NULL; } val = jparse_str_from_obj(haystack, key); #ifdef HAVE_JSON_C_OLD json_object_put(haystack); #else if (json_object_put(haystack) != 1) DPRINTF(E_LOG, L_SPOTIFY, "Memleak: JSON parser did not free object\n"); #endif return val; } // Will find all track Spotify uri's and register them with libspotify. // Returns the number of tracks found in the json input. "total" will be the // total reported by Spotify in the response, and "next" will be an allocated // string with the url of the next page, as reported by Spotify static int jparse_and_register_tracks(int *total, char **next, const char *s) { json_object *haystack; json_object *needle; json_object *items; json_object *item; json_object *track; char *uri; int ret; int len; int i; haystack = json_tokener_parse(s); if (!haystack) { DPRINTF(E_LOG, L_SPOTIFY, "JSON parser returned an error\n"); return -1; } if (json_object_object_get_ex(haystack, "total", &needle) && json_object_get_type(needle) == json_type_int) *total = json_object_get_int(needle); else *total = -1; *next = jparse_str_from_obj(haystack, "next"); if (! (json_object_object_get_ex(haystack, "items", &items) && json_object_get_type(items) == json_type_array) ) { DPRINTF(E_LOG, L_SPOTIFY, "No items in reply from Spotify. See:\n%s\n", s); ret = -1; goto out_free_json; } len = json_object_array_length(items); DPRINTF(E_DBG, L_SPOTIFY, "Got %d saved tracks\n", len); for (i = 0; i < len; i++) { item = json_object_array_get_idx(items, i); if (! (item && json_object_object_get_ex(item, "track", &track) && (uri = jparse_str_from_obj(track, "uri")) )) { DPRINTF(E_LOG, L_SPOTIFY, "Unexpected JSON: Item %d did not have 'track'->'uri'\n", i); len--; continue; } commands_exec_sync(cmdbase, spotify_uri_register, NULL, uri); free(uri); } ret = len; out_free_json: #ifdef HAVE_JSON_C_OLD json_object_put(haystack); #else if (json_object_put(haystack) != 1) DPRINTF(E_LOG, L_SPOTIFY, "Memleak: JSON parser did not free object\n"); #endif return ret; } static int tokens_get(const char *code, const char *redirect_uri, const char **err) { struct http_client_ctx ctx; struct keyval kv; char *param; char *body; int ret; memset(&kv, 0, sizeof(struct keyval)); ret = ( (keyval_add(&kv, "grant_type", "authorization_code") == 0) && (keyval_add(&kv, "code", code) == 0) && (keyval_add(&kv, "client_id", spotify_client_id) == 0) && (keyval_add(&kv, "client_secret", spotify_client_secret) == 0) && (keyval_add(&kv, "redirect_uri", redirect_uri) == 0) ); if (!ret) { *err = "Add parameters to keyval failed"; ret = -1; goto out_clear_kv; } param = http_form_urlencode(&kv); if (!param) { *err = "http_form_uriencode() failed"; ret = -1; goto out_clear_kv; } memset(&ctx, 0, sizeof(struct http_client_ctx)); ctx.url = (char *)spotify_token_uri; ctx.output_body = param; ctx.input_body = evbuffer_new(); ret = http_client_request(&ctx); if (ret < 0) { *err = "Did not get a reply from Spotify"; goto out_free_input_body; } // 0-terminate for safety evbuffer_add(ctx.input_body, "", 1); body = (char *)evbuffer_pullup(ctx.input_body, -1); if (!body || (strlen(body) == 0)) { *err = "The reply from Spotify is empty or invalid"; ret = -1; goto out_free_input_body; } spotify_access_token = jparse_str_from_str(body, "access_token"); spotify_refresh_token = jparse_str_from_str(body, "refresh_token"); if (!spotify_access_token || !spotify_refresh_token) { DPRINTF(E_LOG, L_SPOTIFY, "Could not find token in reply: %s\n", body); *err = "Could not find token in Spotify reply (see log)"; ret = -1; goto out_free_input_body; } ret = 0; out_free_input_body: evbuffer_free(ctx.input_body); free(param); out_clear_kv: keyval_clear(&kv); return ret; } static int saved_tracks_get(int *total, const char **err, const char *uri) { struct http_client_ctx ctx; struct keyval kv; char bearer_token[1024]; char *body; char *next; int ret; int i; *total = -1; snprintf(bearer_token, sizeof(bearer_token), "Bearer %s", spotify_access_token); memset(&kv, 0, sizeof(struct keyval)); if (keyval_add(&kv, "Authorization", bearer_token) < 0) { *err = "Add bearer_token to keyval failed"; return -1; } memset(&ctx, 0, sizeof(struct http_client_ctx)); ctx.output_headers = &kv; ctx.input_body = evbuffer_new(); ctx.url = uri; next = NULL; for (i = 0; i < SPOTIFY_WEB_REQUESTS_MAX; i++) { ret = http_client_request(&ctx); if (ret < 0) { *err = "Request for saved tracks/albums failed"; break; } // 0-terminate for safety evbuffer_add(ctx.input_body, "", 1); body = (char *)evbuffer_pullup(ctx.input_body, -1); if (!body || (strlen(body) == 0)) { *err = "Request for saved tracks/albums failed, response was empty"; ret = -1; break; } if (next) free(next); next = NULL; if (uri == spotify_tracks_uri) ret = jparse_and_register_tracks(total, &next, body); else ret = -1; if (ret < 0) { *err = "Could not parse track/album response from Spotify"; break; } ret += 50 * i; // Equals total number of tracks/albums registered if (!next || (strncmp(next, "null", 4) == 0)) break; ctx.url = next; evbuffer_drain(ctx.input_body, evbuffer_get_length(ctx.input_body)); } if (next) free(next); evbuffer_free(ctx.input_body); keyval_clear(&kv); return ret; } /* -------------------------- PLAYLIST CALLBACKS ------------------------- */ /** * Called when a playlist is updating or is done updating * * This is called before and after a series of changes are applied to the * playlist. It allows e.g. the user interface to defer updating until the * entire operation is complete. * * @param[in] pl Playlist object * @param[in] done True iff the update is completed * @param[in] userdata Userdata passed to sp_playlist_add_callbacks() */ static void playlist_update_in_progress(sp_playlist *pl, bool done, void *userdata) { if (done) { DPRINTF(E_DBG, L_SPOTIFY, "Playlist update (status %d): %s\n", done, fptr_sp_playlist_name(pl)); spotify_playlist_save(pl); } } static void playlist_metadata_updated(sp_playlist *pl, void *userdata) { DPRINTF(E_DBG, L_SPOTIFY, "Playlist metadata updated: %s\n", fptr_sp_playlist_name(pl)); spotify_playlist_save(pl); } /** * The callbacks we are interested in for individual playlists. */ static sp_playlist_callbacks pl_callbacks = { .playlist_update_in_progress = &playlist_update_in_progress, .playlist_metadata_updated = &playlist_metadata_updated, }; /* -------------------- PLAYLIST CONTAINER CALLBACKS --------------------- */ /** * Callback from libspotify, telling us a playlist was added to the playlist container. * * We add our playlist callbacks to the newly added playlist. * * @param pc The playlist container handle * @param pl The playlist handle * @param position Index of the added playlist * @param userdata The opaque pointer */ static void playlist_added(sp_playlistcontainer *pc, sp_playlist *pl, int position, void *userdata) { DPRINTF(E_INFO, L_SPOTIFY, "Playlist added: %s (%d tracks)\n", fptr_sp_playlist_name(pl), fptr_sp_playlist_num_tracks(pl)); fptr_sp_playlist_add_callbacks(pl, &pl_callbacks, NULL); spotify_playlist_save(pl); } /** * Callback from libspotify, telling us a playlist was removed from the playlist container. * * This is the place to remove our playlist callbacks. * * @param pc The playlist container handle * @param pl The playlist handle * @param position Index of the removed playlist * @param userdata The opaque pointer */ static void playlist_removed(sp_playlistcontainer *pc, sp_playlist *pl, int position, void *userdata) { struct playlist_info *pli; sp_link *link; char url[1024]; int plid; int ret; DPRINTF(E_INFO, L_SPOTIFY, "Playlist removed: %s\n", fptr_sp_playlist_name(pl)); fptr_sp_playlist_remove_callbacks(pl, &pl_callbacks, NULL); link = fptr_sp_link_create_from_playlist(pl); if (!link) { DPRINTF(E_LOG, L_SPOTIFY, "Could not find link for deleted playlist\n"); return; } ret = fptr_sp_link_as_string(link, url, sizeof(url)); if (ret == sizeof(url)) { DPRINTF(E_DBG, L_SPOTIFY, "Spotify link truncated: %s\n", url); } fptr_sp_link_release(link); pli = db_pl_fetch_bypath(url); if (!pli) { DPRINTF(E_DBG, L_SPOTIFY, "Playlist %s not found, can't delete\n", url); return; } plid = pli->id; free_pli(pli, 0); db_spotify_pl_delete(plid); spotify_cleanup_files(); } /** * Callback from libspotify, telling us the rootlist is fully synchronized * * @param pc The playlist container handle * @param userdata The opaque pointer */ static void container_loaded(sp_playlistcontainer *pc, void *userdata) { int num; num = fptr_sp_playlistcontainer_num_playlists(pc); DPRINTF(E_INFO, L_SPOTIFY, "Rootlist synchronized (%d playlists)\n", num); } /** * The playlist container callbacks */ static sp_playlistcontainer_callbacks pc_callbacks = { .playlist_added = &playlist_added, .playlist_removed = &playlist_removed, .container_loaded = &container_loaded, }; /* --------------------- INTERNAL PLAYBACK AND AUDIO ----------------------- */ /* Should only be called from within the spotify thread */ static void mk_reltime(struct timespec *ts, time_t sec) { #if _POSIX_TIMERS > 0 clock_gettime(CLOCK_REALTIME, ts); #else struct timeval tv; gettimeofday(&tv, NULL); TIMEVAL_TO_TIMESPEC(&tv, ts); #endif ts->tv_sec += sec; } static void audio_fifo_flush(void) { audio_fifo_data_t *afd; DPRINTF(E_DBG, L_SPOTIFY, "Flushing audio fifo\n"); CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&g_audio_fifo->mutex)); while((afd = TAILQ_FIRST(&g_audio_fifo->q))) { TAILQ_REMOVE(&g_audio_fifo->q, afd, link); free(afd); } g_audio_fifo->qlen = 0; g_audio_fifo->fullcount = 0; CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&g_audio_fifo->mutex)); } static enum command_state playback_setup(void *arg, int *retval) { sp_link *link; sp_track *track; sp_error err; DPRINTF(E_DBG, L_SPOTIFY, "Setting up for playback\n"); link = (sp_link *) arg; if (SP_CONNECTION_STATE_LOGGED_IN != fptr_sp_session_connectionstate(g_sess)) { DPRINTF(E_LOG, L_SPOTIFY, "Can't play music, not connected and logged in to Spotify\n"); *retval = -1; return COMMAND_END; } if (!link) { DPRINTF(E_LOG, L_SPOTIFY, "Playback setup failed, no Spotify link\n"); *retval = -1; return COMMAND_END; } track = fptr_sp_link_as_track(link); if (!track) { DPRINTF(E_LOG, L_SPOTIFY, "Playback setup failed, invalid Spotify track\n"); *retval = -1; return COMMAND_END; } err = fptr_sp_session_player_load(g_sess, track); if (SP_ERROR_OK != err) { DPRINTF(E_LOG, L_SPOTIFY, "Playback setup failed: %s\n", fptr_sp_error_message(err)); *retval = -1; return COMMAND_END; } audio_fifo_flush(); *retval = 0; return COMMAND_END; } static enum command_state playback_play(void *arg, int *retval) { sp_error err; DPRINTF(E_DBG, L_SPOTIFY, "Starting playback\n"); err = fptr_sp_session_player_play(g_sess, 1); if (SP_ERROR_OK != err) { DPRINTF(E_LOG, L_SPOTIFY, "Playback failed: %s\n", fptr_sp_error_message(err)); *retval = -1; return COMMAND_END; } g_state = SPOTIFY_STATE_PLAYING; *retval = 0; return COMMAND_END; } static enum command_state playback_pause(void *arg, int *retval) { sp_error err; DPRINTF(E_DBG, L_SPOTIFY, "Pausing playback\n"); err = fptr_sp_session_player_play(g_sess, 0); DPRINTF(E_DBG, L_SPOTIFY, "Playback paused\n"); if (SP_ERROR_OK != err) { DPRINTF(E_LOG, L_SPOTIFY, "Playback pause failed: %s\n", fptr_sp_error_message(err)); *retval = -1; return COMMAND_END; } g_state = SPOTIFY_STATE_PAUSED; *retval = 0; return COMMAND_END; } static enum command_state playback_stop(void *arg, int *retval) { sp_error err; DPRINTF(E_DBG, L_SPOTIFY, "Stopping playback\n"); err = fptr_sp_session_player_unload(g_sess); if (SP_ERROR_OK != err) { DPRINTF(E_LOG, L_SPOTIFY, "Playback stop failed: %s\n", fptr_sp_error_message(err)); *retval = -1; return COMMAND_END; } g_state = SPOTIFY_STATE_STOPPED; *retval = 0; return COMMAND_END; } static enum command_state playback_seek(void *arg, int *retval) { int seek_ms; sp_error err; DPRINTF(E_DBG, L_SPOTIFY, "Playback seek\n"); seek_ms = *((int *) arg); err = fptr_sp_session_player_seek(g_sess, seek_ms); if (SP_ERROR_OK != err) { DPRINTF(E_LOG, L_SPOTIFY, "Could not seek: %s\n", fptr_sp_error_message(err)); *retval = -1; return COMMAND_END; } audio_fifo_flush(); *retval = 0; return COMMAND_END; } static enum command_state playback_eot(void *arg, int *retval) { sp_error err; DPRINTF(E_DBG, L_SPOTIFY, "Playback end of track\n"); err = fptr_sp_session_player_unload(g_sess); if (SP_ERROR_OK != err) { DPRINTF(E_LOG, L_SPOTIFY, "Playback end of track failed: %s\n", fptr_sp_error_message(err)); *retval = -1; return COMMAND_END; } g_state = SPOTIFY_STATE_STOPPING; *retval = 0; return COMMAND_END; } static enum command_state audio_get(void *arg, int *retval) { struct audio_get_param *audio; struct timespec ts; audio_fifo_data_t *afd; int processed; int timeout; int ret; int err; int s; audio = (struct audio_get_param *) arg; afd = NULL; processed = 0; // If spotify was paused begin by resuming playback if (g_state == SPOTIFY_STATE_PAUSED) playback_play(NULL, retval); CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&g_audio_fifo->mutex)); while ((processed < audio->wanted) && (g_state != SPOTIFY_STATE_STOPPED)) { // If track has ended and buffer is empty if ((g_state == SPOTIFY_STATE_STOPPING) && (g_audio_fifo->qlen <= 0)) { DPRINTF(E_DBG, L_SPOTIFY, "Track finished\n"); g_state = SPOTIFY_STATE_STOPPED; break; } // If buffer is empty, wait for audio, but use timed wait so we don't // risk waiting forever (maybe the player stopped while we were waiting) timeout = 0; while ( !(afd = TAILQ_FIRST(&g_audio_fifo->q)) && (g_state != SPOTIFY_STATE_STOPPED) && (g_state != SPOTIFY_STATE_STOPPING) && (timeout < SPOTIFY_TIMEOUT) ) { DPRINTF(E_DBG, L_SPOTIFY, "Waiting for audio\n"); timeout += 5; mk_reltime(&ts, 5); CHECK_ERR_EXCEPT(L_SPOTIFY, pthread_cond_timedwait(&g_audio_fifo->cond, &g_audio_fifo->mutex, &ts), err, ETIMEDOUT); } if ((!afd) && (timeout >= SPOTIFY_TIMEOUT)) { DPRINTF(E_LOG, L_SPOTIFY, "Timeout waiting for audio (waited %d sec)\n", timeout); spotify_playback_stop_nonblock(); } if (!afd) break; TAILQ_REMOVE(&g_audio_fifo->q, afd, link); g_audio_fifo->qlen -= afd->nsamples; s = afd->nsamples * sizeof(int16_t) * 2; ret = evbuffer_add(audio->evbuf, afd->samples, s); free(afd); afd = NULL; if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Out of memory for evbuffer (tried to add %d bytes)\n", s); CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&g_audio_fifo->mutex)); *retval = -1; return COMMAND_END; } processed += s; } CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&g_audio_fifo->mutex)); *retval = processed; return COMMAND_END; } static void artwork_loaded_cb(sp_image *image, void *userdata) { struct artwork_get_param *artwork; artwork = userdata; CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&artwork->mutex)); artwork->is_loaded = 1; CHECK_ERR(L_SPOTIFY, pthread_cond_signal(&artwork->cond)); CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&artwork->mutex)); } static enum command_state artwork_get_bh(void *arg, int *retval) { struct artwork_get_param *artwork; sp_imageformat imageformat; sp_error err; const void *data; size_t data_size; int ret; artwork = arg; sp_image *image = artwork->image; char *path = artwork->path; err = fptr_sp_image_error(image); if (err != SP_ERROR_OK) { DPRINTF(E_WARN, L_SPOTIFY, "Getting artwork (%s) failed, Spotify error: %s\n", path, fptr_sp_error_message(err)); goto fail; } if (!fptr_sp_image_is_loaded(image)) { DPRINTF(E_LOG, L_SPOTIFY, "Load callback returned, but no image? Possible bug: %s\n", path); goto fail; } imageformat = fptr_sp_image_format(image); if (imageformat != SP_IMAGE_FORMAT_JPEG) { DPRINTF(E_WARN, L_SPOTIFY, "Getting artwork failed, invalid image format from Spotify: %s\n", path); goto fail; } data = fptr_sp_image_data(image, &data_size); if (!data || (data_size == 0)) { DPRINTF(E_LOG, L_SPOTIFY, "Getting artwork failed, no image data from Spotify: %s\n", path); goto fail; } ret = evbuffer_expand(artwork->evbuf, data_size); if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Out of memory for artwork\n"); goto fail; } ret = evbuffer_add(artwork->evbuf, data, data_size); if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Could not add Spotify image to event buffer\n"); goto fail; } DPRINTF(E_DBG, L_SPOTIFY, "Spotify artwork loaded ok\n"); fptr_sp_image_release(image); *retval = 0; return COMMAND_END; fail: fptr_sp_image_release(image); *retval = -1; return COMMAND_END; } static enum command_state artwork_get(void *arg, int *retval) { struct artwork_get_param *artwork; char *path; sp_link *link; sp_track *track; sp_album *album; const byte *image_id; sp_image *image; sp_image_size image_size; sp_error err; artwork = arg; path = artwork->path; // Now begins: path -> link -> track -> album -> image_id -> image -> format -> data link = fptr_sp_link_create_from_string(path); if (!link) { DPRINTF(E_WARN, L_SPOTIFY, "Getting artwork failed, invalid Spotify link: %s\n", path); goto level1_exit; } track = fptr_sp_link_as_track(link); if (!track) { DPRINTF(E_WARN, L_SPOTIFY, "Getting artwork failed, invalid Spotify track: %s\n", path); goto level2_exit; } album = fptr_sp_track_album(track); if (!album) { DPRINTF(E_WARN, L_SPOTIFY, "Getting artwork failed, invalid Spotify album: %s\n", path); goto level2_exit; } // Get an image at least the same size as requested image_size = SP_IMAGE_SIZE_SMALL; // 64x64 if ((artwork->max_w > 64) || (artwork->max_h > 64)) image_size = SP_IMAGE_SIZE_NORMAL; // 300x300 if ((artwork->max_w > 300) || (artwork->max_h > 300)) image_size = SP_IMAGE_SIZE_LARGE; // 640x640 image_id = fptr_sp_album_cover(album, image_size); if (!image_id) { DPRINTF(E_DBG, L_SPOTIFY, "Getting artwork failed, no Spotify image id: %s\n", path); goto level2_exit; } image = fptr_sp_image_create(g_sess, image_id); if (!image) { DPRINTF(E_DBG, L_SPOTIFY, "Getting artwork failed, no Spotify image: %s\n", path); goto level2_exit; } fptr_sp_link_release(link); artwork->image = image; artwork->is_loaded = fptr_sp_image_is_loaded(image); /* If the image is ready we can return it straight away, otherwise we will * let the calling thread wait, since the Spotify thread should not wait */ if (artwork->is_loaded) return artwork_get_bh(artwork, retval); DPRINTF(E_SPAM, L_SPOTIFY, "Will wait for Spotify to call artwork_loaded_cb\n"); /* Async - we will return to spotify_artwork_get which will wait for callback */ err = fptr_sp_image_add_load_callback(image, artwork_loaded_cb, artwork); if (err != SP_ERROR_OK) { DPRINTF(E_WARN, L_SPOTIFY, "Adding artwork cb failed, Spotify error: %s\n", fptr_sp_error_message(err)); *retval = -1; return COMMAND_END; } *retval = 0; return COMMAND_END; level2_exit: fptr_sp_link_release(link); level1_exit: *retval = -1; return COMMAND_END; } /* --------------------------- SESSION CALLBACKS ------------------------- */ /** * This callback is called when an attempt to login has succeeded or failed. * * @sa sp_session_callbacks#logged_in */ static void logged_in(sp_session *sess, sp_error error) { cfg_t *spotify_cfg; sp_playlist *pl; sp_playlistcontainer *pc; struct playlist_info pli; int ret; int i; if (SP_ERROR_OK != error) { DPRINTF(E_LOG, L_SPOTIFY, "Login failed: %s\n", fptr_sp_error_message(error)); return; } DPRINTF(E_LOG, L_SPOTIFY, "Login to Spotify succeeded, reloading playlists\n"); db_directory_enable_bypath("/spotify:"); pl = fptr_sp_session_starred_create(sess); fptr_sp_playlist_add_callbacks(pl, &pl_callbacks, NULL); spotify_cfg = cfg_getsec(cfg, "spotify"); if (! cfg_getbool(spotify_cfg, "base_playlist_disable")) { memset(&pli, 0, sizeof(struct playlist_info)); pli.title = "Spotify"; pli.type = PL_FOLDER; pli.path = "spotify:playlistfolder"; ret = db_pl_add(&pli, &spotify_base_plid); if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Error adding base playlist\n"); return; } } else spotify_base_plid = 0; pc = fptr_sp_session_playlistcontainer(sess); fptr_sp_playlistcontainer_add_callbacks(pc, &pc_callbacks, NULL); DPRINTF(E_DBG, L_SPOTIFY, "Found %d playlists\n", fptr_sp_playlistcontainer_num_playlists(pc)); for (i = 0; i < fptr_sp_playlistcontainer_num_playlists(pc); i++) { pl = fptr_sp_playlistcontainer_playlist(pc, i); fptr_sp_playlist_add_callbacks(pl, &pl_callbacks, NULL); } } /** * Called when logout has been processed. * Either called explicitly if you initialize a logout operation, or implicitly * if there is a permanent connection error * * @sa sp_session_callbacks#logged_out */ static void logged_out(sp_session *sess) { DPRINTF(E_INFO, L_SPOTIFY, "Logout complete\n"); CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&login_lck)); CHECK_ERR(L_SPOTIFY, pthread_cond_signal(&login_cond)); CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&login_lck)); } /** * This callback is used from libspotify whenever there is PCM data available. * * @sa sp_session_callbacks#music_delivery */ static int music_delivery(sp_session *sess, const sp_audioformat *format, const void *frames, int num_frames) { audio_fifo_data_t *afd; size_t s; /* No support for resampling right now */ if ((format->sample_rate != 44100) || (format->channels != 2)) { DPRINTF(E_LOG, L_SPOTIFY, "Got music with unsupported samplerate or channels, stopping playback\n"); spotify_playback_stop_nonblock(); return num_frames; } if (num_frames == 0) return 0; // Audio discontinuity, do nothing CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&g_audio_fifo->mutex)); /* Buffer three seconds of audio */ if (g_audio_fifo->qlen > (3 * format->sample_rate)) { // If the buffer has been full the last 300 times (~about a minute) we // assume the player thread paused/died without telling us, so we signal pause if (g_audio_fifo->fullcount < 300) g_audio_fifo->fullcount++; else { DPRINTF(E_WARN, L_SPOTIFY, "Buffer full more than 300 times, pausing\n"); spotify_playback_pause_nonblock(); g_audio_fifo->fullcount = 0; } CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&g_audio_fifo->mutex)); return 0; } else g_audio_fifo->fullcount = 0; s = num_frames * sizeof(int16_t) * format->channels; afd = malloc(sizeof(*afd) + s); memcpy(afd->samples, frames, s); afd->nsamples = num_frames; TAILQ_INSERT_TAIL(&g_audio_fifo->q, afd, link); g_audio_fifo->qlen += num_frames; CHECK_ERR(L_SPOTIFY, pthread_cond_signal(&g_audio_fifo->cond)); CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&g_audio_fifo->mutex)); return num_frames; } /** * This callback is called from an internal libspotify thread to ask us to * reiterate the main loop. This must not block. * * @sa sp_session_callbacks#notify_main_thread */ static void notify_main_thread(sp_session *sess) { int dummy = 42; int ret; ret = write(g_notify_pipe[1], &dummy, sizeof(dummy)); if (ret != sizeof(dummy)) DPRINTF(E_LOG, L_SPOTIFY, "Could not write to notify fd: %s\n", strerror(errno)); } /** * Called whenever metadata has been updated * * If you have metadata cached outside of libspotify, you should purge * your caches and fetch new versions. * * @param[in] session Session */ static void metadata_updated(sp_session *session) { DPRINTF(E_DBG, L_SPOTIFY, "Session metadata updated\n"); commands_exec_async(cmdbase, spotify_pending_process, NULL); } /* Misc connection error callbacks */ static void play_token_lost(sp_session *sess) { DPRINTF(E_LOG, L_SPOTIFY, "Music interrupted - some other session is playing on the account\n"); spotify_playback_stop_nonblock(); } static void connectionstate_updated(sp_session *session) { struct reload_list *reload; int ret; if (SP_CONNECTION_STATE_LOGGED_IN == fptr_sp_session_connectionstate(session)) { DPRINTF(E_LOG, L_SPOTIFY, "Connection to Spotify (re)established, reloading saved tracks\n"); while ((reload = spotify_reload_list)) { spotify_uri_register(reload->uri, &ret); spotify_reload_list = reload->next; free(reload->uri); free(reload); } } else if (g_state == SPOTIFY_STATE_PLAYING) { DPRINTF(E_LOG, L_SPOTIFY, "Music interrupted - connection error or logged out\n"); spotify_playback_stop_nonblock(); } } /** * This callback is used from libspotify when the current track has ended * * @sa sp_session_callbacks#end_of_track */ static void end_of_track(sp_session *sess) { DPRINTF(E_DBG, L_SPOTIFY, "End of track\n"); commands_exec_async(cmdbase, playback_eot, NULL); } /** * The session callbacks */ static sp_session_callbacks session_callbacks = { .logged_in = &logged_in, .logged_out = &logged_out, .connectionstate_updated = &connectionstate_updated, .notify_main_thread = ¬ify_main_thread, .music_delivery = &music_delivery, .metadata_updated = &metadata_updated, .play_token_lost = &play_token_lost, .log_message = NULL, .end_of_track = &end_of_track, }; /** * The session configuration. */ static sp_session_config spconfig = { .api_version = SPOTIFY_API_VERSION, .cache_location = NULL, .settings_location = NULL, .application_key = g_appkey, .application_key_size = sizeof(g_appkey), .user_agent = "forked-daapd", .callbacks = &session_callbacks, NULL, }; /* ------------------------------- MAIN LOOP ------------------------------- */ /* Thread: spotify */ static struct reload_list * reload_list_create(int plid) { struct query_params qp; struct db_media_file_info dbmfi; struct reload_list *head; struct reload_list *reload; int ret; memset(&qp, 0, sizeof(struct query_params)); qp.type = Q_PLITEMS; qp.sort = S_NONE; qp.id = plid; ret = db_query_start(&qp); if (ret < 0) { db_query_end(&qp); return NULL; } head = NULL; while (((ret = db_query_fetch_file(&qp, &dbmfi)) == 0) && (dbmfi.path)) { reload = malloc(sizeof(struct reload_list)); reload->uri = strdup(dbmfi.path); reload->next = head; head = reload; } db_query_end(&qp); return head; } static void * spotify(void *arg) { int ret; DPRINTF(E_DBG, L_SPOTIFY, "Main loop initiating\n"); ret = db_perthread_init(); if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Error: DB init failed\n"); pthread_exit(NULL); } g_state = SPOTIFY_STATE_WAIT; event_base_dispatch(evbase_spotify); if (g_state != SPOTIFY_STATE_INACTIVE) { DPRINTF(E_LOG, L_SPOTIFY, "Spotify event loop terminated ahead of time!\n"); g_state = SPOTIFY_STATE_INACTIVE; } db_perthread_deinit(); DPRINTF(E_DBG, L_SPOTIFY, "Main loop terminating\n"); pthread_exit(NULL); } static void exit_cb() { fptr_sp_session_player_unload(g_sess); fptr_sp_session_logout(g_sess); g_state = SPOTIFY_STATE_INACTIVE; } /* Process events when timeout expires or triggered by libspotify's notify_main_thread */ static void notify_cb(int fd, short what, void *arg) { struct timeval tv; int next_timeout; int dummy; int ret; if (what & EV_READ) { ret = read(g_notify_pipe[0], &dummy, sizeof(dummy)); if (ret != sizeof(dummy)) DPRINTF(E_LOG, L_SPOTIFY, "Error reading from notify pipe\n"); } do { fptr_sp_session_process_events(g_sess, &next_timeout); } while (next_timeout == 0); tv.tv_sec = next_timeout / 1000; tv.tv_usec = (next_timeout % 1000) * 1000; event_add(g_notifyev, &tv); } /* ---------------------------- Our Spotify API --------------------------- */ /* Thread: player */ int spotify_playback_setup(const char *path) { sp_link *link; DPRINTF(E_DBG, L_SPOTIFY, "Playback setup request\n"); link = fptr_sp_link_create_from_string(path); if (!link) { DPRINTF(E_LOG, L_SPOTIFY, "Playback setup failed, invalid Spotify link: %s\n", path); return -1; } return commands_exec_sync(cmdbase, playback_setup, NULL, link); } int spotify_playback_play() { DPRINTF(E_DBG, L_SPOTIFY, "Playback request\n"); return commands_exec_sync(cmdbase, playback_play, NULL, NULL); } int spotify_playback_pause() { DPRINTF(E_DBG, L_SPOTIFY, "Pause request\n"); return commands_exec_sync(cmdbase, playback_pause, NULL, NULL); } /* Thread: libspotify */ void spotify_playback_pause_nonblock(void) { DPRINTF(E_DBG, L_SPOTIFY, "Nonblock pause request\n"); commands_exec_async(cmdbase, playback_pause, NULL); } /* Thread: player and libspotify */ int spotify_playback_stop(void) { DPRINTF(E_DBG, L_SPOTIFY, "Stop request\n"); return commands_exec_sync(cmdbase, playback_stop, NULL, NULL); } /* Thread: player and libspotify */ void spotify_playback_stop_nonblock(void) { DPRINTF(E_DBG, L_SPOTIFY, "Nonblock stop request\n"); commands_exec_async(cmdbase, playback_stop, NULL); } /* Thread: player */ int spotify_playback_seek(int ms) { int ret; ret = commands_exec_sync(cmdbase, playback_seek, NULL, &ms); if (ret == 0) return ms; else return -1; } /* Thread: player */ int spotify_audio_get(struct evbuffer *evbuf, int wanted) { struct audio_get_param audio; audio.evbuf = evbuf; audio.wanted = wanted; return commands_exec_sync(cmdbase, audio_get, NULL, &audio); } /* Thread: httpd (artwork) and worker */ int spotify_artwork_get(struct evbuffer *evbuf, char *path, int max_w, int max_h) { struct artwork_get_param artwork; struct timespec ts; int ret; int err; artwork.evbuf = evbuf; artwork.path = path; artwork.max_w = max_w; artwork.max_h = max_h; CHECK_ERR(L_SPOTIFY, mutex_init(&artwork.mutex)); CHECK_ERR(L_SPOTIFY, pthread_cond_init(&artwork.cond, NULL)); ret = commands_exec_sync(cmdbase, artwork_get, NULL, &artwork); // Artwork was not ready, wait for callback from libspotify if (ret == 0) { CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&artwork.mutex)); mk_reltime(&ts, SPOTIFY_ARTWORK_TIMEOUT); if (!artwork.is_loaded) CHECK_ERR_EXCEPT(L_SPOTIFY, pthread_cond_timedwait(&artwork.cond, &artwork.mutex, &ts), err, ETIMEDOUT); CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&artwork.mutex)); ret = commands_exec_sync(cmdbase, artwork_get_bh, NULL, &artwork); } return ret; } /* Thread: httpd */ void spotify_oauth_interface(struct evbuffer *evbuf, const char *redirect_uri) { struct keyval kv; char *param; int ret; memset(&kv, 0, sizeof(struct keyval)); ret = ( (keyval_add(&kv, "client_id", spotify_client_id) == 0) && (keyval_add(&kv, "response_type", "code") == 0) && (keyval_add(&kv, "redirect_uri", redirect_uri) == 0) && (keyval_add(&kv, "scope", "playlist-read-private user-library-read") == 0) && (keyval_add(&kv, "show_dialog", "false") == 0) ); if (!ret) { DPRINTF(E_LOG, L_SPOTIFY, "Cannot display Spotify oath interface (error adding parameters to keyval)\n"); goto out_clear_kv; } param = http_form_urlencode(&kv); if (!param) { DPRINTF(E_LOG, L_SPOTIFY, "Cannot display Spotify oath interface (http_form_uriencode() failed)\n"); goto out_clear_kv; } evbuffer_add_printf(evbuf, "Click here to authorize forked-daapd with Spotify\n", spotify_auth_uri, param); free(param); out_clear_kv: keyval_clear(&kv); } /* Thread: httpd */ void spotify_oauth_callback(struct evbuffer *evbuf, struct evkeyvalq *param, const char *redirect_uri) { const char *code; const char *err = ""; int total; int ret; code = evhttp_find_header(param, "code"); if (!code) { evbuffer_add_printf(evbuf, "Error: Didn't receive a code from Spotify\n"); return; } DPRINTF(E_DBG, L_SPOTIFY, "Received OAuth code: %s\n", code); evbuffer_add_printf(evbuf, "

Requesting access token from Spotify...\n"); ret = tokens_get(code, redirect_uri, &err); if (ret < 0) { evbuffer_add_printf(evbuf, "failed

\n

Error: %s

\n", err); return; } commands_exec_sync(cmdbase, spotify_saved_pl_clear_items, NULL, NULL); evbuffer_add_printf(evbuf, "ok

\n

Retrieving saved tracks...\n"); ret = saved_tracks_get(&total, &err, spotify_tracks_uri); if (ret < 0) { evbuffer_add_printf(evbuf, "failed

\n

Error: %s

\n", err); return; } evbuffer_add_printf(evbuf, "ok, got %d out of %d tracks

\n", ret, total); evbuffer_add_printf(evbuf, "

Purging removed tracks/albums...\n"); // TODO release links to the items we are going to clean up commands_exec_sync(cmdbase, spotify_cleanup_wrapper, NULL, NULL); evbuffer_add_printf(evbuf, "ok, all done

\n"); return; } /* Thread: filescanner */ void spotify_login(char *path) { struct playlist_info *pli; sp_error err; char *username; char *password; int ret; if (!g_sess) { if (!g_libhandle) DPRINTF(E_LOG, L_SPOTIFY, "Can't login! - could not find libspotify\n"); else DPRINTF(E_LOG, L_SPOTIFY, "Can't login! - no valid Spotify session\n"); return; } if (SP_CONNECTION_STATE_LOGGED_IN == fptr_sp_session_connectionstate(g_sess)) { CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&login_lck)); DPRINTF(E_LOG, L_SPOTIFY, "Logging out of Spotify (current state is %d)\n", g_state); fptr_sp_session_player_unload(g_sess); err = fptr_sp_session_logout(g_sess); if (SP_ERROR_OK != err) { DPRINTF(E_LOG, L_SPOTIFY, "Could not logout of Spotify: %s\n", fptr_sp_error_message(err)); CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&login_lck)); return; } CHECK_ERR(L_SPOTIFY, pthread_cond_wait(&login_cond, &login_lck)); CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&login_lck)); } DPRINTF(E_INFO, L_SPOTIFY, "Logging into Spotify\n"); if (path) { db_spotify_purge(); spotify_saved_plid = 0; ret = spotify_file_read(path, &username, &password); if (ret < 0) return; err = fptr_sp_session_login(g_sess, username, password, 1, NULL); free(username); free(password); } else { pli = db_pl_fetch_bypath("spotify:savedtracks"); if (pli) { spotify_reload_list = reload_list_create(pli->id); free_pli(pli, 0); } db_spotify_purge(); spotify_saved_plid = 0; err = fptr_sp_session_relogin(g_sess); } if (SP_ERROR_OK != err) { DPRINTF(E_LOG, L_SPOTIFY, "Could not login into Spotify: %s\n", fptr_sp_error_message(err)); return; } } /* Thread: main */ int spotify_init(void) { cfg_t *spotify_cfg; sp_session *sp; sp_error err; int ret; /* Initialize libspotify */ g_libhandle = dlopen("libspotify.so", RTLD_LAZY); if (!g_libhandle) { DPRINTF(E_INFO, L_SPOTIFY, "libspotify.so not installed or not found\n"); goto libspotify_fail; } ret = fptr_assign_all(); if (ret < 0) goto assign_fail; #ifdef HAVE_PIPE2 ret = pipe2(g_notify_pipe, O_CLOEXEC); #else ret = pipe(g_notify_pipe); #endif if (ret < 0) { DPRINTF(E_LOG, L_SPOTIFY, "Could not notify command pipe: %s\n", strerror(errno)); goto notify_fail; } evbase_spotify = event_base_new(); if (!evbase_spotify) { DPRINTF(E_LOG, L_SPOTIFY, "Could not create an event base\n"); goto evbase_fail; } g_notifyev = event_new(evbase_spotify, g_notify_pipe[0], EV_READ | EV_TIMEOUT, notify_cb, NULL); if (!g_notifyev) { DPRINTF(E_LOG, L_SPOTIFY, "Could not create notify event\n"); goto evnew_fail; } event_add(g_notifyev, NULL); cmdbase = commands_base_new(evbase_spotify, exit_cb); if (!cmdbase) { DPRINTF(E_LOG, L_SPOTIFY, "Could not create command base\n"); goto cmd_fail; } DPRINTF(E_INFO, L_SPOTIFY, "Spotify session init\n"); spotify_cfg = cfg_getsec(cfg, "spotify"); spconfig.settings_location = cfg_getstr(spotify_cfg, "settings_dir"); spconfig.cache_location = cfg_getstr(spotify_cfg, "cache_dir"); DPRINTF(E_DBG, L_SPOTIFY, "Creating Spotify session\n"); err = fptr_sp_session_create(&spconfig, &sp); if (SP_ERROR_OK != err) { DPRINTF(E_LOG, L_SPOTIFY, "Could not create Spotify session: %s\n", fptr_sp_error_message(err)); goto session_fail; } g_sess = sp; g_state = SPOTIFY_STATE_INACTIVE; switch (cfg_getint(spotify_cfg, "bitrate")) { case 1: fptr_sp_session_preferred_bitrate(g_sess, SP_BITRATE_96k); break; case 2: fptr_sp_session_preferred_bitrate(g_sess, SP_BITRATE_160k); break; case 3: fptr_sp_session_preferred_bitrate(g_sess, SP_BITRATE_320k); break; } /* Prepare audio buffer */ g_audio_fifo = (audio_fifo_t *)malloc(sizeof(audio_fifo_t)); if (!g_audio_fifo) { DPRINTF(E_LOG, L_SPOTIFY, "Out of memory for audio buffer\n"); goto audio_fifo_fail; } TAILQ_INIT(&g_audio_fifo->q); g_audio_fifo->qlen = 0; CHECK_ERR(L_SPOTIFY, mutex_init(&g_audio_fifo->mutex)); CHECK_ERR(L_SPOTIFY, pthread_cond_init(&g_audio_fifo->cond, NULL)); CHECK_ERR(L_SPOTIFY, mutex_init(&login_lck)); CHECK_ERR(L_SPOTIFY, pthread_cond_init(&login_cond, NULL)); /* Spawn thread */ ret = pthread_create(&tid_spotify, NULL, spotify, NULL); if (ret < 0) { DPRINTF(E_FATAL, L_SPOTIFY, "Could not spawn Spotify thread: %s\n", strerror(errno)); goto thread_fail; } #if defined(HAVE_PTHREAD_SETNAME_NP) pthread_setname_np(tid_spotify, "spotify"); #elif defined(HAVE_PTHREAD_SET_NAME_NP) pthread_set_name_np(tid_spotify, "spotify"); #endif DPRINTF(E_DBG, L_SPOTIFY, "Spotify init complete\n"); return 0; thread_fail: CHECK_ERR(L_SPOTIFY, pthread_cond_destroy(&login_cond)); CHECK_ERR(L_SPOTIFY, pthread_mutex_destroy(&login_lck)); CHECK_ERR(L_SPOTIFY, pthread_cond_destroy(&g_audio_fifo->cond)); CHECK_ERR(L_SPOTIFY, pthread_mutex_destroy(&g_audio_fifo->mutex)); free(g_audio_fifo); audio_fifo_fail: fptr_sp_session_release(g_sess); g_sess = NULL; session_fail: cmd_fail: evnew_fail: commands_base_free(cmdbase); event_base_free(evbase_spotify); evbase_spotify = NULL; evbase_fail: close(g_notify_pipe[0]); close(g_notify_pipe[1]); notify_fail: assign_fail: dlclose(g_libhandle); g_libhandle = NULL; libspotify_fail: return -1; } void spotify_deinit(void) { int ret; if (!g_libhandle) return; /* Send exit signal to thread (if active) */ if (g_state != SPOTIFY_STATE_INACTIVE) { commands_base_destroy(cmdbase); g_state = SPOTIFY_STATE_INACTIVE; ret = pthread_join(tid_spotify, NULL); if (ret != 0) { DPRINTF(E_FATAL, L_SPOTIFY, "Could not join Spotify thread: %s\n", strerror(errno)); return; } } /* Release session */ fptr_sp_session_release(g_sess); /* Free event base (should free events too) */ event_base_free(evbase_spotify); /* Close pipes */ close(g_notify_pipe[0]); close(g_notify_pipe[1]); /* Destroy locks */ CHECK_ERR(L_SPOTIFY, pthread_cond_destroy(&login_cond)); CHECK_ERR(L_SPOTIFY, pthread_mutex_destroy(&login_lck)); /* Clear audio fifo */ CHECK_ERR(L_SPOTIFY, pthread_cond_destroy(&g_audio_fifo->cond)); CHECK_ERR(L_SPOTIFY, pthread_mutex_destroy(&g_audio_fifo->mutex)); free(g_audio_fifo); /* Release libspotify handle */ dlclose(g_libhandle); }