mirror of
https://github.com/anope/anope.git
synced 2026-06-23 18:16:36 +02:00
307 lines
7.3 KiB
C++
307 lines
7.3 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 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> ¶ms) 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)
|