2
* ircd-ratbox: A slightly useful ircd.
3
* m_kick.c: Kicks a user from a channel.
5
* Copyright (C) 1990 Jarkko Oikarinen and University of Oulu, Co Center
6
* Copyright (C) 1996-2002 Hybrid Development Team
7
* Copyright (C) 2002-2005 ircd-ratbox development team
9
* This program is free software; you can redistribute it and/or modify
10
* it under the terms of the GNU General Public License as published by
11
* the Free Software Foundation; either version 2 of the License, or
12
* (at your option) any later version.
14
* This program is distributed in the hope that it will be useful,
15
* but WITHOUT ANY WARRANTY; without even the implied warranty of
16
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17
* GNU General Public License for more details.
19
* You should have received a copy of the GNU General Public License
20
* along with this program; if not, write to the Free Software
21
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
24
* $Id: m_kick.c 3317 2007-03-28 23:17:06Z jilles $
42
static int m_kick(struct Client *, struct Client *, int, const char **);
43
#define mg_kick { m_kick, 3 }
45
struct Message kick_msgtab = {
46
"KICK", 0, 0, 0, MFLG_SLOW,
47
{mg_unreg, mg_kick, mg_kick, mg_kick, mg_ignore, mg_kick}
50
mapi_clist_av1 kick_clist[] = { &kick_msgtab, NULL };
52
DECLARE_MODULE_AV1(kick, NULL, NULL, kick_clist, NULL, NULL, "$Revision: 3317 $");
57
** parv[2] = client to kick
58
** parv[3] = kick comment
61
m_kick(struct Client *client_p, struct Client *source_p, int parc, const char *parv[])
63
struct membership *msptr;
65
struct Channel *chptr;
71
static char buf[BUFSIZE];
73
if(MyClient(source_p) && !IsFloodDone(source_p))
74
flood_endgrace(source_p);
77
if((p = strchr(parv[1], ',')))
82
chptr = find_channel(name);
85
sendto_one_numeric(source_p, ERR_NOSUCHCHANNEL, form_str(ERR_NOSUCHCHANNEL), name);
89
if(!IsServer(source_p))
91
msptr = find_channel_membership(chptr, source_p);
93
if((msptr == NULL) && MyConnect(source_p))
95
sendto_one_numeric(source_p, ERR_NOTONCHANNEL,
96
form_str(ERR_NOTONCHANNEL), name);
100
if(!is_chanop(msptr))
102
if(MyConnect(source_p))
104
sendto_one(source_p, form_str(ERR_CHANOPRIVSNEEDED),
105
me.name, source_p->name, name);
109
/* If its a TS 0 channel, do it the old way */
110
if(chptr->channelts == 0)
112
sendto_one(source_p, form_str(ERR_CHANOPRIVSNEEDED),
113
get_id(&me, source_p), get_id(source_p, source_p), name);
118
/* Its a user doing a kick, but is not showing as chanop locally
119
* its also not a user ON -my- server, and the channel has a TS.
120
* There are two cases we can get to this point then...
122
* 1) connect burst is happening, and for some reason a legit
123
* op has sent a KICK, but the SJOIN hasn't happened yet or
124
* been seen. (who knows.. due to lag...)
126
* 2) The channel is desynced. That can STILL happen with TS
128
* Now, the old code roger wrote, would allow the KICK to
129
* go through. Thats quite legit, but lets weird things like
130
* KICKS by users who appear not to be chanopped happen,
131
* or even neater, they appear not to be on the channel.
132
* This fits every definition of a desync, doesn't it? ;-)
133
* So I will allow the KICK, otherwise, things are MUCH worse.
134
* But I will warn it as a possible desync.
140
if((p = strchr(parv[2], ',')))
143
user = parv[2]; /* strtoken(&p2, parv[2], ","); */
145
if(!(who = find_chasing(source_p, user, &chasing)))
150
msptr = find_channel_membership(chptr, who);
154
if(MyClient(source_p) && IsService(who))
156
sendto_one(source_p, form_str(ERR_ISCHANSERVICE),
157
me.name, source_p->name, who->name, chptr->chname);
161
if(MyClient(source_p))
163
hook_data_channel_approval hookdata;
165
hookdata.client = source_p;
166
hookdata.chptr = chptr;
167
hookdata.target = who;
168
hookdata.approved = 1;
170
call_hook(h_can_kick, &hookdata);
172
if (!hookdata.approved)
176
comment = LOCAL_COPY((EmptyString(parv[3])) ? who->name : parv[3]);
177
if(strlen(comment) > (size_t) REASONLEN)
178
comment[REASONLEN] = '\0';
181
* - In the case of a server kicking a user (i.e. CLEARCHAN),
182
* the kick should show up as coming from the server which did
184
* - Personally, flame and I believe that server kicks shouldn't
185
* be sent anyways. Just waiting for some oper to abuse it...
187
if(IsServer(source_p))
188
sendto_channel_local(ALL_MEMBERS, chptr, ":%s KICK %s %s :%s",
189
source_p->name, name, who->name, comment);
191
sendto_channel_local(ALL_MEMBERS, chptr,
192
":%s!%s@%s KICK %s %s :%s",
193
source_p->name, source_p->username,
194
source_p->host, name, who->name, comment);
196
sendto_server(client_p, chptr, CAP_TS6, NOCAPS,
197
":%s KICK %s %s :%s",
198
use_id(source_p), chptr->chname, use_id(who), comment);
199
remove_user_from_channel(msptr);
201
else if (MyClient(source_p))
202
sendto_one_numeric(source_p, ERR_USERNOTINCHANNEL,
203
form_str(ERR_USERNOTINCHANNEL), user, name);