mirror of
https://github.com/anope/anope.git
synced 2026-06-25 11:56:37 +02:00
206 lines
5.1 KiB
C++
206 lines
5.1 KiB
C++
/*
|
|
* Copyright (C) 2008-2010 Robin Burchell <w00t@inspircd.org>
|
|
* Copyright (C) 2008-2010 Anope Team <team@anope.org>
|
|
*
|
|
* Please read COPYING and README for further details.
|
|
*/
|
|
|
|
#include "services.h"
|
|
#include "modules.h"
|
|
#include "commands.h"
|
|
|
|
botinfo_map BotListByNick;
|
|
botinfo_uid_map BotListByUID;
|
|
|
|
BotInfo *BotServ = NULL;
|
|
BotInfo *ChanServ = NULL;
|
|
BotInfo *Global = NULL;
|
|
BotInfo *HostServ = NULL;
|
|
BotInfo *MemoServ = NULL;
|
|
BotInfo *NickServ = NULL;
|
|
BotInfo *OperServ = NULL;
|
|
|
|
BotInfo::BotInfo(const Anope::string &nnick, const Anope::string &nuser, const Anope::string &nhost, const Anope::string &nreal) : User(nnick, nuser, nhost, ts6_uid_retrieve())
|
|
{
|
|
this->realname = nreal;
|
|
this->server = Me;
|
|
|
|
this->chancount = 0;
|
|
this->lastmsg = this->created = Anope::CurTime;
|
|
|
|
this->SetFlag(BI_CORE);
|
|
if (!Config->s_ChanServ.empty() && nnick.equals_ci(Config->s_ChanServ))
|
|
ChanServ = this;
|
|
else if (!Config->s_BotServ.empty() && nnick.equals_ci(Config->s_BotServ))
|
|
BotServ = this;
|
|
else if (!Config->s_HostServ.empty() && nnick.equals_ci(Config->s_HostServ))
|
|
HostServ = this;
|
|
else if (!Config->s_OperServ.empty() && nnick.equals_ci(Config->s_OperServ))
|
|
OperServ = this;
|
|
else if (!Config->s_MemoServ.empty() && nnick.equals_ci(Config->s_MemoServ))
|
|
MemoServ = this;
|
|
else if (!Config->s_NickServ.empty() && nnick.equals_ci(Config->s_NickServ))
|
|
NickServ = this;
|
|
else if (!Config->s_GlobalNoticer.empty() && nnick.equals_ci(Config->s_GlobalNoticer))
|
|
Global = this;
|
|
else
|
|
this->UnsetFlag(BI_CORE);
|
|
|
|
BotListByNick[this->nick] = this;
|
|
if (!this->uid.empty())
|
|
BotListByUID[this->uid] = this;
|
|
|
|
// If we're synchronised with the uplink already, send the bot.
|
|
if (Me && Me->IsSynced())
|
|
{
|
|
ircdproto->SendClientIntroduction(this, ircd->pseudoclient_mode);
|
|
XLine x(this->nick, "Reserved for services");
|
|
ircdproto->SendSQLine(&x);
|
|
}
|
|
|
|
this->SetModeInternal(ModeManager::FindUserModeByName(UMODE_PROTECTED));
|
|
this->SetModeInternal(ModeManager::FindUserModeByName(UMODE_GOD));
|
|
}
|
|
|
|
BotInfo::~BotInfo()
|
|
{
|
|
for (registered_channel_map::const_iterator it = RegisteredChannelList.begin(), it_end = RegisteredChannelList.end(); it != it_end; ++it)
|
|
{
|
|
ChannelInfo *ci = it->second;
|
|
|
|
if (ci->bi == this)
|
|
ci->bi = NULL;
|
|
}
|
|
|
|
BotListByNick.erase(this->nick);
|
|
if (!this->uid.empty())
|
|
BotListByUID.erase(this->uid);
|
|
}
|
|
|
|
|
|
void BotInfo::SetNewNick(const Anope::string &newnick)
|
|
{
|
|
UserListByNick.erase(this->nick);
|
|
BotListByNick.erase(this->nick);
|
|
|
|
this->nick = newnick;
|
|
|
|
UserListByNick[this->nick] = this;
|
|
BotListByNick[this->nick] = this;
|
|
}
|
|
|
|
void BotInfo::RejoinAll()
|
|
{
|
|
for (registered_channel_map::const_iterator it = RegisteredChannelList.begin(), it_end = RegisteredChannelList.end(); it != it_end; ++it)
|
|
{
|
|
ChannelInfo *ci = it->second;
|
|
|
|
if (ci->bi == this && ci->c && ci->c->users.size() >= Config->BSMinUsers)
|
|
this->Join(ci->c);
|
|
}
|
|
}
|
|
|
|
void BotInfo::Assign(User *u, ChannelInfo *ci)
|
|
{
|
|
EventReturn MOD_RESULT = EVENT_CONTINUE;
|
|
FOREACH_RESULT(I_OnBotAssign, OnBotAssign(u, ci, this));
|
|
if (MOD_RESULT == EVENT_STOP)
|
|
return;
|
|
|
|
if (ci->bi)
|
|
ci->bi->UnAssign(u, ci);
|
|
|
|
++this->chancount;
|
|
|
|
ci->bi = this;
|
|
if (ci->c && ci->c->users.size() >= Config->BSMinUsers)
|
|
this->Join(ci->c);
|
|
}
|
|
|
|
void BotInfo::UnAssign(User *u, ChannelInfo *ci)
|
|
{
|
|
EventReturn MOD_RESULT = EVENT_CONTINUE;
|
|
FOREACH_RESULT(I_OnBotUnAssign, OnBotUnAssign(u, ci));
|
|
if (MOD_RESULT == EVENT_STOP)
|
|
return;
|
|
|
|
if (ci->c && ci->c->FindUser(ci->bi))
|
|
{
|
|
if (u)
|
|
ci->bi->Part(ci->c, "UNASSIGN from " + u->nick);
|
|
else
|
|
ci->bi->Part(ci->c);
|
|
}
|
|
|
|
--this->chancount;
|
|
|
|
ci->bi = NULL;
|
|
}
|
|
|
|
void BotInfo::Join(Channel *c, bool update_ts)
|
|
{
|
|
if (Config->BSSmartJoin)
|
|
{
|
|
/* We check for bans */
|
|
if (c->bans && c->bans->count)
|
|
{
|
|
Entry *ban, *next;
|
|
|
|
for (ban = c->bans->entries; ban; ban = next)
|
|
{
|
|
next = ban->next;
|
|
|
|
if (entry_match(ban, this->nick, this->GetIdent(), this->host, 0))
|
|
c->RemoveMode(NULL, CMODE_BAN, ban->mask);
|
|
}
|
|
|
|
Anope::string Limit;
|
|
int limit = 0;
|
|
if (c->GetParam(CMODE_LIMIT, Limit) && Limit.is_number_only())
|
|
limit = convertTo<int>(Limit);
|
|
|
|
/* Should we be invited? */
|
|
if (c->HasMode(CMODE_INVITE) || (limit && c->users.size() >= limit))
|
|
ircdproto->SendNoticeChanops(this, c, "%s invited %s into the channel.", this->nick.c_str(), this->nick.c_str());
|
|
}
|
|
|
|
ModeManager::ProcessModes();
|
|
}
|
|
|
|
c->JoinUser(this);
|
|
ChannelContainer *cc = this->FindChannel(c);
|
|
for (int i = 0; i < Config->BotModeList.size(); ++i)
|
|
{
|
|
if (!update_ts)
|
|
{
|
|
c->SetMode(this, Config->BotModeList[i], this->nick, false);
|
|
}
|
|
else
|
|
{
|
|
cc->Status->SetFlag(Config->BotModeList[i]->Name);
|
|
c->SetModeInternal(Config->BotModeList[i], this->nick, false);
|
|
}
|
|
}
|
|
if (!update_ts)
|
|
ircdproto->SendJoin(this, c->name, c->creation_time);
|
|
else if (Me && Me->IsSynced())
|
|
{
|
|
ircdproto->SendJoin(this, cc);
|
|
|
|
c->Reset();
|
|
}
|
|
FOREACH_MOD(I_OnBotJoin, OnBotJoin(c->ci, this));
|
|
}
|
|
|
|
void BotInfo::Join(const Anope::string &chname, bool update_ts)
|
|
{
|
|
Channel *c = findchan(chname);
|
|
return this->Join(c ? c : new Channel(chname), update_ts);
|
|
}
|
|
|
|
void BotInfo::Part(Channel *c, const Anope::string &reason)
|
|
{
|
|
ircdproto->SendPart(this, c, "%s", !reason.empty() ? reason.c_str() : "");
|
|
c->DeleteUser(this);
|
|
}
|