1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
|
/* BotServ core functions
*
* (C) 2003-2025 Anope Team
* Contact us at team@anope.org
*
* Please read COPYING and README for further details.
*
* Based on the original code of Epona by Lara.
* Based on the original code of Services by Andy Church.
*/
#include "module.h"
class CommandBSBotList final
: public Command
{
public:
CommandBSBotList(Module *creator) : Command(creator, "botserv/botlist", 0, 1)
{
this->SetDesc(_("Lists available bots"));
this->SetSyntax("[OPERONLY] [UNUSED] [VANITY]");
}
void Execute(CommandSource &source, const std::vector<Anope::string> ¶ms) override
{
const bool is_admin = source.HasCommand("botserv/administration");
auto operonly = false;
auto unused = false;
auto vanity = false;
if (is_admin && !params.empty())
{
spacesepstream keywords(params[0]);
for (Anope::string keyword; keywords.GetToken(keyword); )
{
if (keyword.equals_ci("OPERONLY"))
operonly = true;
if (keyword.equals_ci("UNUSED"))
unused = true;
if (keyword.equals_ci("VANITY"))
vanity = true;
}
}
unsigned count = 0;
ListFormatter list(source.GetAccount());
list.AddColumn(_("Nick")).AddColumn(_("Mask")).AddColumn(_("Real name"));
for (const auto &[_, bi] : *BotListByNick)
{
if (is_admin || !bi->oper_only)
{
if (operonly && !bi->oper_only)
continue;
if (unused && bi->GetChannelCount())
continue;
if (vanity && bi->conf)
continue;
++count;
ListFormatter::ListEntry entry;
entry["Nick"] = bi->nick;
entry["Mask"] = bi->GetIdent() + "@" + bi->host;
entry["Real name"] = bi->realname;
list.AddEntry(entry);
}
}
std::vector<Anope::string> replies;
list.Process(replies);
if (!count)
{
source.Reply(_(
"There are no bots available at this time. "
"Ask a Services Operator to create one!"
));
}
else
{
source.Reply(_("Bot list:"));
for (const auto &reply : replies)
source.Reply(reply);
source.Reply(_("%d bots available."), count);
}
}
bool OnHelp(CommandSource &source, const Anope::string &subcommand) override
{
this->SendSyntax(source);
source.Reply(" ");
source.Reply(_(
"Lists all available bots on this network."
"\n\n"
"If the OPERONLY, UNUSED or VANITY options are given only "
"bots which, respectively, are oper-only, unused or were "
"added at runtime will be displayed. If multiple options are "
"given, all nicks matching at least one option will be "
"displayed."
"\n\n"
"Note that these options are limited to \037Services Operators\037."
));
return true;
}
};
class BSBotList final
: public Module
{
CommandBSBotList commandbsbotlist;
public:
BSBotList(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, VENDOR),
commandbsbotlist(this)
{
}
};
MODULE_INIT(BSBotList)
|