mirror of
https://github.com/anope/anope.git
synced 2026-06-24 18:26:37 +02:00
230 lines
6.0 KiB
C++
230 lines
6.0 KiB
C++
/* ChanServ core functions
|
|
*
|
|
* (C) 2003-2011 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"
|
|
|
|
struct EntryMsg : Serializable<EntryMsg>
|
|
{
|
|
ChannelInfo *ci;
|
|
Anope::string creator;
|
|
Anope::string message;
|
|
time_t when;
|
|
|
|
EntryMsg(ChannelInfo *c, const Anope::string &cname, const Anope::string &cmessage, time_t ct = Anope::CurTime)
|
|
{
|
|
|
|
this->ci = c;
|
|
this->creator = cname;
|
|
this->message = cmessage;
|
|
this->when = ct;
|
|
}
|
|
|
|
serialized_data serialize()
|
|
{
|
|
serialized_data data;
|
|
|
|
data["ci"] << this->ci->name;
|
|
data["creator"] << this->creator;
|
|
data["message"] << this->message;
|
|
data["when"].setType(Serialize::DT_INT) << this->when;
|
|
|
|
return data;
|
|
}
|
|
|
|
static void unserialize(serialized_data &data);
|
|
};
|
|
|
|
static unsigned MaxEntries = 0;
|
|
|
|
struct EntryMessageList : std::vector<EntryMsg>, ExtensibleItem
|
|
{
|
|
};
|
|
|
|
void EntryMsg::unserialize(serialized_data &data)
|
|
{
|
|
ChannelInfo *ci = cs_findchan(data["ci"].astr());
|
|
if (!ci)
|
|
return;
|
|
|
|
EntryMessageList *messages = ci->GetExt<EntryMessageList *>("cs_entrymsg");
|
|
if (messages == NULL)
|
|
{
|
|
messages = new EntryMessageList();
|
|
ci->Extend("cs_entrymsg", messages);
|
|
}
|
|
|
|
messages->push_back(EntryMsg(ci, data["creator"].astr(), data["message"].astr()));
|
|
}
|
|
|
|
class CommandEntryMessage : public Command
|
|
{
|
|
private:
|
|
void DoList(CommandSource &source, ChannelInfo *ci)
|
|
{
|
|
EntryMessageList *messages = ci->GetExt<EntryMessageList *>("cs_entrymsg");
|
|
if (messages == NULL)
|
|
{
|
|
source.Reply(_("Entry message list for \2%s\2:"), ci->name.c_str());
|
|
for (unsigned i = 0; i < messages->size(); ++i)
|
|
source.Reply(CHAN_LIST_ENTRY, i + 1, (*messages)[i].message.c_str(), (*messages)[i].creator.c_str(), do_strftime((*messages)[i].when).c_str());
|
|
source.Reply(_("End of entry message list."));
|
|
}
|
|
else
|
|
source.Reply(_("Entry message list for \2%s\2 is empty."), ci->name.c_str());
|
|
}
|
|
|
|
void DoAdd(CommandSource &source, ChannelInfo *ci, const Anope::string &message)
|
|
{
|
|
EntryMessageList *messages = ci->GetExt<EntryMessageList *>("cs_entrymsg");
|
|
if (messages == NULL)
|
|
{
|
|
messages = new EntryMessageList();
|
|
ci->Extend("cs_entrymsg", messages);
|
|
}
|
|
|
|
if (MaxEntries && messages->size() >= MaxEntries)
|
|
source.Reply(_("The entry message list for \2%s\2 is full."), ci->name.c_str());
|
|
else
|
|
{
|
|
messages->push_back(EntryMsg(ci, source.u->nick, message));
|
|
source.Reply(_("Entry message added to \2%s\2"), ci->name.c_str());
|
|
}
|
|
}
|
|
|
|
void DoDel(CommandSource &source, ChannelInfo *ci, const Anope::string &message)
|
|
{
|
|
EntryMessageList *messages = ci->GetExt<EntryMessageList *>("cs_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 != NULL)
|
|
{
|
|
try
|
|
{
|
|
unsigned i = convertTo<unsigned>(message);
|
|
if (i > 0 && i <= messages->size())
|
|
{
|
|
messages->erase(messages->begin() + i - 1);
|
|
if (messages->empty())
|
|
ci->Shrink("cs_entrymsg");
|
|
source.Reply(_("Entry message \2%i\2 for \2%s\2 deleted."), i, ci->name.c_str());
|
|
}
|
|
else
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
source.Reply(_("Entry message \2%s\2 not found on channel \2%s\2."), message.c_str(), ci->name.c_str());
|
|
}
|
|
}
|
|
else
|
|
source.Reply(_("Entry message list for \2%s\2 is empty."), ci->name.c_str());
|
|
}
|
|
|
|
void DoClear(CommandSource &source, ChannelInfo *ci)
|
|
{
|
|
ci->Shrink("cs_entrymsg");
|
|
source.Reply(_("Entry messages for \2%s\2 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|DEL|LIST|CLEAR} [\037message\037|\037num\037]"));
|
|
}
|
|
|
|
void Execute(CommandSource &source, const std::vector<Anope::string> ¶ms)
|
|
{
|
|
User *u = source.u;
|
|
|
|
ChannelInfo *ci = cs_findchan(params[0]);
|
|
if (ci == NULL)
|
|
{
|
|
source.Reply(CHAN_X_NOT_REGISTERED, params[0].c_str());
|
|
return;
|
|
}
|
|
|
|
if (IsFounder(u, ci) || u->HasCommand("chanserv/entrymsg"))
|
|
{
|
|
bool success = true;
|
|
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)
|
|
{
|
|
success = false;
|
|
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
|
|
{
|
|
success = false;
|
|
this->OnSyntaxError(source, "");
|
|
}
|
|
if (success)
|
|
Log(IsFounder(u, ci) ? LOG_COMMAND : LOG_OVERRIDE, u, this, ci) << params[1];
|
|
}
|
|
else
|
|
source.Reply(ACCESS_DENIED);
|
|
|
|
return;
|
|
}
|
|
|
|
bool OnHelp(CommandSource &source, const Anope::string &subcommand)
|
|
{
|
|
this->SendSyntax(source);
|
|
source.Reply(" ");
|
|
source.Reply(_("Controls what messages will be sent to users when they join the channel."));
|
|
return true;
|
|
}
|
|
};
|
|
|
|
class CSEntryMessage : public Module
|
|
{
|
|
CommandEntryMessage commandentrymsg;
|
|
|
|
public:
|
|
CSEntryMessage(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, CORE), commandentrymsg(this)
|
|
{
|
|
this->SetAuthor("Anope");
|
|
|
|
this->OnReload();
|
|
|
|
Serializable<EntryMsg>::Alloc.Register("EntryMsg");
|
|
}
|
|
|
|
void OnJoinChannel(User *u, Channel *c)
|
|
{
|
|
if (u && c && c->ci && u->server->IsSynced())
|
|
{
|
|
EntryMessageList *messages = c->ci->GetExt<EntryMessageList *>("cs_entrymsg");
|
|
|
|
if (messages != NULL)
|
|
for (unsigned i = 0; i < messages->size(); ++i)
|
|
u->SendMessage(c->ci->WhoSends(), "[%s] %s", c->ci->name.c_str(), (*messages)[i].message.c_str());
|
|
}
|
|
}
|
|
|
|
void OnReload()
|
|
{
|
|
ConfigReader config;
|
|
MaxEntries = config.ReadInteger("cs_entrymsg", "maxentries", "5", 0, true);
|
|
}
|
|
};
|
|
|
|
MODULE_INIT(CSEntryMessage)
|