/* * Copyright 2003-2016 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 "ProxyDatabasePlugin.hxx" #include "db/Interface.hxx" #include "db/DatabasePlugin.hxx" #include "db/DatabaseListener.hxx" #include "db/Selection.hxx" #include "db/DatabaseError.hxx" #include "db/PlaylistInfo.hxx" #include "db/LightDirectory.hxx" #include "db/LightSong.hxx" #include "db/Stats.hxx" #include "SongFilter.hxx" #include "Compiler.h" #include "config/Block.hxx" #include "tag/TagBuilder.hxx" #include "tag/Tag.hxx" #include "util/Error.hxx" #include "util/Domain.hxx" #include "util/ScopeExit.hxx" #include "protocol/Ack.hxx" #include "event/SocketMonitor.hxx" #include "event/IdleMonitor.hxx" #include "Log.hxx" #include #include #include #include #include class LibmpdclientError final : std::runtime_error { enum mpd_error code; public: LibmpdclientError(enum mpd_error _code, const char *_msg) :std::runtime_error(_msg), code(_code) {} enum mpd_error GetCode() const { return code; } }; class ProxySong : public LightSong { Tag tag2; public: explicit ProxySong(const mpd_song *song); }; class AllocatedProxySong : public ProxySong { mpd_song *const song; public: explicit AllocatedProxySong(mpd_song *_song) :ProxySong(_song), song(_song) {} ~AllocatedProxySong() { mpd_song_free(song); } }; class ProxyDatabase final : public Database, SocketMonitor, IdleMonitor { DatabaseListener &listener; std::string host; unsigned port; bool keepalive; struct mpd_connection *connection; /* this is mutable because GetStats() must be "const" */ mutable time_t update_stamp; /** * The libmpdclient idle mask that was removed from the other * MPD. This will be handled by the next OnIdle() call. */ unsigned idle_received; /** * Is the #connection currently "idle"? That is, did we send * the "idle" command to it? */ bool is_idle; public: ProxyDatabase(EventLoop &_loop, DatabaseListener &_listener) :Database(proxy_db_plugin), SocketMonitor(_loop), IdleMonitor(_loop), listener(_listener) {} static Database *Create(EventLoop &loop, DatabaseListener &listener, const ConfigBlock &block, Error &error); virtual void Open() override; virtual void Close() override; const LightSong *GetSong(const char *uri_utf8) const override; void ReturnSong(const LightSong *song) const override; virtual bool Visit(const DatabaseSelection &selection, VisitDirectory visit_directory, VisitSong visit_song, VisitPlaylist visit_playlist, Error &error) const override; virtual bool VisitUniqueTags(const DatabaseSelection &selection, TagType tag_type, tag_mask_t group_mask, VisitTag visit_tag, Error &error) const override; virtual bool GetStats(const DatabaseSelection &selection, DatabaseStats &stats, Error &error) const override; virtual unsigned Update(const char *uri_utf8, bool discard, Error &error) override; virtual time_t GetUpdateStamp() const override { return update_stamp; } private: bool Configure(const ConfigBlock &block, Error &error); void Connect(); void CheckConnection(); void EnsureConnected(); void Disconnect(); /* virtual methods from SocketMonitor */ virtual bool OnSocketReady(unsigned flags) override; /* virtual methods from IdleMonitor */ virtual void OnIdle() override; }; static constexpr Domain libmpdclient_domain("libmpdclient"); static constexpr struct { TagType d; enum mpd_tag_type s; } tag_table[] = { { TAG_ARTIST, MPD_TAG_ARTIST }, { TAG_ALBUM, MPD_TAG_ALBUM }, { TAG_ALBUM_ARTIST, MPD_TAG_ALBUM_ARTIST }, { TAG_TITLE, MPD_TAG_TITLE }, { TAG_TRACK, MPD_TAG_TRACK }, { TAG_NAME, MPD_TAG_NAME }, { TAG_GENRE, MPD_TAG_GENRE }, { TAG_DATE, MPD_TAG_DATE }, { TAG_COMPOSER, MPD_TAG_COMPOSER }, { TAG_PERFORMER, MPD_TAG_PERFORMER }, { TAG_COMMENT, MPD_TAG_COMMENT }, { TAG_DISC, MPD_TAG_DISC }, { TAG_MUSICBRAINZ_ARTISTID, MPD_TAG_MUSICBRAINZ_ARTISTID }, { TAG_MUSICBRAINZ_ALBUMID, MPD_TAG_MUSICBRAINZ_ALBUMID }, { TAG_MUSICBRAINZ_ALBUMARTISTID, MPD_TAG_MUSICBRAINZ_ALBUMARTISTID }, { TAG_MUSICBRAINZ_TRACKID, MPD_TAG_MUSICBRAINZ_TRACKID }, #if LIBMPDCLIENT_CHECK_VERSION(2,10,0) { TAG_MUSICBRAINZ_RELEASETRACKID, MPD_TAG_MUSICBRAINZ_RELEASETRACKID }, #endif { TAG_NUM_OF_ITEM_TYPES, MPD_TAG_COUNT } }; static void Copy(TagBuilder &tag, TagType d_tag, const struct mpd_song *song, enum mpd_tag_type s_tag) { for (unsigned i = 0;; ++i) { const char *value = mpd_song_get_tag(song, s_tag, i); if (value == nullptr) break; tag.AddItem(d_tag, value); } } ProxySong::ProxySong(const mpd_song *song) { directory = nullptr; uri = mpd_song_get_uri(song); real_uri = nullptr; tag = &tag2; mtime = mpd_song_get_last_modified(song); #if LIBMPDCLIENT_CHECK_VERSION(2,3,0) start_time = SongTime::FromS(mpd_song_get_start(song)); end_time = SongTime::FromS(mpd_song_get_end(song)); #else start_time = end_time = SongTime::zero(); #endif TagBuilder tag_builder; const unsigned duration = mpd_song_get_duration(song); if (duration > 0) tag_builder.SetDuration(SignedSongTime::FromS(duration)); for (const auto *i = &tag_table[0]; i->d != TAG_NUM_OF_ITEM_TYPES; ++i) Copy(tag_builder, i->d, song, i->s); tag_builder.Commit(tag2); } gcc_const static enum mpd_tag_type Convert(TagType tag_type) { for (auto i = &tag_table[0]; i->d != TAG_NUM_OF_ITEM_TYPES; ++i) if (i->d == tag_type) return i->s; return MPD_TAG_COUNT; } static void ThrowError(struct mpd_connection *connection) { const auto code = mpd_connection_get_error(connection); AtScopeExit(connection) { mpd_connection_clear_error(connection); }; if (code == MPD_ERROR_SERVER) { /* libmpdclient's "enum mpd_server_error" is the same as our "enum ack" */ const auto server_error = mpd_connection_get_server_error(connection); throw ProtocolError((enum ack)server_error, mpd_connection_get_error_message(connection)); } else { throw LibmpdclientError(code, mpd_connection_get_error_message(connection)); } } static void CheckError(struct mpd_connection *connection) { const auto code = mpd_connection_get_error(connection); if (code != MPD_ERROR_SUCCESS) ThrowError(connection); } static bool SendConstraints(mpd_connection *connection, const SongFilter::Item &item) { switch (item.GetTag()) { mpd_tag_type tag; #if LIBMPDCLIENT_CHECK_VERSION(2,9,0) case LOCATE_TAG_BASE_TYPE: if (mpd_connection_cmp_server_version(connection, 0, 18, 0) < 0) /* requires MPD 0.18 */ return true; return mpd_search_add_base_constraint(connection, MPD_OPERATOR_DEFAULT, item.GetValue()); #endif case LOCATE_TAG_FILE_TYPE: return mpd_search_add_uri_constraint(connection, MPD_OPERATOR_DEFAULT, item.GetValue()); case LOCATE_TAG_ANY_TYPE: return mpd_search_add_any_tag_constraint(connection, MPD_OPERATOR_DEFAULT, item.GetValue()); default: tag = Convert(TagType(item.GetTag())); if (tag == MPD_TAG_COUNT) return true; return mpd_search_add_tag_constraint(connection, MPD_OPERATOR_DEFAULT, tag, item.GetValue()); } } static bool SendConstraints(mpd_connection *connection, const SongFilter &filter) { for (const auto &i : filter.GetItems()) if (!SendConstraints(connection, i)) return false; return true; } static bool SendConstraints(mpd_connection *connection, const DatabaseSelection &selection) { #if LIBMPDCLIENT_CHECK_VERSION(2,9,0) if (!selection.uri.empty() && mpd_connection_cmp_server_version(connection, 0, 18, 0) >= 0) { /* requires MPD 0.18 */ if (!mpd_search_add_base_constraint(connection, MPD_OPERATOR_DEFAULT, selection.uri.c_str())) return false; } #endif if (selection.filter != nullptr && !SendConstraints(connection, *selection.filter)) return false; return true; } Database * ProxyDatabase::Create(EventLoop &loop, DatabaseListener &listener, const ConfigBlock &block, Error &error) { ProxyDatabase *db = new ProxyDatabase(loop, listener); if (!db->Configure(block, error)) { delete db; db = nullptr; } return db; } bool ProxyDatabase::Configure(const ConfigBlock &block, gcc_unused Error &error) { host = block.GetBlockValue("host", ""); port = block.GetBlockValue("port", 0u); keepalive = block.GetBlockValue("keepalive", false); return true; } void ProxyDatabase::Open() { Connect(); update_stamp = 0; } void ProxyDatabase::Close() { if (connection != nullptr) Disconnect(); } void ProxyDatabase::Connect() { const char *_host = host.empty() ? nullptr : host.c_str(); connection = mpd_connection_new(_host, port, 0); if (connection == nullptr) throw LibmpdclientError(MPD_ERROR_OOM, "Out of memory"); try { CheckError(connection); } catch (...) { mpd_connection_free(connection); connection = nullptr; throw; } #if LIBMPDCLIENT_CHECK_VERSION(2, 10, 0) mpd_connection_set_keepalive(connection, keepalive); #endif idle_received = unsigned(-1); is_idle = false; SocketMonitor::Open(mpd_async_get_fd(mpd_connection_get_async(connection))); IdleMonitor::Schedule(); } void ProxyDatabase::CheckConnection() { assert(connection != nullptr); if (!mpd_connection_clear_error(connection)) { Disconnect(); Connect(); return; } if (is_idle) { unsigned idle = mpd_run_noidle(connection); if (idle == 0) { try { CheckError(connection); } catch (...) { Disconnect(); throw; } } idle_received |= idle; is_idle = false; IdleMonitor::Schedule(); } } void ProxyDatabase::EnsureConnected() { if (connection != nullptr) CheckConnection(); else Connect(); } void ProxyDatabase::Disconnect() { assert(connection != nullptr); IdleMonitor::Cancel(); SocketMonitor::Steal(); mpd_connection_free(connection); connection = nullptr; } bool ProxyDatabase::OnSocketReady(gcc_unused unsigned flags) { assert(connection != nullptr); if (!is_idle) { // TODO: can this happen? IdleMonitor::Schedule(); return false; } unsigned idle = (unsigned)mpd_recv_idle(connection, false); if (idle == 0) { try { CheckError(connection); } catch (const std::runtime_error &error) { LogError(error); Disconnect(); return false; } } /* let OnIdle() handle this */ idle_received |= idle; is_idle = false; IdleMonitor::Schedule(); return false; } void ProxyDatabase::OnIdle() { assert(connection != nullptr); /* handle previous idle events */ if (idle_received & MPD_IDLE_DATABASE) listener.OnDatabaseModified(); idle_received = 0; /* send a new idle command to the other MPD */ if (is_idle) // TODO: can this happen? return; if (!mpd_send_idle_mask(connection, MPD_IDLE_DATABASE)) { try { ThrowError(connection); } catch (const std::runtime_error &error) { LogError(error); } SocketMonitor::Steal(); mpd_connection_free(connection); connection = nullptr; return; } is_idle = true; SocketMonitor::ScheduleRead(); } const LightSong * ProxyDatabase::GetSong(const char *uri) const { // TODO: eliminate the const_cast const_cast(this)->EnsureConnected(); if (!mpd_send_list_meta(connection, uri)) ThrowError(connection); struct mpd_song *song = mpd_recv_song(connection); if (!mpd_response_finish(connection)) { if (song != nullptr) mpd_song_free(song); ThrowError(connection); } if (song == nullptr) throw DatabaseError(DatabaseErrorCode::NOT_FOUND, "No such song"); return new AllocatedProxySong(song); } void ProxyDatabase::ReturnSong(const LightSong *_song) const { assert(_song != nullptr); AllocatedProxySong *song = (AllocatedProxySong *) const_cast(_song); delete song; } static bool Visit(struct mpd_connection *connection, const char *uri, bool recursive, const SongFilter *filter, VisitDirectory visit_directory, VisitSong visit_song, VisitPlaylist visit_playlist, Error &error); static bool Visit(struct mpd_connection *connection, bool recursive, const SongFilter *filter, const struct mpd_directory *directory, VisitDirectory visit_directory, VisitSong visit_song, VisitPlaylist visit_playlist, Error &error) { const char *path = mpd_directory_get_path(directory); #if LIBMPDCLIENT_CHECK_VERSION(2,9,0) time_t mtime = mpd_directory_get_last_modified(directory); #else time_t mtime = 0; #endif if (visit_directory && !visit_directory(LightDirectory(path, mtime), error)) return false; if (recursive && !Visit(connection, path, recursive, filter, visit_directory, visit_song, visit_playlist, error)) return false; return true; } gcc_pure static bool Match(const SongFilter *filter, const LightSong &song) { return filter == nullptr || filter->Match(song); } static bool Visit(const SongFilter *filter, const mpd_song *_song, VisitSong visit_song, Error &error) { if (!visit_song) return true; const ProxySong song(_song); return !Match(filter, song) || visit_song(song, error); } static bool Visit(const struct mpd_playlist *playlist, VisitPlaylist visit_playlist, Error &error) { if (!visit_playlist) return true; PlaylistInfo p(mpd_playlist_get_path(playlist), mpd_playlist_get_last_modified(playlist)); return visit_playlist(p, LightDirectory::Root(), error); } class ProxyEntity { struct mpd_entity *entity; public: explicit ProxyEntity(struct mpd_entity *_entity) :entity(_entity) {} ProxyEntity(const ProxyEntity &other) = delete; ProxyEntity(ProxyEntity &&other) :entity(other.entity) { other.entity = nullptr; } ~ProxyEntity() { if (entity != nullptr) mpd_entity_free(entity); } ProxyEntity &operator=(const ProxyEntity &other) = delete; operator const struct mpd_entity *() const { return entity; } }; static std::list ReceiveEntities(struct mpd_connection *connection) { std::list entities; struct mpd_entity *entity; while ((entity = mpd_recv_entity(connection)) != nullptr) entities.push_back(ProxyEntity(entity)); mpd_response_finish(connection); return entities; } static bool Visit(struct mpd_connection *connection, const char *uri, bool recursive, const SongFilter *filter, VisitDirectory visit_directory, VisitSong visit_song, VisitPlaylist visit_playlist, Error &error) { if (!mpd_send_list_meta(connection, uri)) ThrowError(connection); std::list entities(ReceiveEntities(connection)); CheckError(connection); for (const auto &entity : entities) { switch (mpd_entity_get_type(entity)) { case MPD_ENTITY_TYPE_UNKNOWN: break; case MPD_ENTITY_TYPE_DIRECTORY: if (!Visit(connection, recursive, filter, mpd_entity_get_directory(entity), visit_directory, visit_song, visit_playlist, error)) return false; break; case MPD_ENTITY_TYPE_SONG: if (!Visit(filter, mpd_entity_get_song(entity), visit_song, error)) return false; break; case MPD_ENTITY_TYPE_PLAYLIST: if (!Visit(mpd_entity_get_playlist(entity), visit_playlist, error)) return false; break; } } return true; } static bool SearchSongs(struct mpd_connection *connection, const DatabaseSelection &selection, VisitSong visit_song, Error &error) { assert(selection.recursive); assert(visit_song); const bool exact = selection.filter == nullptr || !selection.filter->HasFoldCase(); if (!mpd_search_db_songs(connection, exact) || !SendConstraints(connection, selection) || !mpd_search_commit(connection)) ThrowError(connection); bool result = true; struct mpd_song *song; while (result && (song = mpd_recv_song(connection)) != nullptr) { AllocatedProxySong song2(song); result = !Match(selection.filter, song2) || visit_song(song2, error); } if (!mpd_response_finish(connection) && result) ThrowError(connection); return result; } /** * Check whether we can use the "base" constraint. Requires * libmpdclient 2.9 and MPD 0.18. */ gcc_pure static bool ServerSupportsSearchBase(const struct mpd_connection *connection) { #if LIBMPDCLIENT_CHECK_VERSION(2,9,0) return mpd_connection_cmp_server_version(connection, 0, 18, 0) >= 0; #else (void)connection; return false; #endif } bool ProxyDatabase::Visit(const DatabaseSelection &selection, VisitDirectory visit_directory, VisitSong visit_song, VisitPlaylist visit_playlist, Error &error) const { // TODO: eliminate the const_cast const_cast(this)->EnsureConnected(); if (!visit_directory && !visit_playlist && selection.recursive && (ServerSupportsSearchBase(connection) ? !selection.IsEmpty() : selection.HasOtherThanBase())) /* this optimized code path can only be used under certain conditions */ return ::SearchSongs(connection, selection, visit_song, error); /* fall back to recursive walk (slow!) */ return ::Visit(connection, selection.uri.c_str(), selection.recursive, selection.filter, visit_directory, visit_song, visit_playlist, error); } bool ProxyDatabase::VisitUniqueTags(const DatabaseSelection &selection, TagType tag_type, gcc_unused tag_mask_t group_mask, VisitTag visit_tag, Error &error) const { // TODO: eliminate the const_cast const_cast(this)->EnsureConnected(); enum mpd_tag_type tag_type2 = Convert(tag_type); if (tag_type2 == MPD_TAG_COUNT) { error.Set(libmpdclient_domain, "Unsupported tag"); return false; } if (!mpd_search_db_tags(connection, tag_type2) || !SendConstraints(connection, selection)) ThrowError(connection); // TODO: use group_mask if (!mpd_search_commit(connection)) ThrowError(connection); bool result = true; struct mpd_pair *pair; while (result && (pair = mpd_recv_pair_tag(connection, tag_type2)) != nullptr) { AtScopeExit(this, pair) { mpd_return_pair(connection, pair); }; TagBuilder tag; tag.AddItem(tag_type, pair->value); if (tag.IsEmpty()) /* if no tag item has been added, then the given value was not acceptable (e.g. empty); forcefully insert an empty tag in this case, as the caller expects the given tag type to be present */ tag.AddEmptyItem(tag_type); result = visit_tag(tag.Commit(), error); } if (!mpd_response_finish(connection) && result) ThrowError(connection); return result; } bool ProxyDatabase::GetStats(const DatabaseSelection &selection, DatabaseStats &stats, gcc_unused Error &error) const { // TODO: match (void)selection; // TODO: eliminate the const_cast const_cast(this)->EnsureConnected(); struct mpd_stats *stats2 = mpd_run_stats(connection); if (stats2 == nullptr) ThrowError(connection); update_stamp = (time_t)mpd_stats_get_db_update_time(stats2); stats.song_count = mpd_stats_get_number_of_songs(stats2); stats.total_duration = std::chrono::seconds(mpd_stats_get_db_play_time(stats2)); stats.artist_count = mpd_stats_get_number_of_artists(stats2); stats.album_count = mpd_stats_get_number_of_albums(stats2); mpd_stats_free(stats2); return true; } unsigned ProxyDatabase::Update(const char *uri_utf8, bool discard, gcc_unused Error &error) { EnsureConnected(); unsigned id = discard ? mpd_run_rescan(connection, uri_utf8) : mpd_run_update(connection, uri_utf8); if (id == 0) CheckError(connection); return id; } const DatabasePlugin proxy_db_plugin = { "proxy", DatabasePlugin::FLAG_REQUIRE_STORAGE, ProxyDatabase::Create, };