2014-03-11 18:20:29 -04:00
|
|
|
/*
|
2016-11-05 08:44:23 -04:00
|
|
|
* Copyright (C) 2016 Espen Jürgensen <espenjurgensen@gmail.com>
|
2014-03-11 18:20:29 -04:00
|
|
|
*
|
|
|
|
* 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 <config.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdint.h>
|
2014-03-23 16:45:06 -04:00
|
|
|
#include <fcntl.h>
|
2014-03-11 18:20:29 -04:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <sys/queue.h>
|
2015-12-27 01:16:50 -05:00
|
|
|
#include <time.h>
|
2014-03-11 18:20:29 -04:00
|
|
|
#include <pthread.h>
|
2016-05-05 13:39:15 -04:00
|
|
|
#ifdef HAVE_PTHREAD_NP_H
|
2016-03-05 07:19:55 -05:00
|
|
|
# include <pthread_np.h>
|
|
|
|
#endif
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
#include <dlfcn.h>
|
|
|
|
#include <libspotify/api.h>
|
2017-01-06 03:44:18 -05:00
|
|
|
#include <json.h>
|
2016-11-05 08:44:23 -04:00
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
#include "spotify.h"
|
|
|
|
#include "logger.h"
|
2016-11-05 08:44:23 -04:00
|
|
|
#include "misc.h"
|
|
|
|
#include "http.h"
|
2014-03-11 18:20:29 -04:00
|
|
|
#include "conffile.h"
|
|
|
|
#include "filescanner.h"
|
2016-01-16 11:02:03 -05:00
|
|
|
#include "cache.h"
|
2016-05-08 03:54:58 -04:00
|
|
|
#include "commands.h"
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
/* TODO for the web api:
|
|
|
|
* - UI should be prettier
|
2016-11-13 14:28:29 -05:00
|
|
|
* - map "added_at" to time_added
|
2016-11-05 08:44:23 -04:00
|
|
|
* - what to do about the lack of push?
|
2016-11-13 14:28:29 -05:00
|
|
|
* - 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.
|
2016-11-05 08:44:23 -04:00
|
|
|
*/
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
// How long to wait for audio (in sec) before giving up
|
2014-11-11 14:22:39 -05:00
|
|
|
#define SPOTIFY_TIMEOUT 20
|
2016-11-05 08:44:23 -04:00
|
|
|
// How long to wait for artwork (in sec) before giving up
|
2016-01-08 18:36:30 -05:00
|
|
|
#define SPOTIFY_ARTWORK_TIMEOUT 3
|
2016-11-05 08:44:23 -04:00
|
|
|
// An upper limit on sequential requests to Spotify's web api
|
|
|
|
// - each request will return 50 objects (tracks)
|
|
|
|
#define SPOTIFY_WEB_REQUESTS_MAX 20
|
2014-11-11 05:59:40 -05:00
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
/* --- Types --- */
|
2014-03-23 16:45:06 -04:00
|
|
|
typedef struct audio_fifo_data
|
|
|
|
{
|
2014-03-11 18:20:29 -04:00
|
|
|
TAILQ_ENTRY(audio_fifo_data) link;
|
|
|
|
int nsamples;
|
|
|
|
int16_t samples[0];
|
|
|
|
} audio_fifo_data_t;
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
typedef struct audio_fifo
|
|
|
|
{
|
2014-03-11 18:20:29 -04:00
|
|
|
TAILQ_HEAD(, audio_fifo_data) q;
|
|
|
|
int qlen;
|
|
|
|
int fullcount;
|
|
|
|
pthread_mutex_t mutex;
|
|
|
|
pthread_cond_t cond;
|
|
|
|
} audio_fifo_t;
|
|
|
|
|
|
|
|
enum spotify_state
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2014-04-04 17:14:43 -04:00
|
|
|
struct artwork_get_param
|
|
|
|
{
|
|
|
|
struct evbuffer *evbuf;
|
|
|
|
char *path;
|
|
|
|
int max_w;
|
|
|
|
int max_h;
|
|
|
|
|
2016-01-08 18:36:30 -05:00
|
|
|
sp_image *image;
|
2016-01-07 16:25:25 -05:00
|
|
|
pthread_mutex_t mutex;
|
|
|
|
pthread_cond_t cond;
|
2016-01-08 18:36:30 -05:00
|
|
|
int is_loaded;
|
2016-01-07 16:25:25 -05:00
|
|
|
};
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
struct pending_metadata
|
|
|
|
{
|
|
|
|
sp_link *link;
|
|
|
|
sp_track *track;
|
|
|
|
struct pending_metadata *next;
|
|
|
|
};
|
2014-03-23 16:45:06 -04:00
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
struct reload_list
|
|
|
|
{
|
|
|
|
char *uri;
|
|
|
|
struct reload_list *next;
|
|
|
|
};
|
2014-03-23 16:45:06 -04:00
|
|
|
|
|
|
|
/* --- Globals --- */
|
|
|
|
// Spotify thread
|
|
|
|
static pthread_t tid_spotify;
|
|
|
|
|
2014-12-28 15:08:52 -05:00
|
|
|
// 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;
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
// Event base, pipes and events
|
|
|
|
struct event_base *evbase_spotify;
|
|
|
|
static int g_notify_pipe[2];
|
|
|
|
static struct event *g_notifyev;
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
static struct commands_base *cmdbase;
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
// The session handle
|
2014-03-23 16:45:06 -04:00
|
|
|
static sp_session *g_sess;
|
2016-11-05 08:44:23 -04:00
|
|
|
// The library handle
|
2014-03-23 16:45:06 -04:00
|
|
|
static void *g_libhandle;
|
2016-11-05 08:44:23 -04:00
|
|
|
// The state telling us what the thread is currently doing
|
2014-03-23 16:45:06 -04:00
|
|
|
static enum spotify_state g_state;
|
2016-11-13 14:28:29 -05:00
|
|
|
// 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;
|
2016-11-05 08:44:23 -04:00
|
|
|
// Linked list of tracks where we are waiting for metadata
|
|
|
|
static struct pending_metadata *spotify_pending_metadata;
|
2016-11-13 14:28:29 -05:00
|
|
|
// Linked list of saved tracks which we want to reload at startup
|
|
|
|
static struct reload_list *spotify_reload_list;
|
2014-03-23 16:45:06 -04:00
|
|
|
|
|
|
|
// Audio fifo
|
|
|
|
static audio_fifo_t *g_audio_fifo;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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,
|
|
|
|
};
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
// 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";
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
// 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);
|
2015-03-07 13:31:24 -05:00
|
|
|
typedef sp_playlist* (*fptr_sp_session_starred_create_t)(sp_session *session);
|
2014-03-11 18:20:29 -04:00
|
|
|
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);
|
2014-03-23 16:45:06 -04:00
|
|
|
typedef sp_connectionstate (*fptr_sp_session_connectionstate_t)(sp_session *session);
|
2014-03-29 17:26:46 -04:00
|
|
|
typedef sp_error (*fptr_sp_session_preferred_bitrate_t)(sp_session *session, sp_bitrate bitrate);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
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);
|
2015-05-29 13:47:53 -04:00
|
|
|
typedef int (*fptr_sp_playlist_track_create_time_t)(sp_playlist *playlist, int index);
|
2016-03-12 01:39:26 -05:00
|
|
|
typedef sp_user* (*fptr_sp_playlist_owner_t)(sp_playlist *playlist);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
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);
|
2014-04-03 14:44:36 -04:00
|
|
|
typedef sp_track_availability (*fptr_sp_track_get_availability_t)(sp_session *session, sp_track *track);
|
2015-03-07 12:30:32 -05:00
|
|
|
typedef bool (*fptr_sp_track_is_starred_t)(sp_session *session, sp_track *track);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
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);
|
2014-04-04 17:14:43 -04:00
|
|
|
typedef const byte* (*fptr_sp_album_cover_t)(sp_album *album, sp_image_size size);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
typedef const char* (*fptr_sp_artist_name_t)(sp_artist *artist);
|
|
|
|
|
2014-04-04 17:14:43 -04:00
|
|
|
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);
|
2016-01-07 16:25:25 -05:00
|
|
|
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);
|
2014-04-04 17:14:43 -04:00
|
|
|
|
2016-03-12 01:39:26 -05:00
|
|
|
typedef const char* (*fptr_sp_user_display_name_t)(sp_user *user);
|
|
|
|
typedef const char* (*fptr_sp_user_canonical_name_t)(sp_user *user);
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
/* 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;
|
2015-03-07 13:31:24 -05:00
|
|
|
fptr_sp_session_starred_create_t fptr_sp_session_starred_create;
|
2014-03-11 18:20:29 -04:00
|
|
|
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;
|
2014-03-23 16:45:06 -04:00
|
|
|
fptr_sp_session_connectionstate_t fptr_sp_session_connectionstate;
|
2014-03-29 17:26:46 -04:00
|
|
|
fptr_sp_session_preferred_bitrate_t fptr_sp_session_preferred_bitrate;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
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;
|
2015-05-29 13:47:53 -04:00
|
|
|
fptr_sp_playlist_track_create_time_t fptr_sp_playlist_track_create_time;
|
2016-03-12 01:39:26 -05:00
|
|
|
fptr_sp_playlist_owner_t fptr_sp_playlist_owner;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
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;
|
2014-04-03 14:44:36 -04:00
|
|
|
fptr_sp_track_get_availability_t fptr_sp_track_get_availability;
|
2015-03-07 12:30:32 -05:00
|
|
|
fptr_sp_track_is_starred_t fptr_sp_track_is_starred;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
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;
|
2014-04-04 17:14:43 -04:00
|
|
|
fptr_sp_album_cover_t fptr_sp_album_cover;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
fptr_sp_artist_name_t fptr_sp_artist_name;
|
|
|
|
|
2014-04-04 17:14:43 -04:00
|
|
|
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;
|
2016-01-07 16:25:25 -05:00
|
|
|
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;
|
2014-04-04 17:14:43 -04:00
|
|
|
|
2016-03-12 01:39:26 -05:00
|
|
|
fptr_sp_user_display_name_t fptr_sp_user_display_name;
|
|
|
|
fptr_sp_user_canonical_name_t fptr_sp_user_canonical_name;
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
/* 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"))
|
2014-03-23 16:45:06 -04:00
|
|
|
&& (fptr_sp_session_connectionstate = dlsym(h, "sp_session_connectionstate"))
|
2014-03-29 17:26:46 -04:00
|
|
|
&& (fptr_sp_session_preferred_bitrate = dlsym(h, "sp_session_preferred_bitrate"))
|
2014-03-11 18:20:29 -04:00
|
|
|
&& (fptr_sp_playlistcontainer_add_callbacks = dlsym(h, "sp_playlistcontainer_add_callbacks"))
|
|
|
|
&& (fptr_sp_playlistcontainer_num_playlists = dlsym(h, "sp_playlistcontainer_num_playlists"))
|
2015-03-07 13:31:24 -05:00
|
|
|
&& (fptr_sp_session_starred_create = dlsym(h, "sp_session_starred_create"))
|
2014-03-11 18:20:29 -04:00
|
|
|
&& (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"))
|
2015-05-29 13:47:53 -04:00
|
|
|
&& (fptr_sp_playlist_track_create_time = dlsym(h, "sp_playlist_track_create_time"))
|
2016-03-12 01:39:26 -05:00
|
|
|
&& (fptr_sp_playlist_owner = dlsym(h, "sp_playlist_owner"))
|
2014-03-11 18:20:29 -04:00
|
|
|
&& (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"))
|
2014-04-03 14:44:36 -04:00
|
|
|
&& (fptr_sp_track_get_availability = dlsym(h, "sp_track_get_availability"))
|
2015-03-07 12:30:32 -05:00
|
|
|
&& (fptr_sp_track_is_starred = dlsym(h, "sp_track_is_starred"))
|
2014-03-11 18:20:29 -04:00
|
|
|
&& (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"))
|
2014-04-04 17:14:43 -04:00
|
|
|
&& (fptr_sp_album_cover = dlsym(h, "sp_album_cover"))
|
2014-03-11 18:20:29 -04:00
|
|
|
&& (fptr_sp_artist_name = dlsym(h, "sp_artist_name"))
|
2014-04-04 17:14:43 -04:00
|
|
|
&& (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"))
|
2016-01-07 16:25:25 -05:00
|
|
|
&& (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"))
|
2016-03-12 01:39:26 -05:00
|
|
|
&& (fptr_sp_user_display_name = dlsym(h, "sp_user_display_name"))
|
|
|
|
&& (fptr_sp_user_canonical_name = dlsym(h, "sp_user_canonical_name"))
|
2014-03-11 18:20:29 -04:00
|
|
|
;
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
/* ------------------------------- 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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
/* -------------------------- PLAYLIST HELPERS ------------------------- */
|
2014-03-23 16:45:06 -04:00
|
|
|
/* Should only be called from within the spotify thread */
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
static int
|
2015-05-29 13:47:53 -04:00
|
|
|
spotify_metadata_get(sp_track *track, struct media_file_info *mfi, const char *pltitle, int time_added)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2015-03-08 04:04:26 -04:00
|
|
|
cfg_t *spotify_cfg;
|
|
|
|
bool artist_override;
|
2015-03-08 04:24:23 -04:00
|
|
|
bool album_override;
|
2014-03-11 18:20:29 -04:00
|
|
|
sp_album *album;
|
|
|
|
sp_artist *artist;
|
|
|
|
sp_albumtype albumtype;
|
2015-03-07 12:30:32 -05:00
|
|
|
bool starred;
|
2015-04-01 17:08:34 -04:00
|
|
|
int compilation;
|
2015-03-08 04:24:23 -04:00
|
|
|
char *albumname;
|
2015-03-08 04:04:26 -04:00
|
|
|
|
|
|
|
spotify_cfg = cfg_getsec(cfg, "spotify");
|
|
|
|
artist_override = cfg_getbool(spotify_cfg, "artist_override");
|
2015-03-08 04:24:23 -04:00
|
|
|
album_override = cfg_getbool(spotify_cfg, "album_override");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
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);
|
2015-03-07 12:30:32 -05:00
|
|
|
starred = fptr_sp_track_is_starred(g_sess, track);
|
|
|
|
|
2015-03-08 04:04:26 -04:00
|
|
|
/*
|
|
|
|
* 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)
|
2016-03-05 06:55:34 -05:00
|
|
|
|| artist_override);
|
2015-03-08 04:04:26 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
if (album_override && pltitle)
|
2015-03-08 04:24:23 -04:00
|
|
|
albumname = strdup(pltitle);
|
|
|
|
else
|
|
|
|
albumname = strdup(fptr_sp_album_name(album));
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
mfi->title = strdup(fptr_sp_track_name(track));
|
2015-03-08 04:24:23 -04:00
|
|
|
mfi->album = albumname;
|
2014-03-11 18:20:29 -04:00
|
|
|
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);
|
2015-03-08 04:04:26 -04:00
|
|
|
mfi->compilation = compilation;
|
2014-03-11 18:20:29 -04:00
|
|
|
mfi->artwork = ARTWORK_SPOTIFY;
|
|
|
|
mfi->type = strdup("spotify");
|
|
|
|
mfi->codectype = strdup("wav");
|
|
|
|
mfi->description = strdup("Spotify audio");
|
2015-05-29 13:47:53 -04:00
|
|
|
mfi->time_added = time_added;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2015-03-07 12:30:32 -05:00
|
|
|
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);
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2015-05-29 13:47:53 -04:00
|
|
|
spotify_track_save(int plid, sp_track *track, const char *pltitle, int time_added)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
|
|
|
struct media_file_info mfi;
|
|
|
|
sp_link *link;
|
|
|
|
char url[1024];
|
|
|
|
int ret;
|
2015-12-27 01:16:50 -05:00
|
|
|
int dir_id;
|
|
|
|
char virtual_path[PATH_MAX];
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
if (!fptr_sp_track_is_loaded(track))
|
|
|
|
{
|
2015-05-29 13:47:53 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Track appears to no longer have the proper status\n");
|
|
|
|
return -1;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2014-04-03 14:44:36 -04:00
|
|
|
if (fptr_sp_track_get_availability(g_sess, track) != SP_TRACK_AVAILABILITY_AVAILABLE)
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Track not available for playback: '%s'\n", fptr_sp_track_name(track));
|
2014-04-03 14:44:36 -04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
link = fptr_sp_link_create_from_track(track, 0);
|
|
|
|
if (!link)
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Could not create link for track: '%s'\n", fptr_sp_track_name(track));
|
2014-03-11 18:20:29 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = fptr_sp_link_as_string(link, url, sizeof(url));
|
|
|
|
if (ret == sizeof(url))
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Spotify link truncated: '%s'\n", url);
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
fptr_sp_link_release(link);
|
|
|
|
|
|
|
|
/* Add to playlistitems table */
|
2016-11-05 08:44:23 -04:00
|
|
|
if (plid)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2016-11-05 08:44:23 -04:00
|
|
|
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;
|
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
memset(&mfi, 0, sizeof(struct media_file_info));
|
|
|
|
|
2015-05-29 13:47:53 -04:00
|
|
|
ret = spotify_metadata_get(track, &mfi, pltitle, time_added);
|
2014-03-11 18:20:29 -04:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Metadata missing (but track should be loaded?): '%s'\n", fptr_sp_track_name(track));
|
2014-03-11 18:20:29 -04:00
|
|
|
free_mfi(&mfi, 1);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-01-24 12:59:05 -05:00
|
|
|
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;
|
|
|
|
}
|
2016-01-24 12:45:28 -05:00
|
|
|
dir_id = db_directory_addorupdate(virtual_path, 0, DIR_SPOTIFY);
|
2015-12-27 01:16:50 -05:00
|
|
|
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;
|
|
|
|
}
|
2016-01-24 12:59:05 -05:00
|
|
|
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;
|
|
|
|
}
|
2015-12-27 01:16:50 -05:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
// DPRINTF(E_DBG, L_SPOTIFY, "Saving track '%s': '%s' by %s (%s)\n", url, mfi.title, mfi.artist, mfi.album);
|
|
|
|
|
2015-12-27 01:16:50 -05:00
|
|
|
filescanner_process_media(url, time(NULL), 0, F_SCAN_TYPE_SPOTIFY, &mfi, dir_id);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
free_mfi(&mfi, 1);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-01-16 11:02:03 -05:00
|
|
|
static int
|
2016-11-13 14:28:29 -05:00
|
|
|
spotify_cleanup_files(void)
|
2016-01-16 11:02:03 -05:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
static int
|
|
|
|
spotify_playlist_save(sp_playlist *pl)
|
|
|
|
{
|
|
|
|
struct playlist_info *pli;
|
|
|
|
sp_track *track;
|
|
|
|
sp_link *link;
|
2016-03-12 01:39:26 -05:00
|
|
|
sp_user *owner;
|
2014-03-11 18:20:29 -04:00
|
|
|
char url[1024];
|
|
|
|
const char *name;
|
2016-03-12 01:39:26 -05:00
|
|
|
const char *ownername;
|
2014-03-11 18:20:29 -04:00
|
|
|
int plid;
|
|
|
|
int num_tracks;
|
2014-12-21 14:41:44 -05:00
|
|
|
char virtual_path[PATH_MAX];
|
2015-12-27 01:16:50 -05:00
|
|
|
int created;
|
2014-03-11 18:20:29 -04:00
|
|
|
int ret;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!fptr_sp_playlist_is_loaded(pl))
|
|
|
|
{
|
2015-05-29 13:47:53 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Playlist still not loaded - will wait for next callback\n");
|
2014-03-11 18:20:29 -04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
name = fptr_sp_playlist_name(pl);
|
2015-05-29 13:47:53 -04:00
|
|
|
num_tracks = fptr_sp_playlist_num_tracks(pl);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2015-01-19 15:32:24 -05:00
|
|
|
// The starred playlist has an empty name, set it manually to "Starred"
|
|
|
|
if (*name == '\0')
|
|
|
|
name = "Starred";
|
|
|
|
|
2015-05-29 13:47:53 -04:00
|
|
|
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);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
// Save playlist (playlists table)
|
2014-03-11 18:20:29 -04:00
|
|
|
link = fptr_sp_link_create_from_playlist(pl);
|
|
|
|
if (!link)
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Could not create link for playlist (wait): '%s'\n", name);
|
2014-03-11 18:20:29 -04:00
|
|
|
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);
|
|
|
|
|
2016-03-12 01:39:26 -05:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2015-03-08 04:05:21 -04:00
|
|
|
|
2016-03-12 01:39:26 -05:00
|
|
|
pli = db_pl_fetch_bypath(url);
|
2014-12-21 14:41:44 -05:00
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
if (pli)
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Playlist found ('%s', link %s), updating\n", name, url);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
plid = pli->id;
|
|
|
|
|
2015-03-14 17:34:03 -04:00
|
|
|
free(pli->title);
|
2015-01-19 15:32:24 -05:00
|
|
|
pli->title = strdup(name);
|
2015-03-14 17:34:03 -04:00
|
|
|
free(pli->virtual_path);
|
|
|
|
pli->virtual_path = strdup(virtual_path);
|
2016-03-12 01:39:26 -05:00
|
|
|
pli->directory_id = DIR_SPOTIFY;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2015-03-14 17:34:03 -04:00
|
|
|
ret = db_pl_update(pli);
|
2014-03-11 18:20:29 -04:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Error updating playlist ('%s', link %s)\n", name, url);
|
2015-03-14 17:34:03 -04:00
|
|
|
|
|
|
|
free_pli(pli, 0);
|
2014-03-11 18:20:29 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
db_pl_clear_items(plid);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Adding playlist ('%s', link %s)\n", name, url);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2015-03-14 17:34:03 -04:00
|
|
|
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));
|
2015-01-19 15:32:24 -05:00
|
|
|
|
2015-04-21 13:12:54 -04:00
|
|
|
pli->type = PL_PLAIN;
|
2015-01-19 15:32:24 -05:00
|
|
|
pli->title = strdup(name);
|
2015-03-14 17:34:03 -04:00
|
|
|
pli->path = strdup(url);
|
|
|
|
pli->virtual_path = strdup(virtual_path);
|
2016-11-13 14:28:29 -05:00
|
|
|
pli->parent_id = spotify_base_plid;
|
2016-01-24 12:45:28 -05:00
|
|
|
pli->directory_id = DIR_SPOTIFY;
|
2015-03-14 17:34:03 -04:00
|
|
|
|
|
|
|
ret = db_pl_add(pli, &plid);
|
2014-03-11 18:20:29 -04:00
|
|
|
if ((ret < 0) || (plid < 1))
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Error adding playlist ('%s', link %s, ret %d, plid %d)\n", name, url, ret, plid);
|
2015-03-14 17:34:03 -04:00
|
|
|
|
|
|
|
free_pli(pli, 0);
|
2014-03-11 18:20:29 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-14 17:34:03 -04:00
|
|
|
free_pli(pli, 0);
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
// Save tracks and playlistitems (files and playlistitems table)
|
2015-05-29 13:47:53 -04:00
|
|
|
db_transaction_begin();
|
2014-03-11 18:20:29 -04:00
|
|
|
for (i = 0; i < num_tracks; i++)
|
|
|
|
{
|
|
|
|
track = fptr_sp_playlist_track(pl, i);
|
|
|
|
if (!track)
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Track %d in playlist '%s' (id %d) is invalid\n", i, name, plid);
|
2014-03-11 18:20:29 -04:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-12-27 01:16:50 -05:00
|
|
|
created = fptr_sp_playlist_track_create_time(pl, i);
|
2015-05-29 13:47:53 -04:00
|
|
|
|
2015-12-27 01:16:50 -05:00
|
|
|
ret = spotify_track_save(plid, track, name, created);
|
2014-03-11 18:20:29 -04:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
2014-06-10 16:49:44 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Error saving track %d to playlist '%s' (id %d)\n", i, name, plid);
|
2014-03-11 18:20:29 -04:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2016-01-16 11:02:03 -05:00
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
spotify_cleanup_files();
|
2015-05-29 13:47:53 -04:00
|
|
|
db_transaction_end();
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
return plid;
|
|
|
|
}
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
// 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)
|
|
|
|
{
|
2016-11-13 14:28:29 -05:00
|
|
|
struct playlist_info pli;
|
2016-11-05 08:44:23 -04:00
|
|
|
struct pending_metadata *pm;
|
|
|
|
sp_link *link;
|
|
|
|
sp_track *track;
|
2016-11-13 14:28:29 -05:00
|
|
|
int ret;
|
2016-11-05 08:44:23 -04:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
// 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";
|
2016-12-03 08:50:20 -05:00
|
|
|
pli.virtual_path = "/spotify:/Spotify Saved";
|
2016-11-13 14:28:29 -05:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
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);
|
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
while ((pm = spotify_pending_metadata))
|
2016-11-05 08:44:23 -04:00
|
|
|
{
|
|
|
|
spotify_track_save(0, pm->track, NULL, time(NULL));
|
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
// 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;
|
2016-11-05 08:44:23 -04:00
|
|
|
free(pm);
|
|
|
|
}
|
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
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;
|
2016-11-05 08:44:23 -04:00
|
|
|
|
|
|
|
return COMMAND_END;
|
|
|
|
}
|
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
static enum command_state
|
|
|
|
spotify_cleanup_wrapper(void *arg, int *retval)
|
|
|
|
{
|
|
|
|
*retval = spotify_cleanup_files();
|
|
|
|
|
|
|
|
return COMMAND_END;
|
|
|
|
}
|
2016-11-05 08:44:23 -04:00
|
|
|
|
|
|
|
/*--------------------- 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
|
2016-11-13 14:28:29 -05:00
|
|
|
saved_tracks_get(int *total, const char **err, const char *uri)
|
2016-11-05 08:44:23 -04:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
/* -------------------------- 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)
|
|
|
|
{
|
2014-03-18 04:57:49 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Playlist update (status %d): %s\n", done, fptr_sp_playlist_name(pl));
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
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);
|
2016-11-13 14:28:29 -05:00
|
|
|
|
|
|
|
spotify_cleanup_files();
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* --------------------- INTERNAL PLAYBACK AND AUDIO ----------------------- */
|
|
|
|
/* Should only be called from within the spotify thread */
|
|
|
|
|
2016-01-07 16:25:25 -05:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
static void
|
2014-03-23 16:45:06 -04:00
|
|
|
audio_fifo_flush(void)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
audio_fifo_data_t *afd;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Flushing audio fifo\n");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&g_audio_fifo->mutex));
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
while((afd = TAILQ_FIRST(&g_audio_fifo->q))) {
|
|
|
|
TAILQ_REMOVE(&g_audio_fifo->q, afd, link);
|
|
|
|
free(afd);
|
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
g_audio_fifo->qlen = 0;
|
|
|
|
g_audio_fifo->fullcount = 0;
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&g_audio_fifo->mutex));
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
static enum command_state
|
|
|
|
playback_setup(void *arg, int *retval)
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
2016-05-08 03:54:58 -04:00
|
|
|
sp_link *link;
|
2014-03-23 16:45:06 -04:00
|
|
|
sp_track *track;
|
|
|
|
sp_error err;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2015-09-12 01:45:31 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Setting up for playback\n");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
link = (sp_link *) arg;
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
if (SP_CONNECTION_STATE_LOGGED_IN != fptr_sp_session_connectionstate(g_sess))
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Can't play music, not connected and logged in to Spotify\n");
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
if (!link)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Playback setup failed, no Spotify link\n");
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
track = fptr_sp_link_as_track(link);
|
2014-03-23 16:45:06 -04:00
|
|
|
if (!track)
|
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Playback setup failed, invalid Spotify track\n");
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
2016-11-05 08:44:23 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
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));
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
audio_fifo_flush();
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = 0;
|
|
|
|
return COMMAND_END;
|
2015-09-12 01:45:31 -04:00
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
static enum command_state
|
|
|
|
playback_play(void *arg, int *retval)
|
2015-09-12 01:45:31 -04:00
|
|
|
{
|
|
|
|
sp_error err;
|
|
|
|
|
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Starting playback\n");
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
err = fptr_sp_session_player_play(g_sess, 1);
|
|
|
|
if (SP_ERROR_OK != err)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Playback failed: %s\n", fptr_sp_error_message(err));
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
g_state = SPOTIFY_STATE_PLAYING;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = 0;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
static enum command_state
|
|
|
|
playback_pause(void *arg, int *retval)
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
|
|
|
sp_error err;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Pausing playback\n");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
err = fptr_sp_session_player_play(g_sess, 0);
|
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Playback paused\n");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
if (SP_ERROR_OK != err)
|
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Playback pause failed: %s\n", fptr_sp_error_message(err));
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
g_state = SPOTIFY_STATE_PAUSED;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = 0;
|
|
|
|
return COMMAND_END;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
static enum command_state
|
|
|
|
playback_stop(void *arg, int *retval)
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
|
|
|
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));
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
g_state = SPOTIFY_STATE_STOPPED;
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
|
|
|
|
*retval = 0;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
static enum command_state
|
|
|
|
playback_seek(void *arg, int *retval)
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
2016-05-08 03:54:58 -04:00
|
|
|
int seek_ms;
|
2014-03-23 16:45:06 -04:00
|
|
|
sp_error err;
|
|
|
|
|
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Playback seek\n");
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
seek_ms = *((int *) arg);
|
|
|
|
|
|
|
|
err = fptr_sp_session_player_seek(g_sess, seek_ms);
|
2014-03-23 16:45:06 -04:00
|
|
|
if (SP_ERROR_OK != err)
|
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Could not seek: %s\n", fptr_sp_error_message(err));
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
audio_fifo_flush();
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = 0;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
static enum command_state
|
|
|
|
playback_eot(void *arg, int *retval)
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
|
|
|
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));
|
2016-08-06 00:26:38 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
g_state = SPOTIFY_STATE_STOPPING;
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = 0;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
static enum command_state
|
|
|
|
audio_get(void *arg, int *retval)
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
2016-05-08 03:54:58 -04:00
|
|
|
struct audio_get_param *audio;
|
2014-03-23 16:45:06 -04:00
|
|
|
struct timespec ts;
|
|
|
|
audio_fifo_data_t *afd;
|
|
|
|
int processed;
|
2014-11-11 05:59:40 -05:00
|
|
|
int timeout;
|
2014-03-23 16:45:06 -04:00
|
|
|
int ret;
|
2017-01-21 10:11:20 -05:00
|
|
|
int err;
|
2014-03-23 16:45:06 -04:00
|
|
|
int s;
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
audio = (struct audio_get_param *) arg;
|
2014-03-23 16:45:06 -04:00
|
|
|
afd = NULL;
|
|
|
|
processed = 0;
|
|
|
|
|
|
|
|
// If spotify was paused begin by resuming playback
|
|
|
|
if (g_state == SPOTIFY_STATE_PAUSED)
|
2016-05-08 03:54:58 -04:00
|
|
|
playback_play(NULL, retval);
|
2014-03-23 16:45:06 -04:00
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&g_audio_fifo->mutex));
|
2014-03-23 16:45:06 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
while ((processed < audio->wanted) && (g_state != SPOTIFY_STATE_STOPPED))
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
|
|
|
// 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)
|
2014-11-11 05:59:40 -05:00
|
|
|
timeout = 0;
|
2014-03-23 16:45:06 -04:00
|
|
|
while ( !(afd = TAILQ_FIRST(&g_audio_fifo->q)) &&
|
|
|
|
(g_state != SPOTIFY_STATE_STOPPED) &&
|
2014-11-11 05:59:40 -05:00
|
|
|
(g_state != SPOTIFY_STATE_STOPPING) &&
|
|
|
|
(timeout < SPOTIFY_TIMEOUT) )
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Waiting for audio\n");
|
2014-11-11 05:59:40 -05:00
|
|
|
timeout += 5;
|
2016-01-07 16:25:25 -05:00
|
|
|
mk_reltime(&ts, 5);
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR_EXCEPT(L_SPOTIFY, pthread_cond_timedwait(&g_audio_fifo->cond, &g_audio_fifo->mutex, &ts), err, ETIMEDOUT);
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
|
|
|
|
2014-11-11 05:59:40 -05:00
|
|
|
if ((!afd) && (timeout >= SPOTIFY_TIMEOUT))
|
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Timeout waiting for audio (waited %d sec)\n", timeout);
|
|
|
|
|
|
|
|
spotify_playback_stop_nonblock();
|
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
if (!afd)
|
|
|
|
break;
|
|
|
|
|
|
|
|
TAILQ_REMOVE(&g_audio_fifo->q, afd, link);
|
|
|
|
g_audio_fifo->qlen -= afd->nsamples;
|
|
|
|
|
|
|
|
s = afd->nsamples * sizeof(int16_t) * 2;
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
ret = evbuffer_add(audio->evbuf, afd->samples, s);
|
2014-03-23 16:45:06 -04:00
|
|
|
free(afd);
|
|
|
|
afd = NULL;
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Out of memory for evbuffer (tried to add %d bytes)\n", s);
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&g_audio_fifo->mutex));
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
processed += s;
|
|
|
|
}
|
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&g_audio_fifo->mutex));
|
2014-03-23 16:45:06 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
|
|
|
|
*retval = processed;
|
|
|
|
return COMMAND_END;
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
|
|
|
|
2016-01-07 16:25:25 -05:00
|
|
|
static void
|
|
|
|
artwork_loaded_cb(sp_image *image, void *userdata)
|
|
|
|
{
|
2016-05-08 03:54:58 -04:00
|
|
|
struct artwork_get_param *artwork;
|
2017-01-13 17:32:59 -05:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
artwork = userdata;
|
2017-01-13 17:32:59 -05:00
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&artwork->mutex));
|
2016-01-08 18:36:30 -05:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
artwork->is_loaded = 1;
|
2016-01-08 18:36:30 -05:00
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_signal(&artwork->cond));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&artwork->mutex));
|
2016-01-08 18:36:30 -05:00
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
static enum command_state
|
|
|
|
artwork_get_bh(void *arg, int *retval)
|
2016-01-08 18:36:30 -05:00
|
|
|
{
|
2016-05-08 03:54:58 -04:00
|
|
|
struct artwork_get_param *artwork;
|
2016-01-08 18:36:30 -05:00
|
|
|
sp_imageformat imageformat;
|
2016-01-07 16:25:25 -05:00
|
|
|
sp_error err;
|
2016-01-08 18:36:30 -05:00
|
|
|
const void *data;
|
|
|
|
size_t data_size;
|
|
|
|
int ret;
|
2016-01-07 16:25:25 -05:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
artwork = arg;
|
|
|
|
sp_image *image = artwork->image;
|
|
|
|
char *path = artwork->path;
|
2016-01-07 16:25:25 -05:00
|
|
|
|
|
|
|
err = fptr_sp_image_error(image);
|
|
|
|
if (err != SP_ERROR_OK)
|
|
|
|
{
|
2016-01-08 18:36:30 -05:00
|
|
|
DPRINTF(E_WARN, L_SPOTIFY, "Getting artwork (%s) failed, Spotify error: %s\n", path, fptr_sp_error_message(err));
|
|
|
|
goto fail;
|
2016-01-07 16:25:25 -05:00
|
|
|
}
|
|
|
|
|
2016-01-08 18:36:30 -05:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
ret = evbuffer_expand(artwork->evbuf, data_size);
|
2016-01-08 18:36:30 -05:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Out of memory for artwork\n");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
ret = evbuffer_add(artwork->evbuf, data, data_size);
|
2016-01-08 18:36:30 -05:00
|
|
|
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);
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = 0;
|
|
|
|
return COMMAND_END;
|
2016-01-08 18:36:30 -05:00
|
|
|
|
|
|
|
fail:
|
|
|
|
fptr_sp_image_release(image);
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2016-01-07 16:25:25 -05:00
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
static enum command_state
|
|
|
|
artwork_get(void *arg, int *retval)
|
2014-04-04 17:14:43 -04:00
|
|
|
{
|
2016-05-08 03:54:58 -04:00
|
|
|
struct artwork_get_param *artwork;
|
2014-04-04 17:14:43 -04:00
|
|
|
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;
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
artwork = arg;
|
|
|
|
path = artwork->path;
|
2014-04-04 17:14:43 -04:00
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
2014-11-11 05:37:03 -05:00
|
|
|
// Get an image at least the same size as requested
|
2014-04-04 17:14:43 -04:00
|
|
|
image_size = SP_IMAGE_SIZE_SMALL; // 64x64
|
2016-05-08 03:54:58 -04:00
|
|
|
if ((artwork->max_w > 64) || (artwork->max_h > 64))
|
2014-04-04 17:14:43 -04:00
|
|
|
image_size = SP_IMAGE_SIZE_NORMAL; // 300x300
|
2016-05-08 03:54:58 -04:00
|
|
|
if ((artwork->max_w > 300) || (artwork->max_h > 300))
|
2014-04-04 17:14:43 -04:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2016-01-08 18:36:30 -05:00
|
|
|
fptr_sp_link_release(link);
|
2014-04-04 17:14:43 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
artwork->image = image;
|
|
|
|
artwork->is_loaded = fptr_sp_image_is_loaded(image);
|
2014-04-04 17:14:43 -04:00
|
|
|
|
2016-01-08 18:36:30 -05:00
|
|
|
/* 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
|
|
|
|
*/
|
2016-05-08 03:54:58 -04:00
|
|
|
if (artwork->is_loaded)
|
|
|
|
return artwork_get_bh(artwork, retval);
|
2014-04-04 17:14:43 -04:00
|
|
|
|
2016-01-08 18:36:30 -05:00
|
|
|
DPRINTF(E_SPAM, L_SPOTIFY, "Will wait for Spotify to call artwork_loaded_cb\n");
|
2014-04-04 17:14:43 -04:00
|
|
|
|
2016-01-08 18:36:30 -05:00
|
|
|
/* Async - we will return to spotify_artwork_get which will wait for callback */
|
2016-05-08 03:54:58 -04:00
|
|
|
err = fptr_sp_image_add_load_callback(image, artwork_loaded_cb, artwork);
|
2016-01-08 18:36:30 -05:00
|
|
|
if (err != SP_ERROR_OK)
|
2014-04-04 17:14:43 -04:00
|
|
|
{
|
2016-01-08 18:36:30 -05:00
|
|
|
DPRINTF(E_WARN, L_SPOTIFY, "Adding artwork cb failed, Spotify error: %s\n", fptr_sp_error_message(err));
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-04-04 17:14:43 -04:00
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = 0;
|
|
|
|
return COMMAND_END;
|
2014-04-04 17:14:43 -04:00
|
|
|
|
|
|
|
level2_exit:
|
|
|
|
fptr_sp_link_release(link);
|
|
|
|
|
|
|
|
level1_exit:
|
2016-05-08 03:54:58 -04:00
|
|
|
*retval = -1;
|
|
|
|
return COMMAND_END;
|
2014-04-04 17:14:43 -04:00
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
|
|
|
|
/* --------------------------- 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)
|
|
|
|
{
|
2015-03-29 18:59:55 -04:00
|
|
|
cfg_t *spotify_cfg;
|
2014-03-23 16:45:06 -04:00
|
|
|
sp_playlist *pl;
|
|
|
|
sp_playlistcontainer *pc;
|
2015-01-19 15:32:24 -05:00
|
|
|
struct playlist_info pli;
|
|
|
|
int ret;
|
2014-03-23 16:45:06 -04:00
|
|
|
int i;
|
|
|
|
|
|
|
|
if (SP_ERROR_OK != error)
|
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Login failed: %s\n", fptr_sp_error_message(error));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Login to Spotify succeeded, reloading playlists\n");
|
2014-03-23 16:45:06 -04:00
|
|
|
|
2016-01-30 01:27:20 -05:00
|
|
|
db_directory_enable_bypath("/spotify:");
|
|
|
|
|
2015-03-07 13:31:24 -05:00
|
|
|
pl = fptr_sp_session_starred_create(sess);
|
|
|
|
fptr_sp_playlist_add_callbacks(pl, &pl_callbacks, NULL);
|
|
|
|
|
2015-03-29 18:59:55 -04:00
|
|
|
spotify_cfg = cfg_getsec(cfg, "spotify");
|
|
|
|
if (! cfg_getbool(spotify_cfg, "base_playlist_disable"))
|
2015-01-19 15:32:24 -05:00
|
|
|
{
|
2015-03-29 18:59:55 -04:00
|
|
|
memset(&pli, 0, sizeof(struct playlist_info));
|
|
|
|
pli.title = "Spotify";
|
|
|
|
pli.type = PL_FOLDER;
|
|
|
|
pli.path = "spotify:playlistfolder";
|
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
ret = db_pl_add(&pli, &spotify_base_plid);
|
2015-03-29 18:59:55 -04:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Error adding base playlist\n");
|
|
|
|
return;
|
|
|
|
}
|
2015-01-19 15:32:24 -05:00
|
|
|
}
|
2015-03-29 18:59:55 -04:00
|
|
|
else
|
2016-11-13 14:28:29 -05:00
|
|
|
spotify_base_plid = 0;
|
2015-01-19 15:32:24 -05:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-28 15:08:52 -05:00
|
|
|
/**
|
|
|
|
* 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");
|
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&login_lck));
|
2014-12-28 15:08:52 -05:00
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_signal(&login_cond));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&login_lck));
|
2014-12-28 15:08:52 -05:00
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/**
|
|
|
|
* 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)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
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
|
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&g_audio_fifo->mutex));
|
2014-03-23 16:45:06 -04:00
|
|
|
|
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&g_audio_fifo->mutex));
|
2014-03-23 16:45:06 -04:00
|
|
|
|
|
|
|
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;
|
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_signal(&g_audio_fifo->cond));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&g_audio_fifo->mutex));
|
2014-03-23 16:45:06 -04:00
|
|
|
|
|
|
|
return num_frames;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-03-23 16:45:06 -04:00
|
|
|
* This callback is called from an internal libspotify thread to ask us to
|
|
|
|
* reiterate the main loop. This must not block.
|
2014-03-11 18:20:29 -04:00
|
|
|
*
|
2014-03-23 16:45:06 -04:00
|
|
|
* @sa sp_session_callbacks#notify_main_thread
|
2014-03-11 18:20:29 -04:00
|
|
|
*/
|
2014-03-23 16:45:06 -04:00
|
|
|
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");
|
2016-11-05 08:44:23 -04:00
|
|
|
|
|
|
|
commands_exec_async(cmdbase, spotify_pending_process, NULL);
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Misc connection error callbacks */
|
2014-03-11 18:20:29 -04:00
|
|
|
static void play_token_lost(sp_session *sess)
|
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
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)
|
|
|
|
{
|
2016-11-13 14:28:29 -05:00
|
|
|
struct reload_list *reload;
|
|
|
|
int ret;
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
if (SP_CONNECTION_STATE_LOGGED_IN == fptr_sp_session_connectionstate(session))
|
|
|
|
{
|
2016-11-13 14:28:29 -05:00
|
|
|
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);
|
|
|
|
}
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
2014-06-22 16:27:52 -04:00
|
|
|
else if (g_state == SPOTIFY_STATE_PLAYING)
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Music interrupted - connection error or logged out\n");
|
|
|
|
spotify_playback_stop_nonblock();
|
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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)
|
|
|
|
{
|
2014-03-18 04:57:49 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "End of track\n");
|
2014-03-23 16:45:06 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
commands_exec_async(cmdbase, playback_eot, NULL);
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The session callbacks
|
|
|
|
*/
|
|
|
|
static sp_session_callbacks session_callbacks = {
|
|
|
|
.logged_in = &logged_in,
|
2014-12-28 15:08:52 -05:00
|
|
|
.logged_out = &logged_out,
|
2014-03-23 16:45:06 -04:00
|
|
|
.connectionstate_updated = &connectionstate_updated,
|
2014-03-11 18:20:29 -04:00
|
|
|
.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,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* ------------------------------- MAIN LOOP ------------------------------- */
|
|
|
|
/* Thread: spotify */
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
static void *
|
|
|
|
spotify(void *arg)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
int ret;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Main loop initiating\n");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
ret = db_perthread_init();
|
|
|
|
if (ret < 0)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Error: DB init failed\n");
|
|
|
|
pthread_exit(NULL);
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
g_state = SPOTIFY_STATE_WAIT;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
event_base_dispatch(evbase_spotify);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
if (g_state != SPOTIFY_STATE_INACTIVE)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Spotify event loop terminated ahead of time!\n");
|
|
|
|
g_state = SPOTIFY_STATE_INACTIVE;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
db_perthread_deinit();
|
|
|
|
|
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Main loop terminating\n");
|
|
|
|
|
|
|
|
pthread_exit(NULL);
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
static void
|
2016-05-23 00:41:43 -04:00
|
|
|
exit_cb()
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
fptr_sp_session_player_unload(g_sess);
|
|
|
|
fptr_sp_session_logout(g_sess);
|
2016-06-11 04:00:23 -04:00
|
|
|
g_state = SPOTIFY_STATE_INACTIVE;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* 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;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
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");
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
do
|
|
|
|
{
|
|
|
|
fptr_sp_session_process_events(g_sess, &next_timeout);
|
|
|
|
}
|
|
|
|
while (next_timeout == 0);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
tv.tv_sec = next_timeout / 1000;
|
|
|
|
tv.tv_usec = (next_timeout % 1000) * 1000;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
event_add(g_notifyev, &tv);
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* ---------------------------- Our Spotify API --------------------------- */
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
/* Thread: player */
|
|
|
|
int
|
2016-10-26 13:37:41 -04:00
|
|
|
spotify_playback_setup(const char *path)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
|
|
|
sp_link *link;
|
|
|
|
|
2015-09-12 01:45:31 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Playback setup request\n");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-10-26 13:37:41 -04:00
|
|
|
link = fptr_sp_link_create_from_string(path);
|
2014-03-11 18:20:29 -04:00
|
|
|
if (!link)
|
|
|
|
{
|
2016-10-26 13:37:41 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Playback setup failed, invalid Spotify link: %s\n", path);
|
2014-03-11 18:20:29 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
return commands_exec_sync(cmdbase, playback_setup, NULL, link);
|
2014-03-23 16:45:06 -04:00
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2015-09-12 01:45:31 -04:00
|
|
|
int
|
|
|
|
spotify_playback_play()
|
|
|
|
{
|
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Playback request\n");
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
return commands_exec_sync(cmdbase, playback_play, NULL, NULL);
|
2015-09-12 01:45:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
spotify_playback_pause()
|
|
|
|
{
|
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Pause request\n");
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
return commands_exec_sync(cmdbase, playback_pause, NULL, NULL);
|
2015-09-12 01:45:31 -04:00
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* Thread: libspotify */
|
|
|
|
void
|
|
|
|
spotify_playback_pause_nonblock(void)
|
|
|
|
{
|
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Nonblock pause request\n");
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
commands_exec_async(cmdbase, playback_pause, NULL);
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* Thread: player and libspotify */
|
2014-03-11 18:20:29 -04:00
|
|
|
int
|
|
|
|
spotify_playback_stop(void)
|
|
|
|
{
|
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Stop request\n");
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
return commands_exec_sync(cmdbase, playback_stop, NULL, NULL);
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2014-11-11 05:59:40 -05:00
|
|
|
/* Thread: player and libspotify */
|
2014-03-23 16:45:06 -04:00
|
|
|
void
|
|
|
|
spotify_playback_stop_nonblock(void)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2014-03-23 16:45:06 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Nonblock stop request\n");
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
commands_exec_async(cmdbase, playback_stop, NULL);
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Thread: player */
|
|
|
|
int
|
2014-03-23 16:45:06 -04:00
|
|
|
spotify_playback_seek(int ms)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
ret = commands_exec_sync(cmdbase, playback_seek, NULL, &ms);
|
2014-03-17 17:25:47 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
if (ret == 0)
|
|
|
|
return ms;
|
|
|
|
else
|
|
|
|
return -1;
|
|
|
|
}
|
2014-03-17 17:25:47 -04:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* Thread: player */
|
|
|
|
int
|
|
|
|
spotify_audio_get(struct evbuffer *evbuf, int wanted)
|
|
|
|
{
|
2016-05-08 03:54:58 -04:00
|
|
|
struct audio_get_param audio;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
audio.evbuf = evbuf;
|
|
|
|
audio.wanted = wanted;
|
2014-04-04 17:14:43 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
return commands_exec_sync(cmdbase, audio_get, NULL, &audio);
|
2014-04-04 17:14:43 -04:00
|
|
|
}
|
|
|
|
|
2016-01-08 18:36:30 -05:00
|
|
|
/* Thread: httpd (artwork) and worker */
|
2014-04-04 17:14:43 -04:00
|
|
|
int
|
|
|
|
spotify_artwork_get(struct evbuffer *evbuf, char *path, int max_w, int max_h)
|
|
|
|
{
|
2016-05-08 03:54:58 -04:00
|
|
|
struct artwork_get_param artwork;
|
2016-01-08 18:36:30 -05:00
|
|
|
struct timespec ts;
|
2014-04-04 17:14:43 -04:00
|
|
|
int ret;
|
2017-01-21 10:11:20 -05:00
|
|
|
int err;
|
2014-04-04 17:14:43 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
artwork.evbuf = evbuf;
|
|
|
|
artwork.path = path;
|
|
|
|
artwork.max_w = max_w;
|
|
|
|
artwork.max_h = max_h;
|
2014-04-04 17:14:43 -04:00
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, mutex_init(&artwork.mutex));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_init(&artwork.cond, NULL));
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
ret = commands_exec_sync(cmdbase, artwork_get, NULL, &artwork);
|
2017-01-13 17:32:59 -05:00
|
|
|
|
2016-01-08 18:36:30 -05:00
|
|
|
// Artwork was not ready, wait for callback from libspotify
|
|
|
|
if (ret == 0)
|
|
|
|
{
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&artwork.mutex));
|
2016-01-08 18:36:30 -05:00
|
|
|
mk_reltime(&ts, SPOTIFY_ARTWORK_TIMEOUT);
|
2016-05-08 03:54:58 -04:00
|
|
|
if (!artwork.is_loaded)
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR_EXCEPT(L_SPOTIFY, pthread_cond_timedwait(&artwork.cond, &artwork.mutex, &ts), err, ETIMEDOUT);
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&artwork.mutex));
|
2016-01-08 18:36:30 -05:00
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
ret = commands_exec_sync(cmdbase, artwork_get_bh, NULL, &artwork);
|
2016-01-08 18:36:30 -05:00
|
|
|
}
|
2017-01-13 17:32:59 -05:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
return ret;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
/* Thread: httpd */
|
|
|
|
void
|
|
|
|
spotify_oauth_interface(struct evbuffer *evbuf, const char *redirect_uri)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2016-11-05 08:44:23 -04:00
|
|
|
struct keyval kv;
|
|
|
|
char *param;
|
|
|
|
int ret;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
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)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2016-11-05 08:44:23 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Cannot display Spotify oath interface (error adding parameters to keyval)\n");
|
|
|
|
goto out_clear_kv;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
param = http_form_urlencode(&kv);
|
|
|
|
if (!param)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2016-11-05 08:44:23 -04:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Cannot display Spotify oath interface (http_form_uriencode() failed)\n");
|
|
|
|
goto out_clear_kv;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
evbuffer_add_printf(evbuf, "<a href=\"%s/?%s\">Click here to authorize forked-daapd with Spotify</a>\n", spotify_auth_uri, param);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
free(param);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
out_clear_kv:
|
|
|
|
keyval_clear(&kv);
|
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
/* Thread: httpd */
|
|
|
|
void
|
|
|
|
spotify_oauth_callback(struct evbuffer *evbuf, struct evkeyvalq *param, const char *redirect_uri)
|
|
|
|
{
|
|
|
|
const char *code;
|
2017-01-21 10:11:20 -05:00
|
|
|
const char *err = "";
|
2016-11-05 08:44:23 -04:00
|
|
|
int total;
|
|
|
|
int ret;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
code = evhttp_find_header(param, "code");
|
|
|
|
if (!code)
|
|
|
|
{
|
|
|
|
evbuffer_add_printf(evbuf, "Error: Didn't receive a code from Spotify\n");
|
|
|
|
return;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Received OAuth code: %s\n", code);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
evbuffer_add_printf(evbuf, "<p>Requesting access token from Spotify...\n");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
ret = tokens_get(code, redirect_uri, &err);
|
|
|
|
if (ret < 0)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2016-11-05 08:44:23 -04:00
|
|
|
evbuffer_add_printf(evbuf, "failed</p>\n<p>Error: %s</p>\n", err);
|
|
|
|
return;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
commands_exec_sync(cmdbase, spotify_saved_pl_clear_items, NULL, NULL);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
evbuffer_add_printf(evbuf, "ok</p>\n<p>Retrieving saved tracks...\n");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
ret = saved_tracks_get(&total, &err, spotify_tracks_uri);
|
2016-11-05 08:44:23 -04:00
|
|
|
if (ret < 0)
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2016-11-05 08:44:23 -04:00
|
|
|
evbuffer_add_printf(evbuf, "failed</p>\n<p>Error: %s</p>\n", err);
|
|
|
|
return;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
evbuffer_add_printf(evbuf, "ok, got %d out of %d tracks</p>\n", ret, total);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
evbuffer_add_printf(evbuf, "<p>Purging removed tracks/albums...\n");
|
2014-06-10 16:49:44 -04:00
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
// TODO release links to the items we are going to clean up
|
2014-06-10 16:49:44 -04:00
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
commands_exec_sync(cmdbase, spotify_cleanup_wrapper, NULL, NULL);
|
2014-06-10 16:49:44 -04:00
|
|
|
|
2016-11-13 14:28:29 -05:00
|
|
|
evbuffer_add_printf(evbuf, "ok, all done</p>\n");
|
2014-06-10 16:49:44 -04:00
|
|
|
|
2016-11-05 08:44:23 -04:00
|
|
|
return;
|
2014-06-10 16:49:44 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Thread: filescanner */
|
|
|
|
void
|
|
|
|
spotify_login(char *path)
|
|
|
|
{
|
2016-11-13 14:28:29 -05:00
|
|
|
struct playlist_info *pli;
|
2014-12-28 15:08:52 -05:00
|
|
|
sp_error err;
|
2014-06-10 16:49:44 -04:00
|
|
|
char *username;
|
|
|
|
char *password;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!g_sess)
|
|
|
|
{
|
2015-01-28 13:59:38 -05:00
|
|
|
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");
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-12-28 15:08:52 -05:00
|
|
|
if (SP_CONNECTION_STATE_LOGGED_IN == fptr_sp_session_connectionstate(g_sess))
|
2014-03-11 18:20:29 -04:00
|
|
|
{
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_lock(&login_lck));
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-12-28 15:08:52 -05:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Logging out of Spotify (current state is %d)\n", g_state);
|
2014-03-23 16:45:06 -04:00
|
|
|
|
2014-12-28 15:08:52 -05:00
|
|
|
fptr_sp_session_player_unload(g_sess);
|
|
|
|
err = fptr_sp_session_logout(g_sess);
|
|
|
|
|
|
|
|
if (SP_ERROR_OK != err)
|
2014-03-23 16:45:06 -04:00
|
|
|
{
|
2014-12-28 15:08:52 -05:00
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Could not logout of Spotify: %s\n", fptr_sp_error_message(err));
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&login_lck));
|
2014-03-23 16:45:06 -04:00
|
|
|
return;
|
|
|
|
}
|
2014-12-28 15:08:52 -05:00
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_wait(&login_cond, &login_lck));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_unlock(&login_lck));
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
2014-12-28 15:08:52 -05:00
|
|
|
DPRINTF(E_INFO, L_SPOTIFY, "Logging into Spotify\n");
|
2016-11-13 14:28:29 -05:00
|
|
|
|
2014-06-10 16:49:44 -04:00
|
|
|
if (path)
|
|
|
|
{
|
2016-11-13 14:28:29 -05:00
|
|
|
db_spotify_purge();
|
|
|
|
spotify_saved_plid = 0;
|
|
|
|
|
2014-06-10 16:49:44 -04:00
|
|
|
ret = spotify_file_read(path, &username, &password);
|
|
|
|
if (ret < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
err = fptr_sp_session_login(g_sess, username, password, 1, NULL);
|
2014-12-28 15:08:52 -05:00
|
|
|
free(username);
|
|
|
|
free(password);
|
2014-06-10 16:49:44 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-11-13 14:28:29 -05:00
|
|
|
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;
|
|
|
|
|
2014-06-10 16:49:44 -04:00
|
|
|
err = fptr_sp_session_relogin(g_sess);
|
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
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)
|
|
|
|
{
|
2014-03-29 17:26:46 -04:00
|
|
|
cfg_t *spotify_cfg;
|
2014-03-11 18:20:29 -04:00
|
|
|
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");
|
2014-03-23 16:45:06 -04:00
|
|
|
goto libspotify_fail;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = fptr_assign_all();
|
|
|
|
if (ret < 0)
|
2014-03-23 16:45:06 -04:00
|
|
|
goto assign_fail;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-03-17 17:20:16 -04:00
|
|
|
#ifdef HAVE_PIPE2
|
2014-03-23 16:45:06 -04:00
|
|
|
ret = pipe2(g_notify_pipe, O_CLOEXEC);
|
2016-03-17 17:20:16 -04:00
|
|
|
#else
|
|
|
|
ret = pipe(g_notify_pipe);
|
|
|
|
#endif
|
2014-03-23 16:45:06 -04:00
|
|
|
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);
|
|
|
|
|
2016-05-08 03:54:58 -04:00
|
|
|
|
2016-05-23 00:41:43 -04:00
|
|
|
cmdbase = commands_base_new(evbase_spotify, exit_cb);
|
2016-05-08 03:54:58 -04:00
|
|
|
if (!cmdbase)
|
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Could not create command base\n");
|
|
|
|
goto cmd_fail;
|
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
DPRINTF(E_INFO, L_SPOTIFY, "Spotify session init\n");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-29 17:26:46 -04:00
|
|
|
spotify_cfg = cfg_getsec(cfg, "spotify");
|
|
|
|
spconfig.settings_location = cfg_getstr(spotify_cfg, "settings_dir");
|
|
|
|
spconfig.cache_location = cfg_getstr(spotify_cfg, "cache_dir");
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
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));
|
2014-03-23 16:45:06 -04:00
|
|
|
goto session_fail;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
g_sess = sp;
|
2014-03-23 16:45:06 -04:00
|
|
|
g_state = SPOTIFY_STATE_INACTIVE;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-03-29 17:26:46 -04:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* Prepare audio buffer */
|
2014-03-11 18:20:29 -04:00
|
|
|
g_audio_fifo = (audio_fifo_t *)malloc(sizeof(audio_fifo_t));
|
2014-03-23 16:45:06 -04:00
|
|
|
if (!g_audio_fifo)
|
|
|
|
{
|
|
|
|
DPRINTF(E_LOG, L_SPOTIFY, "Out of memory for audio buffer\n");
|
|
|
|
goto audio_fifo_fail;
|
|
|
|
}
|
2014-03-11 18:20:29 -04:00
|
|
|
TAILQ_INIT(&g_audio_fifo->q);
|
|
|
|
g_audio_fifo->qlen = 0;
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, mutex_init(&g_audio_fifo->mutex));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_init(&g_audio_fifo->cond, NULL));
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, mutex_init(&login_lck));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_init(&login_cond, NULL));
|
2014-12-28 15:08:52 -05:00
|
|
|
|
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
|
2016-05-05 13:39:15 -04:00
|
|
|
#if defined(HAVE_PTHREAD_SETNAME_NP)
|
2016-03-05 07:19:55 -05:00
|
|
|
pthread_setname_np(tid_spotify, "spotify");
|
2016-05-05 13:39:15 -04:00
|
|
|
#elif defined(HAVE_PTHREAD_SET_NAME_NP)
|
2016-03-05 07:19:55 -05:00
|
|
|
pthread_set_name_np(tid_spotify, "spotify");
|
|
|
|
#endif
|
|
|
|
|
2014-03-11 18:20:29 -04:00
|
|
|
DPRINTF(E_DBG, L_SPOTIFY, "Spotify init complete\n");
|
|
|
|
return 0;
|
2014-03-23 16:45:06 -04:00
|
|
|
|
2014-12-28 15:08:52 -05:00
|
|
|
thread_fail:
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_destroy(&login_cond));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_destroy(&login_lck));
|
2014-12-28 15:08:52 -05:00
|
|
|
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_destroy(&g_audio_fifo->cond));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_destroy(&g_audio_fifo->mutex));
|
2014-12-28 15:08:52 -05:00
|
|
|
free(g_audio_fifo);
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
audio_fifo_fail:
|
|
|
|
fptr_sp_session_release(g_sess);
|
|
|
|
g_sess = NULL;
|
|
|
|
|
|
|
|
session_fail:
|
2016-05-08 03:54:58 -04:00
|
|
|
cmd_fail:
|
2014-03-23 16:45:06 -04:00
|
|
|
evnew_fail:
|
2016-05-08 03:54:58 -04:00
|
|
|
commands_base_free(cmdbase);
|
2014-03-23 16:45:06 -04:00
|
|
|
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;
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
spotify_deinit(void)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
if (!g_libhandle)
|
2014-03-11 18:20:29 -04:00
|
|
|
return;
|
|
|
|
|
|
|
|
/* Send exit signal to thread (if active) */
|
|
|
|
if (g_state != SPOTIFY_STATE_INACTIVE)
|
|
|
|
{
|
2016-06-11 04:00:23 -04:00
|
|
|
commands_base_destroy(cmdbase);
|
2016-05-23 00:41:43 -04:00
|
|
|
g_state = SPOTIFY_STATE_INACTIVE;
|
2014-03-11 18:20:29 -04:00
|
|
|
|
|
|
|
ret = pthread_join(tid_spotify, NULL);
|
|
|
|
if (ret != 0)
|
|
|
|
{
|
|
|
|
DPRINTF(E_FATAL, L_SPOTIFY, "Could not join Spotify thread: %s\n", strerror(errno));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* Release session */
|
2014-03-11 18:20:29 -04:00
|
|
|
fptr_sp_session_release(g_sess);
|
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* Free event base (should free events too) */
|
|
|
|
event_base_free(evbase_spotify);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2016-06-11 04:00:23 -04:00
|
|
|
/* Close pipes */
|
2014-03-23 16:45:06 -04:00
|
|
|
close(g_notify_pipe[0]);
|
|
|
|
close(g_notify_pipe[1]);
|
2014-03-11 18:20:29 -04:00
|
|
|
|
2014-12-28 15:08:52 -05:00
|
|
|
/* Destroy locks */
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_destroy(&login_cond));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_destroy(&login_lck));
|
2014-12-28 15:08:52 -05:00
|
|
|
|
2014-03-23 16:45:06 -04:00
|
|
|
/* Clear audio fifo */
|
2017-01-21 10:11:20 -05:00
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_cond_destroy(&g_audio_fifo->cond));
|
|
|
|
CHECK_ERR(L_SPOTIFY, pthread_mutex_destroy(&g_audio_fifo->mutex));
|
2014-03-11 18:20:29 -04:00
|
|
|
free(g_audio_fifo);
|
|
|
|
|
|
|
|
/* Release libspotify handle */
|
2014-03-23 16:45:06 -04:00
|
|
|
dlclose(g_libhandle);
|
2014-03-11 18:20:29 -04:00
|
|
|
}
|