mirror of
https://github.com/owntone/owntone-server.git
synced 2025-01-01 01:53:23 -05:00
Merge pull request #1286 from chme/spotify_podcasts
[spotify] Scan spotify podcast subscriptions into library
This commit is contained in:
commit
29cd5a7a19
@ -1232,6 +1232,7 @@ jsonapi_reply_spotify(struct httpd_request *hreq)
|
|||||||
spotify_status_get(&sp_status);
|
spotify_status_get(&sp_status);
|
||||||
json_object_object_add(jreply, "spotify_installed", json_object_new_boolean(sp_status.installed));
|
json_object_object_add(jreply, "spotify_installed", json_object_new_boolean(sp_status.installed));
|
||||||
json_object_object_add(jreply, "spotify_logged_in", json_object_new_boolean(sp_status.logged_in));
|
json_object_object_add(jreply, "spotify_logged_in", json_object_new_boolean(sp_status.logged_in));
|
||||||
|
json_object_object_add(jreply, "has_podcast_support", json_object_new_boolean(sp_status.has_podcast_support));
|
||||||
safe_json_add_string(jreply, "libspotify_user", sp_status.username);
|
safe_json_add_string(jreply, "libspotify_user", sp_status.username);
|
||||||
|
|
||||||
spotifywebapi_status_info_get(&webapi_info);
|
spotifywebapi_status_info_get(&webapi_info);
|
||||||
|
@ -9,6 +9,7 @@ struct spotify_status
|
|||||||
bool installed;
|
bool installed;
|
||||||
bool logged_in;
|
bool logged_in;
|
||||||
char username[128];
|
char username[128];
|
||||||
|
bool has_podcast_support;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct spotify_backend
|
struct spotify_backend
|
||||||
|
@ -738,6 +738,7 @@ status_get(struct spotify_status *status)
|
|||||||
memcpy(status->username, ctx->status.username, sizeof(status->username));
|
memcpy(status->username, ctx->status.username, sizeof(status->username));
|
||||||
status->logged_in = ctx->status.logged_in;
|
status->logged_in = ctx->status.logged_in;
|
||||||
status->installed = true;
|
status->installed = true;
|
||||||
|
status->has_podcast_support = true;
|
||||||
|
|
||||||
pthread_mutex_unlock(&ctx->lock);
|
pthread_mutex_unlock(&ctx->lock);
|
||||||
}
|
}
|
||||||
|
@ -116,6 +116,7 @@ status_get(struct spotify_status *status)
|
|||||||
|
|
||||||
status->installed = info.libspotify_installed;
|
status->installed = info.libspotify_installed;
|
||||||
status->logged_in = info.libspotify_logged_in;
|
status->logged_in = info.libspotify_logged_in;
|
||||||
|
status->has_podcast_support = false;
|
||||||
snprintf(status->username, sizeof(status->username), "%s", info.libspotify_user);
|
snprintf(status->username, sizeof(status->username), "%s", info.libspotify_user);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,6 +44,17 @@ enum spotify_request_type {
|
|||||||
SPOTIFY_REQUEST_TYPE_METARESCAN,
|
SPOTIFY_REQUEST_TYPE_METARESCAN,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum spotify_item_type {
|
||||||
|
SPOTIFY_ITEM_TYPE_ALBUM,
|
||||||
|
SPOTIFY_ITEM_TYPE_ARTIST,
|
||||||
|
SPOTIFY_ITEM_TYPE_TRACK,
|
||||||
|
SPOTIFY_ITEM_TYPE_PLAYLIST,
|
||||||
|
SPOTIFY_ITEM_TYPE_SHOW,
|
||||||
|
SPOTIFY_ITEM_TYPE_EPISODE,
|
||||||
|
|
||||||
|
SPOTIFY_ITEM_TYPE_UNKNOWN,
|
||||||
|
};
|
||||||
|
|
||||||
struct spotify_album
|
struct spotify_album
|
||||||
{
|
{
|
||||||
const char *added_at;
|
const char *added_at;
|
||||||
@ -62,6 +73,7 @@ struct spotify_album
|
|||||||
int release_year;
|
int release_year;
|
||||||
const char *uri;
|
const char *uri;
|
||||||
const char *artwork_url;
|
const char *artwork_url;
|
||||||
|
const char *type;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct spotify_track
|
struct spotify_track
|
||||||
@ -89,6 +101,7 @@ struct spotify_track
|
|||||||
bool is_playable;
|
bool is_playable;
|
||||||
const char *restrictions;
|
const char *restrictions;
|
||||||
const char *linked_from_uri;
|
const char *linked_from_uri;
|
||||||
|
const char *type;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct spotify_playlist
|
struct spotify_playlist
|
||||||
@ -149,8 +162,43 @@ static const char *spotify_album_tracks_uri = "https://api.spotify.com/v1/albums
|
|||||||
static const char *spotify_playlists_uri = "https://api.spotify.com/v1/me/playlists?limit=50";
|
static const char *spotify_playlists_uri = "https://api.spotify.com/v1/me/playlists?limit=50";
|
||||||
static const char *spotify_playlist_tracks_uri = "https://api.spotify.com/v1/playlists/%s/tracks";
|
static const char *spotify_playlist_tracks_uri = "https://api.spotify.com/v1/playlists/%s/tracks";
|
||||||
static const char *spotify_artist_albums_uri = "https://api.spotify.com/v1/artists/%s/albums?include_groups=album,single";
|
static const char *spotify_artist_albums_uri = "https://api.spotify.com/v1/artists/%s/albums?include_groups=album,single";
|
||||||
|
static const char *spotify_shows_uri = "https://api.spotify.com/v1/me/shows?limit=50";
|
||||||
|
static const char *spotify_shows_episodes_uri = "https://api.spotify.com/v1/shows/%s/episodes";
|
||||||
|
static const char *spotify_episode_uri = "https://api.spotify.com/v1/episodes/%s";
|
||||||
|
|
||||||
|
|
||||||
|
static enum spotify_item_type
|
||||||
|
parse_type_from_uri(const char *uri)
|
||||||
|
{
|
||||||
|
if (strncasecmp(uri, "spotify:track:", strlen("spotify:track:")) == 0)
|
||||||
|
{
|
||||||
|
return SPOTIFY_ITEM_TYPE_TRACK;
|
||||||
|
}
|
||||||
|
else if (strncasecmp(uri, "spotify:artist:", strlen("spotify:artist:")) == 0)
|
||||||
|
{
|
||||||
|
return SPOTIFY_ITEM_TYPE_ARTIST;
|
||||||
|
}
|
||||||
|
else if (strncasecmp(uri, "spotify:album:", strlen("spotify:album:")) == 0)
|
||||||
|
{
|
||||||
|
return SPOTIFY_ITEM_TYPE_ALBUM;
|
||||||
|
}
|
||||||
|
else if (strncasecmp(uri, "spotify:show:", strlen("spotify:show:")) == 0)
|
||||||
|
{
|
||||||
|
return SPOTIFY_ITEM_TYPE_SHOW;
|
||||||
|
}
|
||||||
|
else if (strncasecmp(uri, "spotify:episode:", strlen("spotify:episode:")) == 0)
|
||||||
|
{
|
||||||
|
return SPOTIFY_ITEM_TYPE_EPISODE;
|
||||||
|
}
|
||||||
|
else if (strncasecmp(uri, "spotify:", strlen("spotify:")) == 0 && strstr(uri, "playlist:"))
|
||||||
|
{
|
||||||
|
return SPOTIFY_ITEM_TYPE_PLAYLIST;
|
||||||
|
}
|
||||||
|
|
||||||
|
DPRINTF(E_WARN, L_SPOTIFY, "Could not parse item type from Spotify uri: %s\n", uri);
|
||||||
|
return SPOTIFY_ITEM_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
free_credentials(void)
|
free_credentials(void)
|
||||||
{
|
{
|
||||||
@ -687,6 +735,7 @@ parse_metadata_track(json_object *jsontrack, struct spotify_track *track, int ma
|
|||||||
track->track_number = jparse_int_from_obj(jsontrack, "track_number");
|
track->track_number = jparse_int_from_obj(jsontrack, "track_number");
|
||||||
track->uri = jparse_str_from_obj(jsontrack, "uri");
|
track->uri = jparse_str_from_obj(jsontrack, "uri");
|
||||||
track->id = jparse_str_from_obj(jsontrack, "id");
|
track->id = jparse_str_from_obj(jsontrack, "id");
|
||||||
|
track->type = jparse_str_from_obj(jsontrack, "type");
|
||||||
|
|
||||||
// "is_playable" is only returned for a request with a market parameter, default to true if it is not in the response
|
// "is_playable" is only returned for a request with a market parameter, default to true if it is not in the response
|
||||||
track->is_playable = true;
|
track->is_playable = true;
|
||||||
@ -731,6 +780,7 @@ parse_metadata_album(json_object *jsonalbum, struct spotify_album *album, int ma
|
|||||||
album->name = jparse_str_from_obj(jsonalbum, "name");
|
album->name = jparse_str_from_obj(jsonalbum, "name");
|
||||||
album->uri = jparse_str_from_obj(jsonalbum, "uri");
|
album->uri = jparse_str_from_obj(jsonalbum, "uri");
|
||||||
album->id = jparse_str_from_obj(jsonalbum, "id");
|
album->id = jparse_str_from_obj(jsonalbum, "id");
|
||||||
|
album->type = jparse_str_from_obj(jsonalbum, "type");
|
||||||
|
|
||||||
album->album_type = jparse_str_from_obj(jsonalbum, "album_type");
|
album->album_type = jparse_str_from_obj(jsonalbum, "album_type");
|
||||||
album->is_compilation = (album->album_type && 0 == strcmp(album->album_type, "compilation"));
|
album->is_compilation = (album->album_type && 0 == strcmp(album->album_type, "compilation"));
|
||||||
@ -772,6 +822,52 @@ parse_metadata_playlist(json_object *jsonplaylist, struct spotify_playlist *play
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
parse_metadata_show(json_object *jsonshow, struct spotify_album *show)
|
||||||
|
{
|
||||||
|
memset(show, 0, sizeof(struct spotify_album));
|
||||||
|
|
||||||
|
show->name = jparse_str_from_obj(jsonshow, "name");
|
||||||
|
show->artist = jparse_str_from_obj(jsonshow, "publisher");
|
||||||
|
show->uri = jparse_str_from_obj(jsonshow, "uri");
|
||||||
|
show->id = jparse_str_from_obj(jsonshow, "id");
|
||||||
|
show->type = jparse_str_from_obj(jsonshow, "type");
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
parse_metadata_episode(json_object *jsonepisode, struct spotify_track *episode, int max_w)
|
||||||
|
{
|
||||||
|
json_object *jsonshow;
|
||||||
|
|
||||||
|
memset(episode, 0, sizeof(struct spotify_track));
|
||||||
|
|
||||||
|
if (json_object_object_get_ex(jsonepisode, "show", &jsonshow))
|
||||||
|
{
|
||||||
|
episode->album = jparse_str_from_obj(jsonshow, "name");
|
||||||
|
episode->artwork_url = get_album_image(jsonshow, max_w);
|
||||||
|
}
|
||||||
|
|
||||||
|
episode->name = jparse_str_from_obj(jsonepisode, "name");
|
||||||
|
episode->uri = jparse_str_from_obj(jsonepisode, "uri");
|
||||||
|
episode->id = jparse_str_from_obj(jsonepisode, "id");
|
||||||
|
episode->type = jparse_str_from_obj(jsonepisode, "type");
|
||||||
|
episode->duration_ms = jparse_int_from_obj(jsonepisode, "duration_ms");
|
||||||
|
|
||||||
|
episode->release_date = jparse_str_from_obj(jsonepisode, "release_date");
|
||||||
|
episode->release_date_precision = jparse_str_from_obj(jsonepisode, "release_date_precision");
|
||||||
|
if (episode->release_date_precision && strcmp(episode->release_date_precision, "day") == 0)
|
||||||
|
episode->release_date_time = jparse_time_from_obj(jsonepisode, "release_date");
|
||||||
|
episode->release_year = get_year_from_date(episode->release_date);
|
||||||
|
episode->mtime = episode->release_date_time;
|
||||||
|
|
||||||
|
// "is_playable" is only returned for a request with a market parameter, default to true if it is not in the response
|
||||||
|
episode->is_playable = true;
|
||||||
|
if (json_object_object_get_ex(jsonepisode, "is_playable", NULL))
|
||||||
|
{
|
||||||
|
episode->is_playable = jparse_bool_from_obj(jsonepisode, "is_playable");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Creates a new string for the playlist API endpoint for the given playist-uri.
|
* Creates a new string for the playlist API endpoint for the given playist-uri.
|
||||||
* The returned string needs to be freed by the caller.
|
* The returned string needs to be freed by the caller.
|
||||||
@ -921,6 +1017,27 @@ get_artist_albums_endpoint_uri(const char *uri)
|
|||||||
return endpoint_uri;
|
return endpoint_uri;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static char *
|
||||||
|
get_episode_endpoint_uri(const char *uri)
|
||||||
|
{
|
||||||
|
char *endpoint_uri = NULL;
|
||||||
|
char *id = NULL;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = get_id_from_uri(uri, &id);
|
||||||
|
if (ret < 0)
|
||||||
|
{
|
||||||
|
DPRINTF(E_LOG, L_SPOTIFY, "Error extracting id from track uri '%s'\n", uri);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
endpoint_uri = safe_asprintf(spotify_episode_uri, id);
|
||||||
|
|
||||||
|
out:
|
||||||
|
free(id);
|
||||||
|
return endpoint_uri;
|
||||||
|
}
|
||||||
|
|
||||||
static json_object *
|
static json_object *
|
||||||
request_track(const char *path)
|
request_track(const char *path)
|
||||||
{
|
{
|
||||||
@ -934,6 +1051,19 @@ request_track(const char *path)
|
|||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static json_object *
|
||||||
|
request_episode(const char *path)
|
||||||
|
{
|
||||||
|
char *endpoint_uri;
|
||||||
|
json_object *response;
|
||||||
|
|
||||||
|
endpoint_uri = get_episode_endpoint_uri(path);
|
||||||
|
response = request_endpoint_with_token_refresh(endpoint_uri);
|
||||||
|
free(endpoint_uri);
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
/* Thread: httpd */
|
/* Thread: httpd */
|
||||||
char *
|
char *
|
||||||
spotifywebapi_oauth_uri_get(const char *redirect_uri)
|
spotifywebapi_oauth_uri_get(const char *redirect_uri)
|
||||||
@ -1284,22 +1414,25 @@ queue_add_playlist(const char *uri, int position, char reshuffle, uint32_t item_
|
|||||||
static int
|
static int
|
||||||
queue_item_add(const char *uri, int position, char reshuffle, uint32_t item_id, int *count, int *new_item_id)
|
queue_item_add(const char *uri, int position, char reshuffle, uint32_t item_id, int *count, int *new_item_id)
|
||||||
{
|
{
|
||||||
if (strncasecmp(uri, "spotify:track:", strlen("spotify:track:")) == 0)
|
enum spotify_item_type type;
|
||||||
|
|
||||||
|
type = parse_type_from_uri(uri);
|
||||||
|
if (type == SPOTIFY_ITEM_TYPE_TRACK)
|
||||||
{
|
{
|
||||||
queue_add_track(uri, position, reshuffle, item_id, count, new_item_id);
|
queue_add_track(uri, position, reshuffle, item_id, count, new_item_id);
|
||||||
return LIBRARY_OK;
|
return LIBRARY_OK;
|
||||||
}
|
}
|
||||||
else if (strncasecmp(uri, "spotify:artist:", strlen("spotify:artist:")) == 0)
|
else if (type == SPOTIFY_ITEM_TYPE_ARTIST)
|
||||||
{
|
{
|
||||||
queue_add_artist(uri, position, reshuffle, item_id, count, new_item_id);
|
queue_add_artist(uri, position, reshuffle, item_id, count, new_item_id);
|
||||||
return LIBRARY_OK;
|
return LIBRARY_OK;
|
||||||
}
|
}
|
||||||
else if (strncasecmp(uri, "spotify:album:", strlen("spotify:album:")) == 0)
|
else if (type == SPOTIFY_ITEM_TYPE_ALBUM)
|
||||||
{
|
{
|
||||||
queue_add_album(uri, position, reshuffle, item_id, count, new_item_id);
|
queue_add_album(uri, position, reshuffle, item_id, count, new_item_id);
|
||||||
return LIBRARY_OK;
|
return LIBRARY_OK;
|
||||||
}
|
}
|
||||||
else if (strncasecmp(uri, "spotify:", strlen("spotify:")) == 0)
|
else if (type == SPOTIFY_ITEM_TYPE_PLAYLIST)
|
||||||
{
|
{
|
||||||
queue_add_playlist(uri, position, reshuffle, item_id, count, new_item_id);
|
queue_add_playlist(uri, position, reshuffle, item_id, count, new_item_id);
|
||||||
return LIBRARY_OK;
|
return LIBRARY_OK;
|
||||||
@ -1399,6 +1532,9 @@ map_track_to_mfi(struct media_file_info *mfi, const struct spotify_track *track,
|
|||||||
mfi->track = track->track_number;
|
mfi->track = track->track_number;
|
||||||
|
|
||||||
mfi->data_kind = DATA_KIND_SPOTIFY;
|
mfi->data_kind = DATA_KIND_SPOTIFY;
|
||||||
|
if (strcmp(track->type, "episode") == 0)
|
||||||
|
mfi->media_kind = MEDIA_KIND_PODCAST;
|
||||||
|
else
|
||||||
mfi->media_kind = MEDIA_KIND_MUSIC;
|
mfi->media_kind = MEDIA_KIND_MUSIC;
|
||||||
mfi->type = strdup("spotify");
|
mfi->type = strdup("spotify");
|
||||||
mfi->codectype = strdup("wav");
|
mfi->codectype = strdup("wav");
|
||||||
@ -1433,6 +1569,12 @@ map_track_to_mfi(struct media_file_info *mfi, const struct spotify_track *track,
|
|||||||
mfi->compilation = track->is_compilation;
|
mfi->compilation = track->is_compilation;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (mfi->media_kind == MEDIA_KIND_PODCAST)
|
||||||
|
{
|
||||||
|
// For podcasts we want the tracks/episodes release date
|
||||||
|
mfi->date_released = track->release_date_time;
|
||||||
|
mfi->year = track->release_year;
|
||||||
|
}
|
||||||
snprintf(virtual_path, PATH_MAX, "/spotify:/%s/%s/%s", mfi->album_artist, mfi->album, mfi->title);
|
snprintf(virtual_path, PATH_MAX, "/spotify:/%s/%s/%s", mfi->album_artist, mfi->album, mfi->title);
|
||||||
mfi->virtual_path = strdup(virtual_path);
|
mfi->virtual_path = strdup(virtual_path);
|
||||||
}
|
}
|
||||||
@ -1581,6 +1723,83 @@ scan_saved_albums(enum spotify_request_type request_type)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Add a saved podcast show to the library
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
saved_episodes_add(json_object *item, int index, int total, enum spotify_request_type request_type, void *arg)
|
||||||
|
{
|
||||||
|
struct spotify_album *show = arg;
|
||||||
|
struct spotify_track episode;
|
||||||
|
int dir_id;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
DPRINTF(E_DBG, L_SPOTIFY, "saved_episodes_add: %s\n", json_object_to_json_string(item));
|
||||||
|
|
||||||
|
// Map episode information
|
||||||
|
parse_metadata_episode(item, &episode, 0);
|
||||||
|
|
||||||
|
// Get or create the directory structure for this album
|
||||||
|
dir_id = prepare_directories(show->artist, show->name);
|
||||||
|
|
||||||
|
ret = track_add(&episode, show, NULL, dir_id, request_type);
|
||||||
|
|
||||||
|
if (ret == 0 && spotify_saved_plid)
|
||||||
|
db_pl_add_item_bypath(spotify_saved_plid, episode.uri);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Add a saved podcast show to the library
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
saved_show_add(json_object *item, int index, int total, enum spotify_request_type request_type, void *arg)
|
||||||
|
{
|
||||||
|
json_object *jsonshow;
|
||||||
|
struct spotify_album show;
|
||||||
|
char *endpoint_uri;
|
||||||
|
|
||||||
|
DPRINTF(E_DBG, L_SPOTIFY, "saved_show_add: %s\n", json_object_to_json_string(item));
|
||||||
|
|
||||||
|
if (!json_object_object_get_ex(item, "show", &jsonshow))
|
||||||
|
{
|
||||||
|
DPRINTF(E_LOG, L_SPOTIFY, "Unexpected JSON: Item %d is missing the 'show' field\n", index);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Map show information
|
||||||
|
parse_metadata_show(jsonshow, &show);
|
||||||
|
show.added_at = jparse_str_from_obj(item, "added_at");
|
||||||
|
show.mtime = jparse_time_from_obj(item, "added_at");
|
||||||
|
|
||||||
|
|
||||||
|
// Now map the show episodes and insert/update them in the files database
|
||||||
|
endpoint_uri = safe_asprintf(spotify_shows_episodes_uri, show.id);
|
||||||
|
request_pagingobject_endpoint(endpoint_uri, saved_episodes_add, transaction_start, transaction_end, true, request_type, &show);
|
||||||
|
free(endpoint_uri);
|
||||||
|
|
||||||
|
if ((index + 1) >= total || ((index + 1) % 10 == 0))
|
||||||
|
DPRINTF(E_LOG, L_SPOTIFY, "Scanned %d of %d saved albums\n", (index + 1), total);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Thread: library
|
||||||
|
*
|
||||||
|
* Scan users saved podcast shows into the library
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
scan_saved_shows(enum spotify_request_type request_type)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = request_pagingobject_endpoint(spotify_shows_uri, saved_show_add, NULL, NULL, true, request_type, NULL);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Add a saved playlist tracks to the library
|
* Add a saved playlist tracks to the library
|
||||||
@ -1770,6 +1989,7 @@ create_base_playlist(void)
|
|||||||
static void
|
static void
|
||||||
scan(enum spotify_request_type request_type)
|
scan(enum spotify_request_type request_type)
|
||||||
{
|
{
|
||||||
|
struct spotify_status sp_status;
|
||||||
time_t start;
|
time_t start;
|
||||||
time_t end;
|
time_t end;
|
||||||
|
|
||||||
@ -1787,6 +2007,9 @@ scan(enum spotify_request_type request_type)
|
|||||||
create_saved_tracks_playlist();
|
create_saved_tracks_playlist();
|
||||||
scan_saved_albums(request_type);
|
scan_saved_albums(request_type);
|
||||||
scan_playlists(request_type);
|
scan_playlists(request_type);
|
||||||
|
spotify_status_get(&sp_status);
|
||||||
|
if (sp_status.has_podcast_support)
|
||||||
|
scan_saved_shows(request_type);
|
||||||
|
|
||||||
scanning = false;
|
scanning = false;
|
||||||
end = time(NULL);
|
end = time(NULL);
|
||||||
@ -1997,18 +2220,35 @@ spotifywebapi_pl_remove(const char *uri)
|
|||||||
char *
|
char *
|
||||||
spotifywebapi_artwork_url_get(const char *uri, int max_w, int max_h)
|
spotifywebapi_artwork_url_get(const char *uri, int max_w, int max_h)
|
||||||
{
|
{
|
||||||
|
enum spotify_item_type type;
|
||||||
json_object *response;
|
json_object *response;
|
||||||
struct spotify_track track;
|
struct spotify_track track;
|
||||||
char *artwork_url;
|
char *artwork_url;
|
||||||
|
|
||||||
|
type = parse_type_from_uri(uri);
|
||||||
|
if (type == SPOTIFY_ITEM_TYPE_TRACK)
|
||||||
|
{
|
||||||
response = request_track(uri);
|
response = request_track(uri);
|
||||||
|
if (response)
|
||||||
|
parse_metadata_track(response, &track, max_w);
|
||||||
|
}
|
||||||
|
else if (type == SPOTIFY_ITEM_TYPE_EPISODE)
|
||||||
|
{
|
||||||
|
response = request_episode(uri);
|
||||||
|
if (response)
|
||||||
|
parse_metadata_episode(response, &track, max_w);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
DPRINTF(E_WARN, L_SPOTIFY, "Unsupported Spotify type for artwork request: '%s'\n", uri);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
if (!response)
|
if (!response)
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
parse_metadata_track(response, &track, max_w);
|
|
||||||
|
|
||||||
DPRINTF(E_DBG, L_SPOTIFY, "Got track artwork url: '%s' (%s) \n", track.artwork_url, track.uri);
|
DPRINTF(E_DBG, L_SPOTIFY, "Got track artwork url: '%s' (%s) \n", track.artwork_url, track.uri);
|
||||||
|
|
||||||
artwork_url = safe_strdup(track.artwork_url);
|
artwork_url = safe_strdup(track.artwork_url);
|
||||||
|
Loading…
Reference in New Issue
Block a user