907af2ad02
This replaces the output parameter (which is bad API design). As a
side effect, it fixes the bad [[gnu::pure]] attribute added by commit
a636d2127
which caused optimizing compilers to miscompile calls to
that function. "Pure" functions can be assumed to have no output
arguments, so the compiler can assume the function doesn't modify
them.
Closes https://github.com/MusicPlayerDaemon/MPD/issues/1282
189 lines
4.5 KiB
C++
189 lines
4.5 KiB
C++
/*
|
|
* 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 <cassert>
|
|
#include <cstring>
|
|
#include <map>
|
|
#include <string>
|
|
#include <utility>
|
|
|
|
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<std::string, unsigned> permission_passwords;
|
|
|
|
static unsigned permission_default;
|
|
|
|
#ifdef HAVE_UN
|
|
static unsigned local_permissions;
|
|
#endif
|
|
|
|
#ifdef HAVE_TCP
|
|
static std::map<std::string, unsigned> 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_PLAYER |
|
|
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
|
|
|
|
std::optional<unsigned>
|
|
GetPermissionFromPassword(const char *password) noexcept
|
|
{
|
|
auto i = permission_passwords.find(password);
|
|
if (i == permission_passwords.end())
|
|
return std::nullopt;
|
|
|
|
return i->second;
|
|
}
|
|
|
|
unsigned
|
|
getDefaultPermissions() noexcept
|
|
{
|
|
return permission_default;
|
|
}
|
|
|
|
#ifdef HAVE_UN
|
|
|
|
unsigned
|
|
GetLocalPermissions() noexcept
|
|
{
|
|
return local_permissions;
|
|
}
|
|
|
|
#endif
|