// Anope IRC Services
//
// Copyright (C) 2003-2026 Anope Contributors
//
// Anope is free software. You can use, modify, and/or distribute it under the
// terms of version 2 of the GNU General Public License. See docs/LICENSE.txt
// for the complete terms of this license and docs/AUTHORS.txt for a list of
// contributors.
//
// Based on the original code of Epona by Lara
// Based on the original code of Services by Andy Church
//
// SPDX-License-Identifier: GPL-2.0-only
#include "module.h"
#include "modules/memoserv/service.h"
class MemoServCore final
: public Module
, public MemoServ::Service
{
Reference MemoServ;
static bool SendMemoMail(NickCore *nc, MemoInfo *mi, Memo *m)
{
Anope::map vars = {
{ "receiver", nc->display },
{ "sender", m->sender },
{ "number", Anope::ToString(mi->GetIndex(m) + 1) },
{ "text", m->text },
{ "network", Config->GetBlock("networkinfo").Get("networkname") },
};
auto subject = Anope::Template(Language::Translate(nc, Config->GetBlock("mail").Get("memo_subject").c_str()), vars);
auto message = Anope::Template(Language::Translate(nc, Config->GetBlock("mail").Get("memo_message").c_str()), vars);
return Mail::Send(nc, subject, message);
}
public:
MemoServCore(const Anope::string &modname, const Anope::string &creator)
: Module(modname, creator, PSEUDOCLIENT | VENDOR)
, MemoServ::Service(this)
{
}
MemoServ::MemoResult Send(const Anope::string &source, const Anope::string &target, const Anope::string &message, bool force) override
{
bool ischan;
MemoInfo *mi = MemoInfo::GetMemoInfo(target, ischan);
if (mi == NULL)
return MemoServ::MEMO_INVALID_TARGET;
Anope::string sender_display = source;
User *sender = User::Find(source, true);
if (sender != NULL)
{
if (!sender->HasPriv("memoserv/no-limit") && !force)
{
time_t send_delay = Config->GetModule("memoserv").Get("senddelay");
if (send_delay > 0 && sender->lastmemosend + send_delay > Anope::CurTime)
return MemoServ::MEMO_TOO_FAST;
else if (!mi->memomax)
return MemoServ::MEMO_TARGET_FULL;
else if (mi->memomax > 0 && mi->memos->size() >= static_cast(mi->memomax))
return MemoServ::MEMO_TARGET_FULL;
else if (mi->HasIgnore(sender))
return MemoServ::MEMO_SUCCESS;
}
NickCore *acc = sender->Account();
if (acc != NULL)
{
sender_display = acc->display;
}
}
if (sender != NULL)
sender->lastmemosend = Anope::CurTime;
auto *m = new Memo();
m->mi = mi;
mi->memos->push_back(m);
m->owner = target;
m->sender = sender_display;
m->time = Anope::CurTime;
m->text = message;
m->unread = true;
FOREACH_MOD(OnMemoSend, (source, target, mi, m));
if (ischan)
{
ChannelInfo *ci = ChannelInfo::Find(target);
if (ci->c)
{
for (const auto &[_, cu] : ci->c->users)
{
if (ci->AccessFor(cu->user).HasPriv("MEMO"))
{
if (cu->user->IsIdentified() && cu->user->Account()->HasExt("MEMO_RECEIVE"))
{
cu->user->SendMessage(MemoServ, MEMO_NEW_X_MEMO_ARRIVED, ci->name.c_str(), MemoServ->GetQueryCommand("memoserv/read").c_str(),
ci->name.c_str(), mi->memos->size());
}
}
}
}
}
else
{
NickCore *nc = NickAlias::Find(target)->nc;
if (nc->HasExt("MEMO_RECEIVE"))
{
for (auto *na : *nc->aliases)
{
User *user = User::Find(na->nick, true);
if (user && user->IsIdentified())
user->SendMessage(MemoServ, MEMO_NEW_MEMO_ARRIVED, source.c_str(), MemoServ->GetQueryCommand("memoserv/read").c_str(), mi->memos->size());
}
}
/* let's get out the mail if set in the nickcore - certus */
if (nc->HasExt("MEMO_MAIL"))
SendMemoMail(nc, mi, m);
}
return MemoServ::MEMO_SUCCESS;
}
void Check(User *u) override
{
const NickCore *nc = u->Account();
if (!nc)
return;
unsigned i = 0, end = nc->memos.memos->size(), newcnt = 0;
for (; i < end; ++i)
if (nc->memos.GetMemo(i)->unread)
++newcnt;
if (newcnt > 0)
u->SendMessage(MemoServ, newcnt, N_("You have %d new memo.", "You have %d new memos."), newcnt);
if (nc->memos.memomax > 0 && nc->memos.memos->size() >= static_cast(nc->memos.memomax))
{
if (nc->memos.memos->size() > static_cast(nc->memos.memomax))
u->SendMessage(MemoServ, _("You are over your maximum number of memos (%d). You will be unable to receive any new memos until you delete some of your current ones."), nc->memos.memomax);
else
u->SendMessage(MemoServ, _("You have reached your maximum number of memos (%d). You will be unable to receive any new memos until you delete some of your current ones."), nc->memos.memomax);
}
}
void OnReload(Configuration::Conf &conf) override
{
const Anope::string &msnick = conf.GetModule(this).Get("client");
if (msnick.empty())
throw ConfigException(Module::name + ": must be defined");
BotInfo *bi = BotInfo::Find(msnick, true);
if (!bi)
throw ConfigException(Module::name + ": no bot named " + msnick);
MemoServ = bi;
}
void OnNickCoreCreate(NickCore *nc) override
{
nc->memos.memomax = Config->GetModule(this).Get("maxmemos");
}
void OnCreateChan(ChannelInfo *ci) override
{
ci->memos.memomax = Config->GetModule(this).Get("maxmemos");
}
void OnBotDelete(BotInfo *bi) override
{
if (bi == MemoServ)
MemoServ = NULL;
}
void OnNickIdentify(User *u) override
{
this->Check(u);
}
void OnJoinChannel(User *u, Channel *c) override
{
if (c->ci && !c->ci->memos.memos->empty() && c->ci->AccessFor(u).HasPriv("MEMO"))
{
auto memocount = c->ci->memos.memos->size();
u->SendMessage(MemoServ, memocount, N_("There is \002%zu\002 memo on channel %s.", "There are \002%zu\002 memos on channel %s."),
memocount, c->ci->name.c_str());
}
}
void OnUserAway(User *u, const Anope::string &message) override
{
if (message.empty())
this->Check(u);
}
void OnNickUpdate(User *u) override
{
this->Check(u);
}
void OnUserConnect(User *user, bool &exempt) override
{
this->Check(user);
}
EventReturn OnPreHelp(CommandSource &source, const std::vector ¶ms) override
{
if (!params.empty() || source.c || source.service != *MemoServ)
return EVENT_CONTINUE;
source.Reply(_(
"\002%s\002 is a utility allowing IRC users to send short "
"messages to other IRC users, whether they are online at "
"the time or not, or to channels(*). Both the sender's "
"nickname and the target nickname or channel must be "
"registered in order to send a memo."
"\n\n"
"%s's commands include:"
),
MemoServ->nick.c_str(),
MemoServ->nick.c_str());
return EVENT_CONTINUE;
}
void OnPostHelp(CommandSource &source, const std::vector ¶ms) override
{
if (!params.empty() || source.c || source.service != *MemoServ)
return;
source.Reply(" ");
source.Reply(_("Type \002%s\033\037command\037\002 for help on any of the above commands."),
MemoServ->GetQueryCommand("generic/help").c_str());
}
};
MODULE_INIT(MemoServCore)