/* * Copyright 2003-2017 The Music Player Daemon Project * http://www.musicpd.org * * 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., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "config.h" #include "SoundCloudPlaylistPlugin.hxx" #include "../PlaylistPlugin.hxx" #include "../MemorySongEnumerator.hxx" #include "config/Block.hxx" #include "input/InputStream.hxx" #include "tag/TagBuilder.hxx" #include "util/StringCompare.hxx" #include "util/Alloc.hxx" #include "util/Domain.hxx" #include "util/ScopeExit.hxx" #include "Log.hxx" #include #include #include #include static struct { std::string apikey; } soundcloud_config; static constexpr Domain soundcloud_domain("soundcloud"); static bool soundcloud_init(const ConfigBlock &block) { // APIKEY for MPD application, registered under DarkFox' account. soundcloud_config.apikey = block.GetBlockValue("apikey", "a25e51780f7f86af0afa91f241d091f8"); if (soundcloud_config.apikey.empty()) { LogDebug(soundcloud_domain, "disabling the soundcloud playlist plugin " "because API key is not set"); return false; } return true; } /** * Construct a full soundcloud resolver URL from the given fragment. * @param uri uri of a soundcloud page (or just the path) * @return Constructed URL. Must be freed with free(). */ static char * soundcloud_resolve(const char* uri) { char *u, *ru; if (StringStartsWith(uri, "https://")) { u = xstrdup(uri); } else if (StringStartsWith(uri, "soundcloud.com")) { u = xstrcatdup("https://", uri); } else { /* assume it's just a path on soundcloud.com */ u = xstrcatdup("https://soundcloud.com/", uri); } ru = xstrcatdup("https://api.soundcloud.com/resolve.json?url=", u, "&client_id=", soundcloud_config.apikey.c_str()); free(u); return ru; } /* YAJL parser for track data from both /tracks/ and /playlists/ JSON */ enum key { Duration, Title, Stream_URL, Other, }; static const char *const key_str[] = { "duration", "title", "stream_url", nullptr, }; struct SoundCloudJsonData { int key; std::string stream_url; long duration; std::string title; int got_url = 0; /* nesting level of last stream_url */ std::forward_list songs; }; static int handle_integer(void *ctx, long long intval) { auto *data = (SoundCloudJsonData *) ctx; switch (data->key) { case Duration: data->duration = intval; break; default: break; } return 1; } static int handle_string(void *ctx, const unsigned char *stringval, size_t stringlen) { auto *data = (SoundCloudJsonData *) ctx; const char *s = (const char *) stringval; switch (data->key) { case Title: data->title.assign(s, stringlen); break; case Stream_URL: data->stream_url.assign(s, stringlen); data->got_url = 1; break; default: break; } return 1; } static int handle_mapkey(void *ctx, const unsigned char *stringval, size_t stringlen) { auto *data = (SoundCloudJsonData *) ctx; int i; data->key = Other; for (i = 0; i < Other; ++i) { if (memcmp((const char *)stringval, key_str[i], stringlen) == 0) { data->key = i; break; } } return 1; } static int handle_start_map(void *ctx) { auto *data = (SoundCloudJsonData *) ctx; if (data->got_url > 0) data->got_url++; return 1; } static int handle_end_map(void *ctx) { auto *data = (SoundCloudJsonData *) ctx; if (data->got_url > 1) { data->got_url--; return 1; } if (data->got_url == 0) return 1; /* got_url == 1, track finished, make it into a song */ data->got_url = 0; const std::string u = data->stream_url + "?client_id=" + soundcloud_config.apikey; TagBuilder tag; tag.SetDuration(SignedSongTime::FromMS(data->duration)); if (!data->title.empty()) tag.AddItem(TAG_NAME, data->title.c_str()); data->songs.emplace_front(u.c_str(), tag.Commit()); return 1; } static constexpr yajl_callbacks parse_callbacks = { nullptr, nullptr, handle_integer, nullptr, nullptr, handle_string, handle_start_map, handle_mapkey, handle_end_map, nullptr, nullptr, }; /** * Read JSON data and parse it using the given YAJL parser. * @param url URL of the JSON data. * @param hand YAJL parser handle. * @return -1 on error, 0 on success. */ static int soundcloud_parse_json(const char *url, yajl_handle hand, Mutex &mutex, Cond &cond) try { auto input_stream = InputStream::OpenReady(url, mutex, cond); const std::lock_guard protect(mutex); yajl_status stat; bool done = false; while (!done) { char buffer[4096]; unsigned char *ubuffer = (unsigned char *)buffer; const size_t nbytes = input_stream->Read(buffer, sizeof(buffer)); if (nbytes == 0) done = true; if (done) { stat = yajl_complete_parse(hand); } else stat = yajl_parse(hand, ubuffer, nbytes); if (stat != yajl_status_ok) { unsigned char *str = yajl_get_error(hand, 1, ubuffer, nbytes); LogError(soundcloud_domain, (const char *)str); yajl_free_error(hand, str); break; } } return 0; } catch (const std::exception &e) { LogError(e); return -1; } /** * Parse a soundcloud:// URL and create a playlist. * @param uri A soundcloud URL. Accepted forms: * soundcloud://track/ * soundcloud://playlist/ * soundcloud://url/ */ static SongEnumerator * soundcloud_open_uri(const char *uri, Mutex &mutex, Cond &cond) { assert(strncmp(uri, "soundcloud://", 13) == 0); uri += 13; char *u = nullptr; if (strncmp(uri, "track/", 6) == 0) { const char *rest = uri + 6; u = xstrcatdup("https://api.soundcloud.com/tracks/", rest, ".json?client_id=", soundcloud_config.apikey.c_str()); } else if (strncmp(uri, "playlist/", 9) == 0) { const char *rest = uri + 9; u = xstrcatdup("https://api.soundcloud.com/playlists/", rest, ".json?client_id=", soundcloud_config.apikey.c_str()); } else if (strncmp(uri, "user/", 5) == 0) { const char *rest = uri + 5; u = xstrcatdup("https://api.soundcloud.com/users/", rest, "/tracks.json?client_id=", soundcloud_config.apikey.c_str()); } else if (strncmp(uri, "search/", 7) == 0) { const char *rest = uri + 7; u = xstrcatdup("https://api.soundcloud.com/tracks.json?q=", rest, "&client_id=", soundcloud_config.apikey.c_str()); } else if (strncmp(uri, "url/", 4) == 0) { const char *rest = uri + 4; /* Translate to soundcloud resolver call. libcurl will automatically follow the redirect to the right resource. */ u = soundcloud_resolve(rest); } AtScopeExit(u) { free(u); }; if (u == nullptr) { LogWarning(soundcloud_domain, "unknown soundcloud URI"); return nullptr; } SoundCloudJsonData data; yajl_handle hand = yajl_alloc(&parse_callbacks, nullptr, &data); AtScopeExit(hand, &data) { yajl_free(hand); }; int ret = soundcloud_parse_json(u, hand, mutex, cond); if (ret == -1) return nullptr; data.songs.reverse(); return new MemorySongEnumerator(std::move(data.songs)); } static const char *const soundcloud_schemes[] = { "soundcloud", nullptr }; const struct playlist_plugin soundcloud_playlist_plugin = { "soundcloud", soundcloud_init, nullptr, soundcloud_open_uri, nullptr, soundcloud_schemes, nullptr, nullptr, };