summaryrefslogtreecommitdiff
path: root/modules/hostserv/hs_request.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'modules/hostserv/hs_request.cpp')
-rw-r--r--modules/hostserv/hs_request.cpp397
1 files changed, 397 insertions, 0 deletions
diff --git a/modules/hostserv/hs_request.cpp b/modules/hostserv/hs_request.cpp
new file mode 100644
index 000000000..e6f30fd70
--- /dev/null
+++ b/modules/hostserv/hs_request.cpp
@@ -0,0 +1,397 @@
+/* hs_request.c - Add request and activate functionality to HostServ
+ *
+ * (C) 2003-2024 Anope Team
+ * Contact us at team@anope.org
+ *
+ * Based on the original module by Rob <rob@anope.org>
+ * Included in the Anope module pack since Anope 1.7.11
+ * Anope Coder: GeniusDex <geniusdex@anope.org>
+ *
+ * Please read COPYING and README for further details.
+ *
+ * Send bug reports to the Anope Coder instead of the module
+ * author, because any changes since the inclusion into anope
+ * are not supported by the original author.
+ */
+
+#include "module.h"
+#include "modules/hs_request.h"
+
+static ServiceReference<MemoServService> memoserv("MemoServService", "MemoServ");
+
+static void req_send_memos(Module *me, CommandSource &source, const Anope::string &vident, const Anope::string &vhost);
+
+struct HostRequestImpl final
+ : HostRequest
+ , Serializable
+{
+ HostRequestImpl(Extensible *)
+ : Serializable("HostRequest")
+ {
+ }
+
+ void Serialize(Serialize::Data &data) const override
+ {
+ data.Store("nick", this->nick);
+ data.Store("ident", this->ident);
+ data.Store("host", this->host);
+ data.Store("time", this->time);
+ }
+
+ static Serializable *Unserialize(Serializable *obj, Serialize::Data &data)
+ {
+ Anope::string snick;
+ data["nick"] >> snick;
+
+ NickAlias *na = NickAlias::Find(snick);
+ if (na == NULL)
+ return NULL;
+
+ HostRequestImpl *req;
+ if (obj)
+ req = anope_dynamic_static_cast<HostRequestImpl *>(obj);
+ else
+ req = na->Extend<HostRequestImpl>("hostrequest");
+ if (req)
+ {
+ req->nick = na->nick;
+ data["ident"] >> req->ident;
+ data["host"] >> req->host;
+ data["time"] >> req->time;
+ }
+
+ return req;
+ }
+};
+
+class CommandHSRequest final
+ : public Command
+{
+ static bool isvalidchar(char c)
+ {
+ return (c >= 'A' && c <= 'Z') || (c >= 'a' && c <= 'z') || (c >= '0' && c <= '9') || c == '.' || c == '-';
+ }
+
+public:
+ CommandHSRequest(Module *creator) : Command(creator, "hostserv/request", 1, 1)
+ {
+ this->SetDesc(_("Request a vhost for your nick"));
+ this->SetSyntax(_("vhost"));
+ }
+
+ void Execute(CommandSource &source, const std::vector<Anope::string> &params) override
+ {
+ if (Anope::ReadOnly)
+ {
+ source.Reply(READ_ONLY_MODE);
+ return;
+ }
+
+ User *u = source.GetUser();
+ NickAlias *na = NickAlias::Find(source.GetNick());
+ if (!na || na->nc != source.GetAccount())
+ {
+ source.Reply(ACCESS_DENIED);
+ return;
+ }
+
+ if (source.GetAccount()->HasExt("UNCONFIRMED"))
+ {
+ source.Reply(_("You must confirm your account before you may request a vhost."));
+ return;
+ }
+
+ Anope::string rawhostmask = params[0];
+
+ Anope::string user, host;
+ size_t a = rawhostmask.find('@');
+
+ if (a == Anope::string::npos)
+ host = rawhostmask;
+ else
+ {
+ user = rawhostmask.substr(0, a);
+ host = rawhostmask.substr(a + 1);
+ }
+
+ if (host.empty())
+ {
+ this->OnSyntaxError(source, "");
+ return;
+ }
+
+ if (!user.empty())
+ {
+ if (user.length() > IRCD->MaxUser)
+ {
+ source.Reply(HOST_SET_IDENTTOOLONG, IRCD->MaxUser);
+ return;
+ }
+ else if (!IRCD->CanSetVIdent)
+ {
+ source.Reply(HOST_NO_VIDENT);
+ return;
+ }
+ for (const auto &chr : user)
+ {
+ if (!isvalidchar(chr))
+ {
+ source.Reply(HOST_SET_IDENT_ERROR);
+ return;
+ }
+ }
+ }
+
+ if (host.length() > IRCD->MaxHost)
+ {
+ source.Reply(HOST_SET_TOOLONG, IRCD->MaxHost);
+ return;
+ }
+
+ if (!IRCD->IsHostValid(host))
+ {
+ source.Reply(HOST_SET_ERROR);
+ return;
+ }
+
+ time_t send_delay = Config->GetModule("memoserv")->Get<time_t>("senddelay");
+ if (Config->GetModule(this->owner)->Get<bool>("memooper") && send_delay > 0 && u && u->lastmemosend + send_delay > Anope::CurTime)
+ {
+ source.Reply(_("Please wait %lu seconds before requesting a new vhost."), (unsigned long)send_delay);
+ u->lastmemosend = Anope::CurTime;
+ return;
+ }
+
+ HostRequestImpl req(na);
+ req.nick = source.GetNick();
+ req.ident = user;
+ req.host = host;
+ req.time = Anope::CurTime;
+ na->Extend<HostRequestImpl>("hostrequest", req);
+
+ source.Reply(_("Your vhost has been requested."));
+ req_send_memos(owner, source, user, host);
+ Log(LOG_COMMAND, source, this) << "to request new vhost " << (!user.empty() ? user + "@" : "") << host;
+ }
+
+ bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
+ {
+ this->SendSyntax(source);
+ source.Reply(" ");
+ source.Reply(_("Request the given vhost to be activated for your nick by the\n"
+ "network administrators. Please be patient while your request\n"
+ "is being considered."));
+ return true;
+ }
+};
+
+class CommandHSActivate final
+ : public Command
+{
+public:
+ CommandHSActivate(Module *creator) : Command(creator, "hostserv/activate", 1, 1)
+ {
+ this->SetDesc(_("Approve the requested vhost of a user"));
+ this->SetSyntax(_("\037nick\037"));
+ }
+
+ void Execute(CommandSource &source, const std::vector<Anope::string> &params) override
+ {
+ if (Anope::ReadOnly)
+ {
+ source.Reply(READ_ONLY_MODE);
+ return;
+ }
+
+ const Anope::string &nick = params[0];
+
+ NickAlias *na = NickAlias::Find(nick);
+ HostRequestImpl *req = na ? na->GetExt<HostRequestImpl>("hostrequest") : NULL;
+ if (req)
+ {
+ na->SetVHost(req->ident, req->host, source.GetNick(), req->time);
+ FOREACH_MOD(OnSetVHost, (na));
+
+ if (Config->GetModule(this->owner)->Get<bool>("memouser") && memoserv)
+ memoserv->Send(source.service->nick, na->nick, _("[auto memo] Your requested vhost has been approved."), true);
+
+ source.Reply(_("VHost for %s has been activated."), na->nick.c_str());
+ Log(LOG_COMMAND, source, this) << "for " << na->nick << " for vhost " << (!req->ident.empty() ? req->ident + "@" : "") << req->host;
+ na->Shrink<HostRequestImpl>("hostrequest");
+ }
+ else
+ source.Reply(_("No request for nick %s found."), nick.c_str());
+ }
+
+ bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
+ {
+ this->SendSyntax(source);
+ source.Reply(" ");
+ source.Reply(_("Activate the requested vhost for the given nick."));
+ if (Config->GetModule(this->owner)->Get<bool>("memouser"))
+ source.Reply(_("A memo informing the user will also be sent."));
+
+ return true;
+ }
+};
+
+class CommandHSReject final
+ : public Command
+{
+public:
+ CommandHSReject(Module *creator) : Command(creator, "hostserv/reject", 1, 2)
+ {
+ this->SetDesc(_("Reject the requested vhost of a user"));
+ this->SetSyntax(_("\037nick\037 [\037reason\037]"));
+ }
+
+ void Execute(CommandSource &source, const std::vector<Anope::string> &params) override
+ {
+ if (Anope::ReadOnly)
+ {
+ source.Reply(READ_ONLY_MODE);
+ return;
+ }
+
+ const Anope::string &nick = params[0];
+ const Anope::string &reason = params.size() > 1 ? params[1] : "";
+
+ NickAlias *na = NickAlias::Find(nick);
+ HostRequestImpl *req = na ? na->GetExt<HostRequestImpl>("hostrequest") : NULL;
+ if (req)
+ {
+ na->Shrink<HostRequestImpl>("hostrequest");
+
+ if (Config->GetModule(this->owner)->Get<bool>("memouser") && memoserv)
+ {
+ Anope::string message;
+ if (!reason.empty())
+ message = Anope::printf(_("[auto memo] Your requested vhost has been rejected. Reason: %s"), reason.c_str());
+ else
+ message = _("[auto memo] Your requested vhost has been rejected.");
+
+ memoserv->Send(source.service->nick, nick, Language::Translate(source.GetAccount(), message.c_str()), true);
+ }
+
+ source.Reply(_("VHost for %s has been rejected."), nick.c_str());
+ Log(LOG_COMMAND, source, this) << "to reject vhost for " << nick << " (" << (!reason.empty() ? reason : "no reason") << ")";
+ }
+ else
+ source.Reply(_("No request for nick %s found."), nick.c_str());
+ }
+
+ bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
+ {
+ this->SendSyntax(source);
+ source.Reply(" ");
+ source.Reply(_("Reject the requested vhost for the given nick."));
+ if (Config->GetModule(this->owner)->Get<bool>("memouser"))
+ source.Reply(_("A memo informing the user will also be sent, which includes the reason for the rejection if supplied."));
+
+ return true;
+ }
+};
+
+class CommandHSWaiting final
+ : public Command
+{
+public:
+ CommandHSWaiting(Module *creator) : Command(creator, "hostserv/waiting", 0, 0)
+ {
+ this->SetDesc(_("Retrieves the vhost requests"));
+ }
+
+ void Execute(CommandSource &source, const std::vector<Anope::string> &params) override
+ {
+ unsigned counter = 0;
+ unsigned display_counter = 0, listmax = Config->GetModule(this->owner)->Get<unsigned>("listmax");
+ ListFormatter list(source.GetAccount());
+
+ list.AddColumn(_("Number")).AddColumn(_("Nick")).AddColumn(_("VHost")).AddColumn(_("Created"));
+
+ for (const auto &[nick, na] : *NickAliasList)
+ {
+ HostRequestImpl *hr = na->GetExt<HostRequestImpl>("hostrequest");
+ if (!hr)
+ continue;
+
+ if (!listmax || display_counter < listmax)
+ {
+ ++display_counter;
+
+ ListFormatter::ListEntry entry;
+ entry["Number"] = Anope::ToString(display_counter);
+ entry["Nick"] = nick;
+ if (!hr->ident.empty())
+ entry["VHost"] = hr->ident + "@" + hr->host;
+ else
+ entry["VHost"] = hr->host;
+ entry["Created"] = Anope::strftime(hr->time, NULL, true);
+ list.AddEntry(entry);
+ }
+ ++counter;
+ }
+
+ std::vector<Anope::string> replies;
+ list.Process(replies);
+
+ for (const auto &reply : replies)
+ source.Reply(reply);
+
+ source.Reply(_("Displayed \002%d\002 records (\002%d\002 total)."), display_counter, counter);
+ }
+
+ bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
+ {
+ this->SendSyntax(source);
+ source.Reply(" ");
+ source.Reply(_("This command retrieves the vhost requests."));
+
+ return true;
+ }
+};
+
+class HSRequest final
+ : public Module
+{
+ CommandHSRequest commandhsrequest;
+ CommandHSActivate commandhsactive;
+ CommandHSReject commandhsreject;
+ CommandHSWaiting commandhswaiting;
+ ExtensibleItem<HostRequestImpl> hostrequest;
+ Serialize::Type request_type;
+
+public:
+ HSRequest(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, VENDOR),
+ commandhsrequest(this), commandhsactive(this),
+ commandhsreject(this), commandhswaiting(this), hostrequest(this, "hostrequest"), request_type("HostRequest", HostRequestImpl::Unserialize)
+ {
+ if (!IRCD || !IRCD->CanSetVHost)
+ throw ModuleException("Your IRCd does not support vhosts");
+ }
+};
+
+static void req_send_memos(Module *me, CommandSource &source, const Anope::string &vident, const Anope::string &vhost)
+{
+ Anope::string host;
+ if (!vident.empty())
+ host = vident + "@" + vhost;
+ else
+ host = vhost;
+
+ if (Config->GetModule(me)->Get<bool>("memooper") && memoserv)
+ {
+ for (auto *o : Oper::opers)
+ {
+ const NickAlias *na = NickAlias::Find(o->name);
+ if (!na)
+ continue;
+
+ Anope::string message = Anope::printf(_("[auto memo] VHost \002%s\002 has been requested by %s."), host.c_str(), source.GetNick().c_str());
+
+ memoserv->Send(source.service->nick, na->nick, message, true);
+ }
+ }
+}
+
+MODULE_INIT(HSRequest)