source: irc_channel.c @ 441a67e

Last change on this file since 441a67e was 7d4ffc2, checked in by Wilmer van der Gaast <wilmer@…>, at 2012-02-17T10:20:28Z

Fixing NULL pointer dereference in irc_channel_free(). This seems to happen
for example when the user gets invited to a channel that already exists.
Separately, I should handle invites like that better. Will file a bug for
that.

  • Property mode set to 100644
File size: 19.6 KB
RevLine 
[4be8239]1  /********************************************************************\
2  * BitlBee -- An IRC to other IM-networks gateway                     *
3  *                                                                    *
4  * Copyright 2002-2010 Wilmer van der Gaast and others                *
5  \********************************************************************/
6
7/* The IRC-based UI - Representing (virtual) channels.                  */
8
9/*
10  This program is free software; you can redistribute it and/or modify
11  it under the terms of the GNU General Public License as published by
12  the Free Software Foundation; either version 2 of the License, or
13  (at your option) any later version.
14
15  This program is distributed in the hope that it will be useful,
16  but WITHOUT ANY WARRANTY; without even the implied warranty of
17  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  GNU General Public License for more details.
19
20  You should have received a copy of the GNU General Public License with
21  the Debian GNU/Linux distribution in /usr/share/common-licenses/GPL;
22  if not, write to the Free Software Foundation, Inc., 59 Temple Place,
23  Suite 330, Boston, MA  02111-1307  USA
24*/
25
26#include "bitlbee.h"
27
[2b8473c]28static char *set_eval_channel_type( set_t *set, char *value );
[e54112f]29static gint irc_channel_user_cmp( gconstpointer a_, gconstpointer b_ );
[280c56a]30static const struct irc_channel_funcs control_channel_funcs;
[5a75d15]31
32extern const struct irc_channel_funcs irc_channel_im_chat_funcs;
[280c56a]33
[4be8239]34irc_channel_t *irc_channel_new( irc_t *irc, const char *name )
35{
36        irc_channel_t *ic;
37       
[9438323]38        if( !irc_channel_name_ok( name ) || irc_channel_by_name( irc, name ) )
[4be8239]39                return NULL;
40       
41        ic = g_new0( irc_channel_t, 1 );
42        ic->irc = irc;
43        ic->name = g_strdup( name );
44        strcpy( ic->mode, CMODE );
45       
46        irc_channel_add_user( ic, irc->root );
47       
[36562b0]48        irc->channels = g_slist_append( irc->channels, ic );
[4be8239]49       
[c8eeadd]50        set_add( &ic->set, "auto_join", "false", set_eval_bool, ic );
[2b8473c]51        set_add( &ic->set, "type", "control", set_eval_channel_type, ic );
52       
[eb37735]53        if( name[0] == '&' )
[2b8473c]54                set_setstr( &ic->set, "type", "control" );
[eb37735]55        else /* if( name[0] == '#' ) */
[2b8473c]56                set_setstr( &ic->set, "type", "chat" );
[9ac3ed1]57       
[4be8239]58        return ic;
59}
60
[b9e020a]61irc_channel_t *irc_channel_by_name( irc_t *irc, const char *name )
62{
63        GSList *l;
64       
65        for( l = irc->channels; l; l = l->next )
66        {
67                irc_channel_t *ic = l->data;
68               
[6b90431]69                if( irc_channel_name_cmp( name, ic->name ) == 0 )
[b9e020a]70                        return ic;
71        }
72       
73        return NULL;
74}
75
[36562b0]76irc_channel_t *irc_channel_get( irc_t *irc, char *id )
77{
78        irc_channel_t *ic, *ret = NULL;
79        GSList *l;
80        int nr;
81       
82        if( sscanf( id, "%d", &nr ) == 1 && nr < 1000 )
83        {
84                for( l = irc->channels; l; l = l->next )
85                {
86                        ic = l->data;
87                        if( ( nr-- ) == 0 ) 
88                                return ic;
89                }
90               
91                return NULL;
92        }
93       
94        /* Exact match first: Partial match only sucks if there's a channel
95           #aa and #aabb */
96        if( ( ret = irc_channel_by_name( irc, id ) ) )
97                return ret;
98       
99        for( l = irc->channels; l; l = l->next )
100        {
101                ic = l->data;
102               
103                if( strstr( ic->name, id ) )
104                {
105                        /* Make sure it's a unique match. */
106                        if( !ret )
107                                ret = ic;
108                        else
109                                return NULL;
110                }
111        }
112       
113        return ret;
114}
115
[63a520b]116int irc_channel_free( irc_channel_t *ic )
117{
[7d4ffc2]118        irc_t *irc;
[2fe5eb9]119        GSList *l;
[63a520b]120       
[7d4ffc2]121        if( ic == NULL )
122                return 0;
123        irc = ic->irc;
124       
[63a520b]125        if( ic->flags & IRC_CHANNEL_JOINED )
[006a84f]126                irc_channel_del_user( ic, irc->user, IRC_CDU_KICK, "Cleaning up channel" );
[63a520b]127       
[d7db346]128        if( ic->f->_free )
129                ic->f->_free( ic );
130       
131        while( ic->set )
132                set_del( &ic->set, ic->set->key );
133       
[63a520b]134        irc->channels = g_slist_remove( irc->channels, ic );
[e54112f]135        while( ic->users )
136        {
137                g_free( ic->users->data );
138                ic->users = g_slist_remove( ic->users, ic->users->data );
139        }
[63a520b]140       
[2fe5eb9]141        for( l = irc->users; l; l = l->next )
142        {
143                irc_user_t *iu = l->data;
144               
145                if( iu->last_channel == ic )
146                        iu->last_channel = irc->default_channel;
147        }
148       
[cc20520]149        if( ic->pastebuf_timer ) b_event_remove( ic->pastebuf_timer );
150       
[63a520b]151        g_free( ic->name );
152        g_free( ic->topic );
[d7db346]153        g_free( ic->topic_who );
[63a520b]154        g_free( ic );
155       
156        return 1;
157}
158
[ab6006c]159struct irc_channel_free_data
160{
161        irc_t *irc;
162        irc_channel_t *ic;
163        char *name;
164};
165
166static gboolean irc_channel_free_callback( gpointer data, gint fd, b_input_condition cond )
167{
168        struct irc_channel_free_data *d = data;
169       
170        if( g_slist_find( irc_connection_list, d->irc ) &&
171            irc_channel_by_name( d->irc, d->name ) == d->ic &&
172            !( d->ic->flags & IRC_CHANNEL_JOINED ) )
173                irc_channel_free( d->ic );
174
175        g_free( d->name );
176        g_free( d );
177        return FALSE;
178}
179
180/* Free the channel, but via the event loop, so after finishing whatever event
181   we're currently handling. */
182void irc_channel_free_soon( irc_channel_t *ic )
183{
184        struct irc_channel_free_data *d = g_new0( struct irc_channel_free_data, 1 );
185       
186        d->irc = ic->irc;
187        d->ic = ic;
188        d->name = g_strdup( ic->name );
189       
190        b_timeout_add( 0, irc_channel_free_callback, d );
191}
192
[2b8473c]193static char *set_eval_channel_type( set_t *set, char *value )
194{
195        struct irc_channel *ic = set->data;
196        const struct irc_channel_funcs *new;
197       
198        if( strcmp( value, "control" ) == 0 )
199                new = &control_channel_funcs;
[05bf2a0]200        else if( ic != ic->irc->default_channel && strcmp( value, "chat" ) == 0 )
[5a75d15]201                new = &irc_channel_im_chat_funcs;
[2b8473c]202        else
203                return SET_INVALID;
204       
205        /* TODO: Return values. */
206        if( ic->f && ic->f->_free )
207                ic->f->_free( ic );
208       
209        ic->f = new;
210       
211        if( ic->f && ic->f->_init )
212                ic->f->_init( ic );
213       
214        return value;
215}
216
[4be8239]217int irc_channel_add_user( irc_channel_t *ic, irc_user_t *iu )
218{
[e54112f]219        irc_channel_user_t *icu;
220       
[57c96f7]221        if( irc_channel_has_user( ic, iu ) )
[4be8239]222                return 0;
223       
[e54112f]224        icu = g_new0( irc_channel_user_t, 1 );
225        icu->iu = iu;
226       
227        ic->users = g_slist_insert_sorted( ic->users, icu, irc_channel_user_cmp );
[4be8239]228       
[c5aefa4]229        irc_channel_update_ops( ic, set_getstr( &ic->irc->b->set, "ops" ) );
230       
[4be8239]231        if( iu == ic->irc->user || ic->flags & IRC_CHANNEL_JOINED )
232        {
233                ic->flags |= IRC_CHANNEL_JOINED;
234                irc_send_join( ic, iu );
235        }
236       
237        return 1;
238}
239
[006a84f]240int irc_channel_del_user( irc_channel_t *ic, irc_user_t *iu, irc_channel_del_user_type_t type, const char *msg )
[4be8239]241{
[e54112f]242        irc_channel_user_t *icu;
243       
244        if( !( icu = irc_channel_has_user( ic, iu ) ) )
[4be8239]245                return 0;
246       
[e54112f]247        ic->users = g_slist_remove( ic->users, icu );
248        g_free( icu );
[4be8239]249       
[006a84f]250        if( !( ic->flags & IRC_CHANNEL_JOINED ) || type == IRC_CDU_SILENT ) {}
251                /* Do nothing. The caller should promise it won't screw
252                   up state of the IRC client. :-) */
253        else if( type == IRC_CDU_PART )
[18da20b]254                irc_send_part( ic, iu, msg );
[006a84f]255        else if( type == IRC_CDU_KICK )
256                irc_send_kick( ic, iu, ic->irc->root, msg );
[4be8239]257       
258        if( iu == ic->irc->user )
[1c40aa7]259        {
[4be8239]260                ic->flags &= ~IRC_CHANNEL_JOINED;
[1c40aa7]261               
[06f9548]262                if( ic->irc->status & USTATUS_SHUTDOWN )
263                {
264                        /* Don't do anything fancy when we're shutting down anyway. */
265                }
266                else if( ic->flags & IRC_CHANNEL_TEMP )
267                {
[ab6006c]268                        irc_channel_free_soon( ic );
[06f9548]269                }
[9052bc1]270                else
271                {
272                        /* Flush userlist now. The user won't see it anyway. */
273                        while( ic->users )
274                        {
275                                g_free( ic->users->data );
276                                ic->users = g_slist_remove( ic->users, ic->users->data );
277                        }
278                        irc_channel_add_user( ic, ic->irc->root );
279                }
[1c40aa7]280        }
[4be8239]281       
282        return 1;
283}
284
[e54112f]285irc_channel_user_t *irc_channel_has_user( irc_channel_t *ic, irc_user_t *iu )
[0b5cc72]286{
[e54112f]287        GSList *l;
288       
289        for( l = ic->users; l; l = l->next )
290        {
291                irc_channel_user_t *icu = l->data;
292               
293                if( icu->iu == iu )
294                        return icu;
295        }
296       
297        return NULL;
[0b5cc72]298}
299
[bb151f7]300/* Find a channel we're currently in, that currently has iu in it. */
301struct irc_channel *irc_channel_with_user( irc_t *irc, irc_user_t *iu )
302{
303        GSList *l;
304       
305        for( l = irc->channels; l; l = l->next )
306        {
307                irc_channel_t *ic = l->data;
308               
309                if( strcmp( set_getstr( &ic->set, "type" ), "control" ) != 0 )
310                        continue;
311               
312                if( ( ic->flags & IRC_CHANNEL_JOINED ) &&
313                    irc_channel_has_user( ic, iu ) )
314                        return ic;
315        }
316       
317        /* If there was no match, try once more but just see if the user
318           *would* be in the channel, i.e. if s/he were online. */
319        if( iu->bu == NULL )
320                return NULL;
321       
322        for( l = irc->channels; l; l = l->next )
323        {
324                irc_channel_t *ic = l->data;
325               
326                if( strcmp( set_getstr( &ic->set, "type" ), "control" ) != 0 )
327                        continue;
328               
329                if( ( ic->flags & IRC_CHANNEL_JOINED ) &&
330                    irc_channel_wants_user( ic, iu ) )
331                        return ic;
332        }
333       
334        return NULL;
335}
336
[83e92bf]337int irc_channel_set_topic( irc_channel_t *ic, const char *topic, const irc_user_t *iu )
[4be8239]338{
339        g_free( ic->topic );
340        ic->topic = g_strdup( topic );
341       
[83e92bf]342        g_free( ic->topic_who );
343        if( iu )
344                ic->topic_who = g_strdup_printf( "%s!%s@%s", iu->nick, iu->user, iu->host );
345        else
346                ic->topic_who = NULL;
347       
348        ic->topic_time = time( NULL );
349       
[4be8239]350        if( ic->flags & IRC_CHANNEL_JOINED )
[83e92bf]351                irc_send_topic( ic, TRUE );
[4be8239]352       
353        return 1;
354}
[b919363]355
[6a9d068]356void irc_channel_user_set_mode( irc_channel_t *ic, irc_user_t *iu, irc_channel_user_flags_t flags )
357{
358        irc_channel_user_t *icu = irc_channel_has_user( ic, iu );
359       
[c5aefa4]360        if( !icu || icu->flags == flags )
[6a9d068]361                return;
362       
363        if( ic->flags & IRC_CHANNEL_JOINED )
364                irc_send_channel_user_mode_diff( ic, iu, icu->flags, flags );
365       
366        icu->flags = flags;
367}
368
[65016a6]369void irc_channel_set_mode( irc_channel_t *ic, const char *s )
370{
371        irc_t *irc = ic->irc;
372        char m[128], st = 1;
373        const char *t;
374        int i;
375        char changes[512], *p, st2 = 2;
376       
377        memset( m, 0, sizeof( m ) );
378       
379        for( t = ic->mode; *t; t ++ )
380                if( *t < sizeof( m ) )
381                        m[(int)*t] = 1;
382       
383        p = changes;
384        for( t = s; *t; t ++ )
385        {
386                if( *t == '+' || *t == '-' )
387                        st = *t == '+';
388                else if( strchr( CMODES, *t ) )
389                {
390                        if( m[(int)*t] != st)
391                        {
392                                if( st != st2 )
393                                        st2 = st, *p++ = st ? '+' : '-';
394                                *p++ = *t;
395                        }
396                        m[(int)*t] = st;
397                }
398        }
399        *p = '\0';
400       
401        memset( ic->mode, 0, sizeof( ic->mode ) );
402       
403        for( i = 'A'; i <= 'z' && strlen( ic->mode ) < ( sizeof( ic->mode ) - 1 ); i ++ )
404                if( m[i] )
405                        ic->mode[strlen(ic->mode)] = i;
406       
407        if( *changes && ( ic->flags & IRC_CHANNEL_JOINED ) )
408                irc_write( irc, ":%s!%s@%s MODE %s :%s", irc->root->nick,
409                           irc->root->user, irc->root->host, ic->name,
410                           changes );
411}
412
[c8eeadd]413void irc_channel_auto_joins( irc_t *irc, account_t *acc )
414{
415        GSList *l;
416       
417        for( l = irc->channels; l; l = l->next )
418        {
419                irc_channel_t *ic = l->data;
420                gboolean aj = set_getbool( &ic->set, "auto_join" );
421                char *type;
422               
423                if( acc &&
424                    ( type = set_getstr( &ic->set, "chat_type" ) ) &&
425                    strcmp( type, "room" ) == 0 )
426                {
427                        /* Bit of an ugly special case: Handle chatrooms here, we
428                           can only auto-join them if their account is online. */
429                        char *acc_s;
430                       
431                        if( !aj && !( ic->flags & IRC_CHANNEL_JOINED ) )
432                                /* Only continue if this one's marked as auto_join
433                                   or if we're in it already. (Possible if the
434                                   client auto-rejoined it before identyfing.) */
435                                continue;
436                        else if( !( acc_s = set_getstr( &ic->set, "account" ) ) )
437                                continue;
438                        else if( account_get( irc->b, acc_s ) != acc )
439                                continue;
440                        else if( acc->ic == NULL || !( acc->ic->flags & OPT_LOGGED_IN ) )
441                                continue;
442                        else
443                                ic->f->join( ic );
444                }
445                else if( aj )
446                {
447                        irc_channel_add_user( ic, irc->user );
448                }
449        }
450}
451
[9893da3]452void irc_channel_printf( irc_channel_t *ic, char *format, ... )
453{
454        va_list params;
455        char *text;
456       
457        va_start( params, format );
458        text = g_strdup_vprintf( format, params );
459        va_end( params );
460       
461        irc_send_msg( ic->irc->root, "PRIVMSG", ic->name, text, NULL );
462        g_free( text );
463}
464
[6b90431]465gboolean irc_channel_name_ok( const char *name_ )
[b919363]466{
[6b90431]467        const unsigned char *name = (unsigned char*) name_;
468        int i;
[4c17d19]469       
[a670aeb]470        if( name_[0] == '\0' )
471                return FALSE;
472       
[4c17d19]473        /* Check if the first character is in CTYPES (#&) */
[6b90431]474        if( strchr( CTYPES, name_[0] ) == NULL )
[4c17d19]475                return FALSE;
476       
[6b90431]477        /* RFC 1459 keeps amazing me: While only a "few" chars are allowed
478           in nicknames, channel names can be pretty much anything as long
479           as they start with # or &. I'll be a little bit more strict and
480           disallow all non-printable characters. */
481        for( i = 1; name[i]; i ++ )
482                if( name[i] <= ' ' || name[i] == ',' )
483                        return FALSE;
484       
485        return TRUE;
486}
487
[134a02c]488void irc_channel_name_strip( char *name )
489{
490        int i, j;
491       
492        for( i = j = 0; name[i]; i ++ )
493                if( name[i] > ' ' && name[i] != ',' )
494                        name[j++] = name[i];
495       
496        name[j] = '\0';
497}
498
[6b90431]499int irc_channel_name_cmp( const char *a_, const char *b_ )
500{
501        static unsigned char case_map[256];
502        const unsigned char *a = (unsigned char*) a_, *b = (unsigned char*) b_;
503        int i;
504       
505        if( case_map['A'] == '\0' )
506        {
507                for( i = 33; i < 256; i ++ )
508                        if( i != ',' )
509                                case_map[i] = i;
510               
511                for( i = 0; i < 26; i ++ )
512                        case_map['A'+i] = 'a' + i;
513               
514                case_map['['] = '{';
515                case_map[']'] = '}';
516                case_map['~'] = '`';
517                case_map['\\'] = '|';
518        }
519       
520        if( !irc_channel_name_ok( a_ ) || !irc_channel_name_ok( b_ ) )
521                return -1;
522       
523        for( i = 0; a[i] && b[i] && case_map[a[i]] && case_map[b[i]]; i ++ )
524        {
525                if( case_map[a[i]] == case_map[b[i]] )
526                        continue;
527                else
528                        return case_map[a[i]] - case_map[b[i]];
529        }
530       
531        return case_map[a[i]] - case_map[b[i]];
[b919363]532}
[280c56a]533
[e54112f]534static gint irc_channel_user_cmp( gconstpointer a_, gconstpointer b_ )
535{
536        const irc_channel_user_t *a = a_, *b = b_;
537       
538        return irc_user_cmp( a->iu, b->iu );
539}
540
[c5aefa4]541void irc_channel_update_ops( irc_channel_t *ic, char *value )
542{
543        irc_channel_user_set_mode( ic, ic->irc->root,
544                ( strcmp( value, "both" ) == 0 ||
545                  strcmp( value, "root" ) == 0 ) ? IRC_CHANNEL_USER_OP : 0 );
546        irc_channel_user_set_mode( ic, ic->irc->user,
547                ( strcmp( value, "both" ) == 0 ||
548                  strcmp( value, "user" ) == 0 ) ? IRC_CHANNEL_USER_OP : 0 );
549}
550
551char *set_eval_irc_channel_ops( set_t *set, char *value )
552{
553        irc_t *irc = set->data;
554        GSList *l;
555       
556        if( strcmp( value, "both" ) != 0 && strcmp( value, "none" ) != 0 && 
557            strcmp( value, "user" ) != 0 && strcmp( value, "root" ) != 0 )
558                return SET_INVALID;
559       
560        for( l = irc->channels; l; l = l->next )
561                irc_channel_update_ops( l->data, value );
562       
563        return value;
564}
565
[280c56a]566/* Channel-type dependent functions, for control channels: */
567static gboolean control_channel_privmsg( irc_channel_t *ic, const char *msg )
568{
[bce78c8]569        irc_t *irc = ic->irc;
[f7ca587]570        irc_user_t *iu;
[bce78c8]571        const char *s;
[fb117aee]572       
[bce78c8]573        /* Scan for non-whitespace chars followed by a colon: */
[b0364dc]574        for( s = msg; *s && !isspace( *s ) && *s != ':' && *s != ','; s ++ ) {}
[74f1cde]575       
[b0364dc]576        if( *s == ':' || *s == ',' )
[bce78c8]577        {
578                char to[s-msg+1];
579               
[1a3ba05]580                memset( to, 0, sizeof( to ) );
[bce78c8]581                strncpy( to, msg, s - msg );
582                while( *(++s) && isspace( *s ) ) {}
[f7ca587]583                msg = s;
[bce78c8]584               
[f7ca587]585                if( !( iu = irc_user_by_name( irc, to ) ) )
[9893da3]586                        irc_channel_printf( ic, "User does not exist: %s", to );
[f7ca587]587                else
588                        ic->last_target = iu;
[bce78c8]589        }
[f7ca587]590        else if( g_strcasecmp( set_getstr( &irc->b->set, "default_target" ), "last" ) == 0 &&
591                 ic->last_target && g_slist_find( irc->users, ic->last_target ) )
592                iu = ic->last_target;
[bce78c8]593        else
[f7ca587]594                iu = irc->root;
595       
596        if( iu && iu->f->privmsg )
[bce78c8]597        {
[f7ca587]598                iu->last_channel = ic;
599                iu->f->privmsg( iu, msg );
[bce78c8]600        }
[280c56a]601       
602        return TRUE;
603}
604
[46d215d]605static gboolean control_channel_invite( irc_channel_t *ic, irc_user_t *iu )
606{
607        struct irc_control_channel *icc = ic->data;
608        bee_user_t *bu = iu->bu;
609       
610        if( bu == NULL )
611                return FALSE;
612       
613        if( icc->type != IRC_CC_TYPE_GROUP )
614        {
615                irc_send_num( ic->irc, 482, "%s :Invitations are only possible to fill_by=group channels", ic->name );
616                return FALSE;
617        }
618       
619        bu->ic->acc->prpl->add_buddy( bu->ic, bu->handle,
620                                      icc->group ? icc->group->name : NULL );
621       
622        return TRUE;
623}
624
[2b8473c]625static char *set_eval_by_account( set_t *set, char *value );
626static char *set_eval_fill_by( set_t *set, char *value );
627static char *set_eval_by_group( set_t *set, char *value );
[7a6ba50]628static char *set_eval_by_protocol( set_t *set, char *value );
[94d5da9c]629static char *set_eval_show_users( set_t *set, char *value );
[2b8473c]630
[9ac3ed1]631static gboolean control_channel_init( irc_channel_t *ic )
632{
633        struct irc_control_channel *icc;
634       
[2b8473c]635        set_add( &ic->set, "account", NULL, set_eval_by_account, ic );
636        set_add( &ic->set, "fill_by", "all", set_eval_fill_by, ic );
637        set_add( &ic->set, "group", NULL, set_eval_by_group, ic );
[7a6ba50]638        set_add( &ic->set, "protocol", NULL, set_eval_by_protocol, ic );
[5a61bf59]639       
640        /* When changing the default, also change it below. */
[94d5da9c]641        set_add( &ic->set, "show_users", "online+,away", set_eval_show_users, ic );
[2b8473c]642       
[9ac3ed1]643        ic->data = icc = g_new0( struct irc_control_channel, 1 );
644        icc->type = IRC_CC_TYPE_DEFAULT;
645       
[94d5da9c]646        /* Have to run the evaluator to initialize icc->modes. */
[5a61bf59]647        set_setstr( &ic->set, "show_users", "online+,away" );
[94d5da9c]648       
[65016a6]649        /* For scripts that care. */
650        irc_channel_set_mode( ic, "+C" );
651       
[2b8473c]652        return TRUE;
653}
654
[c8eeadd]655static gboolean control_channel_join( irc_channel_t *ic )
656{
657        bee_irc_channel_update( ic->irc, ic, NULL );
658       
659        return TRUE;
660}
661
[2b8473c]662static char *set_eval_by_account( set_t *set, char *value )
663{
664        struct irc_channel *ic = set->data;
665        struct irc_control_channel *icc = ic->data;
666        account_t *acc;
667       
668        if( !( acc = account_get( ic->irc->b, value ) ) )
669                return SET_INVALID;
670       
671        icc->account = acc;
[cf1a979]672        if( icc->type == IRC_CC_TYPE_ACCOUNT )
673                bee_irc_channel_update( ic->irc, ic, NULL );
[7a6ba50]674       
[e135cd09]675        return g_strdup( acc->tag );
[2b8473c]676}
677
678static char *set_eval_fill_by( set_t *set, char *value )
679{
680        struct irc_channel *ic = set->data;
681        struct irc_control_channel *icc = ic->data;
682       
683        if( strcmp( value, "all" ) == 0 )
684                icc->type = IRC_CC_TYPE_DEFAULT;
685        else if( strcmp( value, "rest" ) == 0 )
686                icc->type = IRC_CC_TYPE_REST;
687        else if( strcmp( value, "group" ) == 0 )
[13c1a9f]688                icc->type = IRC_CC_TYPE_GROUP;
[2b8473c]689        else if( strcmp( value, "account" ) == 0 )
[a067771]690                icc->type = IRC_CC_TYPE_ACCOUNT;
[7a6ba50]691        else if( strcmp( value, "protocol" ) == 0 )
692                icc->type = IRC_CC_TYPE_PROTOCOL;
[2b8473c]693        else
694                return SET_INVALID;
695       
[cf1a979]696        bee_irc_channel_update( ic->irc, ic, NULL );
[2b8473c]697        return value;
698}
699
700static char *set_eval_by_group( set_t *set, char *value )
701{
702        struct irc_channel *ic = set->data;
703        struct irc_control_channel *icc = ic->data;
[13c1a9f]704       
[2b8473c]705        icc->group = bee_group_by_name( ic->irc->b, value, TRUE );
[cf1a979]706        if( icc->type == IRC_CC_TYPE_GROUP )
707                bee_irc_channel_update( ic->irc, ic, NULL );
[7a6ba50]708       
[2b8473c]709        return g_strdup( icc->group->name );
710}
711
[7a6ba50]712static char *set_eval_by_protocol( set_t *set, char *value )
713{
714        struct irc_channel *ic = set->data;
715        struct irc_control_channel *icc = ic->data;
716        struct prpl *prpl;
717       
718        if( !( prpl = find_protocol( value ) ) )
719                return SET_INVALID;
720       
721        icc->protocol = prpl;
722        if( icc->type == IRC_CC_TYPE_PROTOCOL )
723                bee_irc_channel_update( ic->irc, ic, NULL );
724       
725        return value;
726}
727
[94d5da9c]728static char *set_eval_show_users( set_t *set, char *value )
729{
730        struct irc_channel *ic = set->data;
731        struct irc_control_channel *icc = ic->data;
732        char **parts = g_strsplit( value, ",", 0 ), **part;
733        char modes[4];
734       
735        memset( modes, 0, 4 );
736        for( part = parts; *part; part ++ )
737        {
738                char last, modechar = IRC_CHANNEL_USER_NONE;
739               
740                if( **part == '\0' )
741                        goto fail;
742               
743                last = (*part)[strlen(*part+1)];
744                if( last == '+' )
745                        modechar = IRC_CHANNEL_USER_VOICE;
746                else if( last == '%' )
747                        modechar = IRC_CHANNEL_USER_HALFOP;
748                else if( last == '@' )
749                        modechar = IRC_CHANNEL_USER_OP;
750               
751                if( strncmp( *part, "offline", 7 ) == 0 )
752                        modes[0] = modechar;
753                else if( strncmp( *part, "away", 4 ) == 0 )
754                        modes[1] = modechar;
755                else if( strncmp( *part, "online", 6 ) == 0 )
756                        modes[2] = modechar;
757                else
758                        goto fail;
759        }
760        memcpy( icc->modes, modes, 4 );
761        bee_irc_channel_update( ic->irc, ic, NULL );
762       
763        g_strfreev( parts );
764        return value;
765       
766fail:
767        g_strfreev( parts );
768        return SET_INVALID;     
769}
770
[ac2717b]771/* Figure out if a channel is supposed to have the user, assuming s/he is
772   online or otherwise also selected by the show_users setting. Only works
773   for control channels, but does *not* check if this channel is of that
774   type. Beware! */
775gboolean irc_channel_wants_user( irc_channel_t *ic, irc_user_t *iu )
776{
777        struct irc_control_channel *icc = ic->data;
778       
779        if( iu->bu == NULL )
780                return FALSE;
781       
782        switch( icc->type )
783        {
784        case IRC_CC_TYPE_GROUP:
785                return iu->bu->group == icc->group;
786        case IRC_CC_TYPE_ACCOUNT:
787                return iu->bu->ic->acc == icc->account;
788        case IRC_CC_TYPE_PROTOCOL:
789                return iu->bu->ic->acc->prpl == icc->protocol;
790        case IRC_CC_TYPE_DEFAULT:
791        default:
792                return TRUE;
793        }
794}
795
[2b8473c]796static gboolean control_channel_free( irc_channel_t *ic )
797{
798        struct irc_control_channel *icc = ic->data;
799       
800        set_del( &ic->set, "account" );
801        set_del( &ic->set, "fill_by" );
802        set_del( &ic->set, "group" );
[9052bc1]803        set_del( &ic->set, "protocol" );
[180ab31]804        set_del( &ic->set, "show_users" );
[2b8473c]805       
806        g_free( icc );
807        ic->data = NULL;
[13c1a9f]808       
[65016a6]809        /* For scripts that care. */
810        irc_channel_set_mode( ic, "-C" );
811       
[9ac3ed1]812        return TRUE;
813}
814
[280c56a]815static const struct irc_channel_funcs control_channel_funcs = {
816        control_channel_privmsg,
[c8eeadd]817        control_channel_join,
[9ac3ed1]818        NULL,
819        NULL,
[46d215d]820        control_channel_invite,
[9ac3ed1]821       
822        control_channel_init,
[2b8473c]823        control_channel_free,
[280c56a]824};
Note: See TracBrowser for help on using the repository browser.