diff options
Diffstat (limited to 'modules/nickserv/list.cpp')
-rw-r--r-- | modules/nickserv/list.cpp | 299 |
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> ¶ms) 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 ®exengine = 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 ¶m) + { + 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> ¶ms) 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> ¶ms) 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) |