mirror of
https://github.com/anope/anope.git
synced 2026-06-22 16:26:38 +02:00
227 lines
6.6 KiB
C++
227 lines
6.6 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
|
|
{
|
|
static unsigned MaxEntries;
|
|
|
|
EntryMsg(const Anope::string &cname, const Anope::string &cmessage, time_t ct = Anope::CurTime)
|
|
{
|
|
this->creator = cname;
|
|
this->message = cmessage;
|
|
this->when = ct;
|
|
}
|
|
|
|
Anope::string creator;
|
|
Anope::string message;
|
|
time_t when;
|
|
};
|
|
unsigned EntryMsg::MaxEntries = 0;
|
|
|
|
class CommandEntryMessage : public Command
|
|
{
|
|
private:
|
|
void DoList(CommandSource &source, ChannelInfo *ci)
|
|
{
|
|
std::vector<EntryMsg> messages;
|
|
if (ci->GetExtRegular("cs_entrymsg", messages))
|
|
{
|
|
source.Reply(_("Entry message list for \2%s\2:"), ci->name.c_str());
|
|
for (unsigned i = 0; i < messages.size(); ++i)
|
|
source.Reply(LanguageString::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)
|
|
{
|
|
std::vector<EntryMsg> messages;
|
|
ci->GetExtRegular("cs_entrymsg", messages);
|
|
|
|
if (EntryMsg::MaxEntries && messages.size() >= EntryMsg::MaxEntries)
|
|
source.Reply(_("The entry message list for \2%s\2 is full."), ci->name.c_str());
|
|
else
|
|
{
|
|
messages.push_back(EntryMsg(source.u->nick, message));
|
|
ci->Extend("cs_entrymsg", new ExtensibleItemRegular<std::vector<EntryMsg> >(messages));
|
|
source.Reply(_("Entry message added to \2%s\2"), ci->name.c_str());
|
|
}
|
|
}
|
|
|
|
void DoDel(CommandSource &source, ChannelInfo *ci, const Anope::string &message)
|
|
{
|
|
std::vector<EntryMsg> messages;
|
|
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 (ci->GetExtRegular("cs_entrymsg", messages))
|
|
{
|
|
try
|
|
{
|
|
unsigned i = convertTo<unsigned>(message);
|
|
if (i <= messages.size())
|
|
{
|
|
messages.erase(messages.begin() + i - 1);
|
|
ci->Extend("cs_entrymsg", new ExtensibleItemRegular<std::vector<EntryMsg> >(messages));
|
|
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(const Anope::string &cname) : Command(cname, 2, 3)
|
|
{
|
|
}
|
|
|
|
CommandReturn Execute(CommandSource &source, const std::vector<Anope::string> ¶ms)
|
|
{
|
|
User *u = source.u;
|
|
ChannelInfo *ci = source.ci;
|
|
|
|
if (ci && (IsFounder(u, ci) || u->Account()->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
|
|
{
|
|
u->SendMessage(ChanServ, LanguageString::ACCESS_DENIED);
|
|
}
|
|
|
|
return MOD_CONT;
|
|
}
|
|
|
|
void OnSyntaxError(CommandSource &source, const Anope::string &)
|
|
{
|
|
SyntaxError(source, "ENTRYMSG", _("ENTRYMSG \037channel\037 {ADD|DEL|LIST|CLEAR} [\037message\037|\037num\037]"));
|
|
}
|
|
|
|
bool OnHelp(CommandSource &source, const Anope::string &subcommand)
|
|
{
|
|
source.Reply(_("Syntax: \002ENTRYMSG \037channel\037 {ADD|DEL|LIST|CLEAR} [\037message\037|\037num\037]\002\n"
|
|
" \n"
|
|
"Controls what messages will be sent to users when they join the channel."));
|
|
return true;
|
|
}
|
|
|
|
void OnServHelp(CommandSource &source)
|
|
{
|
|
source.Reply(_(" ENTRYMSG Manage the channel's entrymsgs"));
|
|
}
|
|
};
|
|
|
|
class CSEntryMessage : public Module
|
|
{
|
|
CommandEntryMessage commandentrymsg;
|
|
|
|
public:
|
|
CSEntryMessage(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator), commandentrymsg("ENTRYMSG")
|
|
{
|
|
this->SetAuthor("Anope");
|
|
this->SetType(CORE);
|
|
|
|
Implementation i[] = { I_OnJoinChannel, I_OnReload, I_OnDatabaseReadMetadata, I_OnDatabaseWriteMetadata };
|
|
ModuleManager::Attach(i, this, 4);
|
|
|
|
this->AddCommand(ChanServ, &commandentrymsg);
|
|
|
|
this->OnReload(false);
|
|
}
|
|
|
|
void OnJoinChannel(User *u, Channel *c)
|
|
{
|
|
if (u && c && c->ci && u->server->IsSynced())
|
|
{
|
|
std::vector<EntryMsg> messages;
|
|
|
|
if (c->ci->GetExtRegular("cs_entrymsg", messages))
|
|
for (unsigned i = 0; i < messages.size(); ++i)
|
|
u->SendMessage(whosends(c->ci), "[%s] %s", c->ci->name.c_str(), messages[i].message.c_str());
|
|
}
|
|
}
|
|
|
|
void OnReload(bool)
|
|
{
|
|
ConfigReader config;
|
|
EntryMsg::MaxEntries = config.ReadInteger("cs_entrymsg", "maxentries", "5", 0, true);
|
|
}
|
|
|
|
EventReturn OnDatabaseReadMetadata(ChannelInfo *ci, const Anope::string &key, const std::vector<Anope::string> ¶ms)
|
|
{
|
|
if (key.find("CS_ENTRYMSG_") == 0 && params.size() > 2)
|
|
{
|
|
Anope::string creator = params[0];
|
|
time_t t = params[1].is_pos_number_only() ? convertTo<time_t>(params[1]) : Anope::CurTime;
|
|
Anope::string message = params[2];
|
|
for (unsigned j = 3; j < params.size(); ++j)
|
|
message += " " + params[j];
|
|
|
|
std::vector<EntryMsg> messages;
|
|
ci->GetExtRegular("cs_entrymsg", messages);
|
|
messages.push_back(EntryMsg(creator, message, t));
|
|
ci->Extend("cs_entrymsg", new ExtensibleItemRegular<std::vector<EntryMsg> >(messages));
|
|
|
|
return EVENT_STOP;
|
|
}
|
|
|
|
return EVENT_CONTINUE;
|
|
}
|
|
|
|
void OnDatabaseWriteMetadata(void (*WriteMetadata)(const Anope::string &, const Anope::string &), ChannelInfo *ci)
|
|
{
|
|
std::vector<EntryMsg> messages;
|
|
if (ci->GetExtRegular("cs_entrymsg", messages))
|
|
for (unsigned i = 0; i < messages.size(); ++i)
|
|
WriteMetadata("CS_ENTRYMSG_" + stringify(i), messages[i].creator + " " + stringify(messages[i].when) + " " + messages[i].message);
|
|
}
|
|
};
|
|
|
|
MODULE_INIT(CSEntryMessage)
|