mirror of
https://github.com/inspircd/inspircd.git
synced 2025-03-10 02:59:01 -04:00
502 lines
12 KiB
C++
502 lines
12 KiB
C++
/*
|
|
* InspIRCd -- Internet Relay Chat Daemon
|
|
*
|
|
* Copyright (C) 2018-2023 Sadie Powell <sadie@witchery.services>
|
|
* Copyright (C) 2015-2016, 2018 Attila Molnar <attilamolnar@hush.com>
|
|
*
|
|
* This file is part of InspIRCd. InspIRCd 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, version 2.
|
|
*
|
|
* 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
#include "inspircd.h"
|
|
#include "modules/reload.h"
|
|
#include "modules/cap.h"
|
|
#include "utility/string.h"
|
|
|
|
enum
|
|
{
|
|
// From IRCv3 capability-negotiation-3.1.
|
|
ERR_INVALIDCAPCMD = 410
|
|
};
|
|
|
|
namespace Cap
|
|
{
|
|
class ManagerImpl;
|
|
}
|
|
|
|
static Cap::ManagerImpl* managerimpl;
|
|
|
|
class Cap::ManagerImpl final
|
|
: public Cap::Manager
|
|
, public ReloadModule::EventListener
|
|
{
|
|
/** Stores the cap state of a module being reloaded
|
|
*/
|
|
struct CapModData final
|
|
{
|
|
struct Data final
|
|
{
|
|
std::string name;
|
|
std::vector<std::string> users;
|
|
|
|
Data(Capability* cap)
|
|
: name(cap->GetName())
|
|
{
|
|
}
|
|
};
|
|
std::vector<Data> caps;
|
|
};
|
|
|
|
typedef insp::flat_map<std::string, Capability*, irc::insensitive_swo> CapMap;
|
|
|
|
ExtItem capext;
|
|
CapMap caps;
|
|
Events::ModuleEventProvider& evprov;
|
|
|
|
static bool CanRequest(LocalUser* user, Ext usercaps, Capability* cap, bool adding)
|
|
{
|
|
const bool hascap = ((usercaps & cap->GetMask()) != 0);
|
|
if (hascap == adding)
|
|
return true;
|
|
|
|
return cap->OnRequest(user, adding);
|
|
}
|
|
|
|
Capability::Bit AllocateBit() const
|
|
{
|
|
Capability::Bit used = 0;
|
|
for (const auto& [_, cap] : caps)
|
|
used |= cap->GetMask();
|
|
|
|
for (size_t i = 0; i < MAX_CAPS; i++)
|
|
{
|
|
Capability::Bit bit = (static_cast<Capability::Bit>(1) << i);
|
|
if (!(used & bit))
|
|
return bit;
|
|
}
|
|
throw ModuleException(creator, "Too many caps");
|
|
}
|
|
|
|
void OnReloadModuleSave(Module* mod, ReloadModule::CustomData& cd) override
|
|
{
|
|
ServerInstance->Logs.Debug(MODNAME, "OnReloadModuleSave()");
|
|
if (mod == creator)
|
|
return;
|
|
|
|
auto* capmoddata = new CapModData();
|
|
cd.add(this, capmoddata);
|
|
|
|
for (const auto& [_, cap] : caps)
|
|
{
|
|
// Only save users of caps that belong to the module being reloaded
|
|
if (cap->creator != mod)
|
|
continue;
|
|
|
|
ServerInstance->Logs.Debug(MODNAME, "Module being reloaded implements cap {}, saving cap users", cap->GetName());
|
|
capmoddata->caps.emplace_back(cap);
|
|
CapModData::Data& capdata = capmoddata->caps.back();
|
|
|
|
// Populate list with uuids of users who are using the cap
|
|
for (auto* user : ServerInstance->Users.GetLocalUsers())
|
|
{
|
|
if (cap->IsEnabled(user))
|
|
capdata.users.push_back(user->uuid);
|
|
}
|
|
}
|
|
}
|
|
|
|
void OnReloadModuleRestore(Module* mod, void* data) override
|
|
{
|
|
CapModData* capmoddata = static_cast<CapModData*>(data);
|
|
for (const auto& capdata : capmoddata->caps)
|
|
{
|
|
Capability* cap = ManagerImpl::Find(capdata.name);
|
|
if (!cap)
|
|
{
|
|
ServerInstance->Logs.Debug(MODNAME, "Cap {} is no longer available after reload", capdata.name);
|
|
continue;
|
|
}
|
|
|
|
// Set back the cap for all users who were using it before the reload
|
|
for (const auto& uuid : capdata.users)
|
|
{
|
|
auto* user = ServerInstance->Users.FindUUID(uuid);
|
|
if (!user)
|
|
{
|
|
ServerInstance->Logs.Debug(MODNAME, "User {} is gone when trying to restore cap {}", uuid, capdata.name);
|
|
continue;
|
|
}
|
|
|
|
cap->Set(user, true);
|
|
}
|
|
}
|
|
delete capmoddata;
|
|
}
|
|
|
|
public:
|
|
ManagerImpl(Module* mod, Events::ModuleEventProvider& evprovref)
|
|
: Cap::Manager(mod)
|
|
, ReloadModule::EventListener(mod)
|
|
, capext(mod)
|
|
, evprov(evprovref)
|
|
{
|
|
managerimpl = this;
|
|
}
|
|
|
|
~ManagerImpl() override
|
|
{
|
|
for (const auto& [_, cap] : caps)
|
|
cap->Unregister();
|
|
}
|
|
|
|
void AddCap(Cap::Capability* cap) override
|
|
{
|
|
// No-op if the cap is already registered.
|
|
// This allows modules to call SetActive() on a cap without checking if it's active first.
|
|
if (cap->IsRegistered())
|
|
return;
|
|
|
|
ServerInstance->Logs.Debug(MODNAME, "Registering cap {}", cap->GetName());
|
|
cap->bit = AllocateBit();
|
|
cap->extitem = &capext;
|
|
caps.emplace(cap->GetName(), cap);
|
|
ServerInstance->Modules.AddReferent("cap/" + cap->GetName(), cap);
|
|
|
|
evprov.Call(&Cap::EventListener::OnCapAddDel, cap, true);
|
|
}
|
|
|
|
void DelCap(Cap::Capability* cap) override
|
|
{
|
|
// No-op if the cap is not registered, see AddCap() above
|
|
if (!cap->IsRegistered())
|
|
return;
|
|
|
|
ServerInstance->Logs.Debug(MODNAME, "Unregistering cap {}", cap->GetName());
|
|
|
|
// Fire the event first so modules can still see who is using the cap which is being unregistered
|
|
evprov.Call(&Cap::EventListener::OnCapAddDel, cap, false);
|
|
|
|
// Turn off the cap for all users
|
|
for (auto* user : ServerInstance->Users.GetLocalUsers())
|
|
cap->Set(user, false);
|
|
|
|
ServerInstance->Modules.DelReferent(cap);
|
|
cap->Unregister();
|
|
caps.erase(cap->GetName());
|
|
}
|
|
|
|
Capability* Find(const std::string& capname) const override
|
|
{
|
|
CapMap::const_iterator it = caps.find(capname);
|
|
if (it != caps.end())
|
|
return it->second;
|
|
return nullptr;
|
|
}
|
|
|
|
void NotifyValueChange(Capability* cap) override
|
|
{
|
|
ServerInstance->Logs.Debug(MODNAME, "Cap {} changed value", cap->GetName());
|
|
evprov.Call(&Cap::EventListener::OnCapValueChange, cap);
|
|
}
|
|
|
|
Protocol GetProtocol(LocalUser* user) const
|
|
{
|
|
return ((capext.Get(user) & CAP_302_BIT) ? CAP_302 : CAP_LEGACY);
|
|
}
|
|
|
|
void Set302Protocol(LocalUser* user)
|
|
{
|
|
capext.Set(user, capext.Get(user) | CAP_302_BIT);
|
|
}
|
|
|
|
bool HandleReq(LocalUser* user, const std::string& reqlist)
|
|
{
|
|
Ext usercaps = capext.Get(user);
|
|
irc::spacesepstream ss(reqlist);
|
|
for (std::string capname; ss.GetToken(capname); )
|
|
{
|
|
bool remove = (capname[0] == '-');
|
|
if (remove)
|
|
capname.erase(capname.begin());
|
|
|
|
Capability* cap = ManagerImpl::Find(capname);
|
|
if ((!cap) || (!CanRequest(user, usercaps, cap, !remove)))
|
|
return false;
|
|
|
|
if (remove)
|
|
usercaps = cap->DelFromMask(usercaps);
|
|
else
|
|
usercaps = cap->AddToMask(usercaps);
|
|
}
|
|
|
|
capext.Set(user, usercaps);
|
|
return true;
|
|
}
|
|
|
|
void HandleList(std::vector<std::string>& out, LocalUser* user, bool show_all, bool show_values, bool minus_prefix = false) const
|
|
{
|
|
Ext show_caps = (show_all ? ~0 : capext.Get(user));
|
|
|
|
for (const auto& [_, cap] : caps)
|
|
{
|
|
if (!(show_caps & cap->GetMask()))
|
|
continue;
|
|
|
|
if ((show_all) && (!cap->OnList(user)))
|
|
continue;
|
|
|
|
std::string token;
|
|
if (minus_prefix)
|
|
token.push_back('-');
|
|
token.append(cap->GetName());
|
|
|
|
if (show_values)
|
|
{
|
|
const std::string* capvalue = cap->GetValue(user);
|
|
if ((capvalue) && (!capvalue->empty()) && (capvalue->find(' ') == std::string::npos))
|
|
{
|
|
token.push_back('=');
|
|
token.append(*capvalue, 0, MAX_VALUE_LENGTH);
|
|
}
|
|
}
|
|
out.push_back(token);
|
|
}
|
|
}
|
|
|
|
void HandleClear(LocalUser* user, std::vector<std::string>& result)
|
|
{
|
|
HandleList(result, user, false, false, true);
|
|
capext.Unset(user);
|
|
}
|
|
};
|
|
|
|
namespace
|
|
{
|
|
std::string SerializeCaps(const Extensible* container, bool human)
|
|
{
|
|
// XXX: Cast away the const because IS_LOCAL() doesn't handle it
|
|
LocalUser* user = IS_LOCAL(const_cast<User*>(static_cast<const User*>(container)));
|
|
if (!user)
|
|
return {};
|
|
|
|
// List requested caps
|
|
std::vector<std::string> result;
|
|
managerimpl->HandleList(result, user, false, false);
|
|
|
|
// Serialize cap protocol version. If building a human-readable string append a
|
|
// new token, otherwise append only a single character indicating the version.
|
|
std::string version;
|
|
if (human)
|
|
version.append("capversion=3.");
|
|
switch (managerimpl->GetProtocol(user))
|
|
{
|
|
case Cap::CAP_302:
|
|
version.push_back('2');
|
|
break;
|
|
default:
|
|
version.push_back('1');
|
|
break;
|
|
}
|
|
result.push_back(version);
|
|
|
|
return insp::join(result, ' ');
|
|
}
|
|
}
|
|
|
|
Cap::ExtItem::ExtItem(Module* mod)
|
|
: IntExtItem(mod, "caps", ExtensionType::USER)
|
|
{
|
|
}
|
|
|
|
std::string Cap::ExtItem::ToHuman(const Extensible* container, void* item) const noexcept
|
|
{
|
|
return SerializeCaps(container, true);
|
|
}
|
|
|
|
std::string Cap::ExtItem::ToInternal(const Extensible* container, void* item) const noexcept
|
|
{
|
|
return SerializeCaps(container, false);
|
|
}
|
|
|
|
void Cap::ExtItem::FromInternal(Extensible* container, const std::string& value) noexcept
|
|
{
|
|
if (container->extype != this->extype)
|
|
return;
|
|
|
|
LocalUser* user = IS_LOCAL(static_cast<User*>(container));
|
|
if (!user)
|
|
return; // Can't happen
|
|
|
|
// Process the cap protocol version which is a single character at the end of the serialized string
|
|
if (value.back() == '2')
|
|
managerimpl->Set302Protocol(user);
|
|
|
|
// Remove the version indicator from the string passed to HandleReq
|
|
std::string caplist(value, 0, value.size()-1);
|
|
managerimpl->HandleReq(user, caplist);
|
|
}
|
|
|
|
class CapMessage final
|
|
: public Cap::MessageBase
|
|
{
|
|
public:
|
|
CapMessage(LocalUser* user, const std::string& subcmd, const std::string& result, bool asterisk)
|
|
: Cap::MessageBase(subcmd)
|
|
{
|
|
SetUser(user);
|
|
if (asterisk)
|
|
PushParam("*");
|
|
PushParamRef(result);
|
|
}
|
|
};
|
|
|
|
class CommandCap final
|
|
: public SplitCommand
|
|
{
|
|
private:
|
|
Events::ModuleEventProvider evprov;
|
|
Cap::ManagerImpl manager;
|
|
ClientProtocol::EventProvider protoevprov;
|
|
|
|
void DisplayResult(LocalUser* user, const std::string& subcmd, const std::vector<std::string>& result, bool asterisk)
|
|
{
|
|
size_t maxline = ServerInstance->Config->Limits.MaxLine - ServerInstance->Config->ServerName.size() - user->nick.length() - subcmd.length() - 11;
|
|
std::string line;
|
|
for (const auto& cap : result)
|
|
{
|
|
if (line.length() + cap.length() < maxline)
|
|
{
|
|
line.append(cap);
|
|
line.push_back(' ');
|
|
}
|
|
else
|
|
{
|
|
DisplaySingleResult(user, subcmd, line, asterisk);
|
|
line.clear();
|
|
}
|
|
}
|
|
DisplaySingleResult(user, subcmd, line, false);
|
|
}
|
|
|
|
void DisplaySingleResult(LocalUser* user, const std::string& subcmd, const std::string& result, bool asterisk)
|
|
{
|
|
CapMessage msg(user, subcmd, result, asterisk);
|
|
ClientProtocol::Event ev(protoevprov, msg);
|
|
user->Send(ev);
|
|
}
|
|
|
|
public:
|
|
BoolExtItem holdext;
|
|
|
|
CommandCap(Module* mod)
|
|
: SplitCommand(mod, "CAP", 1)
|
|
, evprov(mod, "event/cap")
|
|
, manager(mod, evprov)
|
|
, protoevprov(mod, name)
|
|
, holdext(mod, "cap-hold", ExtensionType::USER)
|
|
{
|
|
works_before_reg = true;
|
|
}
|
|
|
|
CmdResult HandleLocal(LocalUser* user, const Params& parameters) override
|
|
{
|
|
if (!user->IsFullyConnected())
|
|
holdext.Set(user);
|
|
|
|
const std::string& subcommand = parameters[0];
|
|
if (irc::equals(subcommand, "REQ"))
|
|
{
|
|
if (parameters.size() < 2)
|
|
return CmdResult::FAILURE;
|
|
|
|
const std::string replysubcmd = (manager.HandleReq(user, parameters[1]) ? "ACK" : "NAK");
|
|
DisplaySingleResult(user, replysubcmd, parameters[1], false);
|
|
}
|
|
else if (irc::equals(subcommand, "END"))
|
|
{
|
|
holdext.Unset(user);
|
|
}
|
|
else if (irc::equals(subcommand, "LS") || irc::equals(subcommand, "LIST"))
|
|
{
|
|
Cap::Protocol capversion = Cap::CAP_LEGACY;
|
|
const bool is_ls = (subcommand.length() == 2);
|
|
if ((is_ls) && (parameters.size() > 1))
|
|
{
|
|
unsigned int version = ConvToNum<unsigned int>(parameters[1]);
|
|
if (version >= 302)
|
|
{
|
|
capversion = Cap::CAP_302;
|
|
manager.Set302Protocol(user);
|
|
}
|
|
}
|
|
|
|
std::vector<std::string> result;
|
|
// Show values only if supports v3.2 and doing LS
|
|
manager.HandleList(result, user, is_ls, ((is_ls) && (capversion != Cap::CAP_LEGACY)));
|
|
DisplayResult(user, subcommand, result, (capversion != Cap::CAP_LEGACY));
|
|
}
|
|
else if (irc::equals(subcommand, "CLEAR") && (manager.GetProtocol(user) == Cap::CAP_LEGACY))
|
|
{
|
|
std::vector<std::string> result;
|
|
manager.HandleClear(user, result);
|
|
DisplayResult(user, "ACK", result, false);
|
|
}
|
|
else
|
|
{
|
|
user->WriteNumeric(ERR_INVALIDCAPCMD, subcommand.empty() ? "*" : subcommand, "Invalid CAP subcommand");
|
|
return CmdResult::FAILURE;
|
|
}
|
|
|
|
return CmdResult::SUCCESS;
|
|
}
|
|
};
|
|
|
|
class PoisonCap final
|
|
: public Cap::Capability
|
|
{
|
|
public:
|
|
PoisonCap(Module* mod)
|
|
: Cap::Capability(mod, "inspircd.org/poison")
|
|
{
|
|
}
|
|
|
|
bool OnRequest(LocalUser* user, bool adding) override
|
|
{
|
|
// Reject the attempt to enable this capability.
|
|
return false;
|
|
}
|
|
};
|
|
|
|
class ModuleCap final
|
|
: public Module
|
|
{
|
|
private:
|
|
CommandCap cmd;
|
|
PoisonCap poisoncap;
|
|
|
|
public:
|
|
ModuleCap()
|
|
: Module(VF_VENDOR, "Provides support for the IRCv3 Client Capability Negotiation extension.")
|
|
, cmd(this)
|
|
, poisoncap(this)
|
|
{
|
|
}
|
|
|
|
ModResult OnCheckReady(LocalUser* user) override
|
|
{
|
|
return (cmd.holdext.Get(user) ? MOD_RES_DENY : MOD_RES_PASSTHRU);
|
|
}
|
|
};
|
|
|
|
MODULE_INIT(ModuleCap)
|