/* * 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. */ #define __STDC_FORMAT_MACROS /* for PRIu64 */ #include "config.h" #include "StorageCommands.hxx" #include "Request.hxx" #include "CommandError.hxx" #include "util/UriUtil.hxx" #include "util/Error.hxx" #include "util/ConstBuffer.hxx" #include "fs/Traits.hxx" #include "client/Client.hxx" #include "client/Response.hxx" #include "Partition.hxx" #include "Instance.hxx" #include "storage/Registry.hxx" #include "storage/CompositeStorage.hxx" #include "storage/FileInfo.hxx" #include "db/plugins/simple/SimpleDatabasePlugin.hxx" #include "db/update/Service.hxx" #include "TimePrint.hxx" #include "IOThread.hxx" #include "Idle.hxx" #include /* for PRIu64 */ gcc_pure static bool skip_path(const char *name_utf8) { return strchr(name_utf8, '\n') != nullptr; } #if defined(WIN32) && GCC_CHECK_VERSION(4,6) /* PRIu64 causes bogus compiler warning */ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wformat" #pragma GCC diagnostic ignored "-Wformat-extra-args" #endif static bool handle_listfiles_storage(Response &r, StorageDirectoryReader &reader, Error &error) { const char *name_utf8; while ((name_utf8 = reader.Read()) != nullptr) { if (skip_path(name_utf8)) continue; StorageFileInfo info; if (!reader.GetInfo(false, info, error)) continue; switch (info.type) { case StorageFileInfo::Type::OTHER: /* ignore */ continue; case StorageFileInfo::Type::REGULAR: r.Format("file: %s\n" "size: %" PRIu64 "\n", name_utf8, info.size); break; case StorageFileInfo::Type::DIRECTORY: r.Format("directory: %s\n", name_utf8); break; } if (info.mtime != 0) time_print(r, "Last-Modified", info.mtime); } return true; } #if defined(WIN32) && GCC_CHECK_VERSION(4,6) #pragma GCC diagnostic pop #endif static bool handle_listfiles_storage(Response &r, Storage &storage, const char *uri, Error &error) { auto reader = storage.OpenDirectory(uri, error); if (reader == nullptr) return false; bool success = handle_listfiles_storage(r, *reader, error); delete reader; return success; } CommandResult handle_listfiles_storage(Response &r, Storage &storage, const char *uri) { Error error; if (!handle_listfiles_storage(r, storage, uri, error)) return print_error(r, error); return CommandResult::OK; } CommandResult handle_listfiles_storage(Response &r, const char *uri) { Error error; Storage *storage = CreateStorageURI(io_thread_get(), uri, error); if (storage == nullptr) { if (error.IsDefined()) return print_error(r, error); r.Error(ACK_ERROR_ARG, "Unrecognized storage URI"); return CommandResult::ERROR; } bool success = handle_listfiles_storage(r, *storage, "", error); delete storage; if (!success) return print_error(r, error); return CommandResult::OK; } static void print_storage_uri(Client &client, Response &r, const Storage &storage) { std::string uri = storage.MapUTF8(""); if (uri.empty()) return; if (PathTraitsUTF8::IsAbsolute(uri.c_str())) { /* storage points to local directory */ if (!client.IsLocal()) /* only "local" clients may see local paths (same policy as with the "config" command) */ return; } else { /* hide username/passwords from client */ std::string allocated = uri_remove_auth(uri.c_str()); if (!allocated.empty()) uri = std::move(allocated); } r.Format("storage: %s\n", uri.c_str()); } CommandResult handle_listmounts(Client &client, gcc_unused Request args, Response &r) { Storage *_composite = client.partition.instance.storage; if (_composite == nullptr) { r.Error(ACK_ERROR_NO_EXIST, "No database"); return CommandResult::ERROR; } CompositeStorage &composite = *(CompositeStorage *)_composite; const auto visitor = [&client, &r](const char *mount_uri, const Storage &storage){ r.Format("mount: %s\n", mount_uri); print_storage_uri(client, r, storage); }; composite.VisitMounts(visitor); return CommandResult::OK; } CommandResult handle_mount(Client &client, Request args, Response &r) { Storage *_composite = client.partition.instance.storage; if (_composite == nullptr) { r.Error(ACK_ERROR_NO_EXIST, "No database"); return CommandResult::ERROR; } CompositeStorage &composite = *(CompositeStorage *)_composite; const char *const local_uri = args[0]; const char *const remote_uri = args[1]; if (*local_uri == 0) { r.Error(ACK_ERROR_ARG, "Bad mount point"); return CommandResult::ERROR; } if (strchr(local_uri, '/') != nullptr) { /* allow only top-level mounts for now */ /* TODO: eliminate this limitation after ensuring that UpdateQueue::Erase() really gets called for every unmount, and no Directory disappears recursively during database update */ r.Error(ACK_ERROR_ARG, "Bad mount point"); return CommandResult::ERROR; } Error error; Storage *storage = CreateStorageURI(io_thread_get(), remote_uri, error); if (storage == nullptr) { if (error.IsDefined()) return print_error(r, error); r.Error(ACK_ERROR_ARG, "Unrecognized storage URI"); return CommandResult::ERROR; } composite.Mount(local_uri, storage); idle_add(IDLE_MOUNT); #ifdef ENABLE_DATABASE Database *_db = client.partition.instance.database; if (_db != nullptr && _db->IsPlugin(simple_db_plugin)) { SimpleDatabase &db = *(SimpleDatabase *)_db; if (!db.Mount(local_uri, remote_uri, error)) { composite.Unmount(local_uri); return print_error(r, error); } // TODO: call Instance::OnDatabaseModified()? // TODO: trigger database update? idle_add(IDLE_DATABASE); } #endif return CommandResult::OK; } CommandResult handle_unmount(Client &client, Request args, Response &r) { Storage *_composite = client.partition.instance.storage; if (_composite == nullptr) { r.Error(ACK_ERROR_NO_EXIST, "No database"); return CommandResult::ERROR; } CompositeStorage &composite = *(CompositeStorage *)_composite; const char *const local_uri = args.front(); if (*local_uri == 0) { r.Error(ACK_ERROR_ARG, "Bad mount point"); return CommandResult::ERROR; } #ifdef ENABLE_DATABASE if (client.partition.instance.update != nullptr) /* ensure that no database update will attempt to work with the database/storage instances we're about to destroy here */ client.partition.instance.update->CancelMount(local_uri); Database *_db = client.partition.instance.database; if (_db != nullptr && _db->IsPlugin(simple_db_plugin)) { SimpleDatabase &db = *(SimpleDatabase *)_db; if (db.Unmount(local_uri)) // TODO: call Instance::OnDatabaseModified()? idle_add(IDLE_DATABASE); } #endif if (!composite.Unmount(local_uri)) { r.Error(ACK_ERROR_ARG, "Not a mount point"); return CommandResult::ERROR; } idle_add(IDLE_MOUNT); return CommandResult::OK; }