mirror of
https://github.com/anope/anope.git
synced 2026-06-12 18:54:47 +02:00
296 lines
8.4 KiB
C++
296 lines
8.4 KiB
C++
/* ChanServ core functions
|
|
*
|
|
* (C) 2003-2025 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"
|
|
#include "modules/cs_entrymsg.h"
|
|
|
|
struct EntryMsgImpl final
|
|
: EntryMsg
|
|
, Serializable
|
|
{
|
|
EntryMsgImpl() : Serializable("EntryMsg")
|
|
{
|
|
}
|
|
|
|
EntryMsgImpl(ChannelInfo *c, const Anope::string &cname, const Anope::string &cmessage, time_t ct = Anope::CurTime) : Serializable("EntryMsg")
|
|
{
|
|
this->chan = c->name;
|
|
this->creator = cname;
|
|
this->message = cmessage;
|
|
this->when = ct;
|
|
}
|
|
|
|
~EntryMsgImpl() override;
|
|
|
|
void Serialize(Serialize::Data &data) const override
|
|
{
|
|
data.Store("ci", this->chan);
|
|
data.Store("creator", this->creator);
|
|
data.Store("message", this->message);
|
|
data.Store("when", this->when);
|
|
}
|
|
|
|
static Serializable *Unserialize(Serializable *obj, Serialize::Data &data);
|
|
};
|
|
|
|
struct EntryMessageListImpl final
|
|
: EntryMessageList
|
|
{
|
|
EntryMessageListImpl(Extensible *) { }
|
|
|
|
EntryMsg *Create() override
|
|
{
|
|
return new EntryMsgImpl();
|
|
}
|
|
};
|
|
|
|
EntryMsgImpl::~EntryMsgImpl()
|
|
{
|
|
ChannelInfo *ci = ChannelInfo::Find(this->chan);
|
|
if (!ci)
|
|
return;
|
|
|
|
EntryMessageList *messages = ci->GetExt<EntryMessageList>("entrymsg");
|
|
if (!messages)
|
|
return;
|
|
|
|
std::vector<EntryMsg *>::iterator it = std::find((*messages)->begin(), (*messages)->end(), this);
|
|
if (it != (*messages)->end())
|
|
(*messages)->erase(it);
|
|
}
|
|
|
|
|
|
Serializable *EntryMsgImpl::Unserialize(Serializable *obj, Serialize::Data &data)
|
|
{
|
|
Anope::string sci, screator, smessage;
|
|
time_t swhen;
|
|
|
|
data["ci"] >> sci;
|
|
data["creator"] >> screator;
|
|
data["message"] >> smessage;
|
|
|
|
ChannelInfo *ci = ChannelInfo::Find(sci);
|
|
if (!ci)
|
|
return NULL;
|
|
|
|
if (obj)
|
|
{
|
|
EntryMsgImpl *msg = anope_dynamic_static_cast<EntryMsgImpl *>(obj);
|
|
msg->chan = ci->name;
|
|
data["creator"] >> msg->creator;
|
|
data["message"] >> msg->message;
|
|
data["when"] >> msg->when;
|
|
return msg;
|
|
}
|
|
|
|
EntryMessageList *messages = ci->Require<EntryMessageList>("entrymsg");
|
|
|
|
data["when"] >> swhen;
|
|
|
|
auto *m = new EntryMsgImpl(ci, screator, smessage, swhen);
|
|
(*messages)->push_back(m);
|
|
return m;
|
|
}
|
|
|
|
class CommandEntryMessage final
|
|
: public Command
|
|
{
|
|
private:
|
|
static void DoList(CommandSource &source, ChannelInfo *ci)
|
|
{
|
|
EntryMessageList *messages = ci->Require<EntryMessageList>("entrymsg");
|
|
|
|
if ((*messages)->empty())
|
|
{
|
|
source.Reply(_("Entry message list for \002%s\002 is empty."), ci->name.c_str());
|
|
return;
|
|
}
|
|
|
|
source.Reply(_("Entry message list for \002%s\002:"), ci->name.c_str());
|
|
|
|
ListFormatter list(source.GetAccount());
|
|
list.AddColumn(_("Number")).AddColumn(_("Creator")).AddColumn(_("Created")).AddColumn(_("Message"));
|
|
for (unsigned i = 0; i < (*messages)->size(); ++i)
|
|
{
|
|
EntryMsg *msg = (*messages)->at(i);
|
|
|
|
ListFormatter::ListEntry entry;
|
|
entry["Number"] = Anope::ToString(i + 1);
|
|
entry["Creator"] = msg->creator;
|
|
entry["Created"] = Anope::strftime(msg->when, NULL, true);
|
|
entry["Message"] = msg->message;
|
|
list.AddEntry(entry);
|
|
}
|
|
|
|
std::vector<Anope::string> replies;
|
|
list.Process(replies);
|
|
for (const auto &reply : replies)
|
|
source.Reply(reply);
|
|
|
|
source.Reply(_("End of entry message list."));
|
|
}
|
|
|
|
void DoAdd(CommandSource &source, ChannelInfo *ci, const Anope::string &message)
|
|
{
|
|
EntryMessageList *messages = ci->Require<EntryMessageList>("entrymsg");
|
|
|
|
if ((*messages)->size() >= Config->GetModule(this->owner)->Get<unsigned>("maxentries"))
|
|
source.Reply(_("The entry message list for \002%s\002 is full."), ci->name.c_str());
|
|
else
|
|
{
|
|
(*messages)->push_back(new EntryMsgImpl(ci, source.GetNick(), message));
|
|
Log(source.AccessFor(ci).HasPriv("SET") ? LOG_COMMAND : LOG_OVERRIDE, source, this, ci) << "to add a message";
|
|
source.Reply(_("Entry message added to \002%s\002"), ci->name.c_str());
|
|
}
|
|
}
|
|
|
|
void DoDel(CommandSource &source, ChannelInfo *ci, const Anope::string &message)
|
|
{
|
|
EntryMessageList *messages = ci->Require<EntryMessageList>("entrymsg");
|
|
|
|
if (!message.is_pos_number_only())
|
|
source.Reply(("Entry message \002%s\002 not found on channel \002%s\002."), message.c_str(), ci->name.c_str());
|
|
else if ((*messages)->empty())
|
|
source.Reply(_("Entry message list for \002%s\002 is empty."), ci->name.c_str());
|
|
else
|
|
{
|
|
auto i = Anope::Convert<unsigned>(message, 0);
|
|
if (i > 0 && i <= (*messages)->size())
|
|
{
|
|
delete (*messages)->at(i - 1);
|
|
if ((*messages)->empty())
|
|
ci->Shrink<EntryMessageList>("entrymsg");
|
|
Log(source.AccessFor(ci).HasPriv("SET") ? LOG_COMMAND : LOG_OVERRIDE, source, this, ci) << "to remove a message";
|
|
source.Reply(_("Entry message \002%i\002 for \002%s\002 deleted."), i, ci->name.c_str());
|
|
}
|
|
else
|
|
{
|
|
source.Reply(_("Entry message \002%s\002 not found on channel \002%s\002."), message.c_str(), ci->name.c_str());
|
|
}
|
|
}
|
|
}
|
|
|
|
void DoClear(CommandSource &source, ChannelInfo *ci)
|
|
{
|
|
ci->Shrink<EntryMessageList>("entrymsg");
|
|
|
|
Log(source.AccessFor(ci).HasPriv("SET") ? LOG_COMMAND : LOG_OVERRIDE, source, this, ci) << "to remove all messages";
|
|
source.Reply(_("Entry messages for \002%s\002 have been cleared."), ci->name.c_str());
|
|
}
|
|
|
|
public:
|
|
CommandEntryMessage(Module *creator) : Command(creator, "chanserv/entrymsg", 2, 3)
|
|
{
|
|
this->SetDesc(_("Manage the channel's entry messages"));
|
|
this->SetSyntax(_("\037channel\037 ADD \037message\037"));
|
|
this->SetSyntax(_("\037channel\037 DEL \037num\037"));
|
|
this->SetSyntax(_("\037channel\037 LIST"));
|
|
this->SetSyntax(_("\037channel\037 CLEAR"));
|
|
}
|
|
|
|
void Execute(CommandSource &source, const std::vector<Anope::string> ¶ms) override
|
|
{
|
|
ChannelInfo *ci = ChannelInfo::Find(params[0]);
|
|
if (ci == NULL)
|
|
{
|
|
source.Reply(CHAN_X_NOT_REGISTERED, params[0].c_str());
|
|
return;
|
|
}
|
|
|
|
if (Anope::ReadOnly && !params[1].equals_ci("LIST"))
|
|
{
|
|
source.Reply(READ_ONLY_MODE);
|
|
return;
|
|
}
|
|
|
|
if (!source.AccessFor(ci).HasPriv("SET") && !source.HasPriv("chanserv/administration"))
|
|
{
|
|
source.Reply(ACCESS_DENIED);
|
|
return;
|
|
}
|
|
|
|
if (params[1].equals_ci("LIST"))
|
|
this->DoList(source, ci);
|
|
else if (params[1].equals_ci("CLEAR"))
|
|
this->DoClear(source, ci);
|
|
else if (params.size() < 3)
|
|
this->OnSyntaxError(source, "");
|
|
else if (params[1].equals_ci("ADD"))
|
|
this->DoAdd(source, ci, params[2]);
|
|
else if (params[1].equals_ci("DEL"))
|
|
this->DoDel(source, ci, params[2]);
|
|
else
|
|
this->OnSyntaxError(source, "");
|
|
}
|
|
|
|
bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
|
|
{
|
|
this->SendSyntax(source);
|
|
source.Reply(" ");
|
|
source.Reply(_("Controls what messages will be sent to users when they join the channel."));
|
|
source.Reply(" ");
|
|
source.Reply(_("The \002ENTRYMSG ADD\002 command adds the given message to\n"
|
|
"the list of messages shown to users when they join\n"
|
|
"the channel."));
|
|
source.Reply(" ");
|
|
source.Reply(_("The \002ENTRYMSG DEL\002 command removes the specified message from\n"
|
|
"the list of messages shown to users when they join\n"
|
|
"the channel. You can remove a message by specifying its number\n"
|
|
"which you can get by listing the messages as explained below."));
|
|
source.Reply(" ");
|
|
source.Reply(_("The \002ENTRYMSG LIST\002 command displays a listing of messages\n"
|
|
"shown to users when they join the channel."));
|
|
source.Reply(" ");
|
|
source.Reply(_("The \002ENTRYMSG CLEAR\002 command clears all entries from\n"
|
|
"the list of messages shown to users when they join\n"
|
|
"the channel, effectively disabling entry messages."));
|
|
source.Reply(" ");
|
|
source.Reply(_("Adding, deleting, or clearing entry messages requires the\n"
|
|
"SET permission."));
|
|
return true;
|
|
}
|
|
};
|
|
|
|
class CSEntryMessage final
|
|
: public Module
|
|
{
|
|
CommandEntryMessage commandentrymsg;
|
|
ExtensibleItem<EntryMessageListImpl> eml;
|
|
Serialize::Type entrymsg_type;
|
|
|
|
public:
|
|
CSEntryMessage(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, VENDOR),
|
|
commandentrymsg(this),
|
|
eml(this, "entrymsg"), entrymsg_type("EntryMsg", EntryMsgImpl::Unserialize)
|
|
{
|
|
}
|
|
|
|
void OnJoinChannel(User *u, Channel *c) override
|
|
{
|
|
if (u && c && c->ci && u->server->IsSynced())
|
|
{
|
|
EntryMessageList *messages = c->ci->GetExt<EntryMessageList>("entrymsg");
|
|
if (!messages)
|
|
return;
|
|
|
|
for (const auto &message : *(*messages))
|
|
{
|
|
if (u->ShouldPrivmsg())
|
|
IRCD->SendContextPrivmsg(c->ci->WhoSends(), u, c, message->message);
|
|
else
|
|
IRCD->SendContextNotice(c->ci->WhoSends(), u, c, message->message);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
MODULE_INIT(CSEntryMessage)
|