mirror of
https://github.com/anope/anope.git
synced 2026-06-12 17:04:47 +02:00
300 lines
7.8 KiB
C++
300 lines
7.8 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 CommandOSChanList final
|
|
: public Command
|
|
{
|
|
public:
|
|
CommandOSChanList(Module *creator) : Command(creator, "operserv/chanlist", 0, 2)
|
|
{
|
|
this->SetDesc(_("Lists all channel records"));
|
|
this->SetSyntax(_("[{\037pattern\037 | \037nick\037} [\037SECRET\037]]"));
|
|
}
|
|
|
|
void Execute(CommandSource &source, const std::vector<Anope::string> ¶ms) override
|
|
{
|
|
const Anope::string &pattern = !params.empty() ? params[0] : "";
|
|
const Anope::string &opt = params.size() > 1 ? params[1] : "";
|
|
std::set<Anope::string> modes;
|
|
User *u2;
|
|
unsigned int count = 0;
|
|
|
|
if (!pattern.empty())
|
|
Log(LOG_ADMIN, source, this) << "for " << pattern;
|
|
else
|
|
Log(LOG_ADMIN, source, this);
|
|
|
|
if (!opt.empty() && opt.equals_ci("SECRET"))
|
|
{
|
|
modes.insert("SECRET");
|
|
modes.insert("PRIVATE");
|
|
}
|
|
|
|
ListFormatter list(source.GetAccount());
|
|
list.AddColumn(_("Name")).AddColumn(_("Users")).AddColumn(_("Modes")).AddColumn(_("Topic"));
|
|
list.SetFlexible([](ListFormatter::ListEntry &row)
|
|
{
|
|
return row["Topic"].empty()
|
|
? _("\002{name}\002 -- {users} user(s); +{modes}")
|
|
: _("\002{name}\002 -- {users} user(s); +{modes} ({topic})");
|
|
});
|
|
|
|
|
|
if (!pattern.empty() && (u2 = User::Find(pattern, true)))
|
|
{
|
|
source.Reply(_("\002%s\002 channel list:"), u2->nick.c_str());
|
|
|
|
for (const auto &[_, cc]: u2->chans)
|
|
{
|
|
if (!modes.empty())
|
|
{
|
|
for (const auto &mode : modes)
|
|
{
|
|
if (!cc->chan->HasMode(mode))
|
|
continue;
|
|
}
|
|
}
|
|
|
|
ListFormatter::ListEntry entry;
|
|
entry["Name"] = cc->chan->name;
|
|
entry["Users"] = Anope::ToString(cc->chan->users.size());
|
|
entry["Modes"] = cc->chan->GetModes(true, true);
|
|
entry["Topic"] = cc->chan->topic;
|
|
list.AddEntry(entry);
|
|
|
|
++count;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
source.Reply(_("Channel list:"));
|
|
|
|
for (const auto &[_, c] : ChannelList)
|
|
{
|
|
if (!pattern.empty() && !Anope::Match(c->name, pattern, false, true))
|
|
continue;
|
|
if (!modes.empty())
|
|
{
|
|
for (const auto &mode : modes)
|
|
{
|
|
if (!c->HasMode(mode))
|
|
continue;
|
|
}
|
|
}
|
|
|
|
ListFormatter::ListEntry entry;
|
|
entry["Name"] = c->name;
|
|
entry["Users"] = Anope::ToString(c->users.size());
|
|
entry["Modes"] = c->GetModes(true, true);
|
|
entry["Topic"] = c->topic;
|
|
list.AddEntry(entry);
|
|
|
|
++count;
|
|
}
|
|
}
|
|
|
|
list.SendTo(source);
|
|
source.Reply(_("End of channel list. \002%u\002 channels shown."), count);
|
|
}
|
|
|
|
bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
|
|
{
|
|
this->SendSyntax(source);
|
|
source.Reply(" ");
|
|
source.Reply(_(
|
|
"Lists all channels currently in use on the IRC network, whether they "
|
|
"are registered or not."
|
|
"\n\n"
|
|
"If \002pattern\002 is given, lists only channels that match it. If a nickname "
|
|
"is given, lists only the channels the user using it is on. If SECRET is "
|
|
"specified, lists only channels matching \002pattern\002 that have the +s or "
|
|
"+p mode."
|
|
));
|
|
|
|
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 CommandOSUserList final
|
|
: public Command
|
|
{
|
|
public:
|
|
CommandOSUserList(Module *creator) : Command(creator, "operserv/userlist", 0, 2)
|
|
{
|
|
this->SetDesc(_("Lists all user records"));
|
|
this->SetSyntax(_("[{\037pattern\037 | \037channel\037} [\037INVISIBLE\037]]"));
|
|
}
|
|
|
|
void Execute(CommandSource &source, const std::vector<Anope::string> ¶ms) override
|
|
{
|
|
const Anope::string &pattern = !params.empty() ? params[0] : "";
|
|
const Anope::string &opt = params.size() > 1 ? params[1] : "";
|
|
Channel *c;
|
|
std::set<Anope::string> modes;
|
|
unsigned int count = 0;
|
|
|
|
if (!pattern.empty())
|
|
Log(LOG_ADMIN, source, this) << "for " << pattern;
|
|
else
|
|
Log(LOG_ADMIN, source, this);
|
|
|
|
if (!opt.empty() && opt.equals_ci("INVISIBLE"))
|
|
modes.insert("INVIS");
|
|
|
|
ListFormatter list(source.GetAccount());
|
|
list.AddColumn(_("Name")).AddColumn(_("Mask")).AddColumn(_("Realname"));
|
|
list.SetFlexible(_("\002{name}\002 ({mask}) [{realname}]"));
|
|
|
|
if (!pattern.empty() && (c = Channel::Find(pattern)))
|
|
{
|
|
source.Reply(_("\002%s\002 users list:"), pattern.c_str());
|
|
|
|
for (const auto &[_, uc] : c->users)
|
|
{
|
|
if (!modes.empty())
|
|
{
|
|
for (const auto &mode : modes)
|
|
{
|
|
if (!uc->user->HasMode(mode))
|
|
continue;
|
|
}
|
|
}
|
|
|
|
ListFormatter::ListEntry entry;
|
|
entry["Name"] = uc->user->nick;
|
|
entry["Mask"] = uc->user->GetIdent() + "@" + uc->user->GetDisplayedHost();
|
|
entry["Realname"] = uc->user->realname;
|
|
list.AddEntry(entry);
|
|
|
|
++count;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Historically this has been ordered, so... */
|
|
Anope::map<User *> ordered_map;
|
|
for (const auto &[nick, user] : UserListByNick)
|
|
ordered_map[nick] = user;
|
|
|
|
source.Reply(_("Users list:"));
|
|
|
|
for (auto it = ordered_map.begin(); it != ordered_map.end(); ++it)
|
|
{
|
|
User *u2 = it->second;
|
|
|
|
if (!pattern.empty())
|
|
{
|
|
/* check displayed host, host, and ip */
|
|
Anope::string masks[] = {
|
|
u2->nick + "!" + u2->GetIdent() + "@" + u2->GetDisplayedHost(),
|
|
u2->nick + "!" + u2->GetIdent() + "@" + u2->host,
|
|
u2->nick + "!" + u2->GetIdent() + "@" + u2->ip.addr()
|
|
};
|
|
|
|
bool match = false;
|
|
for (const auto &mask : masks)
|
|
{
|
|
/* Check mask with realname included, too */
|
|
if (Anope::Match(mask, pattern, false, true) || Anope::Match(mask + "#" + u2->realname, pattern, false, true))
|
|
{
|
|
match = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!match)
|
|
continue;
|
|
|
|
if (!modes.empty())
|
|
{
|
|
for (const auto &mode : modes)
|
|
{
|
|
if (!u2->HasMode(mode))
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
ListFormatter::ListEntry entry;
|
|
entry["Name"] = u2->nick;
|
|
entry["Mask"] = u2->GetIdent() + "@" + u2->GetDisplayedHost();
|
|
entry["Realname"] = u2->realname;
|
|
list.AddEntry(entry);
|
|
|
|
++count;
|
|
}
|
|
}
|
|
|
|
list.SendTo(source);
|
|
source.Reply(_("End of users list. \002%u\002 users shown."), count);
|
|
}
|
|
|
|
bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
|
|
{
|
|
this->SendSyntax(source);
|
|
source.Reply(" ");
|
|
source.Reply(_(
|
|
"Lists all users currently online on the IRC network, whether their "
|
|
"nick is registered or not."
|
|
"\n\n"
|
|
"If \002pattern\002 is given, lists only users that match it (it must be in "
|
|
"the format nick!user@host[#realname]). If \002channel\002 is given, lists "
|
|
"only users that are on the given channel. If INVISIBLE is specified, only users "
|
|
"with the +i flag will be listed."
|
|
));
|
|
|
|
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 OSList final
|
|
: public Module
|
|
{
|
|
CommandOSChanList commandoschanlist;
|
|
CommandOSUserList commandosuserlist;
|
|
|
|
public:
|
|
OSList(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, VENDOR),
|
|
commandoschanlist(this), commandosuserlist(this)
|
|
{
|
|
|
|
}
|
|
};
|
|
|
|
MODULE_INIT(OSList)
|