/* 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 * Included in the Anope module pack since Anope 1.7.11 * Anope Coder: GeniusDex * * 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 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(obj); else req = na->Extend("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 ¶ms) 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("senddelay"); if (Config->GetModule(this->owner)->Get("memooper") && send_delay > 0 && u && u->lastmemosend + send_delay > Anope::CurTime) { auto waitperiod = (u->lastmemosend + send_delay) - Anope::CurTime; source.Reply(_("Please wait %s before requesting a new vhost."), Anope::Duration(waitperiod, source.GetAccount()).c_str()); u->lastmemosend = Anope::CurTime; return; } HostRequestImpl req(na); req.nick = source.GetNick(); req.ident = user; req.host = host; req.time = Anope::CurTime; na->Extend("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 ¶ms) 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("hostrequest") : NULL; if (req) { na->SetVHost(req->ident, req->host, source.GetNick(), req->time); FOREACH_MOD(OnSetVHost, (na)); if (Config->GetModule(this->owner)->Get("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("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("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 ¶ms) 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("hostrequest") : NULL; if (req) { na->Shrink("hostrequest"); if (Config->GetModule(this->owner)->Get("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("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 ¶ms) override { unsigned counter = 0; unsigned display_counter = 0, listmax = Config->GetModule(this->owner)->Get("listmax"); ListFormatter list(source.GetAccount()); list.AddColumn(_("Number")).AddColumn(_("Nick")).AddColumn(_("VHost")).AddColumn(_("Created")); for (const auto &[nick, na] : *NickAliasList) { HostRequestImpl *hr = na->GetExt("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 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 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("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)