2 9dc44d9b 2002-05-27 alex * ngIRCd -- The Next Generation IRC Daemon
3 9dc44d9b 2002-05-27 alex * Copyright (c)2001,2002 by Alexander Barton (alex@barton.de)
5 490f28ff 2002-12-12 alex * This program is free software; you can redistribute it and/or modify
6 490f28ff 2002-12-12 alex * it under the terms of the GNU General Public License as published by
7 490f28ff 2002-12-12 alex * the Free Software Foundation; either version 2 of the License, or
8 490f28ff 2002-12-12 alex * (at your option) any later version.
9 490f28ff 2002-12-12 alex * Please read the file COPYING, README and AUTHORS for more information.
11 490f28ff 2002-12-12 alex * Management of IRC lists: ban, invite, ...
15 9dc44d9b 2002-05-27 alex #include "portab.h"
17 6b1e3228 2004-04-25 alex static char UNUSED id[] = "$Id: lists.c,v 1.15 2004/04/25 15:40:19 alex Exp $";
19 9dc44d9b 2002-05-27 alex #include "imp.h"
20 9dc44d9b 2002-05-27 alex #include <assert.h>
22 31a8dd2f 2002-06-09 alex #include "defines.h"
23 9dc44d9b 2002-05-27 alex #include "conn.h"
24 9dc44d9b 2002-05-27 alex #include "client.h"
25 9dc44d9b 2002-05-27 alex #include "channel.h"
26 31a8dd2f 2002-06-09 alex #include "log.h"
27 592565ae 2002-09-08 alex #include "match.h"
28 592565ae 2002-09-08 alex #include "messages.h"
29 592565ae 2002-09-08 alex #include "irc-write.h"
31 31a8dd2f 2002-06-09 alex #include <stdlib.h>
32 31a8dd2f 2002-06-09 alex #include <string.h>
33 57c7e236 2004-01-17 alex #include <strings.h>
35 9dc44d9b 2002-05-27 alex #include "exp.h"
36 9dc44d9b 2002-05-27 alex #include "lists.h"
39 592565ae 2002-09-08 alex #define MASK_LEN 2*CLIENT_HOST_LEN
42 9dc44d9b 2002-05-27 alex typedef struct _C2C
44 9dc44d9b 2002-05-27 alex struct _C2C *next;
45 31a8dd2f 2002-06-09 alex CHAR mask[MASK_LEN];
46 9dc44d9b 2002-05-27 alex CHANNEL *channel;
47 31a8dd2f 2002-06-09 alex BOOLEAN onlyonce;
51 9dc44d9b 2002-05-27 alex LOCAL C2C *My_Invites, *My_Bans;
54 31a8dd2f 2002-06-09 alex LOCAL C2C *New_C2C PARAMS(( CHAR *Mask, CHANNEL *Chan, BOOLEAN OnlyOnce ));
56 592565ae 2002-09-08 alex LOCAL BOOLEAN Check_List PARAMS(( C2C **Cl2Chan, CLIENT *Client, CHANNEL *Chan ));
57 05fc4a4c 2002-09-08 alex LOCAL BOOLEAN Already_Registered PARAMS(( C2C *Cl2Chan, CHAR *Mask, CHANNEL *Chan ));
62 9dc44d9b 2002-05-27 alex Lists_Init( VOID )
64 9dc44d9b 2002-05-27 alex /* Modul initialisieren */
66 9dc44d9b 2002-05-27 alex My_Invites = My_Bans = NULL;
67 9dc44d9b 2002-05-27 alex } /* Lists_Init */
71 9dc44d9b 2002-05-27 alex Lists_Exit( VOID )
73 9dc44d9b 2002-05-27 alex /* Modul abmelden */
75 31a8dd2f 2002-06-09 alex C2C *c2c, *next;
77 31a8dd2f 2002-06-09 alex /* Invite-Lists freigeben */
78 31a8dd2f 2002-06-09 alex c2c = My_Invites;
79 31a8dd2f 2002-06-09 alex while( c2c )
81 31a8dd2f 2002-06-09 alex next = c2c->next;
82 31a8dd2f 2002-06-09 alex free( c2c );
86 31a8dd2f 2002-06-09 alex /* Ban-Lists freigeben */
87 31a8dd2f 2002-06-09 alex c2c = My_Bans;
88 31a8dd2f 2002-06-09 alex while( c2c )
90 31a8dd2f 2002-06-09 alex next = c2c->next;
91 31a8dd2f 2002-06-09 alex free( c2c );
94 9dc44d9b 2002-05-27 alex } /* Lists_Exit */
97 9dc44d9b 2002-05-27 alex GLOBAL BOOLEAN
98 9dc44d9b 2002-05-27 alex Lists_CheckInvited( CLIENT *Client, CHANNEL *Chan )
100 592565ae 2002-09-08 alex return Check_List( &My_Invites, Client, Chan );
101 592565ae 2002-09-08 alex } /* Lists_CheckInvited */
104 592565ae 2002-09-08 alex GLOBAL BOOLEAN
105 6b1e3228 2004-04-25 alex Lists_IsInviteEntry( CHAR *Mask, CHANNEL *Chan )
107 6b1e3228 2004-04-25 alex assert( Mask != NULL );
108 6b1e3228 2004-04-25 alex assert( Chan != NULL );
110 6b1e3228 2004-04-25 alex return Already_Registered( My_Invites, Mask, Chan );
111 6b1e3228 2004-04-25 alex } /* Lists_IsInviteEntry */
114 6b1e3228 2004-04-25 alex GLOBAL BOOLEAN
115 64d330b7 2004-04-09 alex Lists_AddInvited( CHAR *Mask, CHANNEL *Chan, BOOLEAN OnlyOnce )
119 592565ae 2002-09-08 alex assert( Mask != NULL );
120 9dc44d9b 2002-05-27 alex assert( Chan != NULL );
122 64d330b7 2004-04-09 alex if( Already_Registered( My_Invites, Mask, Chan )) return TRUE;
124 592565ae 2002-09-08 alex c2c = New_C2C( Mask, Chan, OnlyOnce );
125 592565ae 2002-09-08 alex if( ! c2c )
127 592565ae 2002-09-08 alex Log( LOG_ERR, "Can't add new invite list entry!" );
128 592565ae 2002-09-08 alex return FALSE;
131 592565ae 2002-09-08 alex /* verketten */
132 592565ae 2002-09-08 alex c2c->next = My_Invites;
133 592565ae 2002-09-08 alex My_Invites = c2c;
135 592565ae 2002-09-08 alex Log( LOG_DEBUG, "Added \"%s\" to invite list for \"%s\".", Mask, Channel_Name( Chan ));
136 592565ae 2002-09-08 alex return TRUE;
137 592565ae 2002-09-08 alex } /* Lists_AddInvited */
140 592565ae 2002-09-08 alex GLOBAL VOID
141 592565ae 2002-09-08 alex Lists_DelInvited( CHAR *Mask, CHANNEL *Chan )
143 592565ae 2002-09-08 alex C2C *c2c, *last, *next;
145 592565ae 2002-09-08 alex assert( Mask != NULL );
146 592565ae 2002-09-08 alex assert( Chan != NULL );
148 31a8dd2f 2002-06-09 alex last = NULL;
149 31a8dd2f 2002-06-09 alex c2c = My_Invites;
150 31a8dd2f 2002-06-09 alex while( c2c )
152 592565ae 2002-09-08 alex next = c2c->next;
153 592565ae 2002-09-08 alex if(( c2c->channel == Chan ) && ( strcasecmp( c2c->mask, Mask ) == 0 ))
155 592565ae 2002-09-08 alex /* dieser Eintrag muss geloescht werden */
156 592565ae 2002-09-08 alex Log( LOG_DEBUG, "Deleted \"%s\" from invite list for \"%s\"." , c2c->mask, Channel_Name( Chan ));
157 592565ae 2002-09-08 alex if( last ) last->next = next;
158 592565ae 2002-09-08 alex else My_Invites = next;
159 592565ae 2002-09-08 alex free( c2c );
161 592565ae 2002-09-08 alex else last = c2c;
162 592565ae 2002-09-08 alex c2c = next;
164 592565ae 2002-09-08 alex } /* Lists_DelInvited */
167 592565ae 2002-09-08 alex GLOBAL BOOLEAN
168 592565ae 2002-09-08 alex Lists_ShowInvites( CLIENT *Client, CHANNEL *Channel )
172 592565ae 2002-09-08 alex assert( Client != NULL );
173 592565ae 2002-09-08 alex assert( Channel != NULL );
175 592565ae 2002-09-08 alex c2c = My_Invites;
176 592565ae 2002-09-08 alex while( c2c )
178 592565ae 2002-09-08 alex if( c2c->channel == Channel )
180 592565ae 2002-09-08 alex /* Eintrag fuer Channel gefunden; ausgeben: */
181 592565ae 2002-09-08 alex if( ! IRC_WriteStrClient( Client, RPL_INVITELIST_MSG, Client_ID( Client ), Channel_Name( Channel ), c2c->mask )) return DISCONNECTED;
183 31a8dd2f 2002-06-09 alex c2c = c2c->next;
185 592565ae 2002-09-08 alex return IRC_WriteStrClient( Client, RPL_ENDOFINVITELIST_MSG, Client_ID( Client ), Channel_Name( Channel ));
186 592565ae 2002-09-08 alex } /* Lists_ShowInvites */
189 6b1e3228 2004-04-25 alex GLOBAL BOOLEAN
190 6b1e3228 2004-04-25 alex Lists_SendInvites( CLIENT *Client )
194 6b1e3228 2004-04-25 alex assert( Client != NULL );
196 6b1e3228 2004-04-25 alex c2c = My_Invites;
197 6b1e3228 2004-04-25 alex while( c2c )
199 6b1e3228 2004-04-25 alex if( ! IRC_WriteStrClient( Client, "MODE %s +I %s", Channel_Name( c2c->channel ), c2c->mask )) return DISCONNECTED;
200 6b1e3228 2004-04-25 alex c2c = c2c->next;
202 6b1e3228 2004-04-25 alex return CONNECTED;
203 6b1e3228 2004-04-25 alex } /* Lists_SendInvites */
206 81a26d98 2002-07-15 alex GLOBAL BOOLEAN
207 6b1e3228 2004-04-25 alex Lists_SendBans( CLIENT *Client )
211 6b1e3228 2004-04-25 alex assert( Client != NULL );
213 6b1e3228 2004-04-25 alex c2c = My_Bans;
214 6b1e3228 2004-04-25 alex while( c2c )
216 6b1e3228 2004-04-25 alex if( ! IRC_WriteStrClient( Client, "MODE %s +b %s", Channel_Name( c2c->channel ), c2c->mask )) return DISCONNECTED;
217 6b1e3228 2004-04-25 alex c2c = c2c->next;
219 6b1e3228 2004-04-25 alex return CONNECTED;
220 6b1e3228 2004-04-25 alex } /* Lists_SendBans */
223 6b1e3228 2004-04-25 alex GLOBAL BOOLEAN
224 592565ae 2002-09-08 alex Lists_CheckBanned( CLIENT *Client, CHANNEL *Chan )
226 592565ae 2002-09-08 alex return Check_List( &My_Bans, Client, Chan );
227 592565ae 2002-09-08 alex } /* Lists_CheckBanned */
230 592565ae 2002-09-08 alex GLOBAL BOOLEAN
231 6b1e3228 2004-04-25 alex Lists_IsBanEntry( CHAR *Mask, CHANNEL *Chan )
233 6b1e3228 2004-04-25 alex assert( Mask != NULL );
234 6b1e3228 2004-04-25 alex assert( Chan != NULL );
236 6b1e3228 2004-04-25 alex return Already_Registered( My_Bans, Mask, Chan );
237 6b1e3228 2004-04-25 alex } /* Lists_IsBanEntry */
240 6b1e3228 2004-04-25 alex GLOBAL BOOLEAN
241 64d330b7 2004-04-09 alex Lists_AddBanned( CHAR *Mask, CHANNEL *Chan )
245 592565ae 2002-09-08 alex assert( Mask != NULL );
246 31a8dd2f 2002-06-09 alex assert( Chan != NULL );
248 64d330b7 2004-04-09 alex if( Already_Registered( My_Bans, Mask, Chan )) return TRUE;
250 592565ae 2002-09-08 alex c2c = New_C2C( Mask, Chan, FALSE );
251 31a8dd2f 2002-06-09 alex if( ! c2c )
253 592565ae 2002-09-08 alex Log( LOG_ERR, "Can't add new ban list entry!" );
254 81a26d98 2002-07-15 alex return FALSE;
257 31a8dd2f 2002-06-09 alex /* verketten */
258 592565ae 2002-09-08 alex c2c->next = My_Bans;
259 592565ae 2002-09-08 alex My_Bans = c2c;
261 592565ae 2002-09-08 alex Log( LOG_DEBUG, "Added \"%s\" to ban list for \"%s\".", Mask, Channel_Name( Chan ));
262 81a26d98 2002-07-15 alex return TRUE;
263 592565ae 2002-09-08 alex } /* Lists_AddBanned */
266 592565ae 2002-09-08 alex GLOBAL VOID
267 592565ae 2002-09-08 alex Lists_DelBanned( CHAR *Mask, CHANNEL *Chan )
269 592565ae 2002-09-08 alex C2C *c2c, *last, *next;
271 592565ae 2002-09-08 alex assert( Mask != NULL );
272 9dc44d9b 2002-05-27 alex assert( Chan != NULL );
274 592565ae 2002-09-08 alex last = NULL;
275 592565ae 2002-09-08 alex c2c = My_Bans;
276 592565ae 2002-09-08 alex while( c2c )
278 592565ae 2002-09-08 alex next = c2c->next;
279 592565ae 2002-09-08 alex if(( c2c->channel == Chan ) && ( strcasecmp( c2c->mask, Mask ) == 0 ))
281 592565ae 2002-09-08 alex /* dieser Eintrag muss geloescht werden */
282 592565ae 2002-09-08 alex Log( LOG_DEBUG, "Deleted \"%s\" from ban list for \"%s\"." , c2c->mask, Channel_Name( Chan ));
283 592565ae 2002-09-08 alex if( last ) last->next = next;
284 592565ae 2002-09-08 alex else My_Bans = next;
285 592565ae 2002-09-08 alex free( c2c );
287 592565ae 2002-09-08 alex else last = c2c;
288 592565ae 2002-09-08 alex c2c = next;
290 592565ae 2002-09-08 alex } /* Lists_DelBanned */
293 592565ae 2002-09-08 alex GLOBAL BOOLEAN
294 592565ae 2002-09-08 alex Lists_ShowBans( CLIENT *Client, CHANNEL *Channel )
298 592565ae 2002-09-08 alex assert( Client != NULL );
299 592565ae 2002-09-08 alex assert( Channel != NULL );
301 592565ae 2002-09-08 alex c2c = My_Bans;
302 592565ae 2002-09-08 alex while( c2c )
304 592565ae 2002-09-08 alex if( c2c->channel == Channel )
306 592565ae 2002-09-08 alex /* Eintrag fuer Channel gefunden; ausgeben: */
307 592565ae 2002-09-08 alex if( ! IRC_WriteStrClient( Client, RPL_BANLIST_MSG, Client_ID( Client ), Channel_Name( Channel ), c2c->mask )) return DISCONNECTED;
309 592565ae 2002-09-08 alex c2c = c2c->next;
311 592565ae 2002-09-08 alex return IRC_WriteStrClient( Client, RPL_ENDOFBANLIST_MSG, Client_ID( Client ), Channel_Name( Channel ));
312 592565ae 2002-09-08 alex } /* Lists_ShowBans */
315 31a8dd2f 2002-06-09 alex GLOBAL VOID
316 31a8dd2f 2002-06-09 alex Lists_DeleteChannel( CHANNEL *Chan )
318 31a8dd2f 2002-06-09 alex /* Channel wurde geloescht, Invite- und Ban-Lists aufraeumen */
320 31a8dd2f 2002-06-09 alex C2C *c2c, *last, *next;
322 31a8dd2f 2002-06-09 alex /* Invite-List */
323 31a8dd2f 2002-06-09 alex last = NULL;
324 31a8dd2f 2002-06-09 alex c2c = My_Invites;
325 31a8dd2f 2002-06-09 alex while( c2c )
327 31a8dd2f 2002-06-09 alex next = c2c->next;
328 31a8dd2f 2002-06-09 alex if( c2c->channel == Chan )
330 31a8dd2f 2002-06-09 alex /* dieser Eintrag muss geloescht werden */
331 592565ae 2002-09-08 alex Log( LOG_DEBUG, "Deleted \"%s\" from invite list for \"%s\"." , c2c->mask, Channel_Name( Chan ));
332 31a8dd2f 2002-06-09 alex if( last ) last->next = next;
333 31a8dd2f 2002-06-09 alex else My_Invites = next;
334 31a8dd2f 2002-06-09 alex free( c2c );
336 31a8dd2f 2002-06-09 alex else last = c2c;
337 31a8dd2f 2002-06-09 alex c2c = next;
340 31a8dd2f 2002-06-09 alex /* Ban-List */
341 31a8dd2f 2002-06-09 alex last = NULL;
342 31a8dd2f 2002-06-09 alex c2c = My_Bans;
343 31a8dd2f 2002-06-09 alex while( c2c )
345 31a8dd2f 2002-06-09 alex next = c2c->next;
346 31a8dd2f 2002-06-09 alex if( c2c->channel == Chan )
348 31a8dd2f 2002-06-09 alex /* dieser Eintrag muss geloescht werden */
349 592565ae 2002-09-08 alex Log( LOG_DEBUG, "Deleted \"%s\" from ban list for \"%s\"." , c2c->mask, Channel_Name( Chan ));
350 31a8dd2f 2002-06-09 alex if( last ) last->next = next;
351 31a8dd2f 2002-06-09 alex else My_Bans = next;
352 31a8dd2f 2002-06-09 alex free( c2c );
354 31a8dd2f 2002-06-09 alex else last = c2c;
355 31a8dd2f 2002-06-09 alex c2c = next;
357 31a8dd2f 2002-06-09 alex } /* Lists_DeleteChannel */
360 592565ae 2002-09-08 alex GLOBAL CHAR *
361 592565ae 2002-09-08 alex Lists_MakeMask( CHAR *Pattern )
363 592565ae 2002-09-08 alex /* Hier wird aus einem "beliebigen" Pattern eine gueltige IRC-Mask erzeugt.
364 74ff9828 2002-10-03 alex * Diese ist aber nur bis zum naechsten Aufruf von Lists_MakeMask() gueltig,
365 74ff9828 2002-10-03 alex * da ein einziger globaler Puffer verwendet wird. ->Umkopieren!*/
367 592565ae 2002-09-08 alex STATIC CHAR TheMask[MASK_LEN];
368 592565ae 2002-09-08 alex CHAR *excl, *at;
370 74ff9828 2002-10-03 alex assert( Pattern != NULL );
372 592565ae 2002-09-08 alex excl = strchr( Pattern, '!' );
373 592565ae 2002-09-08 alex at = strchr( Pattern, '@' );
375 592565ae 2002-09-08 alex if(( at ) && ( at < excl )) excl = NULL;
377 592565ae 2002-09-08 alex if(( ! at ) && ( ! excl ))
379 0ced4181 2002-12-26 alex /* weder ! noch @ vorhanden: als Nick annehmen */
380 0ced4181 2002-12-26 alex strlcpy( TheMask, Pattern, sizeof( TheMask ) - 5 );
381 0ced4181 2002-12-26 alex strlcat( TheMask, "!*@*", sizeof( TheMask ));
382 592565ae 2002-09-08 alex return TheMask;
385 592565ae 2002-09-08 alex if(( ! at ) && ( excl ))
387 592565ae 2002-09-08 alex /* Domain fehlt */
388 0ced4181 2002-12-26 alex strlcpy( TheMask, Pattern, sizeof( TheMask ) - 3 );
389 0ced4181 2002-12-26 alex strlcat( TheMask, "@*", sizeof( TheMask ));
390 592565ae 2002-09-08 alex return TheMask;
393 592565ae 2002-09-08 alex if(( at ) && ( ! excl ))
395 592565ae 2002-09-08 alex /* User fehlt */
396 592565ae 2002-09-08 alex *at = '\0'; at++;
397 0ced4181 2002-12-26 alex strlcpy( TheMask, Pattern, sizeof( TheMask ) - strlen( at ) - 4 );
398 0ced4181 2002-12-26 alex strlcat( TheMask, "!*@", sizeof( TheMask ));
399 0ced4181 2002-12-26 alex strlcat( TheMask, at, sizeof( TheMask ));
400 592565ae 2002-09-08 alex return TheMask;
403 592565ae 2002-09-08 alex /* alle Teile vorhanden */
404 0ced4181 2002-12-26 alex strlcpy( TheMask, Pattern, sizeof( TheMask ));
405 592565ae 2002-09-08 alex return TheMask;
406 592565ae 2002-09-08 alex } /* Lists_MakeMask */
409 9dc44d9b 2002-05-27 alex LOCAL C2C *
410 31a8dd2f 2002-06-09 alex New_C2C( CHAR *Mask, CHANNEL *Chan, BOOLEAN OnlyOnce )
414 31a8dd2f 2002-06-09 alex assert( Mask != NULL );
415 9dc44d9b 2002-05-27 alex assert( Chan != NULL );
417 31a8dd2f 2002-06-09 alex /* Speicher fuer Eintrag anfordern */
418 cb76d96e 2004-03-11 alex c2c = (C2C *)malloc( sizeof( C2C ));
419 31a8dd2f 2002-06-09 alex if( ! c2c )
421 31a8dd2f 2002-06-09 alex Log( LOG_EMERG, "Can't allocate memory! [New_C2C]" );
422 31a8dd2f 2002-06-09 alex return NULL;
425 0ced4181 2002-12-26 alex strlcpy( c2c->mask, Mask, sizeof( c2c->mask ));
426 31a8dd2f 2002-06-09 alex c2c->channel = Chan;
427 31a8dd2f 2002-06-09 alex c2c->onlyonce = OnlyOnce;
429 31a8dd2f 2002-06-09 alex return c2c;
430 9dc44d9b 2002-05-27 alex } /* New_C2C */
433 592565ae 2002-09-08 alex LOCAL BOOLEAN
434 592565ae 2002-09-08 alex Check_List( C2C **Cl2Chan, CLIENT *Client, CHANNEL *Chan )
436 592565ae 2002-09-08 alex C2C *c2c, *last;
438 592565ae 2002-09-08 alex assert( Cl2Chan != NULL );
439 592565ae 2002-09-08 alex assert( Client != NULL );
440 592565ae 2002-09-08 alex assert( Chan != NULL );
442 592565ae 2002-09-08 alex c2c = *Cl2Chan;
443 592565ae 2002-09-08 alex last = NULL;
445 592565ae 2002-09-08 alex while( c2c )
447 592565ae 2002-09-08 alex if( c2c->channel == Chan )
449 592565ae 2002-09-08 alex /* Ok, richtiger Channel. Passt die Maske? */
450 592565ae 2002-09-08 alex if( Match( c2c->mask, Client_Mask( Client )))
452 592565ae 2002-09-08 alex /* Treffer! */
453 592565ae 2002-09-08 alex if( c2c->onlyonce )
455 592565ae 2002-09-08 alex /* Eintrag loeschen */
456 592565ae 2002-09-08 alex Log( LOG_DEBUG, "Deleted \"%s\" from %s list for \"%s\".", c2c->mask, *Cl2Chan == My_Invites ? "invite" : "ban", Channel_Name( Chan ));
457 592565ae 2002-09-08 alex if( last ) last->next = c2c->next;
458 592565ae 2002-09-08 alex else *Cl2Chan = c2c->next;
459 592565ae 2002-09-08 alex free( c2c );
461 592565ae 2002-09-08 alex return TRUE;
464 592565ae 2002-09-08 alex last = c2c;
465 592565ae 2002-09-08 alex c2c = c2c->next;
468 592565ae 2002-09-08 alex return FALSE;
469 592565ae 2002-09-08 alex } /* Check_List */
472 05fc4a4c 2002-09-08 alex LOCAL BOOLEAN
473 05fc4a4c 2002-09-08 alex Already_Registered( C2C *List, CHAR *Mask, CHANNEL *Chan )
477 05fc4a4c 2002-09-08 alex c2c = List;
478 05fc4a4c 2002-09-08 alex while( c2c )
480 05fc4a4c 2002-09-08 alex if(( c2c->channel == Chan ) && ( strcasecmp( c2c->mask, Mask ) == 0 )) return TRUE;
481 05fc4a4c 2002-09-08 alex c2c = c2c->next;
483 05fc4a4c 2002-09-08 alex return FALSE;
484 05fc4a4c 2002-09-08 alex } /* Already_Registered */
487 9dc44d9b 2002-05-27 alex /* -eof- */