mirror of
https://github.com/anope/anope.git
synced 2026-06-12 19:14:47 +02:00
314 lines
9.5 KiB
C++
314 lines
9.5 KiB
C++
// Anope IRC Services <https://www.anope.org/>
|
|
//
|
|
// 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"
|
|
|
|
class CommandNSList final
|
|
: public Command
|
|
{
|
|
public:
|
|
CommandNSList(Module *creator) : Command(creator, "nickserv/list", 1, 2)
|
|
{
|
|
this->SetDesc(_("List all registered nicknames that match a given pattern"));
|
|
this->SetSyntax(_("\037pattern\037 [DISPLAY] [NOEXPIRE] [SUSPENDED] [UNCONFIRMED]"));
|
|
}
|
|
|
|
void Execute(CommandSource &source, const std::vector<Anope::string> ¶ms) override
|
|
{
|
|
auto from = 0u, to = 0u;
|
|
auto pattern = params[0];
|
|
if (pattern[0] == '#')
|
|
{
|
|
Anope::string n1, n2;
|
|
sepstream(pattern.substr(1), '-').GetToken(n1, 0);
|
|
sepstream(pattern, '-').GetToken(n2, 1);
|
|
|
|
auto num1 = Anope::TryConvert<int>(n1);
|
|
auto num2 = Anope::TryConvert<int>(n2);
|
|
if (!num1.has_value() || !num2.has_value())
|
|
{
|
|
source.Reply(LIST_INCORRECT_RANGE);
|
|
return;
|
|
}
|
|
|
|
from = num1.value();
|
|
to = num2.value();
|
|
pattern = "*";
|
|
}
|
|
|
|
auto display = false, nsnoexpire = false, suspended = false, unconfirmed = false;
|
|
const auto is_servadmin = source.HasCommand("nickserv/list");
|
|
if (is_servadmin && params.size() > 1)
|
|
{
|
|
Anope::string keyword;
|
|
spacesepstream keywords(params[1]);
|
|
while (keywords.GetToken(keyword))
|
|
{
|
|
if (keyword.equals_ci("DISPLAY"))
|
|
display = true;
|
|
else if (keyword.equals_ci("NOEXPIRE"))
|
|
nsnoexpire = true;
|
|
else if (keyword.equals_ci("SUSPENDED"))
|
|
suspended = true;
|
|
else if (keyword.equals_ci("UNCONFIRMED"))
|
|
unconfirmed = true;
|
|
}
|
|
}
|
|
|
|
ListFormatter list(source.GetAccount());
|
|
list.AddColumn(_("Nick")).AddColumn(_("Account")).AddColumn(_("Status"));
|
|
list.SetFlexible([](ListFormatter::ListEntry &row)
|
|
{
|
|
return row["Status"].empty()
|
|
? _("\002{nick}\002 (account: {account})")
|
|
: _("\002{nick}\002 -- {status} (account: {account})");
|
|
});
|
|
|
|
Anope::map<NickAlias *> ordered_map;
|
|
for (const auto &[nick, na] : *NickAliasList)
|
|
ordered_map[nick] = na;
|
|
|
|
const auto listmax = Config->GetModule(this->owner).Get<unsigned>("listmax", "50");
|
|
const auto *mync = source.GetAccount();
|
|
auto count = 0u, nnicks = 0u;
|
|
for (const auto &[_, na] : ordered_map)
|
|
{
|
|
/* Don't show private nicks to non-services admins. */
|
|
if (na->nc->HasExt("NS_PRIVATE") && !is_servadmin && na->nc != mync)
|
|
continue;
|
|
else if (display && na->nc->na != na)
|
|
continue;
|
|
else if (nsnoexpire && !na->HasExt("NS_NO_EXPIRE"))
|
|
continue;
|
|
else if (suspended && !na->nc->HasExt("NS_SUSPENDED"))
|
|
continue;
|
|
else if (unconfirmed && !na->nc->HasExt("UNCONFIRMED"))
|
|
continue;
|
|
|
|
if (na->nick.equals_ci(pattern) || Anope::Match(na->nick, pattern, false, true))
|
|
{
|
|
if (((count + 1 >= from && count + 1 <= to) || (!from && !to)) && ++nnicks <= listmax)
|
|
{
|
|
bool isnoexpire = false;
|
|
if (is_servadmin && na->HasExt("NS_NO_EXPIRE"))
|
|
isnoexpire = true;
|
|
|
|
ListFormatter::ListEntry entry;
|
|
entry["Nick"] = (isnoexpire ? "!" : "") + na->nick;
|
|
entry["Account"] = na->nc->display;
|
|
|
|
auto &status = entry["Status"];
|
|
if (na->nc->HasExt("NS_SUSPENDED"))
|
|
status = source.Translate(_("Suspended"));
|
|
else if (na->nc->HasExt("UNCONFIRMED"))
|
|
status = source.Translate(_("Unconfirmed"));
|
|
list.AddEntry(entry);
|
|
}
|
|
++count;
|
|
}
|
|
}
|
|
|
|
source.Reply(_("List of entries matching \002%s\002:"), pattern.c_str());
|
|
list.SendTo(source);
|
|
source.Reply(_("End of list - %d/%d matches shown."), nnicks > listmax ? listmax : nnicks, nnicks);
|
|
}
|
|
|
|
bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
|
|
{
|
|
this->SendSyntax(source);
|
|
source.Reply(" ");
|
|
source.Reply(_(
|
|
"Lists all registered nicknames which match the given "
|
|
"pattern, in \037nick!user@host\037 format. Nicks with the \002PRIVATE\002 "
|
|
"option set will only be displayed to Services Operators with the "
|
|
"proper access. Nicks with the \002NOEXPIRE\002 option set will have "
|
|
"a \002!\002 prefixed to the nickname for Services Operators to see."
|
|
"\n\n"
|
|
"Note that a preceding '#' specifies a range."
|
|
"\n\n"
|
|
"If the DISPLAY, NOEXPIRE, SUSPENDED, or UNCONFIRMED options are given "
|
|
"only nicks which, respectively, are display nicks, will not expire, are "
|
|
"suspended, or are unconfirmed will be shown. If multiple options are "
|
|
"given, nicks must match every option to be shown. "
|
|
"Note that these options are limited to \037Services Operators\037."
|
|
));
|
|
|
|
ExampleWrapper examples;
|
|
examples.AddEntry("*Bot*", _(
|
|
"Lists all registered nicks with \037Bot\037 in their name (case insensitive)."
|
|
));
|
|
examples.AddEntry("#51-100", _(
|
|
"Lists all registered nicks within the given range (51-100)."
|
|
));
|
|
examples.AddEntry("* DISPLAY", _(
|
|
"Lists all registered nicks that are the display nickname for their account."
|
|
), "nickserv/list");
|
|
examples.AddEntry("* NOEXPIRE", _(
|
|
"Lists all registered nicks that have been set to not expire."
|
|
), "nickserv/list");
|
|
examples.AddEntry("* SUSPENDED", _(
|
|
"Lists all registered nicks that have been suspended."
|
|
), "nickserv/list");
|
|
examples.AddEntry("* UNCONFIRMED", _(
|
|
"Lists all registered nicks that have not been confirmed yet."
|
|
), "nickserv/list");
|
|
examples.SendTo(source);
|
|
|
|
const Anope::string ®exengine = Config->GetBlock("options").Get<const Anope::string>("regexengine");
|
|
if (!regexengine.empty())
|
|
{
|
|
source.Reply(" ");
|
|
source.Reply(_(
|
|
"Regex matches are also supported using the %s engine. "
|
|
"Enclose your pattern in // if this is desired."
|
|
),
|
|
regexengine.c_str());
|
|
}
|
|
|
|
return true;
|
|
}
|
|
};
|
|
|
|
|
|
class CommandNSSetPrivate
|
|
: public Command
|
|
{
|
|
public:
|
|
CommandNSSetPrivate(Module *creator, const Anope::string &sname = "nickserv/set/private", size_t min = 1) : Command(creator, sname, min, min + 1)
|
|
{
|
|
this->SetDesc(_("Prevent the nickname from appearing in the LIST command"));
|
|
this->SetSyntax("{ON | OFF}");
|
|
}
|
|
|
|
void Run(CommandSource &source, const Anope::string &user, const Anope::string ¶m)
|
|
{
|
|
if (Anope::ReadOnly)
|
|
{
|
|
source.Reply(READ_ONLY_MODE);
|
|
return;
|
|
}
|
|
|
|
const NickAlias *na = NickAlias::Find(user);
|
|
if (!na)
|
|
{
|
|
source.Reply(NICK_X_NOT_REGISTERED, user.c_str());
|
|
return;
|
|
}
|
|
NickCore *nc = na->nc;
|
|
|
|
EventReturn MOD_RESULT;
|
|
FOREACH_RESULT(OnSetNickOption, MOD_RESULT, (source, this, nc, param));
|
|
if (MOD_RESULT == EVENT_STOP)
|
|
return;
|
|
|
|
if (param.equals_ci("ON"))
|
|
{
|
|
Log(nc == source.GetAccount() ? LOG_COMMAND : LOG_ADMIN, source, this) << "to enable private for " << nc->display;
|
|
nc->Extend<bool>("NS_PRIVATE");
|
|
source.Reply(_("Private option is now \002on\002 for \002%s\002."), nc->display.c_str());
|
|
}
|
|
else if (param.equals_ci("OFF"))
|
|
{
|
|
Log(nc == source.GetAccount() ? LOG_COMMAND : LOG_ADMIN, source, this) << "to disable private for " << nc->display;
|
|
nc->Shrink<bool>("NS_PRIVATE");
|
|
source.Reply(_("Private option is now \002off\002 for \002%s\002."), nc->display.c_str());
|
|
}
|
|
else
|
|
this->OnSyntaxError(source, "PRIVATE");
|
|
}
|
|
|
|
void Execute(CommandSource &source, const std::vector<Anope::string> ¶ms) override
|
|
{
|
|
this->Run(source, source.nc->display, params[0]);
|
|
}
|
|
|
|
bool OnHelp(CommandSource &source, const Anope::string &) override
|
|
{
|
|
this->SendSyntax(source);
|
|
source.Reply(" ");
|
|
source.Reply(_(
|
|
"Turns %s's privacy option on or off for your nick. "
|
|
"With \002PRIVATE\002 set, your nickname will not appear in "
|
|
"nickname lists generated with %s's \002LIST\002 command. "
|
|
"(However, anyone who knows your nickname can still get "
|
|
"information on it using the \002INFO\002 command.)"
|
|
),
|
|
source.service->nick.c_str(),
|
|
source.service->nick.c_str());
|
|
return true;
|
|
}
|
|
};
|
|
|
|
class CommandNSSASetPrivate final
|
|
: public CommandNSSetPrivate
|
|
{
|
|
public:
|
|
CommandNSSASetPrivate(Module *creator) : CommandNSSetPrivate(creator, "nickserv/saset/private", 2)
|
|
{
|
|
this->ClearSyntax();
|
|
this->SetSyntax(_("\037nickname\037 {ON | OFF}"));
|
|
}
|
|
|
|
void Execute(CommandSource &source, const std::vector<Anope::string> ¶ms) override
|
|
{
|
|
this->Run(source, params[0], params[1]);
|
|
}
|
|
|
|
bool OnHelp(CommandSource &source, const Anope::string &) override
|
|
{
|
|
this->SendSyntax(source);
|
|
source.Reply(" ");
|
|
source.Reply(_(
|
|
"Turns %s's privacy option on or off for the nick. "
|
|
"With \002PRIVATE\002 set, the nickname will not appear in "
|
|
"nickname lists generated with %s's \002LIST\002 command. "
|
|
"(However, anyone who knows the nickname can still get "
|
|
"information on it using the \002INFO\002 command.)"
|
|
),
|
|
source.service->nick.c_str(),
|
|
source.service->nick.c_str());
|
|
return true;
|
|
}
|
|
};
|
|
|
|
|
|
class NSList final
|
|
: public Module
|
|
{
|
|
CommandNSList commandnslist;
|
|
|
|
CommandNSSetPrivate commandnssetprivate;
|
|
CommandNSSASetPrivate commandnssasetprivate;
|
|
|
|
SerializableExtensibleItem<bool> priv;
|
|
|
|
public:
|
|
NSList(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, VENDOR),
|
|
commandnslist(this), commandnssetprivate(this), commandnssasetprivate(this),
|
|
priv(this, "NS_PRIVATE")
|
|
{
|
|
}
|
|
|
|
void OnNickInfo(CommandSource &source, NickAlias *na, InfoFormatter &info, bool show_all) override
|
|
{
|
|
if (!show_all)
|
|
return;
|
|
|
|
if (priv.HasExt(na->nc))
|
|
info.AddOption(_("Private"));
|
|
}
|
|
};
|
|
|
|
MODULE_INIT(NSList)
|