diff options
Diffstat (limited to 'modules/operserv/os_svs.cpp')
-rw-r--r-- | modules/operserv/os_svs.cpp | 182 |
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> ¶ms) 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> ¶ms) 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> ¶ms) 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) |