mirror of
https://github.com/anope/anope.git
synced 2026-06-24 22:26:37 +02:00
412 lines
10 KiB
C++
412 lines
10 KiB
C++
/* BotServ core functions
|
|
*
|
|
* (C) 2003-2010 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 CommandBSBot : public Command
|
|
{
|
|
private:
|
|
CommandReturn DoAdd(User *u, const std::vector<Anope::string> ¶ms)
|
|
{
|
|
Anope::string nick = params[1];
|
|
Anope::string user = params[2];
|
|
Anope::string host = params[3];
|
|
Anope::string real = params[4];
|
|
BotInfo *bi;
|
|
|
|
if (findbot(nick))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BOT_ALREADY_EXISTS, nick.c_str());
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (nick.length() > Config.NickLen)
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_NICK);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (user.length() > Config.UserLen)
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_LONG_IDENT, Config.UserLen);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (host.length() > Config.HostLen)
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_LONG_HOST, Config.HostLen);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
/* Check the nick is valid re RFC 2812 */
|
|
if (isdigit(nick[0]) || nick[0] == '-')
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_NICK);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
for (unsigned i = 0, end = nick.length(); i < end && i < Config.NickLen; ++i)
|
|
if (!isvalidnick(nick[i]))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_NICK);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
/* check for hardcored ircd forbidden nicks */
|
|
if (!ircdproto->IsNickValid(nick))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_NICK);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
/* Check the host is valid re RFC 2812 */
|
|
if (!isValidHost(host, 3))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_HOST);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
for (unsigned i = 0, end = user.length(); i < end && i < Config.UserLen; ++i)
|
|
if (!isalnum(user[i]))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_IDENT, Config.UserLen);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
/* We check whether the nick is registered, and inform the user
|
|
* if so. You need to drop the nick manually before you can use
|
|
* it as a bot nick from now on -GD
|
|
*/
|
|
if (findnick(nick))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, NICK_ALREADY_REGISTERED, nick.c_str());
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!(bi = new BotInfo(nick, user, host, real)))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BOT_CREATION_FAILED);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
notice_lang(Config.s_BotServ, u, BOT_BOT_ADDED, bi->nick.c_str(), bi->GetIdent().c_str(), bi->host.c_str(), bi->realname.c_str());
|
|
|
|
FOREACH_MOD(I_OnBotCreate, OnBotCreate(bi));
|
|
return MOD_CONT;
|
|
}
|
|
|
|
CommandReturn DoChange(User *u, const std::vector<Anope::string> ¶ms)
|
|
{
|
|
Anope::string oldnick = params[1];
|
|
Anope::string nick = params.size() > 2 ? params[2] : "";
|
|
Anope::string user = params.size() > 3 ? params[3] : "";
|
|
Anope::string host = params.size() > 4 ? params[4] : "";
|
|
Anope::string real = params.size() > 5 ? params[5] : "";
|
|
BotInfo *bi;
|
|
|
|
if (oldnick.empty() || nick.empty())
|
|
{
|
|
this->OnSyntaxError(u, "CHANGE");
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!(bi = findbot(oldnick)))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_DOES_NOT_EXIST, oldnick.c_str());
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!oldnick.equals_ci(nick) && nickIsServices(oldnick, false))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_DOES_NOT_EXIST, oldnick.c_str());
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (nick.length() > Config.NickLen)
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_NICK);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!user.empty() && user.length() > Config.UserLen)
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_LONG_IDENT, Config.UserLen);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!host.empty() && host.length() > Config.HostLen)
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_LONG_HOST, Config.HostLen);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!oldnick.equals_ci(nick) && nickIsServices(nick, false))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_DOES_NOT_EXIST, oldnick.c_str());
|
|
return MOD_CONT;
|
|
}
|
|
|
|
/* Checks whether there *are* changes.
|
|
* Case sensitive because we may want to change just the case.
|
|
* And we must finally check that the nick is not already
|
|
* taken by another bot.
|
|
*/
|
|
if (nick.equals_cs(bi->nick) && (!user.empty() ? user.equals_cs(bi->GetIdent()) : 1) && (!host.empty() ? host.equals_cs(bi->host) : 1) && (!real.empty() ? real.equals_cs(bi->realname) : 1))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BOT_ANY_CHANGES);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
/* Check the nick is valid re RFC 2812 */
|
|
if (isdigit(nick[0]) || nick[0] == '-')
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_NICK);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
for (unsigned i = 0, end = nick.length(); i < end && i < Config.NickLen; ++i)
|
|
if (!isvalidnick(nick[i]))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_NICK);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
/* check for hardcored ircd forbidden nicks */
|
|
if (!ircdproto->IsNickValid(nick))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_NICK);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!host.empty() && !isValidHost(host, 3))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_HOST);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!user.empty())
|
|
for (unsigned i = 0, end = user.length(); i < end && i < Config.UserLen; ++i)
|
|
if (!isalnum(user[i]))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BAD_IDENT, Config.UserLen);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!nick.equals_ci(bi->nick) && findbot(nick))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BOT_ALREADY_EXISTS, nick.c_str());
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!nick.equals_ci(bi->nick))
|
|
{
|
|
/* We check whether the nick is registered, and inform the user
|
|
* if so. You need to drop the nick manually before you can use
|
|
* it as a bot nick from now on -GD
|
|
*/
|
|
if (findnick(nick))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, NICK_ALREADY_REGISTERED, nick.c_str());
|
|
return MOD_CONT;
|
|
}
|
|
|
|
/* The new nick is really different, so we remove the Q line for
|
|
the old nick. */
|
|
if (ircd->sqline)
|
|
{
|
|
XLine x(bi->nick);
|
|
ircdproto->SendSQLineDel(&x);
|
|
}
|
|
|
|
/* We check whether user with this nick is online, and kill it if so */
|
|
EnforceQlinedNick(nick, Config.s_BotServ);
|
|
}
|
|
|
|
if (!user.empty())
|
|
ircdproto->SendQuit(bi, "Quit: Be right back");
|
|
else
|
|
{
|
|
ircdproto->SendChangeBotNick(bi, nick);
|
|
XLine x(bi->nick, "Reserved for services");
|
|
ircdproto->SendSQLine(&x);
|
|
}
|
|
|
|
if (!nick.equals_cs(bi->nick))
|
|
bi->SetNewNick(nick);
|
|
|
|
if (!user.empty() && !user.equals_cs(bi->GetIdent()))
|
|
bi->SetIdent(user);
|
|
if (!host.empty() && !host.equals_cs(bi->host))
|
|
bi->host = host;
|
|
if (!real.empty() && !real.equals_cs(bi->realname))
|
|
bi->realname = real;
|
|
|
|
if (!user.empty())
|
|
{
|
|
ircdproto->SendClientIntroduction(bi->nick, bi->GetIdent(), bi->host, bi->realname, ircd->pseudoclient_mode, bi->GetUID());
|
|
XLine x(bi->nick, "Reserved for services");
|
|
ircdproto->SendSQLine(&x);
|
|
bi->RejoinAll();
|
|
}
|
|
|
|
notice_lang(Config.s_BotServ, u, BOT_BOT_CHANGED, oldnick.c_str(), bi->nick.c_str(), bi->GetIdent().c_str(), bi->host.c_str(), bi->realname.c_str());
|
|
|
|
FOREACH_MOD(I_OnBotChange, OnBotChange(bi));
|
|
return MOD_CONT;
|
|
}
|
|
|
|
CommandReturn DoDel(User *u, const std::vector<Anope::string> ¶ms)
|
|
{
|
|
Anope::string nick = params[1];
|
|
BotInfo *bi;
|
|
|
|
if (nick.empty())
|
|
{
|
|
this->OnSyntaxError(u, "DEL");
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (!(bi = findbot(nick)))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_DOES_NOT_EXIST, nick.c_str());
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (nickIsServices(nick, false))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_DOES_NOT_EXIST, nick.c_str());
|
|
return MOD_CONT;
|
|
}
|
|
|
|
FOREACH_MOD(I_OnBotDelete, OnBotDelete(bi));
|
|
|
|
ircdproto->SendQuit(bi, "Quit: Help! I'm being deleted by %s!", u->nick.c_str());
|
|
XLine x(bi->nick);
|
|
ircdproto->SendSQLineDel(&x);
|
|
|
|
delete bi;
|
|
notice_lang(Config.s_BotServ, u, BOT_BOT_DELETED, nick.c_str());
|
|
return MOD_CONT;
|
|
}
|
|
public:
|
|
CommandBSBot() : Command("BOT", 1, 6)
|
|
{
|
|
this->SetFlag(CFLAG_STRIP_CHANNEL);
|
|
}
|
|
|
|
CommandReturn Execute(User *u, const std::vector<Anope::string> ¶ms)
|
|
{
|
|
Anope::string cmd = params[0];
|
|
|
|
if (readonly)
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_BOT_READONLY);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (cmd.equals_ci("ADD"))
|
|
{
|
|
// ADD nick user host real - 5
|
|
if (!u->Account()->HasCommand("botserv/bot/add"))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, ACCESS_DENIED);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (params.size() < 5)
|
|
{
|
|
this->OnSyntaxError(u, "ADD");
|
|
return MOD_CONT;
|
|
}
|
|
|
|
std::vector<Anope::string> tempparams = params;
|
|
// ADD takes less params than CHANGE, so we need to take 6 if given and append it with a space to 5.
|
|
if (tempparams.size() >= 6)
|
|
tempparams[4] = tempparams[4] + " " + tempparams[5];
|
|
|
|
return this->DoAdd(u, tempparams);
|
|
}
|
|
else if (cmd.equals_ci("CHANGE"))
|
|
{
|
|
// CHANGE oldn newn user host real - 6
|
|
// but only oldn and newn are required
|
|
if (!u->Account()->HasCommand("botserv/bot/change"))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, ACCESS_DENIED);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (params.size() < 3)
|
|
{
|
|
this->OnSyntaxError(u, "CHANGE");
|
|
return MOD_CONT;
|
|
}
|
|
|
|
return this->DoChange(u, params);
|
|
}
|
|
else if (cmd.equals_ci("DEL"))
|
|
{
|
|
// DEL nick
|
|
if (!u->Account()->HasCommand("botserv/bot/del"))
|
|
{
|
|
notice_lang(Config.s_BotServ, u, ACCESS_DENIED);
|
|
return MOD_CONT;
|
|
}
|
|
|
|
if (params.size() < 1)
|
|
{
|
|
this->OnSyntaxError(u, "DEL");
|
|
return MOD_CONT;
|
|
}
|
|
|
|
return this->DoDel(u, params);
|
|
}
|
|
else
|
|
this->OnSyntaxError(u, "");
|
|
|
|
return MOD_CONT;
|
|
}
|
|
|
|
bool OnHelp(User *u, const Anope::string &subcommand)
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_SERVADMIN_HELP_BOT);
|
|
return true;
|
|
}
|
|
|
|
void OnSyntaxError(User *u, const Anope::string &subcommand)
|
|
{
|
|
syntax_error(Config.s_BotServ, u, "BOT", BOT_BOT_SYNTAX);
|
|
}
|
|
|
|
void OnServHelp(User *u)
|
|
{
|
|
notice_lang(Config.s_BotServ, u, BOT_HELP_CMD_BOT);
|
|
}
|
|
};
|
|
|
|
class BSBot : public Module
|
|
{
|
|
CommandBSBot commandbsbot;
|
|
|
|
public:
|
|
BSBot(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator)
|
|
{
|
|
this->SetAuthor("Anope");
|
|
this->SetType(CORE);
|
|
|
|
this->AddCommand(BotServ, &commandbsbot);
|
|
}
|
|
};
|
|
|
|
MODULE_INIT(BSBot)
|