summaryrefslogtreecommitdiff
path: root/modules/operserv/os_svs.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'modules/operserv/os_svs.cpp')
-rw-r--r--modules/operserv/os_svs.cpp182
1 files changed, 182 insertions, 0 deletions
diff --git a/modules/operserv/os_svs.cpp b/modules/operserv/os_svs.cpp
new file mode 100644
index 000000000..854fd4af2
--- /dev/null
+++ b/modules/operserv/os_svs.cpp
@@ -0,0 +1,182 @@
+/* OperServ core functions
+ *
+ * (C) 2003-2024 Anope Team
+ * Contact us at team@anope.org
+ *
+ * Please read COPYING and README for further details.
+ *
+ * Based on the original code of Epona by Lara.
+ * Based on the original code of Services by Andy Church.
+ */
+
+#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);
+ }
+ return;
+ }
+
+ 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)