mirror of https://github.com/pissnet/pissircd.git
374 lines
11 KiB
C
374 lines
11 KiB
C
/*
|
|
* IRC - Internet Relay Chat, src/modules/kick.c
|
|
* (C) 2004 The UnrealIRCd Team
|
|
*
|
|
* See file AUTHORS in IRC package for additional names of
|
|
* the programmers.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 1, or (at your option)
|
|
* any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
#include "unrealircd.h"
|
|
|
|
ModuleHeader MOD_HEADER
|
|
= {
|
|
"kick",
|
|
"5.0",
|
|
"command /kick",
|
|
"UnrealIRCd Team",
|
|
"unrealircd-5",
|
|
};
|
|
|
|
/* Forward declarations */
|
|
CMD_FUNC(cmd_kick);
|
|
void _kick_user(MessageTag *mtags, Channel *channel, Client *client, Client *victim, char *comment);
|
|
|
|
MOD_TEST()
|
|
{
|
|
MARK_AS_OFFICIAL_MODULE(modinfo);
|
|
EfunctionAddVoid(modinfo->handle, EFUNC_KICK_USER, _kick_user);
|
|
return MOD_SUCCESS;
|
|
}
|
|
|
|
MOD_INIT()
|
|
{
|
|
CommandAdd(modinfo->handle, "KICK", cmd_kick, 3, CMD_USER|CMD_SERVER);
|
|
MARK_AS_OFFICIAL_MODULE(modinfo);
|
|
return MOD_SUCCESS;
|
|
}
|
|
|
|
MOD_LOAD()
|
|
{
|
|
return MOD_SUCCESS;
|
|
}
|
|
|
|
MOD_UNLOAD()
|
|
{
|
|
return MOD_SUCCESS;
|
|
}
|
|
|
|
/** Kick a user from a channel.
|
|
* @param initial_mtags Message tags associated with this KICK (can be NULL)
|
|
* @param channel The channel where the KICK should happen
|
|
* @param client The evil user doing the kick, can be &me
|
|
* @param victim The target user that will be kicked
|
|
* @param comment The KICK comment (cannot be NULL)
|
|
* @notes The msgid in initial_mtags is actually used as a prefix.
|
|
* The actual mtag will be "initial_mtags_msgid-suffix_msgid"
|
|
* All this is done in order for message tags to be
|
|
* consistent accross servers.
|
|
* The suffix is necessary to handle multi-target-kicks.
|
|
* If initial_mtags is NULL then we will autogenerate one.
|
|
*/
|
|
void _kick_user(MessageTag *initial_mtags, Channel *channel, Client *client, Client *victim, char *comment)
|
|
{
|
|
MessageTag *mtags = NULL;
|
|
int initial_mtags_generated = 0;
|
|
|
|
if (!initial_mtags)
|
|
{
|
|
/* Yeah, we allow callers to be lazy.. */
|
|
initial_mtags_generated = 1;
|
|
new_message(client, NULL, &initial_mtags);
|
|
}
|
|
|
|
new_message_special(client, initial_mtags, &mtags, ":%s KICK %s %s", client->name, channel->chname, victim->name);
|
|
/* The same message is actually sent at 5 places below (though max 4 at most) */
|
|
|
|
if (MyUser(client))
|
|
RunHook5(HOOKTYPE_LOCAL_KICK, client, victim, channel, mtags, comment);
|
|
else
|
|
RunHook5(HOOKTYPE_REMOTE_KICK, client, victim, channel, mtags, comment);
|
|
|
|
if (invisible_user_in_channel(victim, channel))
|
|
{
|
|
/* Send it only to chanops & victim */
|
|
sendto_channel(channel, client, victim,
|
|
PREFIX_HALFOP|PREFIX_OP|PREFIX_OWNER|PREFIX_ADMIN, 0,
|
|
SEND_LOCAL, mtags,
|
|
":%s KICK %s %s :%s",
|
|
client->name, channel->chname, victim->name, comment);
|
|
|
|
if (MyUser(victim))
|
|
{
|
|
sendto_prefix_one(victim, client, mtags, ":%s KICK %s %s :%s",
|
|
client->name, channel->chname, victim->name, comment);
|
|
}
|
|
} else {
|
|
/* NORMAL */
|
|
sendto_channel(channel, client, NULL, 0, 0, SEND_LOCAL, mtags,
|
|
":%s KICK %s %s :%s",
|
|
client->name, channel->chname, victim->name, comment);
|
|
}
|
|
|
|
sendto_server(client, 0, 0, mtags, ":%s KICK %s %s :%s",
|
|
client->id, channel->chname, victim->id, comment);
|
|
|
|
free_message_tags(mtags);
|
|
if (initial_mtags_generated)
|
|
{
|
|
free_message_tags(initial_mtags);
|
|
initial_mtags = NULL;
|
|
}
|
|
|
|
remove_user_from_channel(victim, channel);
|
|
}
|
|
|
|
/*
|
|
** cmd_kick
|
|
** parv[1] = channel
|
|
** parv[2] = client to kick
|
|
** parv[3] = kick comment
|
|
*/
|
|
|
|
#ifdef PREFIX_AQ
|
|
#define CHFL_ISOP (CHFL_CHANOWNER|CHFL_CHANADMIN|CHFL_CHANOP)
|
|
#else
|
|
#define CHFL_ISOP (CHFL_CHANOP)
|
|
#endif
|
|
|
|
CMD_FUNC(cmd_kick)
|
|
{
|
|
Client *who;
|
|
Channel *channel;
|
|
int chasing = 0;
|
|
char *comment, *name, *p = NULL, *user, *p2 = NULL, *badkick;
|
|
Membership *lp;
|
|
Hook *h;
|
|
int ret;
|
|
int ntargets = 0;
|
|
int maxtargets = max_targets_for_command("KICK");
|
|
MessageTag *mtags;
|
|
|
|
if (parc < 3 || *parv[1] == '\0')
|
|
{
|
|
sendnumeric(client, ERR_NEEDMOREPARAMS, "KICK");
|
|
return;
|
|
}
|
|
|
|
comment = (BadPtr(parv[3])) ? client->name : parv[3];
|
|
|
|
if (!BadPtr(parv[3]) && (strlen(comment) > iConf.kick_length))
|
|
comment[iConf.kick_length] = '\0';
|
|
|
|
for (; (name = strtoken(&p, parv[1], ",")); parv[1] = NULL)
|
|
{
|
|
long client_flags = 0;
|
|
channel = get_channel(client, name, !CREATE);
|
|
if (!channel)
|
|
{
|
|
sendnumeric(client, ERR_NOSUCHCHANNEL, name);
|
|
continue;
|
|
}
|
|
/* Store "client" access flags */
|
|
if (IsUser(client))
|
|
client_flags = get_access(client, channel);
|
|
if (MyUser(client) && !IsULine(client) && !op_can_override("channel:override:kick:no-ops",client,channel,NULL)
|
|
&& !(client_flags & CHFL_ISOP) && !(client_flags & CHFL_HALFOP))
|
|
{
|
|
sendnumeric(client, ERR_CHANOPRIVSNEEDED, channel->chname);
|
|
continue;
|
|
}
|
|
|
|
for (; (user = strtoken(&p2, parv[2], ",")); parv[2] = NULL)
|
|
{
|
|
long who_flags;
|
|
|
|
if (MyUser(client) && (++ntargets > maxtargets))
|
|
{
|
|
sendnumeric(client, ERR_TOOMANYTARGETS, user, maxtargets, "KICK");
|
|
break;
|
|
}
|
|
|
|
if (!(who = find_chasing(client, user, &chasing)))
|
|
continue; /* No such user left! */
|
|
|
|
if (!who->user)
|
|
continue; /* non-user */
|
|
|
|
lp = find_membership_link(who->user->channel, channel);
|
|
if (!lp)
|
|
{
|
|
if (MyUser(client))
|
|
sendnumeric(client, ERR_USERNOTINCHANNEL, user, name);
|
|
continue;
|
|
}
|
|
|
|
if (IsULine(client) || IsServer(client))
|
|
goto attack;
|
|
|
|
/* Note for coders regarding oper override:
|
|
* always let a remote kick (=from a user on another server) through or
|
|
* else we will get desynced. In short this means all the denying should
|
|
* always contain a && MyUser(client) and at the end
|
|
* a remote kick should always be allowed (pass through). -- Syzop
|
|
*/
|
|
|
|
/* Store "who" access flags */
|
|
who_flags = get_access(who, channel);
|
|
|
|
badkick = NULL;
|
|
ret = EX_ALLOW;
|
|
for (h = Hooks[HOOKTYPE_CAN_KICK]; h; h = h->next) {
|
|
int n = (*(h->func.intfunc))(client, who, channel, comment, client_flags, who_flags, &badkick);
|
|
|
|
if (n == EX_DENY)
|
|
ret = n;
|
|
else if (n == EX_ALWAYS_DENY)
|
|
{
|
|
ret = n;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (ret == EX_ALWAYS_DENY)
|
|
{
|
|
if (MyUser(client) && badkick)
|
|
sendto_one(client, NULL, "%s", badkick); /* send error, if any */
|
|
|
|
if (MyUser(client))
|
|
continue; /* reject the kick (note: we never block remote kicks) */
|
|
}
|
|
|
|
if (ret == EX_DENY)
|
|
{
|
|
/* If set it means 'not allowed to kick'.. now check if (s)he can override that.. */
|
|
if (op_can_override("channel:override:kick:no-ops",client,channel,NULL))
|
|
{
|
|
sendto_snomask(SNO_EYES,
|
|
"*** OperOverride -- %s (%s@%s) KICK %s %s (%s)",
|
|
client->name, client->user->username, client->user->realhost,
|
|
channel->chname, who->name, comment);
|
|
ircd_log(LOG_OVERRIDE,"OVERRIDE: %s (%s@%s) KICK %s %s (%s)",
|
|
client->name, client->user->username, client->user->realhost,
|
|
channel->chname, who->name, comment);
|
|
goto attack; /* all other checks don't matter anymore (and could cause double msgs) */
|
|
} else {
|
|
/* Not an oper overriding */
|
|
if (MyUser(client) && badkick)
|
|
sendto_one(client, NULL, "%s", badkick); /* send error, if any */
|
|
|
|
continue; /* reject the kick */
|
|
}
|
|
}
|
|
|
|
/* we are neither +o nor +h, OR..
|
|
* we are +h but victim is +o, OR...
|
|
* we are +h and victim is +h
|
|
*/
|
|
if (op_can_override("channel:override:kick:no-ops",client,channel,NULL))
|
|
{
|
|
if ((!(client_flags & CHFL_ISOP) && !(client_flags & CHFL_HALFOP)) ||
|
|
((client_flags & CHFL_HALFOP) && (who_flags & CHFL_ISOP)) ||
|
|
((client_flags & CHFL_HALFOP) && (who_flags & CHFL_HALFOP)))
|
|
{
|
|
sendto_snomask(SNO_EYES,
|
|
"*** OperOverride -- %s (%s@%s) KICK %s %s (%s)",
|
|
client->name, client->user->username, client->user->realhost,
|
|
channel->chname, who->name, comment);
|
|
|
|
/* Logging Implementation added by XeRXeS */
|
|
ircd_log(LOG_OVERRIDE,"OVERRIDE: %s (%s@%s) KICK %s %s (%s)",
|
|
client->name, client->user->username, client->user->realhost,
|
|
channel->chname, who->name, comment);
|
|
|
|
goto attack;
|
|
} /* is_chan_op */
|
|
|
|
}
|
|
|
|
/* victim is +a or +q, we are not +q */
|
|
if ((who_flags & (CHFL_CHANOWNER|CHFL_CHANADMIN))
|
|
&& !(client_flags & CHFL_CHANOWNER)) {
|
|
if (client == who)
|
|
goto attack; /* kicking self == ok */
|
|
if (op_can_override("channel:override:kick:owner",client,channel,NULL)) /* (and f*ck local ops) */
|
|
{ /* IRCop kicking owner/prot */
|
|
sendto_snomask(SNO_EYES,
|
|
"*** OperOverride -- %s (%s@%s) KICK %s %s (%s)",
|
|
client->name, client->user->username, client->user->realhost,
|
|
channel->chname, who->name, comment);
|
|
|
|
/* Logging Implementation added by XeRXeS */
|
|
ircd_log(LOG_OVERRIDE,"OVERRIDE: %s (%s@%s) KICK %s %s (%s)",
|
|
client->name, client->user->username, client->user->realhost,
|
|
channel->chname, who->name, comment);
|
|
|
|
goto attack;
|
|
}
|
|
else if (!IsULine(client) && (who != client) && MyUser(client))
|
|
{
|
|
char errbuf[NICKLEN+25];
|
|
if (who_flags & CHFL_CHANOWNER)
|
|
ircsnprintf(errbuf, sizeof(errbuf), "%s is a channel owner",
|
|
who->name);
|
|
else
|
|
ircsnprintf(errbuf, sizeof(errbuf), "%s is a channel admin",
|
|
who->name);
|
|
sendnumeric(client, ERR_CANNOTDOCOMMAND, "KICK",
|
|
errbuf);
|
|
goto deny;
|
|
} /* chanadmin/chanowner */
|
|
}
|
|
|
|
/* victim is +o, we are +h [operoverride is already taken care of 2 blocks above] */
|
|
if ((who_flags & CHFL_ISOP) && (client_flags & CHFL_HALFOP)
|
|
&& !(client_flags & CHFL_ISOP) && !IsULine(client) && MyUser(client))
|
|
{
|
|
char errbuf[NICKLEN+30];
|
|
ircsnprintf(errbuf, sizeof(errbuf), "%s is a channel operator", who->name);
|
|
sendnumeric(client, ERR_CANNOTDOCOMMAND, "KICK",
|
|
errbuf);
|
|
goto deny;
|
|
}
|
|
|
|
/* victim is +h, we are +h [operoverride is already taken care of 3 blocks above] */
|
|
if ((who_flags & CHFL_HALFOP) && (client_flags & CHFL_HALFOP)
|
|
&& !(client_flags & CHFL_ISOP) && MyUser(client))
|
|
{
|
|
char errbuf[NICKLEN+15];
|
|
ircsnprintf(errbuf, sizeof(errbuf), "%s is a halfop", who->name);
|
|
sendnumeric(client, ERR_CANNOTDOCOMMAND, "KICK",
|
|
errbuf);
|
|
goto deny;
|
|
} /* halfop */
|
|
|
|
/* allowed (either coz access granted or a remote kick), so attack! */
|
|
goto attack;
|
|
|
|
deny:
|
|
continue;
|
|
|
|
attack:
|
|
if (MyConnect(client)) {
|
|
int breakit = 0;
|
|
Hook *h;
|
|
for (h = Hooks[HOOKTYPE_PRE_LOCAL_KICK]; h; h = h->next) {
|
|
if((*(h->func.intfunc))(client,who,channel,comment) > 0) {
|
|
breakit = 1;
|
|
break;
|
|
}
|
|
}
|
|
if (breakit)
|
|
continue;
|
|
}
|
|
|
|
kick_user(recv_mtags, channel, client, who, comment);
|
|
} /* loop on parv[2] */
|
|
if (MyUser(client))
|
|
break;
|
|
} /* loop on parv[1] */
|
|
}
|