/* * Copyright 2003-2021 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 "Permission.hxx" #include "config/Param.hxx" #include "config/Data.hxx" #include "config/Option.hxx" #include "net/AddressInfo.hxx" #include "net/Resolver.hxx" #include "net/ToString.hxx" #include "util/IterableSplitString.hxx" #include "util/RuntimeError.hxx" #include "util/StringView.hxx" #include #include #include #include #include static constexpr char PERMISSION_PASSWORD_CHAR = '@'; static constexpr char PERMISSION_SEPARATOR = ','; static constexpr struct { const char *name; unsigned value; } permission_names[] = { { "read", PERMISSION_READ }, { "add", PERMISSION_ADD }, { "player", PERMISSION_PLAYER }, { "control", PERMISSION_CONTROL }, { "admin", PERMISSION_ADMIN }, { nullptr, 0 }, }; static std::map permission_passwords; static unsigned permission_default; #ifdef HAVE_UN static unsigned local_permissions; #endif #ifdef HAVE_TCP static std::map host_passwords; #endif static unsigned ParsePermission(StringView s) { for (auto i = permission_names; i->name != nullptr; ++i) if (s.Equals(i->name)) return i->value; throw FormatRuntimeError("unknown permission \"%.*s\"", int(s.size), s.data); } static unsigned parsePermissions(std::string_view string) { unsigned permission = 0; for (const auto i : IterableSplitString(string, PERMISSION_SEPARATOR)) if (!i.empty()) permission |= ParsePermission(i); /* for backwards compatiblity with MPD 0.22 and older, "control" implies "play" */ if (permission & PERMISSION_CONTROL) permission |= PERMISSION_PLAYER; return permission; } void initPermissions(const ConfigData &config) { permission_default = PERMISSION_READ | PERMISSION_ADD | PERMISSION_CONTROL | PERMISSION_ADMIN; for (const auto ¶m : config.GetParamList(ConfigOption::PASSWORD)) { permission_default = 0; param.With([](const char *value){ const char *separator = std::strchr(value, PERMISSION_PASSWORD_CHAR); if (separator == nullptr) throw FormatRuntimeError("\"%c\" not found in password string", PERMISSION_PASSWORD_CHAR); std::string password(value, separator); unsigned permission = parsePermissions(separator + 1); permission_passwords.emplace(std::move(password), permission); }); } config.With(ConfigOption::DEFAULT_PERMS, [](const char *value){ if (value != nullptr) permission_default = parsePermissions(value); }); #ifdef HAVE_UN local_permissions = config.With(ConfigOption::LOCAL_PERMISSIONS, [](const char *value){ return value != nullptr ? parsePermissions(value) : permission_default; }); #endif #ifdef HAVE_TCP for (const auto ¶m : config.GetParamList(ConfigOption::HOST_PERMISSIONS)) { permission_default = 0; param.With([](StringView value){ auto [host_sv, permissions_s] = value.Split(' '); unsigned permissions = parsePermissions(permissions_s); const std::string host_s{host_sv}; for (const auto &i : Resolve(host_s.c_str(), 0, AI_PASSIVE, SOCK_STREAM)) host_passwords.emplace(HostToString(i), permissions); }); } #endif } #ifdef HAVE_TCP int GetPermissionsFromAddress(SocketAddress address) noexcept { if (auto i = host_passwords.find(HostToString(address)); i != host_passwords.end()) return i->second; return -1; } #endif int getPermissionFromPassword(const char *password, unsigned *permission) noexcept { auto i = permission_passwords.find(password); if (i == permission_passwords.end()) return -1; *permission = i->second; return 0; } unsigned getDefaultPermissions() noexcept { return permission_default; } #ifdef HAVE_UN unsigned GetLocalPermissions() noexcept { return local_permissions; } #endif