summaryrefslogtreecommitdiff
path: root/modules/nickserv/list.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'modules/nickserv/list.cpp')
-rw-r--r--modules/nickserv/list.cpp299
1 files changed, 299 insertions, 0 deletions
diff --git a/modules/nickserv/list.cpp b/modules/nickserv/list.cpp
new file mode 100644
index 000000000..4b4a12d36
--- /dev/null
+++ b/modules/nickserv/list.cpp
@@ -0,0 +1,299 @@
+/*
+ * Anope IRC Services
+ *
+ * Copyright (C) 2003-2016 Anope Team <team@anope.org>
+ *
+ * This file is part of Anope. Anope is free software; you can
+ * redistribute it and/or modify it under the terms of the GNU
+ * General Public License as published by the Free Software
+ * Foundation, version 2.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, see see <http://www.gnu.org/licenses/>.
+ */
+
+#include "module.h"
+#include "modules/nickserv/info.h"
+#include "modules/nickserv/set.h"
+
+class CommandNSList : 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 [SUSPENDED] [NOEXPIRE] [UNCONFIRMED]"));
+ }
+
+ void Execute(CommandSource &source, const std::vector<Anope::string> &params) override
+ {
+
+ Anope::string pattern = params[0];
+ const NickServ::Account *mync;
+ unsigned nnicks;
+ bool is_servadmin = source.HasCommand("nickserv/list");
+ int count = 0, from = 0, to = 0;
+ bool suspended, nsnoexpire, unconfirmed;
+ unsigned listmax = Config->GetModule(this->GetOwner())->Get<unsigned>("listmax", "50");
+
+ suspended = nsnoexpire = unconfirmed = false;
+
+ if (pattern[0] == '#')
+ {
+ Anope::string n1, n2;
+ sepstream(pattern.substr(1), '-').GetToken(n1, 0);
+ sepstream(pattern, '-').GetToken(n2, 1);
+ try
+ {
+ from = convertTo<int>(n1);
+ to = convertTo<int>(n2);
+ }
+ catch (const ConvertException &)
+ {
+ source.Reply(_("Incorrect range specified. The correct syntax is \002#\037from\037-\037to\037\002."));
+ return;
+ }
+
+ pattern = "*";
+ }
+
+ nnicks = 0;
+
+ if (is_servadmin && params.size() > 1)
+ {
+ Anope::string keyword;
+ spacesepstream keywords(params[1]);
+ while (keywords.GetToken(keyword))
+ {
+ if (keyword.equals_ci("NOEXPIRE"))
+ nsnoexpire = true;
+ if (keyword.equals_ci("SUSPENDED"))
+ suspended = true;
+ if (keyword.equals_ci("UNCONFIRMED"))
+ unconfirmed = true;
+ }
+ }
+
+ mync = source.nc;
+ ListFormatter list(source.GetAccount());
+
+ list.AddColumn(_("Nick")).AddColumn(_("Last usermask"));
+
+ // XXX wtf
+ Anope::map<NickServ::Nick *> ordered_map;
+ for (NickServ::Nick *na : NickServ::service->GetNickList())
+ ordered_map[na->GetNick()] = na;
+
+ for (Anope::map<NickServ::Nick *>::const_iterator it = ordered_map.begin(), it_end = ordered_map.end(); it != it_end; ++it)
+ {
+ NickServ::Nick *na = it->second;
+
+ /* Don't show private nicks to non-services admins. */
+ if (na->GetAccount()->HasFieldS("NS_PRIVATE") && !is_servadmin && na->GetAccount() != mync)
+ continue;
+ else if (nsnoexpire && !na->HasFieldS("NS_NO_EXPIRE"))
+ continue;
+ else if (suspended && !na->GetAccount()->HasFieldS("NS_SUSPENDED"))
+ continue;
+ else if (unconfirmed && !na->GetAccount()->HasFieldS("UNCONFIRMED"))
+ continue;
+
+ /* We no longer compare the pattern against the output buffer.
+ * Instead we build a nice nick!user@host buffer to compare.
+ * The output is then generated separately. -TheShadow */
+ Anope::string buf = Anope::printf("%s!%s", na->GetNick().c_str(), !na->GetLastUsermask().empty() ? na->GetLastUsermask().c_str() : "*@*");
+ if (na->GetNick().equals_ci(pattern) || Anope::Match(buf, pattern, false, true))
+ {
+ if (((count + 1 >= from && count + 1 <= to) || (!from && !to)) && ++nnicks <= listmax)
+ {
+ bool isnoexpire = false;
+ if (is_servadmin && na->HasFieldS("NS_NO_EXPIRE"))
+ isnoexpire = true;
+
+ ListFormatter::ListEntry entry;
+ entry["Nick"] = (isnoexpire ? "!" : "") + na->GetNick();
+ if (na->GetAccount()->HasFieldS("HIDE_MASK") && !is_servadmin && na->GetAccount() != mync)
+ entry["Last usermask"] = Language::Translate(source.GetAccount(), _("[Hostname hidden]"));
+ else if (na->GetAccount()->HasFieldS("NS_SUSPENDED"))
+ entry["Last usermask"] = Language::Translate(source.GetAccount(), _("[Suspended]"));
+ else if (na->GetAccount()->HasFieldS("UNCONFIRMED"))
+ entry["Last usermask"] = Language::Translate(source.GetAccount(), _("[Unconfirmed]"));
+ else
+ entry["Last usermask"] = na->GetLastUsermask();
+ list.AddEntry(entry);
+ }
+ ++count;
+ }
+ }
+
+ source.Reply(_("List of entries matching \002{0}\002:"), pattern);
+
+ std::vector<Anope::string> replies;
+ list.Process(replies);
+
+ for (unsigned i = 0; i < replies.size(); ++i)
+ source.Reply(replies[i]);
+
+ source.Reply(_("End of list - \002{0}\002/\002{1}\002 matches shown."), nnicks > listmax ? listmax : nnicks, nnicks);
+ return;
+ }
+
+ bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
+ {
+ 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 SUSPENDED, UNCONFIRMED or NOEXPIRE options are given, only\n"
+ "nicks which, respectively, are SUSPENDED, UNCONFIRMED or have the\n"
+ "NOEXPIRE flag set will be displayed. If multiple options are\n"
+ "given, all nicks matching at least one option will be displayed.\n"
+ "Note that these options are limited to \037Services Operators\037.\n"
+ "\n"
+ "Examples:\n"
+ "\n"
+ " {0} *!joeuser@foo.com\n"
+ " Lists all registered nicks owned by joeuser@foo.com.\n"
+ "\n"
+ " {0} *Bot*!*@*\n"
+ " Lists all registered nicks with \002Bot\002 in their names (case insensitive).\n"
+ "\n"
+ " {0} * NOEXPIRE\n"
+ " Lists all registered nicks which have been set to not expire.\n"
+ "\n"
+ " {0} #51-100\n"
+ " Lists all registered nicks within the given range (51-100)."));
+
+ const Anope::string &regexengine = Config->GetBlock("options")->Get<Anope::string>("regexengine");
+ if (!regexengine.empty())
+ {
+ source.Reply(" ");
+ source.Reply(_("Regex matches are also supported using the {0} engine. Enclose your pattern in // if this is desired."), regexengine);
+ }
+
+ 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 your account from appearing in the LIST command"));
+ this->SetSyntax("{ON | OFF}");
+ }
+
+ void Run(CommandSource &source, const Anope::string &user, const Anope::string &param)
+ {
+ if (Anope::ReadOnly)
+ {
+ source.Reply(_("Services are in read-only mode."));
+ return;
+ }
+
+ NickServ::Nick *na = NickServ::FindNick(user);
+ if (!na)
+ {
+ source.Reply(_("\002{0}\002 isn't registered."), user);
+ return;
+ }
+ NickServ::Account *nc = na->GetAccount();
+
+ EventReturn MOD_RESULT = EventManager::Get()->Dispatch(&Event::SetNickOption::OnSetNickOption, 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->GetDisplay();
+ nc->SetS<bool>("NS_PRIVATE", true);
+ source.Reply(_("Private option is now \002on\002 for \002{0}\002."), nc->GetDisplay());
+ }
+ else if (param.equals_ci("OFF"))
+ {
+ Log(nc == source.GetAccount() ? LOG_COMMAND : LOG_ADMIN, source, this) << "to disable private for " << nc->GetDisplay();
+ nc->UnsetS<bool>("NS_PRIVATE");
+ source.Reply(_("Private option is now \002off\002 for \002{0}\002."), nc->GetDisplay());
+ }
+ else
+ this->OnSyntaxError(source, "PRIVATE");
+ }
+
+ void Execute(CommandSource &source, const std::vector<Anope::string> &params) override
+ {
+ this->Run(source, source.nc->GetDisplay(), params[0]);
+ }
+
+ bool OnHelp(CommandSource &source, const Anope::string &) override
+ {
+ source.Reply(_("Turns the privacy option on or off for your account."
+ " When \002PRIVATE\002 is set, your account will not appear in the account list."
+ " However, anyone who knows your account can still request information about it."));
+ return true;
+ }
+};
+
+class CommandNSSASetPrivate : public CommandNSSetPrivate
+{
+ public:
+ CommandNSSASetPrivate(Module *creator) : CommandNSSetPrivate(creator, "nickserv/saset/private", 2)
+ {
+ this->ClearSyntax();
+ this->SetSyntax(_("\037account\037 {ON | OFF}"));
+ }
+
+ void Execute(CommandSource &source, const std::vector<Anope::string> &params) override
+ {
+ this->Run(source, params[0], params[1]);
+ }
+
+ bool OnHelp(CommandSource &source, const Anope::string &) override
+ {
+ source.Reply(_("Turns the privacy option on or off for \037account\037."
+ " When \002PRIVATE\002 is set, the account will not appear in the account list."
+ " However, anyone who knows your account can still request information about it."));
+ return true;
+ }
+};
+
+
+class NSList : public Module
+ , public EventHook<Event::NickInfo>
+{
+ CommandNSList commandnslist;
+
+ CommandNSSetPrivate commandnssetprivate;
+ CommandNSSASetPrivate commandnssasetprivate;
+
+ Serialize::Field<NickServ::Account, bool> priv;
+
+ public:
+ NSList(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, VENDOR)
+ , EventHook<Event::NickInfo>(this)
+ , commandnslist(this)
+ , commandnssetprivate(this)
+ , commandnssasetprivate(this)
+ , priv(this, "NS_PRIVATE")
+ {
+ }
+
+ void OnNickInfo(CommandSource &source, NickServ::Nick *na, InfoFormatter &info, bool show_all) override
+ {
+ if (!show_all)
+ return;
+
+ if (priv.HasExt(na->GetAccount()))
+ info.AddOption(_("Private"));
+ }
+};
+
+MODULE_INIT(NSList)