mirror of
https://github.com/anope/anope.git
synced 2026-06-12 18:54:47 +02:00
e66063e630
up into seperate files for each pseudo client. Also reorganized how the modules are stored, and made most of the old "extra" modules "core"
1012 lines
30 KiB
C++
1012 lines
30 KiB
C++
/* BotServ core functions
|
|
*
|
|
* (C) 2003-2011 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 CommandBSKick : public Command
|
|
{
|
|
public:
|
|
CommandBSKick(Module *creator) : Command(creator, "botserv/kick", 3, 6)
|
|
{
|
|
this->SetDesc(_("Configures kickers"));
|
|
this->SetSyntax(_("\037channel\037 \037option\037 {\037ON|\037} [\037settings\037]"));
|
|
}
|
|
|
|
void Execute(CommandSource &source, const std::vector<Anope::string> ¶ms)
|
|
{
|
|
const Anope::string &chan = params[0];
|
|
const Anope::string &option = params[1];
|
|
const Anope::string &value = params[2];
|
|
const Anope::string &ttb = params.size() > 3 ? params[3] : "";
|
|
|
|
User *u = source.u;
|
|
ChannelInfo *ci = cs_findchan(params[0]);
|
|
|
|
if (readonly)
|
|
source.Reply(_("Sorry, kicker configuration is temporarily disabled."));
|
|
else if (ci == NULL)
|
|
source.Reply(CHAN_X_NOT_REGISTERED, params[0].c_str());
|
|
else if (chan.empty() || option.empty() || value.empty())
|
|
this->OnSyntaxError(source, "");
|
|
else if (!value.equals_ci("ON") && !value.equals_ci("OFF"))
|
|
this->OnSyntaxError(source, "");
|
|
else if (!ci->HasPriv(u, CA_SET) && !u->HasPriv("botserv/administration"))
|
|
source.Reply(ACCESS_DENIED);
|
|
else if (!ci->bi)
|
|
source.Reply(BOT_NOT_ASSIGNED);
|
|
else
|
|
{
|
|
bool override = !ci->HasPriv(u, CA_SET);
|
|
Log(override ? LOG_OVERRIDE : LOG_COMMAND, u, this, ci) << option << " " << value;
|
|
|
|
if (option.equals_ci("BADWORDS"))
|
|
{
|
|
if (value.equals_ci("ON"))
|
|
{
|
|
if (!ttb.empty())
|
|
{
|
|
try
|
|
{
|
|
ci->ttb[TTB_BADWORDS] = convertTo<int16>(ttb);
|
|
if (ci->ttb[TTB_BADWORDS] < 0)
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
/* reset the value back to 0 - TSL */
|
|
ci->ttb[TTB_BADWORDS] = 0;
|
|
source.Reply(_("\002%s\002 cannot be taken as times to ban."), ttb.c_str());
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
ci->ttb[TTB_BADWORDS] = 0;
|
|
|
|
ci->botflags.SetFlag(BS_KICK_BADWORDS);
|
|
if (ci->ttb[TTB_BADWORDS])
|
|
source.Reply(_("Bot will now kick \002bad words\002, and will place a ban after \n"
|
|
"%d kicks for the same user. Use the BADWORDS command\n"
|
|
"to add or remove a bad word."), ci->ttb[TTB_BADWORDS]);
|
|
else
|
|
source.Reply(_("Bot will now kick \002bad words\002. Use the BADWORDS command\n"
|
|
"to add or remove a bad word."));
|
|
}
|
|
else
|
|
{
|
|
ci->botflags.UnsetFlag(BS_KICK_BADWORDS);
|
|
source.Reply(_("Bot won't kick \002bad words\002 anymore."));
|
|
}
|
|
}
|
|
else if (option.equals_ci("BOLDS"))
|
|
{
|
|
if (value.equals_ci("ON"))
|
|
{
|
|
if (!ttb.empty())
|
|
{
|
|
try
|
|
{
|
|
ci->ttb[TTB_BOLDS] = convertTo<int16>(ttb);
|
|
if (ci->ttb[TTB_BOLDS] < 0)
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
ci->ttb[TTB_BOLDS] = 0;
|
|
source.Reply(_("\002%s\002 cannot be taken as times to ban."), ttb.c_str());
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
ci->ttb[TTB_BOLDS] = 0;
|
|
ci->botflags.SetFlag(BS_KICK_BOLDS);
|
|
if (ci->ttb[TTB_BOLDS])
|
|
source.Reply(_("Bot will now kick \002bolds\002, and will place a ban after\n%d kicks to the same user."), ci->ttb[TTB_BOLDS]);
|
|
else
|
|
source.Reply(_("Bot will now kick \002bolds\002."));
|
|
}
|
|
else
|
|
{
|
|
ci->botflags.UnsetFlag(BS_KICK_BOLDS);
|
|
source.Reply(_("Bot won't kick \002bolds\002 anymore."));
|
|
}
|
|
}
|
|
else if (option.equals_ci("CAPS"))
|
|
{
|
|
if (value.equals_ci("ON"))
|
|
{
|
|
Anope::string min = params.size() > 4 ? params[4] : "";
|
|
Anope::string percent = params.size() > 5 ? params[5] : "";
|
|
|
|
if (!ttb.empty())
|
|
{
|
|
try
|
|
{
|
|
ci->ttb[TTB_CAPS] = convertTo<int16>(ttb);
|
|
if (ci->ttb[TTB_CAPS] < 0)
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
ci->ttb[TTB_CAPS] = 0;
|
|
source.Reply(_("\002%s\002 cannot be taken as times to ban."), ttb.c_str());
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
ci->ttb[TTB_CAPS] = 0;
|
|
|
|
ci->capsmin = 10;
|
|
try
|
|
{
|
|
ci->capsmin = convertTo<int16>(min);
|
|
}
|
|
catch (const ConvertException &) { }
|
|
if (ci->capsmin < 1)
|
|
ci->capsmin = 10;
|
|
|
|
ci->capspercent = 25;
|
|
try
|
|
{
|
|
ci->capspercent = convertTo<int16>(percent);
|
|
}
|
|
catch (const ConvertException &) { }
|
|
if (ci->capspercent < 1 || ci->capspercent > 100)
|
|
ci->capspercent = 25;
|
|
|
|
ci->botflags.SetFlag(BS_KICK_CAPS);
|
|
if (ci->ttb[TTB_CAPS])
|
|
source.Reply(_("Bot will now kick \002caps\002 (they must constitute at least\n"
|
|
"%d characters and %d%% of the entire message), and will \n"
|
|
"place a ban after %d kicks for the same user."), ci->capsmin, ci->capspercent, ci->ttb[TTB_CAPS]);
|
|
else
|
|
source.Reply(_("Bot will now kick \002caps\002 (they must constitute at least\n"
|
|
"%d characters and %d%% of the entire message)."), ci->capsmin, ci->capspercent);
|
|
}
|
|
else
|
|
{
|
|
ci->botflags.UnsetFlag(BS_KICK_CAPS);
|
|
source.Reply(_("Bot won't kick \002caps\002 anymore."));
|
|
}
|
|
}
|
|
else if (option.equals_ci("COLORS"))
|
|
{
|
|
if (value.equals_ci("ON"))
|
|
{
|
|
if (!ttb.empty())
|
|
{
|
|
try
|
|
{
|
|
ci->ttb[TTB_COLORS] = convertTo<int16>(ttb);
|
|
if (ci->ttb[TTB_COLORS] < 1)
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
ci->ttb[TTB_COLORS] = 0;
|
|
source.Reply(_("\002%s\002 cannot be taken as times to ban."), ttb.c_str());
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
ci->ttb[TTB_COLORS] = 0;
|
|
|
|
ci->botflags.SetFlag(BS_KICK_COLORS);
|
|
if (ci->ttb[TTB_COLORS])
|
|
source.Reply(_("Bot will now kick \002colors\002, and will place a ban after %d\nkicks for the same user."), ci->ttb[TTB_COLORS]);
|
|
else
|
|
source.Reply(_("Bot will now kick \002colors\002."));
|
|
}
|
|
else
|
|
{
|
|
ci->botflags.UnsetFlag(BS_KICK_COLORS);
|
|
source.Reply(_("Bot won't kick \002colors\002 anymore."));
|
|
}
|
|
}
|
|
else if (option.equals_ci("FLOOD"))
|
|
{
|
|
if (value.equals_ci("ON"))
|
|
{
|
|
Anope::string lines = params.size() > 4 ? params[4] : "";
|
|
Anope::string secs = params.size() > 5 ? params[5] : "";
|
|
|
|
if (!ttb.empty())
|
|
{
|
|
try
|
|
{
|
|
ci->ttb[TTB_FLOOD] = convertTo<int16>(ttb);
|
|
if (ci->ttb[TTB_FLOOD] < 1)
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
ci->ttb[TTB_FLOOD] = 0;
|
|
source.Reply(_("\002%s\002 cannot be taken as times to ban."), ttb.c_str());
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
ci->ttb[TTB_FLOOD] = 0;
|
|
|
|
ci->floodlines = 6;
|
|
try
|
|
{
|
|
ci->floodlines = convertTo<int16>(lines);
|
|
}
|
|
catch (const ConvertException &) { }
|
|
if (ci->floodlines < 2)
|
|
ci->floodlines = 6;
|
|
|
|
ci->floodsecs = 10;
|
|
try
|
|
{
|
|
ci->floodsecs = convertTo<int16>(secs);
|
|
}
|
|
catch (const ConvertException &) { }
|
|
if (ci->floodsecs < 1)
|
|
ci->floodsecs = 10;
|
|
if (ci->floodsecs > Config->BSKeepData)
|
|
ci->floodsecs = Config->BSKeepData;
|
|
|
|
ci->botflags.SetFlag(BS_KICK_FLOOD);
|
|
if (ci->ttb[TTB_FLOOD])
|
|
source.Reply(_("Bot will now kick \002flood\002 (%d lines in %d seconds and\nwill place a ban after %d kicks for the same user."), ci->floodlines, ci->floodsecs, ci->ttb[TTB_FLOOD]);
|
|
else
|
|
source.Reply(_("Bot will now kick \002flood\002 (%d lines in %d seconds)."), ci->floodlines, ci->floodsecs);
|
|
}
|
|
else
|
|
{
|
|
ci->botflags.UnsetFlag(BS_KICK_FLOOD);
|
|
source.Reply(_("Bot won't kick \002flood\002 anymore."));
|
|
}
|
|
}
|
|
else if (option.equals_ci("REPEAT"))
|
|
{
|
|
if (value.equals_ci("ON"))
|
|
{
|
|
Anope::string times = params.size() > 4 ? params[4] : "";
|
|
|
|
if (!ttb.empty())
|
|
{
|
|
try
|
|
{
|
|
ci->ttb[TTB_REPEAT] = convertTo<int16>(ttb);
|
|
if (ci->ttb[TTB_REPEAT] < 0)
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
ci->ttb[TTB_REPEAT] = 0;
|
|
source.Reply(_("\002%s\002 cannot be taken as times to ban."), ttb.c_str());
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
ci->ttb[TTB_REPEAT] = 0;
|
|
|
|
ci->repeattimes = 3;
|
|
try
|
|
{
|
|
ci->repeattimes = convertTo<int16>(times);
|
|
}
|
|
catch (const ConvertException &) { }
|
|
if (ci->repeattimes < 2)
|
|
ci->repeattimes = 3;
|
|
|
|
ci->botflags.SetFlag(BS_KICK_REPEAT);
|
|
if (ci->ttb[TTB_REPEAT])
|
|
source.Reply(_("Bot will now kick \002repeats\002 (users that say %d times\n"
|
|
"the same thing), and will place a ban after %d \n"
|
|
"kicks for the same user."), ci->repeattimes, ci->ttb[TTB_REPEAT]);
|
|
else
|
|
source.Reply(_("Bot will now kick \002repeats\002 (users that say %d times\n"
|
|
"the same thing)."), ci->repeattimes);
|
|
}
|
|
else
|
|
{
|
|
ci->botflags.UnsetFlag(BS_KICK_REPEAT);
|
|
source.Reply(_("Bot won't kick \002repeats\002 anymore."));
|
|
}
|
|
}
|
|
else if (option.equals_ci("REVERSES"))
|
|
{
|
|
if (value.equals_ci("ON"))
|
|
{
|
|
if (!ttb.empty())
|
|
{
|
|
try
|
|
{
|
|
ci->ttb[TTB_REVERSES] = convertTo<int16>(ttb);
|
|
if (ci->ttb[TTB_REVERSES] < 0)
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
ci->ttb[TTB_REVERSES] = 0;
|
|
source.Reply(_("\002%s\002 cannot be taken as times to ban."), ttb.c_str());
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
ci->ttb[TTB_REVERSES] = 0;
|
|
ci->botflags.SetFlag(BS_KICK_REVERSES);
|
|
if (ci->ttb[TTB_REVERSES])
|
|
source.Reply(_("Bot will now kick \002reverses\002, and will place a ban after %d\nkicks for the same user."), ci->ttb[TTB_REVERSES]);
|
|
else
|
|
source.Reply(_("Bot will now kick \002reverses\002."));
|
|
}
|
|
else
|
|
{
|
|
ci->botflags.UnsetFlag(BS_KICK_REVERSES);
|
|
source.Reply(_("Bot won't kick \002reverses\002 anymore."));
|
|
}
|
|
}
|
|
else if (option.equals_ci("UNDERLINES"))
|
|
{
|
|
if (value.equals_ci("ON"))
|
|
{
|
|
if (!ttb.empty())
|
|
{
|
|
try
|
|
{
|
|
ci->ttb[TTB_UNDERLINES] = convertTo<int16>(ttb);
|
|
if (ci->ttb[TTB_REVERSES] < 0)
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
ci->ttb[TTB_UNDERLINES] = 0;
|
|
source.Reply(_("\002%s\002 cannot be taken as times to ban."), ttb.c_str());
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
ci->ttb[TTB_UNDERLINES] = 0;
|
|
|
|
ci->botflags.SetFlag(BS_KICK_UNDERLINES);
|
|
if (ci->ttb[TTB_UNDERLINES])
|
|
source.Reply(_("Bot will now kick \002underlines\002, and will place a ban after %d\nkicks for the same user."), ci->ttb[TTB_UNDERLINES]);
|
|
else
|
|
source.Reply(_("Bot will now kick \002underlines\002."));
|
|
}
|
|
else
|
|
{
|
|
ci->botflags.UnsetFlag(BS_KICK_UNDERLINES);
|
|
source.Reply(_("Bot won't kick \002underlines\002 anymore."));
|
|
}
|
|
}
|
|
else if (option.equals_ci("ITALICS"))
|
|
{
|
|
if (value.equals_ci("ON"))
|
|
{
|
|
if (!ttb.empty())
|
|
{
|
|
try
|
|
{
|
|
ci->ttb[TTB_ITALICS] = convertTo<int16>(ttb);
|
|
if (ci->ttb[TTB_ITALICS] < 0)
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
ci->ttb[TTB_ITALICS] = 0;
|
|
source.Reply(_("\002%s\002 cannot be taken as times to ban."), ttb.c_str());
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
ci->ttb[TTB_ITALICS] = 0;
|
|
|
|
ci->botflags.SetFlag(BS_KICK_ITALICS);
|
|
if (ci->ttb[TTB_ITALICS])
|
|
source.Reply(_("Bot will now kick \002italics\002, and will place a ban after\n%d kicks for the same user."), ci->ttb[TTB_ITALICS]);
|
|
else
|
|
source.Reply(_("Bot will now kick \002italics\002."));
|
|
}
|
|
else
|
|
{
|
|
ci->botflags.UnsetFlag(BS_KICK_ITALICS);
|
|
source.Reply(_("Bot won't kick \002italics\002 anymore."));
|
|
}
|
|
}
|
|
else if (option.equals_ci("AMSGS"))
|
|
{
|
|
if (value.equals_ci("ON"))
|
|
{
|
|
if (!ttb.empty())
|
|
{
|
|
try
|
|
{
|
|
ci->ttb[TTB_AMSGS] = convertTo<int16>(ttb);
|
|
if (ci->ttb[TTB_AMSGS] < 0)
|
|
throw ConvertException();
|
|
}
|
|
catch (const ConvertException &)
|
|
{
|
|
ci->ttb[TTB_AMSGS] = 0;
|
|
source.Reply(_("\002%s\002 cannot be taken as times to ban."), ttb.c_str());
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
ci->ttb[TTB_AMSGS] = 0;
|
|
|
|
ci->botflags.SetFlag(BS_KICK_AMSGS);
|
|
if (ci->ttb[TTB_AMSGS])
|
|
source.Reply(_("Bot will now kick for \002amsgs\002, and will place a ban after %d\nkicks for the same user."), ci->ttb[TTB_AMSGS]);
|
|
else
|
|
source.Reply(_("Bot will now kick for \002amsgs\002"));
|
|
}
|
|
else
|
|
{
|
|
ci->botflags.UnsetFlag(BS_KICK_AMSGS);
|
|
source.Reply(_("Bot won't kick for \002amsgs\002 anymore."));
|
|
}
|
|
}
|
|
else
|
|
source.Reply(UNKNOWN_OPTION, Config->UseStrictPrivMsgString.c_str(), option.c_str(), this->name.c_str());
|
|
}
|
|
return;
|
|
}
|
|
|
|
bool OnHelp(CommandSource &source, const Anope::string &subcommand)
|
|
{
|
|
if (subcommand.empty())
|
|
{
|
|
this->SendSyntax(source);
|
|
source.Reply(" ");
|
|
source.Reply(_("Configures bot kickers. \037option\037 can be one of:\n"
|
|
" \n"
|
|
" AMSGS Sets if the bot kicks for amsgs\n"
|
|
" BOLDS Sets if the bot kicks bolds\n"
|
|
" BADWORDS Sets if the bot kicks bad words\n"
|
|
" CAPS Sets if the bot kicks caps\n"
|
|
" COLORS Sets if the bot kicks colors\n"
|
|
" FLOOD Sets if the bot kicks flooding users\n"
|
|
" REPEAT Sets if the bot kicks users who repeat\n"
|
|
" themselves\n"
|
|
" REVERSES Sets if the bot kicks reverses\n"
|
|
" UNDERLINES Sets if the bot kicks underlines\n"
|
|
" ITALICS Sets if the bot kicks italics\n"
|
|
" \n"
|
|
"Type \002%s%s HELP KICK \037option\037\002 for more information\n"
|
|
"on a specific option.\n"
|
|
" \n"
|
|
"Note: access to this command is controlled by the\n"
|
|
"level SET."), Config->UseStrictPrivMsgString.c_str(), source.owner->nick.c_str());
|
|
}
|
|
else if (subcommand.equals_ci("BADWORDS"))
|
|
source.Reply(_("Syntax: \002\037#channel\037 BADWORDS {\037ON|OFF\037} [\037ttb\037]\002\n"
|
|
"Sets the bad words kicker on or off. When enabled, this\n"
|
|
"option tells the bot to kick users who say certain words\n"
|
|
"on the channels.\n"
|
|
"You can define bad words for your channel using the\n"
|
|
"\002BADWORDS\002 command. Type \002%s%s HELP BADWORDS\002 for\n"
|
|
"more information.\n"
|
|
"ttb is the number of times a user can be kicked\n"
|
|
"before it get banned. Don't give ttb to disable\n"
|
|
"the ban system once activated."), Config->UseStrictPrivMsgString.c_str(), source.owner->nick.c_str());
|
|
else if (subcommand.equals_ci("BOLDS"))
|
|
source.Reply(_("Syntax: \002\037channel\037 BOLDS {\037ON|OFF\037} [\037ttb\037]\002\n"
|
|
"Sets the bolds kicker on or off. When enabled, this\n"
|
|
"option tells the bot to kick users who use bolds.\n"
|
|
"ttb is the number of times a user can be kicked\n"
|
|
"before it get banned. Don't give ttb to disable\n"
|
|
"the ban system once activated."));
|
|
else if (subcommand.equals_ci("CAPS"))
|
|
source.Reply(_("Syntax: \002\037channel\037 CAPS {\037ON|OFF\037} [\037ttb\037 [\037min\037 [\037percent\037]]]\002\n"
|
|
"Sets the caps kicker on or off. When enabled, this\n"
|
|
"option tells the bot to kick users who are talking in\n"
|
|
"CAPS.\n"
|
|
"The bot kicks only if there are at least \002min\002 caps\n"
|
|
"and they constitute at least \002percent\002%% of the total \n"
|
|
"text line (if not given, it defaults to 10 characters\n"
|
|
"and 25%%).\n"
|
|
"ttb is the number of times a user can be kicked\n"
|
|
"before it get banned. Don't give ttb to disable\n"
|
|
"the ban system once activated."));
|
|
else if (subcommand.equals_ci("COLORS"))
|
|
source.Reply(_("Syntax: \002\037channel\037 COLORS {\037ON|OFF\037} [\037ttb\037]\002\n"
|
|
"Sets the colors kicker on or off. When enabled, this\n"
|
|
"option tells the bot to kick users who use colors.\n"
|
|
"ttb is the number of times a user can be kicked\n"
|
|
"before it get banned. Don't give ttb to disable\n"
|
|
"the ban system once activated."));
|
|
else if (subcommand.equals_ci("FLOOD"))
|
|
source.Reply(_("Syntax: \002\037channel\037 FLOOD {\037ON|OFF\037} [\037ttb\037 [\037ln\037 [\037secs\037]]]\002\n"
|
|
"Sets the flood kicker on or off. When enabled, this\n"
|
|
"option tells the bot to kick users who are flooding\n"
|
|
"the channel using at least \002ln\002 lines in \002secs\002 seconds\n"
|
|
"(if not given, it defaults to 6 lines in 10 seconds).\n"
|
|
" \n"
|
|
"ttb is the number of times a user can be kicked\n"
|
|
"before it get banned. Don't give ttb to disable\n"
|
|
"the ban system once activated."));
|
|
else if (subcommand.equals_ci("REPEAT"))
|
|
source.Reply(_("Syntax: \002\037#channel\037 REPEAT {\037ON|OFF\037} [\037ttb\037 [\037num\037]]\002\n"
|
|
"Sets the repeat kicker on or off. When enabled, this\n"
|
|
"option tells the bot to kick users who are repeating\n"
|
|
"themselves \002num\002 times (if num is not given, it\n"
|
|
"defaults to 3).\n"
|
|
"ttb is the number of times a user can be kicked\n"
|
|
"before it get banned. Don't give ttb to disable\n"
|
|
"the ban system once activated."));
|
|
else if (subcommand.equals_ci("REVERSES"))
|
|
source.Reply(_("Syntax: \002\037channel\037 REVERSES {\037ON|OFF\037} [\037ttb\037]\002\n"
|
|
"Sets the reverses kicker on or off. When enabled, this\n"
|
|
"option tells the bot to kick users who use reverses.\n"
|
|
"ttb is the number of times a user can be kicked\n"
|
|
"before it get banned. Don't give ttb to disable\n"
|
|
"the ban system once activated."));
|
|
else if (subcommand.equals_ci("UNDERLINES"))
|
|
source.Reply(_("Syntax: \002\037channel\037 UNDERLINES {\037ON|OFF\037} [\037ttb\037]\002\n"
|
|
"Sets the underlines kicker on or off. When enabled, this\n"
|
|
"option tells the bot to kick users who use underlines.\n"
|
|
"ttb is the number of times a user can be kicked\n"
|
|
"before it get banned. Don't give ttb to disable\n"
|
|
"the ban system once activated."));
|
|
else if (subcommand.equals_ci("ITALICS"))
|
|
source.Reply(_("Syntax: \002\037channel\037 ITALICS {\037ON|OFF\037} [\037ttb\037]\002\n"
|
|
"Sets the italics kicker on or off. When enabled, this\n"
|
|
"option tells the bot to kick users who use italics.\n"
|
|
"ttb is the number of times a user can be kicked\n"
|
|
"before it get banned. Don't give ttb to disable\n"
|
|
"the ban system once activated."));
|
|
else if (subcommand.equals_ci("AMSGS"))
|
|
source.Reply(_("Syntax: \002\037channel\037 AMSGS {\037ON|OFF\037} [\037ttb\037]\002\n"
|
|
"Sets the amsg kicker on or off. When enabled, the bot will\n"
|
|
"kick users who send the same message to multiple channels\n"
|
|
"where BotServ bots are.\n"
|
|
"Ttb is the number of times a user can be kicked\n"
|
|
"before it get banned. Don't give ttb to disable\n"
|
|
"the ban system once activated."));
|
|
else
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
};
|
|
|
|
struct BanData
|
|
{
|
|
Anope::string mask;
|
|
time_t last_use;
|
|
int16 ttb[TTB_SIZE];
|
|
|
|
BanData()
|
|
{
|
|
this->Clear();
|
|
}
|
|
|
|
void Clear()
|
|
{
|
|
last_use = 0;
|
|
for (int i = 0; i < TTB_SIZE; ++i)
|
|
this->ttb[i] = 0;
|
|
}
|
|
};
|
|
|
|
struct UserData
|
|
{
|
|
UserData()
|
|
{
|
|
this->Clear();
|
|
}
|
|
|
|
void Clear()
|
|
{
|
|
last_use = last_start = Anope::CurTime;
|
|
lines = times = 0;
|
|
lastline.clear();
|
|
}
|
|
|
|
/* Data validity */
|
|
time_t last_use;
|
|
|
|
/* for flood kicker */
|
|
int16 lines;
|
|
time_t last_start;
|
|
|
|
/* for repeat kicker */
|
|
Anope::string lastline;
|
|
Anope::string lasttarget;
|
|
int16 times;
|
|
};
|
|
|
|
|
|
class BanDataPurger : public CallBack
|
|
{
|
|
public:
|
|
BanDataPurger(Module *owner) : CallBack(owner, 300, Anope::CurTime, true) { }
|
|
|
|
void Tick(time_t)
|
|
{
|
|
Log(LOG_DEBUG) << "bs_main: Running bandata purger";
|
|
|
|
for (channel_map::iterator it = ChannelList.begin(), it_end = ChannelList.end(); it != it_end; ++it)
|
|
{
|
|
Channel *c = it->second;
|
|
|
|
std::map<Anope::string, BanData> bandata;
|
|
if (c->GetExtRegular("bs_main_bandata", bandata))
|
|
{
|
|
for (std::map<Anope::string, BanData>::iterator it2 = bandata.begin(), it2_end = bandata.end(); it2 != it2_end;)
|
|
{
|
|
const Anope::string &user = it->first;
|
|
BanData *bd = &it2->second;
|
|
++it2;
|
|
|
|
if (Anope::CurTime - bd->last_use > Config->BSKeepData)
|
|
{
|
|
bandata.erase(user);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (bandata.empty())
|
|
c->Shrink("bs_main_bandata");
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
class BSKick : public Module
|
|
{
|
|
CommandBSKick commandbskick;
|
|
BanDataPurger purger;
|
|
|
|
BanData *GetBanData(User *u, Channel *c)
|
|
{
|
|
std::map<Anope::string, BanData> bandatamap;
|
|
if (!c->GetExtRegular("bs_main_bandata", bandatamap));
|
|
c->Extend("bs_main_bandata", new ExtensibleItemRegular<std::map<Anope::string, BanData> >(bandatamap));
|
|
c->GetExtRegular("bs_main_bandata", bandatamap);
|
|
|
|
BanData *bd = &bandatamap[u->GetMask()];
|
|
if (bd->last_use && Anope::CurTime - bd->last_use > Config->BSKeepData)
|
|
bd->Clear();
|
|
bd->last_use = Anope::CurTime;
|
|
return bd;
|
|
}
|
|
|
|
UserData *GetUserData(User *u, Channel *c)
|
|
{
|
|
UserData *ud = NULL;
|
|
UserContainer *uc = c->FindUser(u);
|
|
if (uc != NULL && !uc->GetExtPointer("bs_main_userdata", ud))
|
|
{
|
|
ud = new UserData();
|
|
uc->Extend("bs_main_userdata", new ExtensibleItemPointer<UserData>(ud));
|
|
}
|
|
return ud;
|
|
}
|
|
|
|
void check_ban(ChannelInfo *ci, User *u, int ttbtype)
|
|
{
|
|
/* Don't ban ulines */
|
|
if (u->server->IsULined())
|
|
return;
|
|
|
|
BanData *bd = this->GetBanData(u, ci->c);
|
|
|
|
++bd->ttb[ttbtype];
|
|
if (ci->ttb[ttbtype] && bd->ttb[ttbtype] >= ci->ttb[ttbtype])
|
|
{
|
|
/* Should not use == here because bd->ttb[ttbtype] could possibly be > ci->ttb[ttbtype]
|
|
* if the TTB was changed after it was not set (0) before and the user had already been
|
|
* kicked a few times. Bug #1056 - Adam */
|
|
Anope::string mask;
|
|
|
|
bd->ttb[ttbtype] = 0;
|
|
|
|
get_idealban(ci, u, mask);
|
|
|
|
if (ci->c)
|
|
ci->c->SetMode(NULL, CMODE_BAN, mask);
|
|
FOREACH_MOD(I_OnBotBan, OnBotBan(u, ci, mask));
|
|
}
|
|
}
|
|
|
|
void bot_kick(ChannelInfo *ci, User *u, const char *message, ...)
|
|
{
|
|
va_list args;
|
|
char buf[1024];
|
|
|
|
if (!ci || !ci->bi || !ci->c || !u || u->server->IsULined())
|
|
return;
|
|
|
|
Anope::string fmt = translate(u, message);
|
|
va_start(args, message);
|
|
vsnprintf(buf, sizeof(buf), fmt.c_str(), args);
|
|
va_end(args);
|
|
|
|
ci->c->Kick(ci->bi, u, "%s", buf);
|
|
}
|
|
|
|
public:
|
|
BSKick(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, CORE),
|
|
commandbskick(this), purger(this)
|
|
{
|
|
this->SetAuthor("Anope");
|
|
|
|
ModuleManager::RegisterService(&commandbskick);
|
|
|
|
ModuleManager::Attach(I_OnPrivmsg, this);
|
|
}
|
|
|
|
~BSKick()
|
|
{
|
|
for (channel_map::const_iterator cit = ChannelList.begin(), cit_end = ChannelList.end(); cit != cit_end; ++cit)
|
|
{
|
|
cit->second->Shrink("bs_main_userdata");
|
|
cit->second->Shrink("bs_main_bandata");
|
|
}
|
|
}
|
|
|
|
void OnPrivmsg(User *u, Channel *c, Anope::string &msg)
|
|
{
|
|
/* Now we can make kicker stuff. We try to order the checks
|
|
* from the fastest one to the slowest one, since there's
|
|
* no need to process other kickers if a user is kicked before
|
|
* the last kicker check.
|
|
*
|
|
* But FIRST we check whether the user is protected in any
|
|
* way.
|
|
*/
|
|
ChannelInfo *ci = c->ci;
|
|
if (ci == NULL)
|
|
return;
|
|
|
|
bool Allow = true;
|
|
if (ci->HasPriv(u, CA_NOKICK))
|
|
Allow = false;
|
|
else if (ci->botflags.HasFlag(BS_DONTKICKOPS) && (c->HasUserStatus(u, CMODE_HALFOP) || c->HasUserStatus(u, CMODE_OP) || c->HasUserStatus(u, CMODE_PROTECT) || c->HasUserStatus(u, CMODE_OWNER)))
|
|
Allow = false;
|
|
else if (ci->botflags.HasFlag(BS_DONTKICKVOICES) && c->HasUserStatus(u, CMODE_VOICE))
|
|
Allow = false;
|
|
|
|
if (Allow)
|
|
{
|
|
Anope::string realbuf = msg;
|
|
|
|
/* If it's a /me, cut the CTCP part because the ACTION will cause
|
|
* problems with the caps or badwords kicker
|
|
*/
|
|
if (realbuf.substr(0, 8).equals_ci("\1ACTION ") && realbuf[realbuf.length() - 1] == '\1')
|
|
{
|
|
realbuf.erase(0, 8);
|
|
realbuf.erase(realbuf.length() - 1);
|
|
}
|
|
|
|
if (realbuf.empty())
|
|
return;
|
|
|
|
/* Bolds kicker */
|
|
if (ci->botflags.HasFlag(BS_KICK_BOLDS) && realbuf.find(2) != Anope::string::npos)
|
|
{
|
|
check_ban(ci, u, TTB_BOLDS);
|
|
bot_kick(ci, u, _("Don't use bolds on this channel!"));
|
|
return;
|
|
}
|
|
|
|
/* Color kicker */
|
|
if (ci->botflags.HasFlag(BS_KICK_COLORS) && realbuf.find(3) != Anope::string::npos)
|
|
{
|
|
check_ban(ci, u, TTB_COLORS);
|
|
bot_kick(ci, u, _("Don't use colors on this channel!"));
|
|
return;
|
|
}
|
|
|
|
/* Reverses kicker */
|
|
if (ci->botflags.HasFlag(BS_KICK_REVERSES) && realbuf.find(22) != Anope::string::npos)
|
|
{
|
|
check_ban(ci, u, TTB_REVERSES);
|
|
bot_kick(ci, u, _("Don't use reverses on this channel!"));
|
|
return;
|
|
}
|
|
|
|
/* Italics kicker */
|
|
if (ci->botflags.HasFlag(BS_KICK_ITALICS) && realbuf.find(29) != Anope::string::npos)
|
|
{
|
|
check_ban(ci, u, TTB_ITALICS);
|
|
bot_kick(ci, u, _("Don't use italics on this channel!"));
|
|
return;
|
|
}
|
|
|
|
/* Underlines kicker */
|
|
if (ci->botflags.HasFlag(BS_KICK_UNDERLINES) && realbuf.find(31) != Anope::string::npos)
|
|
{
|
|
check_ban(ci, u, TTB_UNDERLINES);
|
|
bot_kick(ci, u, _("Don't use underlines on this channel!"));
|
|
return;
|
|
}
|
|
|
|
/* Caps kicker */
|
|
if (ci->botflags.HasFlag(BS_KICK_CAPS) && realbuf.length() >= ci->capsmin)
|
|
{
|
|
int i = 0, l = 0;
|
|
|
|
for (unsigned j = 0, end = realbuf.length(); j < end; ++j)
|
|
{
|
|
if (isupper(realbuf[j]))
|
|
++i;
|
|
else if (islower(realbuf[j]))
|
|
++l;
|
|
}
|
|
|
|
/* i counts uppercase chars, l counts lowercase chars. Only
|
|
* alphabetic chars (so islower || isupper) qualify for the
|
|
* percentage of caps to kick for; the rest is ignored. -GD
|
|
*/
|
|
|
|
if ((i || l) && i >= ci->capsmin && i * 100 / (i + l) >= ci->capspercent)
|
|
{
|
|
check_ban(ci, u, TTB_CAPS);
|
|
bot_kick(ci, u, _("Turn caps lock OFF!"));
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* Bad words kicker */
|
|
if (ci->botflags.HasFlag(BS_KICK_BADWORDS))
|
|
{
|
|
bool mustkick = false;
|
|
|
|
/* Normalize the buffer */
|
|
Anope::string nbuf = normalizeBuffer(realbuf);
|
|
|
|
for (unsigned i = 0, end = ci->GetBadWordCount(); i < end; ++i)
|
|
{
|
|
BadWord *bw = ci->GetBadWord(i);
|
|
|
|
if (bw->type == BW_ANY && ((Config->BSCaseSensitive && nbuf.find(bw->word) != Anope::string::npos) || (!Config->BSCaseSensitive && nbuf.find_ci(bw->word) != Anope::string::npos)))
|
|
mustkick = true;
|
|
else if (bw->type == BW_SINGLE)
|
|
{
|
|
size_t len = bw->word.length();
|
|
|
|
if ((Config->BSCaseSensitive && bw->word.equals_cs(nbuf)) || (!Config->BSCaseSensitive && bw->word.equals_ci(nbuf)))
|
|
mustkick = true;
|
|
else if (nbuf.find(' ') == len && ((Config->BSCaseSensitive && bw->word.equals_cs(nbuf)) || (!Config->BSCaseSensitive && bw->word.equals_ci(nbuf))))
|
|
mustkick = true;
|
|
else
|
|
{
|
|
if (nbuf.rfind(' ') == nbuf.length() - len - 1 && ((Config->BSCaseSensitive && nbuf.find(bw->word) == nbuf.length() - len) || (!Config->BSCaseSensitive && nbuf.find_ci(bw->word) == nbuf.length() - len)))
|
|
mustkick = true;
|
|
else
|
|
{
|
|
Anope::string wordbuf = " " + bw->word + " ";
|
|
|
|
if ((Config->BSCaseSensitive && nbuf.find(wordbuf) != Anope::string::npos) || (!Config->BSCaseSensitive && nbuf.find_ci(wordbuf) != Anope::string::npos))
|
|
mustkick = true;
|
|
}
|
|
}
|
|
}
|
|
else if (bw->type == BW_START)
|
|
{
|
|
size_t len = bw->word.length();
|
|
|
|
if ((Config->BSCaseSensitive && nbuf.substr(0, len).equals_cs(bw->word)) || (!Config->BSCaseSensitive && nbuf.substr(0, len).equals_ci(bw->word)))
|
|
mustkick = true;
|
|
else
|
|
{
|
|
Anope::string wordbuf = " " + bw->word;
|
|
|
|
if ((Config->BSCaseSensitive && nbuf.find(wordbuf) != Anope::string::npos) || (!Config->BSCaseSensitive && nbuf.find_ci(wordbuf) != Anope::string::npos))
|
|
mustkick = true;
|
|
}
|
|
}
|
|
else if (bw->type == BW_END)
|
|
{
|
|
size_t len = bw->word.length();
|
|
|
|
if ((Config->BSCaseSensitive && nbuf.substr(nbuf.length() - len).equals_cs(bw->word)) || (!Config->BSCaseSensitive && nbuf.substr(nbuf.length() - len).equals_ci(bw->word)))
|
|
mustkick = true;
|
|
else
|
|
{
|
|
Anope::string wordbuf = bw->word + " ";
|
|
|
|
if ((Config->BSCaseSensitive && nbuf.find(wordbuf) != Anope::string::npos) || (!Config->BSCaseSensitive && nbuf.find_ci(wordbuf) != Anope::string::npos))
|
|
mustkick = true;
|
|
}
|
|
}
|
|
|
|
if (mustkick)
|
|
{
|
|
check_ban(ci, u, TTB_BADWORDS);
|
|
if (Config->BSGentleBWReason)
|
|
bot_kick(ci, u, _("Watch your language!"));
|
|
else
|
|
bot_kick(ci, u, _("Don't use the word \"%s\" on this channel!"), bw->word.c_str());
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
UserData *ud = NULL;
|
|
|
|
/* Flood kicker */
|
|
if (ci->botflags.HasFlag(BS_KICK_FLOOD))
|
|
{
|
|
ud = GetUserData(u, c);
|
|
if (ud)
|
|
{
|
|
if (Anope::CurTime - ud->last_start > ci->floodsecs)
|
|
{
|
|
ud->last_start = Anope::CurTime;
|
|
ud->lines = 0;
|
|
}
|
|
|
|
++ud->lines;
|
|
if (ud->lines >= ci->floodlines)
|
|
{
|
|
check_ban(ci, u, TTB_FLOOD);
|
|
bot_kick(ci, u, _("Stop flooding!"));
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Repeat kicker */
|
|
if (ci->botflags.HasFlag(BS_KICK_REPEAT))
|
|
{
|
|
if (!ud)
|
|
ud = GetUserData(u, c);
|
|
if (ud)
|
|
{
|
|
|
|
if (!ud->lastline.empty() && !ud->lastline.equals_ci(realbuf))
|
|
{
|
|
ud->lastline = realbuf;
|
|
ud->times = 0;
|
|
}
|
|
else
|
|
{
|
|
if (ud->lastline.empty())
|
|
ud->lastline = realbuf;
|
|
++ud->times;
|
|
}
|
|
|
|
if (ud->times >= ci->repeattimes)
|
|
{
|
|
check_ban(ci, u, TTB_REPEAT);
|
|
bot_kick(ci, u, _("Stop repeating yourself!"));
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ud && ud->lastline.equals_ci(realbuf) && !ud->lasttarget.empty() && !ud->lasttarget.equals_ci(ci->name))
|
|
{
|
|
for (UChannelList::iterator it = u->chans.begin(); it != u->chans.end();)
|
|
{
|
|
Channel *chan = (*it)->chan;
|
|
++it;
|
|
|
|
if (chan->ci != NULL && chan->ci->botflags.HasFlag(BS_KICK_AMSGS) && !chan->ci->HasPriv(u, CA_NOKICK))
|
|
{
|
|
check_ban(chan->ci, u, TTB_AMSGS);
|
|
bot_kick(chan->ci, u, _("Don't use AMSGs!"));
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ud)
|
|
ud->lasttarget = ci->name;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
MODULE_INIT(BSKick)
|