Blob


1 /*
2 * ngIRCd -- The Next Generation IRC Daemon
3 * Copyright (c)2001,2002 by Alexander Barton (alex@barton.de)
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 * Please read the file COPYING, README and AUTHORS for more information.
10 *
11 * Management of IRC lists: ban, invite, ...
12 */
15 #include "portab.h"
17 static char UNUSED id[] = "$Id: lists.c,v 1.11 2002/12/26 16:25:43 alex Exp $";
19 #include "imp.h"
20 #include <assert.h>
22 #include "defines.h"
23 #include "conn.h"
24 #include "client.h"
25 #include "channel.h"
26 #include "log.h"
27 #include "match.h"
28 #include "messages.h"
29 #include "irc-write.h"
31 #include <stdlib.h>
32 #include <string.h>
34 #include "exp.h"
35 #include "lists.h"
38 #define MASK_LEN 2*CLIENT_HOST_LEN
41 typedef struct _C2C
42 {
43 struct _C2C *next;
44 CHAR mask[MASK_LEN];
45 CHANNEL *channel;
46 BOOLEAN onlyonce;
47 } C2C;
50 LOCAL C2C *My_Invites, *My_Bans;
53 LOCAL C2C *New_C2C PARAMS(( CHAR *Mask, CHANNEL *Chan, BOOLEAN OnlyOnce ));
55 LOCAL BOOLEAN Check_List PARAMS(( C2C **Cl2Chan, CLIENT *Client, CHANNEL *Chan ));
56 LOCAL BOOLEAN Already_Registered PARAMS(( C2C *Cl2Chan, CHAR *Mask, CHANNEL *Chan ));
60 GLOBAL VOID
61 Lists_Init( VOID )
62 {
63 /* Modul initialisieren */
65 My_Invites = My_Bans = NULL;
66 } /* Lists_Init */
69 GLOBAL VOID
70 Lists_Exit( VOID )
71 {
72 /* Modul abmelden */
74 C2C *c2c, *next;
76 /* Invite-Lists freigeben */
77 c2c = My_Invites;
78 while( c2c )
79 {
80 next = c2c->next;
81 free( c2c );
82 c2c = next;
83 }
85 /* Ban-Lists freigeben */
86 c2c = My_Bans;
87 while( c2c )
88 {
89 next = c2c->next;
90 free( c2c );
91 c2c = next;
92 }
93 } /* Lists_Exit */
96 GLOBAL BOOLEAN
97 Lists_CheckInvited( CLIENT *Client, CHANNEL *Chan )
98 {
99 return Check_List( &My_Invites, Client, Chan );
100 } /* Lists_CheckInvited */
103 GLOBAL BOOLEAN
104 Lists_AddInvited( CLIENT *From, CHAR *Mask, CHANNEL *Chan, BOOLEAN OnlyOnce )
106 C2C *c2c;
108 assert( Mask != NULL );
109 assert( Chan != NULL );
111 if( Already_Registered( My_Invites, Mask, Chan ))
113 /* Eintrag ist bereits vorhanden */
114 IRC_WriteStrClient( From, RPL_INVITELIST_MSG, Client_ID( From ), Channel_Name( Chan ), Mask );
115 return FALSE;
118 c2c = New_C2C( Mask, Chan, OnlyOnce );
119 if( ! c2c )
121 Log( LOG_ERR, "Can't add new invite list entry!" );
122 return FALSE;
125 /* verketten */
126 c2c->next = My_Invites;
127 My_Invites = c2c;
129 Log( LOG_DEBUG, "Added \"%s\" to invite list for \"%s\".", Mask, Channel_Name( Chan ));
130 return TRUE;
131 } /* Lists_AddInvited */
134 GLOBAL VOID
135 Lists_DelInvited( CHAR *Mask, CHANNEL *Chan )
137 C2C *c2c, *last, *next;
139 assert( Mask != NULL );
140 assert( Chan != NULL );
142 last = NULL;
143 c2c = My_Invites;
144 while( c2c )
146 next = c2c->next;
147 if(( c2c->channel == Chan ) && ( strcasecmp( c2c->mask, Mask ) == 0 ))
149 /* dieser Eintrag muss geloescht werden */
150 Log( LOG_DEBUG, "Deleted \"%s\" from invite list for \"%s\"." , c2c->mask, Channel_Name( Chan ));
151 if( last ) last->next = next;
152 else My_Invites = next;
153 free( c2c );
155 else last = c2c;
156 c2c = next;
158 } /* Lists_DelInvited */
161 GLOBAL BOOLEAN
162 Lists_ShowInvites( CLIENT *Client, CHANNEL *Channel )
164 C2C *c2c;
166 assert( Client != NULL );
167 assert( Channel != NULL );
169 c2c = My_Invites;
170 while( c2c )
172 if( c2c->channel == Channel )
174 /* Eintrag fuer Channel gefunden; ausgeben: */
175 if( ! IRC_WriteStrClient( Client, RPL_INVITELIST_MSG, Client_ID( Client ), Channel_Name( Channel ), c2c->mask )) return DISCONNECTED;
177 c2c = c2c->next;
179 return IRC_WriteStrClient( Client, RPL_ENDOFINVITELIST_MSG, Client_ID( Client ), Channel_Name( Channel ));
180 } /* Lists_ShowInvites */
183 GLOBAL BOOLEAN
184 Lists_CheckBanned( CLIENT *Client, CHANNEL *Chan )
186 return Check_List( &My_Bans, Client, Chan );
187 } /* Lists_CheckBanned */
190 GLOBAL BOOLEAN
191 Lists_AddBanned( CLIENT *From, CHAR *Mask, CHANNEL *Chan )
193 C2C *c2c;
195 assert( Mask != NULL );
196 assert( Chan != NULL );
198 if( Already_Registered( My_Bans, Mask, Chan ))
200 /* Eintrag ist bereits vorhanden */
201 IRC_WriteStrClient( From, RPL_BANLIST_MSG, Client_ID( From ), Channel_Name( Chan ), Mask );
202 return FALSE;
205 c2c = New_C2C( Mask, Chan, FALSE );
206 if( ! c2c )
208 Log( LOG_ERR, "Can't add new ban list entry!" );
209 return FALSE;
212 /* verketten */
213 c2c->next = My_Bans;
214 My_Bans = c2c;
216 Log( LOG_DEBUG, "Added \"%s\" to ban list for \"%s\".", Mask, Channel_Name( Chan ));
217 return TRUE;
218 } /* Lists_AddBanned */
221 GLOBAL VOID
222 Lists_DelBanned( CHAR *Mask, CHANNEL *Chan )
224 C2C *c2c, *last, *next;
226 assert( Mask != NULL );
227 assert( Chan != NULL );
229 last = NULL;
230 c2c = My_Bans;
231 while( c2c )
233 next = c2c->next;
234 if(( c2c->channel == Chan ) && ( strcasecmp( c2c->mask, Mask ) == 0 ))
236 /* dieser Eintrag muss geloescht werden */
237 Log( LOG_DEBUG, "Deleted \"%s\" from ban list for \"%s\"." , c2c->mask, Channel_Name( Chan ));
238 if( last ) last->next = next;
239 else My_Bans = next;
240 free( c2c );
242 else last = c2c;
243 c2c = next;
245 } /* Lists_DelBanned */
248 GLOBAL BOOLEAN
249 Lists_ShowBans( CLIENT *Client, CHANNEL *Channel )
251 C2C *c2c;
253 assert( Client != NULL );
254 assert( Channel != NULL );
256 c2c = My_Bans;
257 while( c2c )
259 if( c2c->channel == Channel )
261 /* Eintrag fuer Channel gefunden; ausgeben: */
262 if( ! IRC_WriteStrClient( Client, RPL_BANLIST_MSG, Client_ID( Client ), Channel_Name( Channel ), c2c->mask )) return DISCONNECTED;
264 c2c = c2c->next;
266 return IRC_WriteStrClient( Client, RPL_ENDOFBANLIST_MSG, Client_ID( Client ), Channel_Name( Channel ));
267 } /* Lists_ShowBans */
270 GLOBAL VOID
271 Lists_DeleteChannel( CHANNEL *Chan )
273 /* Channel wurde geloescht, Invite- und Ban-Lists aufraeumen */
275 C2C *c2c, *last, *next;
277 /* Invite-List */
278 last = NULL;
279 c2c = My_Invites;
280 while( c2c )
282 next = c2c->next;
283 if( c2c->channel == Chan )
285 /* dieser Eintrag muss geloescht werden */
286 Log( LOG_DEBUG, "Deleted \"%s\" from invite list for \"%s\"." , c2c->mask, Channel_Name( Chan ));
287 if( last ) last->next = next;
288 else My_Invites = next;
289 free( c2c );
291 else last = c2c;
292 c2c = next;
295 /* Ban-List */
296 last = NULL;
297 c2c = My_Bans;
298 while( c2c )
300 next = c2c->next;
301 if( c2c->channel == Chan )
303 /* dieser Eintrag muss geloescht werden */
304 Log( LOG_DEBUG, "Deleted \"%s\" from ban list for \"%s\"." , c2c->mask, Channel_Name( Chan ));
305 if( last ) last->next = next;
306 else My_Bans = next;
307 free( c2c );
309 else last = c2c;
310 c2c = next;
312 } /* Lists_DeleteChannel */
315 GLOBAL CHAR *
316 Lists_MakeMask( CHAR *Pattern )
318 /* Hier wird aus einem "beliebigen" Pattern eine gueltige IRC-Mask erzeugt.
319 * Diese ist aber nur bis zum naechsten Aufruf von Lists_MakeMask() gueltig,
320 * da ein einziger globaler Puffer verwendet wird. ->Umkopieren!*/
322 STATIC CHAR TheMask[MASK_LEN];
323 CHAR *excl, *at;
325 assert( Pattern != NULL );
327 excl = strchr( Pattern, '!' );
328 at = strchr( Pattern, '@' );
330 if(( at ) && ( at < excl )) excl = NULL;
332 if(( ! at ) && ( ! excl ))
334 /* weder ! noch @ vorhanden: als Nick annehmen */
335 strlcpy( TheMask, Pattern, sizeof( TheMask ) - 5 );
336 strlcat( TheMask, "!*@*", sizeof( TheMask ));
337 return TheMask;
340 if(( ! at ) && ( excl ))
342 /* Domain fehlt */
343 strlcpy( TheMask, Pattern, sizeof( TheMask ) - 3 );
344 strlcat( TheMask, "@*", sizeof( TheMask ));
345 return TheMask;
348 if(( at ) && ( ! excl ))
350 /* User fehlt */
351 *at = '\0'; at++;
352 strlcpy( TheMask, Pattern, sizeof( TheMask ) - strlen( at ) - 4 );
353 strlcat( TheMask, "!*@", sizeof( TheMask ));
354 strlcat( TheMask, at, sizeof( TheMask ));
355 return TheMask;
358 /* alle Teile vorhanden */
359 strlcpy( TheMask, Pattern, sizeof( TheMask ));
360 return TheMask;
361 } /* Lists_MakeMask */
364 LOCAL C2C *
365 New_C2C( CHAR *Mask, CHANNEL *Chan, BOOLEAN OnlyOnce )
367 C2C *c2c;
369 assert( Mask != NULL );
370 assert( Chan != NULL );
372 /* Speicher fuer Eintrag anfordern */
373 c2c = malloc( sizeof( C2C ));
374 if( ! c2c )
376 Log( LOG_EMERG, "Can't allocate memory! [New_C2C]" );
377 return NULL;
380 strlcpy( c2c->mask, Mask, sizeof( c2c->mask ));
381 c2c->channel = Chan;
382 c2c->onlyonce = OnlyOnce;
384 return c2c;
385 } /* New_C2C */
388 LOCAL BOOLEAN
389 Check_List( C2C **Cl2Chan, CLIENT *Client, CHANNEL *Chan )
391 C2C *c2c, *last;
393 assert( Cl2Chan != NULL );
394 assert( Client != NULL );
395 assert( Chan != NULL );
397 c2c = *Cl2Chan;
398 last = NULL;
400 while( c2c )
402 if( c2c->channel == Chan )
404 /* Ok, richtiger Channel. Passt die Maske? */
405 if( Match( c2c->mask, Client_Mask( Client )))
407 /* Treffer! */
408 if( c2c->onlyonce )
410 /* Eintrag loeschen */
411 Log( LOG_DEBUG, "Deleted \"%s\" from %s list for \"%s\".", c2c->mask, *Cl2Chan == My_Invites ? "invite" : "ban", Channel_Name( Chan ));
412 if( last ) last->next = c2c->next;
413 else *Cl2Chan = c2c->next;
414 free( c2c );
416 return TRUE;
419 last = c2c;
420 c2c = c2c->next;
423 return FALSE;
424 } /* Check_List */
427 LOCAL BOOLEAN
428 Already_Registered( C2C *List, CHAR *Mask, CHANNEL *Chan )
430 C2C *c2c;
432 c2c = List;
433 while( c2c )
435 if(( c2c->channel == Chan ) && ( strcasecmp( c2c->mask, Mask ) == 0 )) return TRUE;
436 c2c = c2c->next;
438 return FALSE;
439 } /* Already_Registered */
442 /* -eof- */