mirror of
https://github.com/anope/anope.git
synced 2026-06-25 18:36:37 +02:00
987f371c0b
git-svn-id: http://anope.svn.sourceforge.net/svnroot/anope/trunk@2599 5417fbe8-f217-4b02-8779-1006273d7864
2480 lines
56 KiB
C
2480 lines
56 KiB
C
/* Channel-handling routines.
|
|
*
|
|
* (C) 2003-2009 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.
|
|
*
|
|
* $Id$
|
|
*
|
|
*/
|
|
|
|
#include "services.h"
|
|
#include "language.h"
|
|
#include "modules.h"
|
|
|
|
Channel *chanlist[1024];
|
|
|
|
#define HASH(chan) ((chan)[1] ? ((chan)[1]&31)<<5 | ((chan)[2]&31) : 0)
|
|
|
|
/*************************************************************************/
|
|
|
|
/**
|
|
* See if a channel has a mode
|
|
* @param Name The mode name
|
|
* @return true or false
|
|
*/
|
|
bool Channel::HasMode(ChannelModeName Name)
|
|
{
|
|
return modes[(size_t)Name];
|
|
}
|
|
|
|
/**
|
|
* Set a mode on a channel
|
|
* @param Name The mode name
|
|
*/
|
|
void Channel::SetMode(ChannelModeName Name)
|
|
{
|
|
modes[(size_t)Name] = true;
|
|
FOREACH_MOD(I_OnChannelModeSet, OnChannelModeSet(this, Name));
|
|
}
|
|
|
|
/**
|
|
* Set a mode on a channel
|
|
* @param Mode The mode
|
|
*/
|
|
void Channel::SetMode(char Mode)
|
|
{
|
|
ChannelMode *cm;
|
|
|
|
if ((cm = ModeManager::FindChannelModeByChar(Mode)))
|
|
{
|
|
SetMode(cm->Name);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Remove a mode from a channel
|
|
* @param Name The mode name
|
|
*/
|
|
void Channel::RemoveMode(ChannelModeName Name)
|
|
{
|
|
modes[(size_t)Name] = false;
|
|
FOREACH_MOD(I_OnChannelModeUnset, OnChannelModeUnset(this, Name));
|
|
}
|
|
|
|
/**
|
|
* Remove a mode from a channel
|
|
* @param Mode The mode
|
|
*/
|
|
void Channel::RemoveMode(char Mode)
|
|
{
|
|
ChannelMode *cm;
|
|
|
|
if ((cm = ModeManager::FindChannelModeByChar(Mode)))
|
|
{
|
|
RemoveMode(cm->Name);
|
|
}
|
|
}
|
|
|
|
/** Set a channel mode param on the channel
|
|
* @param Name The mode
|
|
* @param param The param
|
|
* @param true on success
|
|
*/
|
|
bool Channel::SetParam(ChannelModeName Name, std::string &Value)
|
|
{
|
|
return Params.insert(std::make_pair(Name, Value)).second;
|
|
}
|
|
|
|
/** Unset a param from the channel
|
|
* @param Name The mode
|
|
*/
|
|
void Channel::UnsetParam(ChannelModeName Name)
|
|
{
|
|
std::map<ChannelModeName, std::string>::iterator it = Params.find(Name);
|
|
|
|
if (it != Params.end())
|
|
{
|
|
Params.erase(it);
|
|
}
|
|
}
|
|
|
|
/** Get a param from the channel
|
|
* @param Name The mode
|
|
* @param Target a string to put the param into
|
|
* @return true on success
|
|
*/
|
|
const bool Channel::GetParam(ChannelModeName Name, std::string *Target)
|
|
{
|
|
std::map<ChannelModeName, std::string>::iterator it = Params.find(Name);
|
|
|
|
(*Target).clear();
|
|
|
|
if (it != Params.end())
|
|
{
|
|
*Target = it->second;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/** Check if a mode is set and has a param
|
|
* @param Name The mode
|
|
*/
|
|
const bool Channel::HasParam(ChannelModeName Name)
|
|
{
|
|
std::map<ChannelModeName, std::string>::iterator it = Params.find(Name);
|
|
|
|
if (it != Params.end())
|
|
{
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/** Clear all the modes from the channel
|
|
* @param client The client unsetting the modes
|
|
*/
|
|
void Channel::ClearModes(char *client)
|
|
{
|
|
ChannelMode *cm;
|
|
ChannelModeParam *cmp;
|
|
std::string buf, buf2;
|
|
const char *argv[2];
|
|
BotInfo *sender = (client ? findbot(client) : whosends(ci));
|
|
|
|
for (size_t n = CMODE_BEGIN + 1; n != CMODE_END; ++n)
|
|
{
|
|
cm = ModeManager::FindChannelModeByName((ChannelModeName)n);
|
|
|
|
if (cm && this->HasMode(cm->Name))
|
|
{
|
|
if (cm->Type == MODE_REGULAR)
|
|
{
|
|
buf = '-';
|
|
buf += cm->ModeChar;
|
|
|
|
argv[0] = buf.c_str();
|
|
|
|
ircdproto->SendMode(sender, this->name, "-%c", cm->ModeChar);
|
|
chan_set_modes(sender->nick, this, 1, argv, 0);
|
|
}
|
|
else if (cm->Type == MODE_PARAM)
|
|
{
|
|
cmp = static_cast<ChannelModeParam *>(cm);
|
|
|
|
buf = '-';
|
|
buf += cmp->ModeChar;
|
|
buf2.clear();
|
|
|
|
if (!cmp->MinusNoArg)
|
|
{
|
|
buf2 = this->GetParam(cmp->Name, &buf2);
|
|
}
|
|
|
|
argv[0] = buf.c_str();
|
|
if (!buf2.empty())
|
|
{
|
|
argv[1] = buf2.c_str();
|
|
|
|
ircdproto->SendMode(sender, this->name, "-%c %s", cm->ModeChar, buf2.c_str());
|
|
chan_set_modes(sender->nick, this, 2, argv, 0);
|
|
}
|
|
else
|
|
{
|
|
argv[1] = NULL;
|
|
|
|
ircdproto->SendMode(sender, this->name, "-%c", cm->ModeChar);
|
|
chan_set_modes(sender->nick, this, 1, argv, 0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
modes.reset();
|
|
}
|
|
|
|
/** Clear all the bans from the channel
|
|
* @param client The client unsetting the modes
|
|
*/
|
|
void Channel::ClearBans(char *client)
|
|
{
|
|
Entry *entry, *nexte;
|
|
BotInfo *sender = (client ? findbot(client) : whosends(ci));
|
|
ChannelModeList *cml;
|
|
std::string buf;
|
|
const char *argv[2];
|
|
|
|
cml = static_cast<ChannelModeList *>(ModeManager::FindChannelModeByName(CMODE_BAN));
|
|
|
|
if (cml && this->bans && this->bans->count)
|
|
{
|
|
buf = '-';
|
|
buf += cml->ModeChar;
|
|
|
|
for (entry = this->bans->entries; entry; entry = nexte)
|
|
{
|
|
nexte = entry->next;
|
|
|
|
argv[0] = buf.c_str();
|
|
argv[1] = sstrdup(entry->mask);
|
|
ircdproto->SendMode(sender, this->name, "%s %s", buf.c_str(), entry->mask);
|
|
chan_set_modes(sender->nick, this, 2, argv, 0);
|
|
delete [] argv[1];
|
|
}
|
|
}
|
|
}
|
|
|
|
/** Clear all the excepts from the channel
|
|
* @param client The client unsetting the modes
|
|
*/
|
|
void Channel::ClearExcepts(char *client)
|
|
{
|
|
Entry *entry, *nexte;
|
|
BotInfo *sender = (client ? findbot(client) : whosends(ci));
|
|
ChannelModeList *cml;
|
|
std::string buf;
|
|
const char *argv[2];
|
|
|
|
cml = static_cast<ChannelModeList *>(ModeManager::FindChannelModeByName(CMODE_EXCEPT));
|
|
|
|
if (cml && this->excepts && this->excepts->count)
|
|
{
|
|
buf = '+';
|
|
buf += cml->ModeChar;
|
|
|
|
for (entry = this->excepts->entries; entry; entry = nexte)
|
|
{
|
|
nexte = entry->next;
|
|
|
|
argv[0] = buf.c_str();
|
|
argv[1] = sstrdup(entry->mask);
|
|
ircdproto->SendMode(sender, this->name, "%s %s", buf.c_str(), entry->mask);
|
|
chan_set_modes(sender->nick, this, 2, argv, 0);
|
|
delete [] argv[1];
|
|
}
|
|
}
|
|
}
|
|
|
|
/** Clear all the invites from the channel
|
|
* @param client The client unsetting the modes
|
|
*/
|
|
void Channel::ClearInvites(char *client)
|
|
{
|
|
Entry *entry, *nexte;
|
|
BotInfo *sender = (client ? findbot(client) : whosends(ci));
|
|
ChannelModeList *cml;
|
|
std::string buf;
|
|
const char *argv[2];
|
|
|
|
cml = static_cast<ChannelModeList *>(ModeManager::FindChannelModeByName(CMODE_INVITEOVERRIDE));
|
|
|
|
if (cml && this->invites && this->invites->count)
|
|
{
|
|
buf = '-';
|
|
buf += cml->ModeChar;
|
|
|
|
for (entry = this->invites->entries; entry; entry = nexte)
|
|
{
|
|
nexte = entry->next;
|
|
|
|
argv[0] = buf.c_str();
|
|
argv[1] = sstrdup(entry->mask);
|
|
ircdproto->SendMode(sender, this->name, "%s %s", buf.c_str(), entry->mask);
|
|
chan_set_modes(sender->nick, this, 2, argv, 0);
|
|
delete [] argv[1];
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
void chan_deluser(User * user, Channel * c)
|
|
{
|
|
struct c_userlist *u;
|
|
|
|
if (c->ci)
|
|
update_cs_lastseen(user, c->ci);
|
|
|
|
for (u = c->users; u && u->user != user; u = u->next);
|
|
if (!u)
|
|
return;
|
|
|
|
if (u->ud) {
|
|
if (u->ud->lastline)
|
|
delete [] u->ud->lastline;
|
|
delete u->ud;
|
|
}
|
|
|
|
if (u->next)
|
|
u->next->prev = u->prev;
|
|
if (u->prev)
|
|
u->prev->next = u->next;
|
|
else
|
|
c->users = u->next;
|
|
delete u;
|
|
c->usercount--;
|
|
|
|
if (s_BotServ && c->ci && c->ci->bi && c->usercount == BSMinUsers - 1) {
|
|
ircdproto->SendPart(c->ci->bi, c->name, NULL);
|
|
}
|
|
|
|
if (!c->users)
|
|
chan_delete(c);
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Returns a fully featured binary modes string. If complete is 0, the
|
|
* eventual parameters won't be added to the string.
|
|
*/
|
|
|
|
char *chan_get_modes(Channel * chan, int complete, int plus)
|
|
{
|
|
static char res[BUFSIZE];
|
|
char params[BUFSIZE];
|
|
char *end = res, *value, *pend = params, *pend2 = params;
|
|
std::string param;
|
|
ChannelMode *cm;
|
|
ChannelModeParam *cmp;
|
|
std::map<char, ChannelMode *>::iterator it;
|
|
|
|
if (!chan->modes.count())
|
|
{
|
|
for (it = ModeManager::ChannelModesByChar.begin(); it != ModeManager::ChannelModesByChar.end(); ++it)
|
|
{
|
|
cm = it->second;
|
|
|
|
if (chan->HasMode(cm->Name))
|
|
{
|
|
*end++ = it->first;
|
|
|
|
if (complete)
|
|
{
|
|
if (cm->Type == MODE_PARAM)
|
|
{
|
|
cmp = static_cast<ChannelModeParam *>(cm);
|
|
|
|
if (plus || !cmp->MinusNoArg)
|
|
{
|
|
chan->GetParam(cmp->Name, ¶m);
|
|
|
|
if (!param.empty())
|
|
{
|
|
value = const_cast<char *>(param.c_str());
|
|
|
|
*pend++ = ' ';
|
|
|
|
while (*value)
|
|
*pend++ = *value++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
while (*pend2)
|
|
*end++ = *pend2++;
|
|
}
|
|
|
|
*end = 0;
|
|
|
|
return res;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Retrieves the status of an user on a channel */
|
|
|
|
int chan_get_user_status(Channel * chan, User * user)
|
|
{
|
|
struct u_chanlist *uc;
|
|
|
|
for (uc = user->chans; uc; uc = uc->next)
|
|
if (uc->chan == chan)
|
|
return uc->status;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Has the given user the given status on the given channel? :p */
|
|
|
|
int chan_has_user_status(Channel * chan, User * user, int16 status)
|
|
{
|
|
struct u_chanlist *uc;
|
|
|
|
for (uc = user->chans; uc; uc = uc->next) {
|
|
if (uc->chan == chan) {
|
|
if (debug) {
|
|
alog("debug: chan_has_user_status wanted %d the user is %d", status, uc->status);
|
|
}
|
|
return (uc->status & status);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Remove the status of an user on a channel */
|
|
|
|
void chan_remove_user_status(Channel * chan, User * user, int16 status)
|
|
{
|
|
struct u_chanlist *uc;
|
|
|
|
if (debug >= 2)
|
|
alog("debug: removing user status (%d) from %s for %s", status,
|
|
user->nick, chan->name);
|
|
|
|
for (uc = user->chans; uc; uc = uc->next) {
|
|
if (uc->chan == chan) {
|
|
uc->status &= ~status;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
void chan_set_modes(const char *source, Channel *chan, int ac, const char **av, int check)
|
|
{
|
|
int add = 1, i, real_ac = ac;
|
|
ChannelMode *cm;
|
|
ChannelModeStatus *cms;
|
|
ChannelModeList *cml;
|
|
ChannelModeParam *cmp;
|
|
const char *modes, **real_av = av;
|
|
char mode;
|
|
BotInfo *bi;
|
|
unsigned char botmode = 0;
|
|
User *user;
|
|
|
|
modes = av[0];
|
|
|
|
if (debug)
|
|
alog("debug: Changing modes for %s to %s", chan->name, merge_args(ac, av));
|
|
|
|
ac--;
|
|
|
|
while ((mode = *modes++))
|
|
{
|
|
switch (mode)
|
|
{
|
|
case '+':
|
|
add = 1;
|
|
continue;
|
|
case '-':
|
|
add = 0;
|
|
continue;
|
|
}
|
|
|
|
if ((cm = ModeManager::FindChannelModeByChar(mode)))
|
|
{
|
|
/* v/h/o/a/q etc */
|
|
if (cm->Type == MODE_STATUS)
|
|
{
|
|
cms = static_cast<ChannelModeStatus *>(cm);
|
|
|
|
if (ac == 0)
|
|
{
|
|
alog("channel: mode %c%c with no parameter (?) for channel %s", add ? '+' : '-', mode, chan->name);
|
|
continue;
|
|
}
|
|
|
|
ac--;
|
|
av++;
|
|
|
|
if (cms->ProtectBotServ && !add)
|
|
{
|
|
if ((bi = findbot(*av)))
|
|
{
|
|
if (!botmode || botmode != mode)
|
|
{
|
|
ircdproto->SendMode(bi, chan->name, "+%c %s", mode, bi->nick);
|
|
botmode = mode;
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
botmode = mode;
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
else if ((bi = findbot(*av)))
|
|
continue;
|
|
|
|
if (!(user = finduser(*av)) && !(ircd->ts6 && (user = find_byuid(*av))))
|
|
{
|
|
if (debug)
|
|
{
|
|
alog("debug: MODE %s %c%c for nonexistent user %s",
|
|
chan->name, (add ? '+' : '-'), mode, *av);
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (debug)
|
|
alog("debug: Setting %c%c on %s for %s", (add ? '+' : '-'),
|
|
mode, chan->name, user->nick);
|
|
|
|
if (add)
|
|
chan_set_user_status(chan, user, cms->Status);
|
|
/* If this does +o, remove any DEOPPED flag */
|
|
else
|
|
chan_remove_user_status(chan, user, cms->Status);
|
|
}
|
|
/* b/e/I etc */
|
|
else if (cm->Type == MODE_LIST)
|
|
{
|
|
cml = static_cast<ChannelModeList *>(cm);
|
|
|
|
if (ac == 0)
|
|
{
|
|
alog("channel: mode %c%c with no parameter (?) for channel %s", add ? '+' : '-', mode, chan->name);
|
|
continue;
|
|
}
|
|
|
|
ac--;
|
|
av++;
|
|
|
|
if (add)
|
|
cml->AddMask(chan, *av);
|
|
else
|
|
cml->DelMask(chan, *av);
|
|
}
|
|
else
|
|
{
|
|
if (check >= 0)
|
|
{
|
|
if (add)
|
|
chan->SetMode(mode);
|
|
else
|
|
chan->RemoveMode(mode);
|
|
}
|
|
|
|
if (cm->Type == MODE_PARAM)
|
|
{
|
|
cmp = static_cast<ChannelModeParam *>(cm);
|
|
|
|
if (add || !cmp->MinusNoArg)
|
|
{
|
|
if (ac == 0)
|
|
{
|
|
alog("channel: mode %c%c with no parameter (?) for channel %s", add ? '+' : '-', mode, chan->name);
|
|
continue;
|
|
}
|
|
|
|
ac--;
|
|
av++;
|
|
}
|
|
|
|
if (*av && !cmp->IsValid(*av))
|
|
continue;
|
|
if (add)
|
|
{
|
|
std::string Param = *av;
|
|
chan->SetParam(cmp->Name, Param);
|
|
}
|
|
else
|
|
chan->UnsetParam(cmp->Name);
|
|
}
|
|
|
|
if (check < 0)
|
|
{
|
|
if (add)
|
|
chan->SetMode(mode);
|
|
else
|
|
chan->RemoveMode(mode);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Don't bounce modes from u:lined clients or servers, bug #1004
|
|
user = finduser(source);
|
|
if ((user && is_ulined(user->server->name)) || is_ulined(source))
|
|
return;
|
|
|
|
if (check > 0)
|
|
{
|
|
check_modes(chan);
|
|
|
|
if ((check < 2) || (check == 3))
|
|
{
|
|
/* Walk through all users we've set modes for and see if they are
|
|
* valid. Invalid modes (like +o with SECUREOPS on) will be removed
|
|
*/
|
|
real_ac--;
|
|
real_av++;
|
|
|
|
for (i = 0; i < real_ac; i++)
|
|
{
|
|
if ((user = finduser(*real_av)) && is_on_chan(chan, user))
|
|
{
|
|
if (check < 2)
|
|
chan_set_correct_modes(user, chan, 0);
|
|
else if ((chan->ci->flags) && (chan->ci->flags & CI_SECUREOPS))
|
|
{
|
|
/* Fixing bug #1006 oringinally caused by fixing #922
|
|
* we must check for secureops here, not in chan_set_correct_modes
|
|
* because chan_set_corret_modes will also check for usercount == 1
|
|
* where it will deop the user, this way we know the channel was not
|
|
* just created. (check == 3 from /cs (half)op) - Adam
|
|
*/
|
|
chan_set_correct_modes(user, chan, 0);
|
|
}
|
|
}
|
|
|
|
real_av++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Set the status of an user on a channel */
|
|
|
|
void chan_set_user_status(Channel * chan, User * user, int16 status)
|
|
{
|
|
struct u_chanlist *uc;
|
|
UserMode *um;
|
|
char av[3];
|
|
|
|
if (debug >= 2)
|
|
alog("debug: setting user status (%d) on %s for %s", status,
|
|
user->nick, chan->name);
|
|
|
|
if (HelpChannel && ((um = ModeManager::FindUserModeByName(UMODE_HELPOP))) && (status & CUS_OP)
|
|
&& (stricmp(chan->name, HelpChannel) == 0)
|
|
&& (!chan->ci || check_access(user, chan->ci, CA_AUTOOP))) {
|
|
if (debug) {
|
|
alog("debug: %s being given helpop for having %d status in %s",
|
|
user->nick, status, chan->name);
|
|
}
|
|
|
|
snprintf(av, sizeof(av), "+%c", um->ModeChar);
|
|
|
|
common_svsmode(user, av, NULL);
|
|
}
|
|
|
|
for (uc = user->chans; uc; uc = uc->next) {
|
|
if (uc->chan == chan) {
|
|
uc->status |= status;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Return the Channel structure corresponding to the named channel, or NULL
|
|
* if the channel was not found. chan is assumed to be non-NULL and valid
|
|
* (i.e. pointing to a channel name of 2 or more characters). */
|
|
|
|
Channel *findchan(const char *chan)
|
|
{
|
|
Channel *c;
|
|
|
|
if (!chan || !*chan) {
|
|
if (debug) {
|
|
alog("debug: findchan() called with NULL values");
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
if (debug >= 3)
|
|
alog("debug: findchan(%p)", chan);
|
|
c = chanlist[HASH(chan)];
|
|
while (c) {
|
|
if (stricmp(c->name, chan) == 0) {
|
|
if (debug >= 3)
|
|
alog("debug: findchan(%s) -> %p", chan, static_cast<void *>(c));
|
|
return c;
|
|
}
|
|
c = c->next;
|
|
}
|
|
if (debug >= 3)
|
|
alog("debug: findchan(%s) -> %p", chan, static_cast<void *>(c));
|
|
return NULL;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Iterate over all channels in the channel list. Return NULL at end of
|
|
* list.
|
|
*/
|
|
|
|
static Channel *current;
|
|
static int next_index;
|
|
|
|
Channel *firstchan()
|
|
{
|
|
next_index = 0;
|
|
while (next_index < 1024 && current == NULL)
|
|
current = chanlist[next_index++];
|
|
if (debug >= 3)
|
|
alog("debug: firstchan() returning %s",
|
|
current ? current->name : "NULL (end of list)");
|
|
return current;
|
|
}
|
|
|
|
Channel *nextchan()
|
|
{
|
|
if (current)
|
|
current = current->next;
|
|
if (!current && next_index < 1024) {
|
|
while (next_index < 1024 && current == NULL)
|
|
current = chanlist[next_index++];
|
|
}
|
|
if (debug >= 3)
|
|
alog("debug: nextchan() returning %s",
|
|
current ? current->name : "NULL (end of list)");
|
|
return current;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Return statistics. Pointers are assumed to be valid. */
|
|
|
|
void get_channel_stats(long *nrec, long *memuse)
|
|
{
|
|
long count = 0, mem = 0;
|
|
Channel *chan;
|
|
struct c_userlist *cu;
|
|
BanData *bd;
|
|
int i;
|
|
std::string buf;
|
|
|
|
for (i = 0; i < 1024; i++) {
|
|
for (chan = chanlist[i]; chan; chan = chan->next) {
|
|
count++;
|
|
mem += sizeof(*chan);
|
|
if (chan->topic)
|
|
mem += strlen(chan->topic) + 1;
|
|
if (chan->GetParam(CMODE_KEY, &buf))
|
|
mem += buf.length() + 1;
|
|
if (chan->GetParam(CMODE_FLOOD, &buf))
|
|
mem += buf.length() + 1;
|
|
if (chan->GetParam(CMODE_REDIRECT, &buf))
|
|
mem += buf.length() + 1;
|
|
mem += get_memuse(chan->bans);
|
|
if (ModeManager::FindChannelModeByName(CMODE_EXCEPT))
|
|
mem += get_memuse(chan->excepts);
|
|
if (ModeManager::FindChannelModeByName(CMODE_INVITEOVERRIDE))
|
|
mem += get_memuse(chan->invites);
|
|
for (cu = chan->users; cu; cu = cu->next) {
|
|
mem += sizeof(*cu);
|
|
if (cu->ud) {
|
|
mem += sizeof(*cu->ud);
|
|
if (cu->ud->lastline)
|
|
mem += strlen(cu->ud->lastline) + 1;
|
|
}
|
|
}
|
|
for (bd = chan->bd; bd; bd = bd->next) {
|
|
if (bd->mask)
|
|
mem += strlen(bd->mask) + 1;
|
|
mem += sizeof(*bd);
|
|
}
|
|
}
|
|
}
|
|
*nrec = count;
|
|
*memuse = mem;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Is the given nick on the given channel? */
|
|
|
|
int is_on_chan(Channel * c, User * u)
|
|
{
|
|
struct u_chanlist *uc;
|
|
|
|
for (uc = u->chans; uc; uc = uc->next)
|
|
if (uc->chan == c)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Is the given nick on the given channel?
|
|
This function supports links. */
|
|
|
|
User *nc_on_chan(Channel * c, NickCore * nc)
|
|
{
|
|
struct c_userlist *u;
|
|
|
|
if (!c || !nc)
|
|
return NULL;
|
|
|
|
for (u = c->users; u; u = u->next) {
|
|
if (u->user->nc == nc)
|
|
return u->user;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
/*************************** Message Handling ****************************/
|
|
/*************************************************************************/
|
|
|
|
/* Handle a JOIN command.
|
|
* av[0] = channels to join
|
|
*/
|
|
|
|
void do_join(const char *source, int ac, const char **av)
|
|
{
|
|
User *user;
|
|
Channel *chan;
|
|
char *s, *t;
|
|
struct u_chanlist *c, *nextc;
|
|
char *channame;
|
|
time_t ts = time(NULL);
|
|
|
|
if (ircd->ts6) {
|
|
user = find_byuid(source);
|
|
if (!user)
|
|
user = finduser(source);
|
|
} else {
|
|
user = finduser(source);
|
|
}
|
|
if (!user) {
|
|
if (debug) {
|
|
alog("debug: JOIN from nonexistent user %s: %s", source,
|
|
merge_args(ac, av));
|
|
}
|
|
return;
|
|
}
|
|
|
|
t = const_cast<char *>(av[0]); // XXX Unsafe cast, this needs reviewing -- CyberBotX
|
|
while (*(s = t)) {
|
|
t = s + strcspn(s, ",");
|
|
if (*t)
|
|
*t++ = 0;
|
|
|
|
if (*s == '0') {
|
|
c = user->chans;
|
|
while (c) {
|
|
nextc = c->next;
|
|
channame = sstrdup(c->chan->name);
|
|
FOREACH_MOD(I_OnPrePartChannel, OnPrePartChannel(user, c->chan));
|
|
chan_deluser(user, c->chan);
|
|
FOREACH_MOD(I_OnPartChannel, OnPartChannel(user, findchan(channame), channame, ""));
|
|
delete [] channame;
|
|
delete c;
|
|
c = nextc;
|
|
}
|
|
user->chans = NULL;
|
|
continue;
|
|
}
|
|
|
|
chan = findchan(s);
|
|
|
|
/* how about not triggering the JOIN event on an actual /part :) -certus */
|
|
FOREACH_MOD(I_OnPreJoinChannel, OnPreJoinChannel(user, s));
|
|
|
|
/* Make sure check_kick comes before chan_adduser, so banned users
|
|
* don't get to see things like channel keys. */
|
|
/* If channel already exists, check_kick() will use correct TS.
|
|
* Otherwise, we lose. */
|
|
if (check_kick(user, s, ts))
|
|
continue;
|
|
|
|
if (ac == 2) {
|
|
ts = strtoul(av[1], NULL, 10);
|
|
if (debug) {
|
|
alog("debug: recieved a new TS for JOIN: %ld",
|
|
static_cast<long>(ts));
|
|
}
|
|
}
|
|
|
|
chan = join_user_update(user, chan, s, ts);
|
|
chan_set_correct_modes(user, chan, 1);
|
|
|
|
FOREACH_MOD(I_OnJoinChannel, OnJoinChannel(user, chan));
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Handle a KICK command.
|
|
* av[0] = channel
|
|
* av[1] = nick(s) being kicked
|
|
* av[2] = reason
|
|
*/
|
|
|
|
void do_kick(const char *source, int ac, const char **av)
|
|
{
|
|
BotInfo *bi;
|
|
ChannelInfo *ci;
|
|
User *user;
|
|
char *s, *t;
|
|
struct u_chanlist *c;
|
|
|
|
t = const_cast<char *>(av[1]); // XXX unsafe cast, this needs reviewing -- w00t
|
|
while (*(s = t)) {
|
|
t = s + strcspn(s, ",");
|
|
if (*t)
|
|
*t++ = 0;
|
|
|
|
/* If it is the bot that is being kicked, we make it rejoin the
|
|
* channel and stop immediately.
|
|
* --lara
|
|
*/
|
|
if (s_BotServ && (bi = findbot(s)) && (ci = cs_findchan(av[0]))) {
|
|
bot_join(ci);
|
|
continue;
|
|
}
|
|
|
|
if (ircd->ts6) {
|
|
user = find_byuid(s);
|
|
if (!user) {
|
|
user = finduser(s);
|
|
}
|
|
} else {
|
|
user = finduser(s);
|
|
}
|
|
if (!user) {
|
|
if (debug) {
|
|
alog("debug: KICK for nonexistent user %s on %s: %s", s,
|
|
av[0], merge_args(ac - 2, av + 2));
|
|
}
|
|
continue;
|
|
}
|
|
if (debug) {
|
|
alog("debug: kicking %s from %s", user->nick, av[0]);
|
|
}
|
|
for (c = user->chans; c && stricmp(av[0], c->chan->name) != 0;
|
|
c = c->next);
|
|
if (c)
|
|
{
|
|
FOREACH_MOD(I_OnUserKicked, OnUserKicked(c->chan, user, source, merge_args(ac - 2, av + 2)));
|
|
chan_deluser(user, c->chan);
|
|
if (c->next)
|
|
c->next->prev = c->prev;
|
|
if (c->prev)
|
|
c->prev->next = c->next;
|
|
else
|
|
user->chans = c->next;
|
|
delete c;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Handle a PART command.
|
|
* av[0] = channels to leave
|
|
* av[1] = reason (optional)
|
|
*/
|
|
|
|
void do_part(const char *source, int ac, const char **av)
|
|
{
|
|
User *user;
|
|
char *s, *t;
|
|
struct u_chanlist *c;
|
|
|
|
if (ircd->ts6) {
|
|
user = find_byuid(source);
|
|
if (!user)
|
|
user = finduser(source);
|
|
} else {
|
|
user = finduser(source);
|
|
}
|
|
if (!user) {
|
|
if (debug) {
|
|
alog("debug: PART from nonexistent user %s: %s", source,
|
|
merge_args(ac, av));
|
|
}
|
|
return;
|
|
}
|
|
t = const_cast<char *>(av[0]); // XXX Unsafe cast, this needs reviewing -- CyberBotX
|
|
while (*(s = t)) {
|
|
t = s + strcspn(s, ",");
|
|
if (*t)
|
|
*t++ = 0;
|
|
if (debug)
|
|
alog("debug: %s leaves %s", source, s);
|
|
for (c = user->chans; c && stricmp(s, c->chan->name) != 0;
|
|
c = c->next);
|
|
if (c) {
|
|
if (!c->chan) {
|
|
alog("user: BUG parting %s: channel entry found but c->chan NULL", s);
|
|
return;
|
|
}
|
|
FOREACH_MOD(I_OnPrePartChannel, OnPrePartChannel(user, c->chan));
|
|
std::string ChannelName = c->chan->name;
|
|
|
|
chan_deluser(user, c->chan);
|
|
|
|
FOREACH_MOD(I_OnPartChannel, OnPartChannel(user, findchan(ChannelName.c_str()), ChannelName, av[1] ? av[1] : ""));
|
|
|
|
if (c->next)
|
|
c->next->prev = c->prev;
|
|
if (c->prev)
|
|
c->prev->next = c->next;
|
|
else
|
|
user->chans = c->next;
|
|
delete c;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Handle a SJOIN command.
|
|
|
|
On channel creation, syntax is:
|
|
|
|
av[0] = timestamp
|
|
av[1] = channel name
|
|
av[2|3|4] = modes \ depends of whether the modes k and l
|
|
av[3|4|5] = users / are set or not.
|
|
|
|
When a single user joins an (existing) channel, it is:
|
|
|
|
av[0] = timestamp
|
|
av[1] = user
|
|
|
|
============================================================
|
|
|
|
Unreal SJOIN
|
|
|
|
On Services connect there is
|
|
SJOIN !11LkOb #ircops +nt :@Trystan &*!*@*.aol.com "*@*.home.com
|
|
|
|
av[0] = time stamp (base64)
|
|
av[1] = channel
|
|
av[2] = modes
|
|
av[3] = users + bans + exceptions
|
|
|
|
On Channel Creation or a User joins an existing
|
|
Luna.NomadIrc.Net SJOIN !11LkW9 #akill :@Trystan
|
|
Luna.NomadIrc.Net SJOIN !11LkW9 #akill :Trystan`
|
|
|
|
av[0] = time stamp (base64)
|
|
av[1] = channel
|
|
av[2] = users
|
|
|
|
*/
|
|
|
|
void do_sjoin(const char *source, int ac, const char **av)
|
|
{
|
|
Channel *c;
|
|
User *user;
|
|
Server *serv;
|
|
struct c_userlist *cu;
|
|
const char *s = NULL;
|
|
char *buf, *end, cubuf[7], *end2, value;
|
|
const char *modes[6];
|
|
int is_sqlined = 0;
|
|
int ts = 0;
|
|
int is_created = 0;
|
|
int keep_their_modes = 1;
|
|
ChannelModeList *cml;
|
|
|
|
serv = findserver(servlist, source);
|
|
|
|
if (ircd->sjb64) {
|
|
ts = base64dects(av[0]);
|
|
} else {
|
|
ts = strtoul(av[0], NULL, 10);
|
|
}
|
|
c = findchan(av[1]);
|
|
if (c != NULL) {
|
|
if (c->creation_time == 0 || ts == 0)
|
|
c->creation_time = 0;
|
|
else if (c->creation_time > ts) {
|
|
c->creation_time = ts;
|
|
for (cu = c->users; cu; cu = cu->next) {
|
|
/* XXX */
|
|
modes[0] = "-ov";
|
|
modes[1] = cu->user->nick;
|
|
modes[2] = cu->user->nick;
|
|
chan_set_modes(source, c, 3, modes, 2);
|
|
}
|
|
if (c->ci)
|
|
{
|
|
if (c->ci->bi)
|
|
{
|
|
/* This is ugly, but it always works */
|
|
ircdproto->SendPart(c->ci->bi, c->name, "TS reop");
|
|
bot_join(c->ci);
|
|
}
|
|
/* Make sure +r is set */
|
|
if (ModeManager::FindChannelModeByName(CMODE_REGISTERED))
|
|
{
|
|
c->SetMode(CMODE_REGISTERED);
|
|
ircdproto->SendMode(whosends(c->ci), c->name, "+r");
|
|
}
|
|
}
|
|
/* XXX simple modes and bans */
|
|
} else if (c->creation_time < ts)
|
|
keep_their_modes = 0;
|
|
} else
|
|
is_created = 1;
|
|
|
|
/* Double check to avoid unknown modes that need parameters */
|
|
if (ac >= 4) {
|
|
if (ircd->chansqline) {
|
|
if (!c)
|
|
is_sqlined = check_chan_sqline(av[1]);
|
|
}
|
|
|
|
cubuf[0] = '+';
|
|
modes[0] = cubuf;
|
|
|
|
/* We make all the users join */
|
|
s = av[ac - 1]; /* Users are always the last element */
|
|
|
|
while (*s) {
|
|
end = const_cast<char *>(strchr(s, ' '));
|
|
if (end)
|
|
*end = 0;
|
|
|
|
end2 = cubuf + 1;
|
|
|
|
|
|
if (ircd->sjoinbanchar) {
|
|
if (*s == ircd->sjoinbanchar && keep_their_modes) {
|
|
buf = myStrGetToken(s, ircd->sjoinbanchar, 1);
|
|
|
|
cml = static_cast<ChannelModeList *>(ModeManager::FindChannelModeByName(CMODE_BAN));
|
|
if (cml->IsValid(buf))
|
|
cml->AddMask(c, buf);
|
|
|
|
delete [] buf;
|
|
if (!end)
|
|
break;
|
|
s = end + 1;
|
|
continue;
|
|
}
|
|
}
|
|
if (ircd->sjoinexchar) {
|
|
if (*s == ircd->sjoinexchar && keep_their_modes) {
|
|
buf = myStrGetToken(s, ircd->sjoinexchar, 1);
|
|
|
|
cml = static_cast<ChannelModeList *>(ModeManager::FindChannelModeByName(CMODE_EXCEPT));
|
|
if (cml->IsValid(buf))
|
|
cml->AddMask(c, buf);
|
|
|
|
delete [] buf;
|
|
if (!end)
|
|
break;
|
|
s = end + 1;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (ircd->sjoininvchar) {
|
|
if (*s == ircd->sjoininvchar && keep_their_modes) {
|
|
buf = myStrGetToken(s, ircd->sjoininvchar, 1);
|
|
|
|
cml = static_cast<ChannelModeList *>(ModeManager::FindChannelModeByName(CMODE_INVITEOVERRIDE));
|
|
if (cml->IsValid(buf))
|
|
cml->AddMask(c, buf);
|
|
|
|
delete [] buf;
|
|
if (!end)
|
|
break;
|
|
s = end + 1;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
while ((value = ModeManager::GetStatusChar(*s)))
|
|
{
|
|
*end2++ = value;
|
|
*s++;
|
|
}
|
|
*end2 = 0;
|
|
|
|
if (ircd->ts6) {
|
|
user = find_byuid(s);
|
|
if (!user)
|
|
user = finduser(s);
|
|
} else {
|
|
user = finduser(s);
|
|
}
|
|
|
|
if (!user) {
|
|
if (debug) {
|
|
alog("debug: SJOIN for nonexistent user %s on %s", s,
|
|
av[1]);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (is_sqlined && !is_oper(user)) {
|
|
ircdproto->SendKick(findbot(s_OperServ), av[1], s, "Q-Lined");
|
|
} else {
|
|
if (!check_kick(user, av[1], ts)) {
|
|
FOREACH_MOD(I_OnPreJoinChannel, OnPreJoinChannel(user, av[1]));
|
|
|
|
/* Make the user join; if the channel does not exist it
|
|
* will be created there. This ensures that the channel
|
|
* is not created to be immediately destroyed, and
|
|
* that the locked key or topic is not shown to anyone
|
|
* who joins the channel when empty.
|
|
*/
|
|
c = join_user_update(user, c, av[1], ts);
|
|
|
|
/* We update user mode on the channel */
|
|
if (end2 - cubuf > 1 && keep_their_modes) {
|
|
int i;
|
|
|
|
for (i = 1; i < end2 - cubuf; i++)
|
|
modes[i] = user->nick;
|
|
chan_set_modes(source, c, 1 + (end2 - cubuf - 1),
|
|
modes, 2);
|
|
}
|
|
|
|
if (c->ci && (!serv || is_sync(serv))
|
|
&& !c->topic_sync)
|
|
restore_topic(c->name);
|
|
chan_set_correct_modes(user, c, 1);
|
|
|
|
FOREACH_MOD(I_OnJoinChannel, OnJoinChannel(user, c));
|
|
}
|
|
}
|
|
|
|
if (!end)
|
|
break;
|
|
s = end + 1;
|
|
}
|
|
|
|
if (c && keep_their_modes) {
|
|
/* We now update the channel mode. */
|
|
chan_set_modes(source, c, ac - 3, &av[2], 2);
|
|
}
|
|
|
|
/* Unreal just had to be different */
|
|
} else if (ac == 3 && !ircd->ts6) {
|
|
if (ircd->chansqline) {
|
|
if (!c)
|
|
is_sqlined = check_chan_sqline(av[1]);
|
|
}
|
|
|
|
cubuf[0] = '+';
|
|
modes[0] = cubuf;
|
|
|
|
/* We make all the users join */
|
|
s = av[2]; /* Users are always the last element */
|
|
|
|
while (*s) {
|
|
end = const_cast<char *>(strchr(s, ' '));
|
|
if (end)
|
|
*end = 0;
|
|
|
|
end2 = cubuf + 1;
|
|
|
|
while ((value = ModeManager::GetStatusChar(*s)))
|
|
{
|
|
*end2++ = value;
|
|
*s++;
|
|
}
|
|
*end2 = 0;
|
|
|
|
if (ircd->ts6) {
|
|
user = find_byuid(s);
|
|
if (!user)
|
|
user = finduser(s);
|
|
} else {
|
|
user = finduser(s);
|
|
}
|
|
|
|
if (!user) {
|
|
if (debug) {
|
|
alog("debug: SJOIN for nonexistent user %s on %s", s,
|
|
av[1]);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (is_sqlined && !is_oper(user)) {
|
|
ircdproto->SendKick(findbot(s_OperServ), av[1], s, "Q-Lined");
|
|
} else {
|
|
if (!check_kick(user, av[1], ts)) {
|
|
FOREACH_MOD(I_OnPreJoinChannel, OnPreJoinChannel(user, av[1]));
|
|
|
|
/* Make the user join; if the channel does not exist it
|
|
* will be created there. This ensures that the channel
|
|
* is not created to be immediately destroyed, and
|
|
* that the locked key or topic is not shown to anyone
|
|
* who joins the channel when empty.
|
|
*/
|
|
c = join_user_update(user, c, av[1], ts);
|
|
|
|
/* We update user mode on the channel */
|
|
if (end2 - cubuf > 1 && keep_their_modes) {
|
|
int i;
|
|
|
|
for (i = 1; i < end2 - cubuf; i++)
|
|
modes[i] = user->nick;
|
|
chan_set_modes(source, c, 1 + (end2 - cubuf - 1),
|
|
modes, 2);
|
|
}
|
|
|
|
chan_set_correct_modes(user, c, 1);
|
|
|
|
FOREACH_MOD(I_OnJoinChannel, OnJoinChannel(user, c));
|
|
}
|
|
}
|
|
|
|
if (!end)
|
|
break;
|
|
s = end + 1;
|
|
}
|
|
} else if (ac == 3 && ircd->ts6) {
|
|
if (ircd->chansqline) {
|
|
if (!c)
|
|
is_sqlined = check_chan_sqline(av[1]);
|
|
}
|
|
|
|
cubuf[0] = '+';
|
|
modes[0] = cubuf;
|
|
|
|
/* We make all the users join */
|
|
s = sstrdup(source); /* Users are always the last element */
|
|
|
|
while (*s) {
|
|
end = const_cast<char *>(strchr(s, ' '));
|
|
if (end)
|
|
*end = 0;
|
|
|
|
end2 = cubuf + 1;
|
|
|
|
while ((value = ModeManager::GetStatusChar(*s)))
|
|
{
|
|
*end2++ = value;
|
|
*s++;
|
|
}
|
|
*end2 = 0;
|
|
|
|
if (ircd->ts6) {
|
|
user = find_byuid(s);
|
|
if (!user)
|
|
user = finduser(s);
|
|
} else {
|
|
user = finduser(s);
|
|
}
|
|
if (!user) {
|
|
if (debug) {
|
|
alog("debug: SJOIN for nonexistent user %s on %s", s,
|
|
av[1]);
|
|
}
|
|
delete [] s;
|
|
return;
|
|
}
|
|
|
|
if (is_sqlined && !is_oper(user)) {
|
|
ircdproto->SendKick(findbot(s_OperServ), av[1], s, "Q-Lined");
|
|
} else {
|
|
if (!check_kick(user, av[1], ts)) {
|
|
FOREACH_MOD(I_OnPreJoinChannel, OnPreJoinChannel(user, av[1]));
|
|
|
|
/* Make the user join; if the channel does not exist it
|
|
* will be created there. This ensures that the channel
|
|
* is not created to be immediately destroyed, and
|
|
* that the locked key or topic is not shown to anyone
|
|
* who joins the channel when empty.
|
|
*/
|
|
c = join_user_update(user, c, av[1], ts);
|
|
|
|
/* We update user mode on the channel */
|
|
if (end2 - cubuf > 1 && keep_their_modes) {
|
|
int i;
|
|
|
|
for (i = 1; i < end2 - cubuf; i++)
|
|
modes[i] = user->nick;
|
|
chan_set_modes(source, c, 1 + (end2 - cubuf - 1),
|
|
modes, 2);
|
|
}
|
|
|
|
chan_set_correct_modes(user, c, 1);
|
|
|
|
FOREACH_MOD(I_OnJoinChannel, OnJoinChannel(user, c));
|
|
}
|
|
}
|
|
|
|
if (!end)
|
|
break;
|
|
s = end + 1;
|
|
}
|
|
delete [] s;
|
|
} else if (ac == 2) {
|
|
if (ircd->ts6) {
|
|
user = find_byuid(source);
|
|
if (!user)
|
|
user = finduser(source);
|
|
} else {
|
|
user = finduser(source);
|
|
}
|
|
if (!user) {
|
|
if (debug) {
|
|
alog("debug: SJOIN for nonexistent user %s on %s", source,
|
|
av[1]);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (check_kick(user, av[1], ts))
|
|
return;
|
|
|
|
if (ircd->chansqline) {
|
|
if (!c)
|
|
is_sqlined = check_chan_sqline(av[1]);
|
|
}
|
|
|
|
if (is_sqlined && !is_oper(user)) {
|
|
ircdproto->SendKick(findbot(s_OperServ), av[1], user->nick, "Q-Lined");
|
|
} else {
|
|
FOREACH_MOD(I_OnPreJoinChannel, OnPreJoinChannel(user, av[1]));
|
|
|
|
c = join_user_update(user, c, av[1], ts);
|
|
if (is_created && c->ci)
|
|
restore_topic(c->name);
|
|
chan_set_correct_modes(user, c, 1);
|
|
|
|
FOREACH_MOD(I_OnJoinChannel, OnJoinChannel(user, c));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Handle a channel MODE command. */
|
|
|
|
/*
|
|
* av[0]: channel
|
|
* av[1]: (tsmode) channel TS (this should be moved to a protocol module)
|
|
* av[2-]: (tsmode) modes and parameters
|
|
*
|
|
* av[1-]: modes and parameters.
|
|
*
|
|
*/
|
|
void do_cmode(const char *source, int ac, const char **av)
|
|
{
|
|
Channel *chan;
|
|
ChannelInfo *ci = NULL;
|
|
unsigned int i;
|
|
const char *t;
|
|
|
|
if (ircdcap->tsmode) {
|
|
/* TSMODE for bahamut - leave this code out to break MODEs. -GD */
|
|
/* if they don't send it in CAPAB check if we just want to enable it */
|
|
if (uplink_capab & ircdcap->tsmode || UseTSMODE) {
|
|
for (i = 0; i < strlen(av[1]); i++) {
|
|
if (!isdigit(av[1][i]))
|
|
break;
|
|
}
|
|
if (av[1][i] == '\0') {
|
|
/* We have a valid TS field in av[1] now, so we can strip it off */
|
|
/* After we swap av[0] and av[1] ofcourse to not break stuff! :) */
|
|
t = av[0];
|
|
av[0] = av[1];
|
|
av[1] = t;
|
|
ac--;
|
|
av++;
|
|
} else {
|
|
alog("TSMODE enabled but MODE has no valid TS");
|
|
}
|
|
}
|
|
}
|
|
|
|
/* :42XAAAAAO TMODE 1106409026 #ircops +b *!*@*.aol.com */
|
|
if (ircd->ts6) {
|
|
if (isdigit(av[0][0])) {
|
|
ac--;
|
|
av++;
|
|
}
|
|
}
|
|
|
|
chan = findchan(av[0]);
|
|
if (!chan) {
|
|
if (debug) {
|
|
ci = cs_findchan(av[0]);
|
|
if (!(ci && (ci->flags & CI_FORBIDDEN)))
|
|
alog("debug: MODE %s for nonexistent channel %s",
|
|
merge_args(ac - 1, av + 1), av[0]);
|
|
}
|
|
return;
|
|
}
|
|
|
|
/* This shouldn't trigger on +o, etc. */
|
|
if (strchr(source, '.') && !av[1][strcspn(av[1], "bovahq")]) {
|
|
if (time(NULL) != chan->server_modetime) {
|
|
chan->server_modecount = 0;
|
|
chan->server_modetime = time(NULL);
|
|
}
|
|
chan->server_modecount++;
|
|
}
|
|
|
|
ac--;
|
|
av++;
|
|
chan_set_modes(source, chan, ac, av, 1);
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Handle a TOPIC command. */
|
|
|
|
void do_topic(const char *source, int ac, const char **av)
|
|
{
|
|
Channel *c = findchan(av[0]);
|
|
ChannelInfo *ci;
|
|
int ts;
|
|
time_t topic_time;
|
|
char *topicsetter;
|
|
|
|
if (ircd->sjb64) {
|
|
ts = base64dects(av[2]);
|
|
if (debug) {
|
|
alog("debug: encoded TOPIC TS %s converted to %d", av[2], ts);
|
|
}
|
|
} else {
|
|
ts = strtoul(av[2], NULL, 10);
|
|
}
|
|
|
|
topic_time = ts;
|
|
|
|
if (!c) {
|
|
if (debug) {
|
|
alog("debug: TOPIC %s for nonexistent channel %s",
|
|
merge_args(ac - 1, av + 1), av[0]);
|
|
}
|
|
return;
|
|
}
|
|
|
|
/* We can be sure that the topic will be in sync here -GD */
|
|
c->topic_sync = 1;
|
|
|
|
ci = c->ci;
|
|
|
|
/* For Unreal, cut off the ! and any futher part of the topic setter.
|
|
* This way, nick!ident@host setters will only show the nick. -GD
|
|
*/
|
|
topicsetter = myStrGetToken(av[1], '!', 0);
|
|
|
|
/* If the current topic we have matches the last known topic for this
|
|
* channel exactly, there's no need to update anything and we can as
|
|
* well just return silently without updating anything. -GD
|
|
*/
|
|
if ((ac > 3) && *av[3] && ci && ci->last_topic
|
|
&& (strcmp(av[3], ci->last_topic) == 0)
|
|
&& (strcmp(topicsetter, ci->last_topic_setter) == 0)) {
|
|
delete [] topicsetter;
|
|
return;
|
|
}
|
|
|
|
if (check_topiclock(c, topic_time)) {
|
|
delete [] topicsetter;
|
|
return;
|
|
}
|
|
|
|
if (c->topic) {
|
|
delete [] c->topic;
|
|
c->topic = NULL;
|
|
}
|
|
if (ac > 3 && *av[3]) {
|
|
c->topic = sstrdup(av[3]);
|
|
}
|
|
|
|
strscpy(c->topic_setter, topicsetter, sizeof(c->topic_setter));
|
|
c->topic_time = topic_time;
|
|
delete [] topicsetter;
|
|
|
|
record_topic(av[0]);
|
|
|
|
FOREACH_MOD(I_OnTopicUpdated, OnTopicUpdated(c, av[0]));
|
|
}
|
|
|
|
/*************************************************************************/
|
|
/**************************** Internal Calls *****************************/
|
|
/*************************************************************************/
|
|
|
|
/**
|
|
* Set the correct modes, or remove the ones granted without permission,
|
|
* for the specified user on ths specified channel. This doesn't give
|
|
* modes to ignored users, but does remove them if needed.
|
|
* @param user The user to give/remove modes to/from
|
|
* @param c The channel to give/remove modes on
|
|
* @param give_modes Set to 1 to give modes, 0 to not give modes
|
|
* @return void
|
|
**/
|
|
void chan_set_correct_modes(User * user, Channel * c, int give_modes)
|
|
{
|
|
std::string modebuf, userbuf;
|
|
int status;
|
|
int add_modes = 0;
|
|
int rem_modes = 0;
|
|
ChannelInfo *ci;
|
|
ChannelMode *owner, *admin, *op, *halfop, *voice;
|
|
|
|
owner = ModeManager::FindChannelModeByName(CMODE_OWNER);
|
|
admin = ModeManager::FindChannelModeByName(CMODE_PROTECT);
|
|
op = ModeManager::FindChannelModeByName(CMODE_OP);
|
|
halfop = ModeManager::FindChannelModeByName(CMODE_HALFOP);
|
|
voice = ModeManager::FindChannelModeByName(CMODE_VOICE);
|
|
|
|
if (!c || !(ci = c->ci))
|
|
return;
|
|
|
|
if ((ci->flags & CI_FORBIDDEN) || (*(c->name) == '+'))
|
|
return;
|
|
|
|
status = chan_get_user_status(c, user);
|
|
|
|
if (debug)
|
|
alog("debug: Setting correct user modes for %s on %s (current status: %d, %sgiving modes)", user->nick, c->name, status, (give_modes ? "" : "not "));
|
|
|
|
/* Changed the second line of this if a bit, to make sure unregistered
|
|
* users can always get modes (IE: they always have autoop enabled). Before
|
|
* this change, you were required to have a registered nick to be able
|
|
* to receive modes. I wonder who added that... *looks at Rob* ;) -GD
|
|
*/
|
|
if (give_modes && (get_ignore(user->nick) == NULL)
|
|
&& (!user->nc || !(user->nc->flags & NI_AUTOOP))) {
|
|
if (owner && (IsFounder(user, ci) || check_access(user, ci, CA_AUTOOWNER)))
|
|
add_modes |= CUS_OWNER;
|
|
else if (admin && check_access(user, ci, CA_AUTOPROTECT))
|
|
add_modes |= CUS_PROTECT;
|
|
if (op && check_access(user, ci, CA_AUTOOP))
|
|
add_modes |= CUS_OP;
|
|
else if (halfop && check_access(user, ci, CA_AUTOHALFOP))
|
|
add_modes |= CUS_HALFOP;
|
|
else if (voice && check_access(user, ci, CA_AUTOVOICE))
|
|
add_modes |= CUS_VOICE;
|
|
}
|
|
|
|
/* We check if every mode they have is legally acquired here, and remove
|
|
* the modes that they're not allowed to have. But only if SECUREOPS is
|
|
* on, because else every mode is legal. -GD
|
|
* Unless the channel has just been created. -heinz
|
|
* Or the user matches CA_AUTODEOP... -GD
|
|
*/
|
|
if (((ci->flags & CI_SECUREOPS) || (c->usercount == 1)
|
|
|| check_access(user, ci, CA_AUTODEOP))
|
|
&& !is_ulined(user->server->name)) {
|
|
if (owner && (status & CUS_OWNER) && !IsFounder(user, ci))
|
|
rem_modes |= CUS_OWNER;
|
|
if (admin && (status & CUS_PROTECT)
|
|
&& !check_access(user, ci, CA_AUTOPROTECT)
|
|
&& !check_access(user, ci, CA_PROTECTME))
|
|
rem_modes |= CUS_PROTECT;
|
|
if (op && (status & CUS_OP) && !check_access(user, ci, CA_AUTOOP)
|
|
&& !check_access(user, ci, CA_OPDEOPME))
|
|
rem_modes |= CUS_OP;
|
|
if (halfop && (status & CUS_HALFOP)
|
|
&& !check_access(user, ci, CA_AUTOHALFOP)
|
|
&& !check_access(user, ci, CA_HALFOPME))
|
|
rem_modes |= CUS_HALFOP;
|
|
}
|
|
|
|
/* No modes to add or remove, exit function -GD */
|
|
if (!add_modes && !rem_modes)
|
|
return;
|
|
|
|
if (add_modes > 0)
|
|
{
|
|
modebuf += "+";
|
|
|
|
if (owner && (add_modes & CUS_OWNER) && !(status & CUS_OWNER))
|
|
{
|
|
modebuf += owner->ModeChar;
|
|
userbuf += " " + std::string(user->nick);
|
|
}
|
|
else
|
|
add_modes &= ~CUS_OWNER;
|
|
|
|
if (admin && (add_modes & CUS_PROTECT) && !(status & CUS_PROTECT))
|
|
{
|
|
modebuf += admin->ModeChar;
|
|
userbuf += " " + std::string(user->nick);
|
|
}
|
|
else
|
|
add_modes &= ~CUS_PROTECT;
|
|
|
|
if (op && (add_modes & CUS_OP) && !(status & CUS_OP))
|
|
{
|
|
modebuf += op->ModeChar;
|
|
userbuf += " " + std::string(user->nick);
|
|
}
|
|
else
|
|
add_modes &= ~CUS_OP;
|
|
|
|
if (halfop && (add_modes & CUS_HALFOP) && !(status & CUS_HALFOP))
|
|
{
|
|
modebuf += halfop->ModeChar;
|
|
userbuf += " " + std::string(user->nick);
|
|
}
|
|
else
|
|
add_modes &= ~CUS_HALFOP;
|
|
|
|
if (voice && (add_modes & CUS_VOICE) && !(status & CUS_VOICE))
|
|
{
|
|
modebuf += voice->ModeChar;
|
|
userbuf += " " + std::string(user->nick);
|
|
}
|
|
else
|
|
add_modes &= ~CUS_VOICE;
|
|
}
|
|
if (rem_modes > 0)
|
|
{
|
|
modebuf += "-";
|
|
|
|
if (owner && rem_modes & CUS_OWNER)
|
|
{
|
|
modebuf += owner->ModeChar;
|
|
userbuf += " " + std::string(user->nick);
|
|
}
|
|
|
|
if (admin && rem_modes & CUS_PROTECT)
|
|
{
|
|
modebuf += admin->ModeChar;
|
|
userbuf += " " + std::string(user->nick);
|
|
}
|
|
|
|
if (op && rem_modes & CUS_OP)
|
|
{
|
|
modebuf += op->ModeChar;
|
|
userbuf += " " + std::string(user->nick);
|
|
}
|
|
|
|
if (halfop && rem_modes & CUS_HALFOP)
|
|
{
|
|
modebuf += halfop->ModeChar;
|
|
userbuf += " " + std::string(user->nick);
|
|
}
|
|
}
|
|
|
|
/* Here, both can be empty again due to the "isn't it set already?"
|
|
* checks above. -GD
|
|
*/
|
|
if (!add_modes && !rem_modes)
|
|
return;
|
|
|
|
ircdproto->SendMode(whosends(ci), c->name, "%s%s", modebuf.c_str(), userbuf.c_str());
|
|
if (add_modes > 0)
|
|
chan_set_user_status(c, user, add_modes);
|
|
if (rem_modes > 0)
|
|
chan_remove_user_status(c, user, rem_modes);
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* Add/remove a user to/from a channel, creating or deleting the channel as
|
|
* necessary. If creating the channel, restore mode lock and topic as
|
|
* necessary. Also check for auto-opping and auto-voicing.
|
|
*/
|
|
|
|
void chan_adduser2(User * user, Channel * c)
|
|
{
|
|
struct c_userlist *u;
|
|
|
|
u = new c_userlist;
|
|
u->prev = NULL;
|
|
u->next = c->users;
|
|
if (c->users)
|
|
c->users->prev = u;
|
|
c->users = u;
|
|
u->user = user;
|
|
u->ud = NULL;
|
|
c->usercount++;
|
|
|
|
if (get_ignore(user->nick) == NULL) {
|
|
if (c->ci && (check_access(user, c->ci, CA_MEMO))
|
|
&& (c->ci->memos.memos.size() > 0)) {
|
|
if (c->ci->memos.memos.size() == 1) {
|
|
notice_lang(s_MemoServ, user, MEMO_X_ONE_NOTICE,
|
|
c->ci->memos.memos.size(), c->ci->name);
|
|
} else {
|
|
notice_lang(s_MemoServ, user, MEMO_X_MANY_NOTICE,
|
|
c->ci->memos.memos.size(), c->ci->name);
|
|
}
|
|
}
|
|
/* Added channelname to entrymsg - 30.03.2004, Certus */
|
|
/* Also, don't send the entrymsg when bursting -GD */
|
|
if (c->ci && c->ci->entry_message && is_sync(user->server))
|
|
user->SendMessage(whosends(c->ci)->nick, "[%s] %s", c->name, c->ci->entry_message);
|
|
}
|
|
|
|
/**
|
|
* We let the bot join even if it was an ignored user, as if we don't,
|
|
* and the ignored user dosnt just leave, the bot will never
|
|
* make it into the channel, leaving the channel botless even for
|
|
* legit users - Rob
|
|
**/
|
|
if (s_BotServ && c->ci && c->ci->bi) {
|
|
if (c->usercount == BSMinUsers)
|
|
bot_join(c->ci);
|
|
if (c->usercount >= BSMinUsers && (c->ci->botflags & BS_GREET)
|
|
&& user->nc && user->nc->greet
|
|
&& check_access(user, c->ci, CA_GREET)) {
|
|
/* Only display the greet if the main uplink we're connected
|
|
* to has synced, or we'll get greet-floods when the net
|
|
* recovers from a netsplit. -GD
|
|
*/
|
|
if (is_sync(user->server)) {
|
|
ircdproto->SendPrivmsg(c->ci->bi, c->name, "[%s] %s",
|
|
user->nc->display, user->nc->greet);
|
|
c->ci->bi->lastmsg = time(NULL);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* This creates the channel structure (was originally in
|
|
chan_adduser, but splitted to make it more efficient to use for
|
|
SJOINs). */
|
|
|
|
Channel *chan_create(const char *chan, time_t ts)
|
|
{
|
|
Channel *c;
|
|
Channel **list;
|
|
|
|
if (debug)
|
|
alog("debug: Creating channel %s", chan);
|
|
c = new Channel;
|
|
strscpy(c->name, chan, sizeof(c->name));
|
|
list = &chanlist[HASH(c->name)];
|
|
c->prev = NULL;
|
|
c->next = *list;
|
|
if (*list)
|
|
(*list)->prev = c;
|
|
*list = c;
|
|
c->creation_time = ts;
|
|
c->topic = NULL;
|
|
*c->topic_setter = 0;
|
|
c->topic_time = 0;
|
|
c->bans = c->excepts = c->invites = NULL;
|
|
c->users = NULL;
|
|
c->usercount = 0;
|
|
c->bd = NULL;
|
|
c->server_modetime = c->chanserv_modetime = 0;
|
|
c->server_modecount = c->chanserv_modecount = c->bouncy_modes = c->topic_sync = 0;
|
|
/* Store ChannelInfo pointer in channel record */
|
|
c->ci = cs_findchan(chan);
|
|
if (c->ci)
|
|
c->ci->c = c;
|
|
/* Restore locked modes and saved topic */
|
|
if (c->ci) {
|
|
check_modes(c);
|
|
stick_all(c->ci);
|
|
}
|
|
|
|
if (serv_uplink && is_sync(serv_uplink) && (!(c->topic_sync))) {
|
|
restore_topic(chan);
|
|
}
|
|
|
|
FOREACH_MOD(I_OnChannelCreate, OnChannelCreate(c));
|
|
|
|
return c;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/* This destroys the channel structure, freeing everything in it. */
|
|
|
|
void chan_delete(Channel * c)
|
|
{
|
|
BanData *bd, *next;
|
|
|
|
FOREACH_MOD(I_OnChannelDelete, OnChannelDelete(c));
|
|
|
|
if (debug)
|
|
alog("debug: Deleting channel %s", c->name);
|
|
|
|
for (bd = c->bd; bd; bd = next) {
|
|
if (bd->mask)
|
|
delete [] bd->mask;
|
|
next = bd->next;
|
|
delete bd;
|
|
}
|
|
|
|
if (c->ci)
|
|
c->ci->c = NULL;
|
|
|
|
if (c->topic)
|
|
delete [] c->topic;
|
|
|
|
if (c->bans && c->bans->count) {
|
|
while (c->bans->entries) {
|
|
entry_delete(c->bans, c->bans->entries);
|
|
}
|
|
}
|
|
|
|
if (ModeManager::FindChannelModeByName(CMODE_EXCEPT))
|
|
{
|
|
if (c->excepts && c->excepts->count) {
|
|
while (c->excepts->entries) {
|
|
entry_delete(c->excepts, c->excepts->entries);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ModeManager::FindChannelModeByName(CMODE_INVITEOVERRIDE))
|
|
{
|
|
if (c->invites && c->invites->count) {
|
|
while (c->invites->entries) {
|
|
entry_delete(c->invites, c->invites->entries);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (c->next)
|
|
c->next->prev = c->prev;
|
|
if (c->prev)
|
|
c->prev->next = c->next;
|
|
else
|
|
chanlist[HASH(c->name)] = c->next;
|
|
|
|
delete c;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
Channel *join_user_update(User * user, Channel * chan, const char *name,
|
|
time_t chants)
|
|
{
|
|
struct u_chanlist *c;
|
|
|
|
/* If it's a new channel, so we need to create it first. */
|
|
if (!chan)
|
|
chan = chan_create(name, chants);
|
|
else
|
|
{
|
|
// Check chants against 0, as not every ircd sends JOIN with a TS.
|
|
if (chan->creation_time > chants && chants != 0)
|
|
{
|
|
struct c_userlist *cu;
|
|
const char *modes[6];
|
|
|
|
chan->creation_time = chants;
|
|
for (cu = chan->users; cu; cu = cu->next)
|
|
{
|
|
/* XXX */
|
|
modes[0] = "-ov";
|
|
modes[1] = cu->user->nick;
|
|
modes[2] = cu->user->nick;
|
|
chan_set_modes(s_OperServ, chan, 3, modes, 2);
|
|
}
|
|
if (chan->ci)
|
|
{
|
|
if (chan->ci->bi)
|
|
{
|
|
/* This is ugly, but it always works */
|
|
ircdproto->SendPart(chan->ci->bi, chan->name, "TS reop");
|
|
bot_join(chan->ci);
|
|
}
|
|
/* Make sure +r is set */
|
|
if (ModeManager::FindChannelModeByName(CMODE_REGISTERED))
|
|
{
|
|
chan->SetMode(CMODE_REGISTERED);
|
|
ircdproto->SendMode(whosends(chan->ci), chan->name, "+r");
|
|
}
|
|
}
|
|
/* XXX simple modes and bans */
|
|
}
|
|
}
|
|
|
|
if (debug)
|
|
alog("debug: %s joins %s", user->nick, chan->name);
|
|
|
|
c = new u_chanlist;
|
|
c->prev = NULL;
|
|
c->next = user->chans;
|
|
if (user->chans)
|
|
user->chans->prev = c;
|
|
user->chans = c;
|
|
c->chan = chan;
|
|
c->status = 0;
|
|
|
|
chan_adduser2(user, chan);
|
|
|
|
return chan;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
|
|
void do_mass_mode(char *modes)
|
|
{
|
|
int ac;
|
|
const char **av;
|
|
Channel *c;
|
|
char *myModes;
|
|
|
|
if (!modes) {
|
|
return;
|
|
}
|
|
|
|
/* Prevent modes being altered by split_buf */
|
|
myModes = sstrdup(modes);
|
|
ac = split_buf(myModes, &av, 1);
|
|
|
|
for (c = firstchan(); c; c = nextchan()) {
|
|
if (c->bouncy_modes) {
|
|
free(av);
|
|
delete [] myModes;
|
|
return;
|
|
} else {
|
|
ircdproto->SendMode(findbot(s_OperServ), c->name, "%s", modes);
|
|
chan_set_modes(s_OperServ, c, ac, av, 1);
|
|
}
|
|
}
|
|
free(av);
|
|
delete [] myModes;
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
void restore_unsynced_topics()
|
|
{
|
|
Channel *c;
|
|
|
|
for (c = firstchan(); c; c = nextchan()) {
|
|
if (!(c->topic_sync))
|
|
restore_topic(c->name);
|
|
}
|
|
}
|
|
|
|
/*************************************************************************/
|
|
|
|
/**
|
|
* This handles creating a new Entry.
|
|
* This function destroys and free's the given mask as a side effect.
|
|
* @param mask Host/IP/CIDR mask to convert to an entry
|
|
* @return Entry struct for the given mask, NULL if creation failed
|
|
*/
|
|
Entry *entry_create(char *mask)
|
|
{
|
|
Entry *entry;
|
|
char *nick = NULL, *user, *host, *cidrhost;
|
|
uint32 ip, cidr;
|
|
|
|
entry = new Entry;
|
|
entry->type = ENTRYTYPE_NONE;
|
|
entry->prev = NULL;
|
|
entry->next = NULL;
|
|
entry->nick = NULL;
|
|
entry->user = NULL;
|
|
entry->host = NULL;
|
|
entry->mask = sstrdup(mask);
|
|
|
|
host = strchr(mask, '@');
|
|
if (host) {
|
|
*host++ = '\0';
|
|
/* If the user is purely a wildcard, ignore it */
|
|
if (str_is_pure_wildcard(mask))
|
|
user = NULL;
|
|
else {
|
|
|
|
/* There might be a nick too */
|
|
user = strchr(mask, '!');
|
|
if (user) {
|
|
*user++ = '\0';
|
|
/* If the nick is purely a wildcard, ignore it */
|
|
if (str_is_pure_wildcard(mask))
|
|
nick = NULL;
|
|
else
|
|
nick = mask;
|
|
} else {
|
|
nick = NULL;
|
|
user = mask;
|
|
}
|
|
}
|
|
} else {
|
|
/* It is possibly an extended ban/invite mask, but we do
|
|
* not support these at this point.. ~ Viper */
|
|
/* If there's no user in the mask, assume a pure wildcard */
|
|
user = NULL;
|
|
host = mask;
|
|
}
|
|
|
|
if (nick) {
|
|
entry->nick = sstrdup(nick);
|
|
/* Check if we have a wildcard user */
|
|
if (str_is_wildcard(nick))
|
|
entry->type |= ENTRYTYPE_NICK_WILD;
|
|
else
|
|
entry->type |= ENTRYTYPE_NICK;
|
|
}
|
|
|
|
if (user) {
|
|
entry->user = sstrdup(user);
|
|
/* Check if we have a wildcard user */
|
|
if (str_is_wildcard(user))
|
|
entry->type |= ENTRYTYPE_USER_WILD;
|
|
else
|
|
entry->type |= ENTRYTYPE_USER;
|
|
}
|
|
|
|
/* Only check the host if it's not a pure wildcard */
|
|
if (*host && !str_is_pure_wildcard(host)) {
|
|
if (ircd->cidrchanbei && str_is_cidr(host, &ip, &cidr, &cidrhost)) {
|
|
entry->cidr_ip = ip;
|
|
entry->cidr_mask = cidr;
|
|
entry->type |= ENTRYTYPE_CIDR4;
|
|
host = cidrhost;
|
|
} else if (ircd->cidrchanbei && strchr(host, '/')) {
|
|
/* Most IRCd's don't enforce sane bans therefore it is not
|
|
* so unlikely we will encounter this.
|
|
* Currently we only support strict CIDR without taking into
|
|
* account quirks of every single ircd (nef) that ignore everything
|
|
* after the first /cidr. To add this, sanitaze before sending to
|
|
* str_is_cidr() as this expects a standard cidr.
|
|
* Add it to the internal list (so it is included in for example clear)
|
|
* but do not use if during matching.. ~ Viper */
|
|
entry->type = ENTRYTYPE_NONE;
|
|
} else {
|
|
entry->host = sstrdup(host);
|
|
if (str_is_wildcard(host))
|
|
entry->type |= ENTRYTYPE_HOST_WILD;
|
|
else
|
|
entry->type |= ENTRYTYPE_HOST;
|
|
}
|
|
}
|
|
delete [] mask;
|
|
|
|
return entry;
|
|
}
|
|
|
|
|
|
/**
|
|
* Create an entry and add it at the beginning of given list.
|
|
* @param list The List the mask should be added to
|
|
* @param mask The mask to parse and add to the list
|
|
* @return Pointer to newly added entry. NULL if it fails.
|
|
*/
|
|
Entry *entry_add(EList * list, const char *mask)
|
|
{
|
|
Entry *e;
|
|
char *hostmask;
|
|
|
|
hostmask = sstrdup(mask);
|
|
e = entry_create(hostmask);
|
|
|
|
if (!e)
|
|
return NULL;
|
|
|
|
e->next = list->entries;
|
|
e->prev = NULL;
|
|
|
|
if (list->entries)
|
|
list->entries->prev = e;
|
|
list->entries = e;
|
|
list->count++;
|
|
|
|
return e;
|
|
}
|
|
|
|
|
|
/**
|
|
* Delete the given entry from a given list.
|
|
* @param list Linked list from which entry needs to be removed.
|
|
* @param e The entry to be deleted, must be member of list.
|
|
*/
|
|
void entry_delete(EList * list, Entry * e)
|
|
{
|
|
if (!list || !e)
|
|
return;
|
|
|
|
if (e->next)
|
|
e->next->prev = e->prev;
|
|
if (e->prev)
|
|
e->prev->next = e->next;
|
|
|
|
if (list->entries == e)
|
|
list->entries = e->next;
|
|
|
|
if (e->nick)
|
|
delete [] e->nick;
|
|
if (e->user)
|
|
delete [] e->user;
|
|
if (e->host)
|
|
delete [] e->host;
|
|
delete [] e->mask;
|
|
delete e;
|
|
|
|
list->count--;
|
|
}
|
|
|
|
|
|
/**
|
|
* Create and initialize a new entrylist
|
|
* @return Pointer to the created EList object
|
|
**/
|
|
EList *list_create()
|
|
{
|
|
EList *list;
|
|
|
|
list = new EList;
|
|
list->entries = NULL;
|
|
list->count = 0;
|
|
|
|
return list;
|
|
}
|
|
|
|
|
|
/**
|
|
* Match the given Entry to the given user/host and optional IP addy
|
|
* @param e Entry struct to match against
|
|
* @param nick Nick to match against
|
|
* @param user User to match against
|
|
* @param host Host to match against
|
|
* @param ip IP to match against, set to 0 to not match this
|
|
* @return 1 for a match, 0 for no match
|
|
*/
|
|
int entry_match(Entry * e, const char *nick, const char *user, const char *host, uint32 ip)
|
|
{
|
|
/* If we don't get an entry, or it s an invalid one, no match ~ Viper */
|
|
if (!e || e->type == ENTRYTYPE_NONE)
|
|
return 0;
|
|
|
|
if (ircd->cidrchanbei && (e->type & ENTRYTYPE_CIDR4) &&
|
|
(!ip || (ip && ((ip & e->cidr_mask) != e->cidr_ip))))
|
|
return 0;
|
|
if ((e->type & ENTRYTYPE_NICK)
|
|
&& (!nick || stricmp(e->nick, nick) != 0))
|
|
return 0;
|
|
if ((e->type & ENTRYTYPE_USER)
|
|
&& (!user || stricmp(e->user, user) != 0))
|
|
return 0;
|
|
if ((e->type & ENTRYTYPE_HOST)
|
|
&& (!user || stricmp(e->host, host) != 0))
|
|
return 0;
|
|
if ((e->type & ENTRYTYPE_NICK_WILD)
|
|
&& !Anope::Match(nick, e->nick, false))
|
|
return 0;
|
|
if ((e->type & ENTRYTYPE_USER_WILD)
|
|
&& !Anope::Match(user, e->user, false))
|
|
return 0;
|
|
if ((e->type & ENTRYTYPE_HOST_WILD)
|
|
&& !Anope::Match(host, e->host, false))
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* Match the given Entry to the given hostmask and optional IP addy.
|
|
* @param e Entry struct to match against
|
|
* @param mask Hostmask to match against
|
|
* @param ip IP to match against, set to 0 to not match this
|
|
* @return 1 for a match, 0 for no match
|
|
*/
|
|
int entry_match_mask(Entry * e, const char *mask, uint32 ip)
|
|
{
|
|
char *hostmask, *nick, *user, *host;
|
|
int res;
|
|
|
|
hostmask = sstrdup(mask);
|
|
|
|
host = strchr(hostmask, '@');
|
|
if (host) {
|
|
*host++ = '\0';
|
|
user = strchr(hostmask, '!');
|
|
if (user) {
|
|
*user++ = '\0';
|
|
nick = hostmask;
|
|
} else {
|
|
nick = NULL;
|
|
user = hostmask;
|
|
}
|
|
} else {
|
|
nick = NULL;
|
|
user = NULL;
|
|
host = hostmask;
|
|
}
|
|
|
|
res = entry_match(e, nick, user, host, ip);
|
|
|
|
/* Free the destroyed mask. */
|
|
delete [] hostmask;
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* Match a nick, user, host, and ip to a list entry
|
|
* @param e List that should be matched against
|
|
* @param nick The nick to match
|
|
* @param user The user to match
|
|
* @param host The host to match
|
|
* @param ip The ip to match
|
|
* @return Returns the first matching entry, if none, NULL is returned.
|
|
*/
|
|
Entry *elist_match(EList * list, const char *nick, const char *user, const char *host,
|
|
uint32 ip)
|
|
{
|
|
Entry *e;
|
|
|
|
if (!list || !list->entries)
|
|
return NULL;
|
|
|
|
for (e = list->entries; e; e = e->next) {
|
|
if (entry_match(e, nick, user, host, ip))
|
|
return e;
|
|
}
|
|
|
|
/* We matched none */
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* Match a mask and ip to a list.
|
|
* @param list EntryList that should be matched against
|
|
* @param mask The nick!user@host mask to match
|
|
* @param ip The ip to match
|
|
* @return Returns the first matching entry, if none, NULL is returned.
|
|
*/
|
|
Entry *elist_match_mask(EList * list, const char *mask, uint32 ip)
|
|
{
|
|
char *hostmask, *nick, *user, *host;
|
|
Entry *res;
|
|
|
|
if (!list || !list->entries || !mask)
|
|
return NULL;
|
|
|
|
hostmask = sstrdup(mask);
|
|
|
|
host = strchr(hostmask, '@');
|
|
if (host) {
|
|
*host++ = '\0';
|
|
user = strchr(hostmask, '!');
|
|
if (user) {
|
|
*user++ = '\0';
|
|
nick = hostmask;
|
|
} else {
|
|
nick = NULL;
|
|
user = hostmask;
|
|
}
|
|
} else {
|
|
nick = NULL;
|
|
user = NULL;
|
|
host = hostmask;
|
|
}
|
|
|
|
res = elist_match(list, nick, user, host, ip);
|
|
|
|
/* Free the destroyed mask. */
|
|
delete [] hostmask;
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* Check if a user matches an entry on a list.
|
|
* @param list EntryList that should be matched against
|
|
* @param user The user to match against the entries
|
|
* @return Returns the first matching entry, if none, NULL is returned.
|
|
*/
|
|
Entry *elist_match_user(EList * list, User * u)
|
|
{
|
|
Entry *res;
|
|
char *host;
|
|
uint32 ip = 0;
|
|
|
|
if (!list || !list->entries || !u)
|
|
return NULL;
|
|
|
|
if (u->hostip == NULL) {
|
|
host = host_resolve(u->host);
|
|
/* we store the just resolved hostname so we don't
|
|
* need to do this again */
|
|
if (host) {
|
|
u->hostip = sstrdup(host);
|
|
}
|
|
} else {
|
|
host = sstrdup(u->hostip);
|
|
}
|
|
|
|
/* Convert the host to an IP.. */
|
|
if (host)
|
|
ip = str_is_ip(host);
|
|
|
|
/* Match what we ve got against the lists.. */
|
|
res = elist_match(list, u->nick, u->GetIdent().c_str(), u->host, ip);
|
|
if (!res)
|
|
res = elist_match(list, u->nick, u->GetIdent().c_str(), u->GetDisplayedHost().c_str(), ip);
|
|
if (!res && !u->GetCloakedHost().empty() && u->GetCloakedHost() != u->GetDisplayedHost())
|
|
res = elist_match(list, u->nick, u->GetIdent().c_str(), u->GetCloakedHost().c_str(), ip);
|
|
|
|
if (host)
|
|
delete [] host;
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* Find a entry identical to the given mask..
|
|
* @param list EntryList that should be matched against
|
|
* @param mask The *!*@* mask to match
|
|
* @return Returns the first matching entry, if none, NULL is returned.
|
|
*/
|
|
Entry *elist_find_mask(EList * list, const char *mask)
|
|
{
|
|
Entry *e;
|
|
|
|
if (!list || !list->entries || !mask)
|
|
return NULL;
|
|
|
|
for (e = list->entries; e; e = e->next) {
|
|
if (!stricmp(e->mask, mask))
|
|
return e;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* Gets the total memory use of an entrylit.
|
|
* @param list The list we should estimate the mem use of.
|
|
* @return Returns the memory useage of the given list.
|
|
*/
|
|
long get_memuse(EList * list)
|
|
{
|
|
Entry *e;
|
|
long mem = 0;
|
|
|
|
if (!list)
|
|
return 0;
|
|
|
|
mem += sizeof(EList *);
|
|
mem += sizeof(Entry *) * list->count;
|
|
if (list->entries) {
|
|
for (e = list->entries; e; e = e->next) {
|
|
if (e->nick)
|
|
mem += strlen(e->nick) + 1;
|
|
if (e->user)
|
|
mem += strlen(e->user) + 1;
|
|
if (e->host)
|
|
mem += strlen(e->host) + 1;
|
|
if (e->mask)
|
|
mem += strlen(e->mask) + 1;
|
|
}
|
|
}
|
|
|
|
return mem;
|
|
}
|
|
|
|
/*************************************************************************/
|