1
0
mirror of https://github.com/anope/anope.git synced 2026-06-23 08:46:37 +02:00
Files
anope/modules/operserv/os_svs.cpp
T
2025-11-12 19:49:59 +00:00

185 lines
5.7 KiB
C++

// Anope IRC Services <https://www.anope.org/>
//
// Copyright (C) 2003-2025 Anope Contributors
//
// Anope is free software. You can use, modify, and/or distribute it under the
// terms of version 2 of the GNU General Public License. See docs/LICENSE.txt
// for the complete terms of this license and docs/AUTHORS.txt for a list of
// contributors.
//
// Based on the original code of Epona by Lara
// Based on the original code of Services by Andy Church
//
// SPDX-License-Identifier: GPL-2.0-only
#include "module.h"
class CommandOSSVSNick final
: public Command
{
public:
CommandOSSVSNick(Module *creator) : Command(creator, "operserv/svsnick", 2, 2)
{
this->SetDesc(_("Forcefully change a user's nickname"));
this->SetSyntax(_("\037nick\037 \037newnick\037"));
}
void Execute(CommandSource &source, const std::vector<Anope::string> &params) override
{
const Anope::string &nick = params[0];
Anope::string newnick = params[1];
User *u2;
if (!IRCD->CanSVSNick)
{
source.Reply(_("Your IRCd does not support SVSNICK."));
return;
}
/* Truncate long nicknames to nicklen characters */
size_t nicklen = IRCD->MaxNick;
if (newnick.length() > nicklen)
{
source.Reply(_("Nick \002%s\002 was truncated to %zu characters."), newnick.c_str(), nicklen);
newnick = params[1].substr(0, nicklen);
}
/* Check for valid characters */
if (!IRCD->IsNickValid(newnick))
{
source.Reply(_("Nick \002%s\002 is an illegal nickname and cannot be used."), newnick.c_str());
return;
}
/* Check for a nick in use or a forbidden/suspended nick */
if (!(u2 = User::Find(nick, true)))
source.Reply(NICK_X_NOT_IN_USE, nick.c_str());
else if (!nick.equals_ci(newnick) && User::Find(newnick))
source.Reply(_("Nick \002%s\002 is currently in use."), newnick.c_str());
else
{
source.Reply(_("The nick \002%s\002 is now being changed to \002%s\002."), nick.c_str(), newnick.c_str());
Log(LOG_ADMIN, source, this) << "to change " << nick << " to " << newnick;
IRCD->SendForceNickChange(u2, newnick, Anope::CurTime);
}
}
bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
{
this->SendSyntax(source);
source.Reply(" ");
source.Reply(_("Forcefully changes a user's nickname from \037nick\037 to \037newnick\037."));
return true;
}
};
class CommandOSSVSJoin final
: public Command
{
public:
CommandOSSVSJoin(Module *creator) : Command(creator, "operserv/svsjoin", 2, 2)
{
this->SetDesc(_("Forcefully join a user to a channel"));
this->SetSyntax(_("\037nick\037 \037channel\037"));
}
void Execute(CommandSource &source, const std::vector<Anope::string> &params) override
{
if (!IRCD->CanSVSJoin)
{
source.Reply(_("Your IRCd does not support SVSJOIN."));
return;
}
User *target = User::Find(params[0], true);
Channel *c = Channel::Find(params[1]);
if (target == NULL)
source.Reply(NICK_X_NOT_IN_USE, params[0].c_str());
else if (source.GetUser() != target && (target->IsProtected() || target->server == Me))
source.Reply(ACCESS_DENIED);
else if (!IRCD->IsChannelValid(params[1]))
source.Reply(CHAN_X_INVALID, params[1].c_str());
else if (c && c->FindUser(target))
source.Reply(_("\002%s\002 is already in \002%s\002."), target->nick.c_str(), c->name.c_str());
else
{
IRCD->SendSVSJoin(*source.service, target, params[1], "");
Log(LOG_ADMIN, source, this) << "to force " << target->nick << " to join " << params[1];
source.Reply(_("\002%s\002 has been joined to \002%s\002."), target->nick.c_str(), params[1].c_str());
}
}
bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
{
this->SendSyntax(source);
source.Reply(" ");
source.Reply(_("Forcefully join a user to a channel."));
return true;
}
};
class CommandOSSVSPart final
: public Command
{
public:
CommandOSSVSPart(Module *creator) : Command(creator, "operserv/svspart", 2, 3)
{
this->SetDesc(_("Forcefully part a user from a channel"));
this->SetSyntax(_("\037nick\037 \037channel\037 [\037reason\037]"));
}
void Execute(CommandSource &source, const std::vector<Anope::string> &params) override
{
if (!IRCD->CanSVSJoin)
{
source.Reply(_("Your IRCd does not support SVSPART."));
return;
}
User *target = User::Find(params[0], true);
Channel *c = Channel::Find(params[1]);
const Anope::string &reason = params.size() > 2 ? params[2] : "";
if (target == NULL)
source.Reply(NICK_X_NOT_IN_USE, params[0].c_str());
else if (source.GetUser() != target && (target->IsProtected() || target->server == Me))
source.Reply(ACCESS_DENIED);
else if (!c)
source.Reply(CHAN_X_NOT_IN_USE, params[1].c_str());
else if (!c->FindUser(target))
source.Reply(_("\002%s\002 is not in \002%s\002."), target->nick.c_str(), c->name.c_str());
else
{
IRCD->SendSVSPart(*source.service, target, params[1], reason);
if (!reason.empty())
Log(LOG_ADMIN, source, this) << "to force " << target->nick << " to part " << c->name << " with reason " << reason;
else
Log(LOG_ADMIN, source, this) << "to force " << target->nick << " to part " << c->name;
source.Reply(_("\002%s\002 has been parted from \002%s\002."), target->nick.c_str(), c->name.c_str());
}
}
bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
{
this->SendSyntax(source);
source.Reply(" ");
source.Reply(_("Forcefully part a user from a channel."));
return true;
}
};
class OSSVS final
: public Module
{
CommandOSSVSNick commandossvsnick;
CommandOSSVSJoin commandossvsjoin;
CommandOSSVSPart commandossvspart;
public:
OSSVS(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, VENDOR),
commandossvsnick(this), commandossvsjoin(this), commandossvspart(this)
{
}
};
MODULE_INIT(OSSVS)