summaryrefslogtreecommitdiff
path: root/modules/operserv/operserv.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'modules/operserv/operserv.cpp')
-rw-r--r--modules/operserv/operserv.cpp303
1 files changed, 303 insertions, 0 deletions
diff --git a/modules/operserv/operserv.cpp b/modules/operserv/operserv.cpp
new file mode 100644
index 000000000..0881b7654
--- /dev/null
+++ b/modules/operserv/operserv.cpp
@@ -0,0 +1,303 @@
+/* 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 SGLineManager final
+ : public XLineManager
+{
+public:
+ SGLineManager(Module *creator) : XLineManager(creator, "xlinemanager/sgline", 'G') { }
+
+ void OnMatch(User *u, XLine *x) override
+ {
+ this->Send(u, x);
+ }
+
+ void OnExpire(const XLine *x) override
+ {
+ Log(Config->GetClient("OperServ"), "expire/akill") << "AKILL on \002" << x->mask << "\002 has expired";
+ }
+
+ void Send(User *u, XLine *x) override
+ {
+ IRCD->SendAkill(u, x);
+ }
+
+ void SendDel(XLine *x) override
+ {
+ IRCD->SendAkillDel(x);
+ }
+
+ bool Check(User *u, const XLine *x) override
+ {
+ if (x->regex)
+ {
+ Anope::string uh = u->GetIdent() + "@" + u->host, nuhr = u->nick + "!" + uh + "#" + u->realname;
+ return x->regex->Matches(uh) || x->regex->Matches(nuhr);
+ }
+
+ if (!x->GetNick().empty() && !Anope::Match(u->nick, x->GetNick()))
+ return false;
+
+ if (!x->GetUser().empty() && !Anope::Match(u->GetIdent(), x->GetUser()))
+ return false;
+
+ if (!x->GetReal().empty() && !Anope::Match(u->realname, x->GetReal()))
+ return false;
+
+ if (x->c && x->c->match(u->ip))
+ return true;
+
+ if (x->GetHost().empty() || Anope::Match(u->host, x->GetHost()) || Anope::Match(u->ip.addr(), x->GetHost()))
+ return true;
+
+ return false;
+ }
+};
+
+class SQLineManager final
+ : public XLineManager
+{
+ ServiceReference<NickServService> nickserv;
+
+public:
+ SQLineManager(Module *creator) : XLineManager(creator, "xlinemanager/sqline", 'Q'), nickserv("NickServService", "NickServ") { }
+
+ void OnMatch(User *u, XLine *x) override
+ {
+ this->Send(u, x);
+ }
+
+ void OnExpire(const XLine *x) override
+ {
+ Log(Config->GetClient("OperServ"), "expire/sqline") << "SQLINE on \002" << x->mask << "\002 has expired";
+ }
+
+ void Send(User *u, XLine *x) override
+ {
+ if (!IRCD->CanSQLine)
+ {
+ if (!u)
+ ;
+ else if (nickserv)
+ nickserv->Collide(u, NULL);
+ else
+ u->Kill(Config->GetClient("OperServ"), "Q-Lined: " + x->reason);
+ }
+ else if (x->IsRegex())
+ {
+ if (u)
+ u->Kill(Config->GetClient("OperServ"), "Q-Lined: " + x->reason);
+ }
+ else if (x->mask[0] != '#' || IRCD->CanSQLineChannel)
+ {
+ IRCD->SendSQLine(u, x);
+ /* If it is an oper, assume they're walking it, otherwise kill for good measure */
+ if (u && !u->HasMode("OPER"))
+ u->Kill(Config->GetClient("OperServ"), "Q-Lined: " + x->reason);
+ }
+ }
+
+ void SendDel(XLine *x) override
+ {
+ if (!IRCD->CanSQLine || x->IsRegex())
+ ;
+ else if (x->mask[0] != '#' || IRCD->CanSQLineChannel)
+ IRCD->SendSQLineDel(x);
+ }
+
+ bool Check(User *u, const XLine *x) override
+ {
+ if (x->regex)
+ return x->regex->Matches(u->nick);
+ return Anope::Match(u->nick, x->mask);
+ }
+
+ XLine *CheckChannel(Channel *c)
+ {
+ for (auto *x : this->GetList())
+ {
+ if (!Anope::NoExpire && x->expires && x->expires < Anope::CurTime)
+ continue; // Skip expired lines.
+
+ if (x->regex)
+ {
+ if (x->regex->Matches(c->name))
+ return x;
+ }
+ else
+ {
+ if (x->mask.empty() || x->mask[0] != '#')
+ continue;
+
+ if (Anope::Match(c->name, x->mask, false, true))
+ return x;
+ }
+ }
+ return NULL;
+ }
+};
+
+class SNLineManager final
+ : public XLineManager
+{
+public:
+ SNLineManager(Module *creator) : XLineManager(creator, "xlinemanager/snline", 'N') { }
+
+ void OnMatch(User *u, XLine *x) override
+ {
+ this->Send(u, x);
+ }
+
+ void OnExpire(const XLine *x) override
+ {
+ Log(Config->GetClient("OperServ"), "expire/snline") << "SNLINE on \002" << x->mask << "\002 has expired";
+ }
+
+ void Send(User *u, XLine *x) override
+ {
+ if (IRCD->CanSNLine && !x->IsRegex())
+ IRCD->SendSGLine(u, x);
+
+ if (u)
+ u->Kill(Config->GetClient("OperServ"), "SNLined: " + x->reason);
+ }
+
+ void SendDel(XLine *x) override
+ {
+ if (IRCD->CanSNLine && !x->IsRegex())
+ IRCD->SendSGLineDel(x);
+ }
+
+ bool Check(User *u, const XLine *x) override
+ {
+ if (x->regex)
+ return x->regex->Matches(u->realname);
+ return Anope::Match(u->realname, x->mask, false, true);
+ }
+};
+
+class OperServCore final
+ : public Module
+{
+ Reference<BotInfo> OperServ;
+ SGLineManager sglines;
+ SQLineManager sqlines;
+ SNLineManager snlines;
+
+public:
+ OperServCore(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, PSEUDOCLIENT | VENDOR),
+ sglines(this), sqlines(this), snlines(this)
+ {
+
+ /* Yes, these are in this order for a reason. Most violent->least violent. */
+ XLineManager::RegisterXLineManager(&sglines);
+ XLineManager::RegisterXLineManager(&sqlines);
+ XLineManager::RegisterXLineManager(&snlines);
+ }
+
+ ~OperServCore() override
+ {
+ this->sglines.Clear();
+ this->sqlines.Clear();
+ this->snlines.Clear();
+
+ XLineManager::UnregisterXLineManager(&sglines);
+ XLineManager::UnregisterXLineManager(&sqlines);
+ XLineManager::UnregisterXLineManager(&snlines);
+ }
+
+ void OnReload(Configuration::Conf *conf) override
+ {
+ const Anope::string &osnick = conf->GetModule(this)->Get<const Anope::string>("client");
+
+ if (osnick.empty())
+ throw ConfigException(this->name + ": <client> must be defined");
+
+ BotInfo *bi = BotInfo::Find(osnick, true);
+ if (!bi)
+ throw ConfigException(this->name + ": no bot named " + osnick);
+
+ OperServ = bi;
+ }
+
+ EventReturn OnBotPrivmsg(User *u, BotInfo *bi, Anope::string &message, const Anope::map<Anope::string> &tags) override
+ {
+ if (bi == OperServ && !u->HasMode("OPER") && Config->GetModule(this)->Get<bool>("opersonly"))
+ {
+ u->SendMessage(bi, ACCESS_DENIED);
+ Log(bi, "bados") << "Denied access to " << bi->nick << " from " << u->GetMask() << " (non-oper)";
+ return EVENT_STOP;
+ }
+
+ return EVENT_CONTINUE;
+ }
+
+ void OnServerQuit(Server *server) override
+ {
+ if (server->IsJuped())
+ Log(server, "squit", OperServ) << "Received SQUIT for juped server " << server->GetName();
+ }
+
+ void OnUserModeSet(const MessageSource &setter, User *u, const Anope::string &mname) override
+ {
+ if (mname == "OPER")
+ Log(u, "oper", OperServ) << "is now an IRC operator.";
+ }
+
+ void OnUserModeUnset(const MessageSource &setter, User *u, const Anope::string &mname) override
+ {
+ if (mname == "OPER")
+ Log(u, "oper", OperServ) << "is no longer an IRC operator";
+ }
+
+ void OnUserConnect(User *u, bool &exempt) override
+ {
+ if (!u->Quitting() && !exempt)
+ XLineManager::CheckAll(u);
+ }
+
+ void OnUserNickChange(User *u, const Anope::string &oldnick) override
+ {
+ if (!u->HasMode("OPER"))
+ this->sqlines.CheckAllXLines(u);
+ }
+
+ EventReturn OnCheckKick(User *u, Channel *c, Anope::string &mask, Anope::string &reason) override
+ {
+ XLine *x = this->sqlines.CheckChannel(c);
+ if (x)
+ {
+ this->sqlines.OnMatch(u, x);
+ reason = x->reason;
+ return EVENT_STOP;
+ }
+
+ return EVENT_CONTINUE;
+ }
+
+ EventReturn OnPreHelp(CommandSource &source, const std::vector<Anope::string> &params) override
+ {
+ if (!params.empty() || source.c || source.service != *OperServ)
+ return EVENT_CONTINUE;
+ source.Reply(_("%s commands:"), OperServ->nick.c_str());
+ return EVENT_CONTINUE;
+ }
+
+ void OnLog(Log *l) override
+ {
+ if (l->type == LOG_SERVER)
+ l->bi = OperServ;
+ }
+};
+
+MODULE_INIT(OperServCore)