source: irc_channel.c @ e1f3f94

Last change on this file since e1f3f94 was 06f9548, checked in by Wilmer van der Gaast <wilmer@…>, at 2010-06-30T23:30:27Z

Disable the code added in the previous change during shutdown, since it's
pointless at that stage and may cause crashes.

  • Property mode set to 100644
File size: 14.4 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       
[2b8473c]50        set_add( &ic->set, "type", "control", set_eval_channel_type, ic );
51       
[eb37735]52        if( name[0] == '&' )
[2b8473c]53                set_setstr( &ic->set, "type", "control" );
[eb37735]54        else /* if( name[0] == '#' ) */
[2b8473c]55                set_setstr( &ic->set, "type", "chat" );
[9ac3ed1]56       
[4be8239]57        return ic;
58}
59
[b9e020a]60irc_channel_t *irc_channel_by_name( irc_t *irc, const char *name )
61{
62        GSList *l;
63       
64        for( l = irc->channels; l; l = l->next )
65        {
66                irc_channel_t *ic = l->data;
67               
[6b90431]68                if( irc_channel_name_cmp( name, ic->name ) == 0 )
[b9e020a]69                        return ic;
70        }
71       
72        return NULL;
73}
74
[36562b0]75irc_channel_t *irc_channel_get( irc_t *irc, char *id )
76{
77        irc_channel_t *ic, *ret = NULL;
78        GSList *l;
79        int nr;
80       
81        if( sscanf( id, "%d", &nr ) == 1 && nr < 1000 )
82        {
83                for( l = irc->channels; l; l = l->next )
84                {
85                        ic = l->data;
86                        if( ( nr-- ) == 0 ) 
87                                return ic;
88                }
89               
90                return NULL;
91        }
92       
93        /* Exact match first: Partial match only sucks if there's a channel
94           #aa and #aabb */
95        if( ( ret = irc_channel_by_name( irc, id ) ) )
96                return ret;
97       
98        for( l = irc->channels; l; l = l->next )
99        {
100                ic = l->data;
101               
102                if( strstr( ic->name, id ) )
103                {
104                        /* Make sure it's a unique match. */
105                        if( !ret )
106                                ret = ic;
107                        else
108                                return NULL;
109                }
110        }
111       
112        return ret;
113}
114
[63a520b]115int irc_channel_free( irc_channel_t *ic )
116{
117        irc_t *irc = ic->irc;
118       
119        if( ic->flags & IRC_CHANNEL_JOINED )
[18da20b]120                irc_channel_del_user( ic, irc->user, FALSE, "Cleaning up channel" );
[63a520b]121       
[d7db346]122        if( ic->f->_free )
123                ic->f->_free( ic );
124       
125        while( ic->set )
126                set_del( &ic->set, ic->set->key );
127       
[63a520b]128        irc->channels = g_slist_remove( irc->channels, ic );
[e54112f]129        while( ic->users )
130        {
131                g_free( ic->users->data );
132                ic->users = g_slist_remove( ic->users, ic->users->data );
133        }
[63a520b]134       
135        g_free( ic->name );
136        g_free( ic->topic );
[d7db346]137        g_free( ic->topic_who );
[63a520b]138        g_free( ic );
139       
140        return 1;
141}
142
[ab6006c]143struct irc_channel_free_data
144{
145        irc_t *irc;
146        irc_channel_t *ic;
147        char *name;
148};
149
150static gboolean irc_channel_free_callback( gpointer data, gint fd, b_input_condition cond )
151{
152        struct irc_channel_free_data *d = data;
153       
154        if( g_slist_find( irc_connection_list, d->irc ) &&
155            irc_channel_by_name( d->irc, d->name ) == d->ic &&
156            !( d->ic->flags & IRC_CHANNEL_JOINED ) )
157                irc_channel_free( d->ic );
158
159        g_free( d->name );
160        g_free( d );
161        return FALSE;
162}
163
164/* Free the channel, but via the event loop, so after finishing whatever event
165   we're currently handling. */
166void irc_channel_free_soon( irc_channel_t *ic )
167{
168        struct irc_channel_free_data *d = g_new0( struct irc_channel_free_data, 1 );
169       
170        d->irc = ic->irc;
171        d->ic = ic;
172        d->name = g_strdup( ic->name );
173       
174        b_timeout_add( 0, irc_channel_free_callback, d );
175}
176
[2b8473c]177static char *set_eval_channel_type( set_t *set, char *value )
178{
179        struct irc_channel *ic = set->data;
180        const struct irc_channel_funcs *new;
181       
182        if( strcmp( value, "control" ) == 0 )
183                new = &control_channel_funcs;
184        else if( strcmp( value, "chat" ) == 0 )
[5a75d15]185                new = &irc_channel_im_chat_funcs;
[2b8473c]186        else
187                return SET_INVALID;
188       
189        /* TODO: Return values. */
190        if( ic->f && ic->f->_free )
191                ic->f->_free( ic );
192       
193        ic->f = new;
194       
195        if( ic->f && ic->f->_init )
196                ic->f->_init( ic );
197       
198        return value;
199}
200
[4be8239]201int irc_channel_add_user( irc_channel_t *ic, irc_user_t *iu )
202{
[e54112f]203        irc_channel_user_t *icu;
204       
[57c96f7]205        if( irc_channel_has_user( ic, iu ) )
[4be8239]206                return 0;
207       
[e54112f]208        icu = g_new0( irc_channel_user_t, 1 );
209        icu->iu = iu;
210       
211        ic->users = g_slist_insert_sorted( ic->users, icu, irc_channel_user_cmp );
[4be8239]212       
[c5aefa4]213        irc_channel_update_ops( ic, set_getstr( &ic->irc->b->set, "ops" ) );
214       
[4be8239]215        if( iu == ic->irc->user || ic->flags & IRC_CHANNEL_JOINED )
216        {
217                ic->flags |= IRC_CHANNEL_JOINED;
218                irc_send_join( ic, iu );
219        }
220       
221        return 1;
222}
223
[18da20b]224int irc_channel_del_user( irc_channel_t *ic, irc_user_t *iu, gboolean silent, const char *msg )
[4be8239]225{
[e54112f]226        irc_channel_user_t *icu;
227       
228        if( !( icu = irc_channel_has_user( ic, iu ) ) )
[4be8239]229                return 0;
230       
[e54112f]231        ic->users = g_slist_remove( ic->users, icu );
232        g_free( icu );
[4be8239]233       
[18da20b]234        if( ic->flags & IRC_CHANNEL_JOINED && !silent )
235                irc_send_part( ic, iu, msg );
[4be8239]236       
237        if( iu == ic->irc->user )
[1c40aa7]238        {
[4be8239]239                ic->flags &= ~IRC_CHANNEL_JOINED;
[1c40aa7]240               
[06f9548]241                if( ic->irc->status & USTATUS_SHUTDOWN )
242                {
243                        /* Don't do anything fancy when we're shutting down anyway. */
244                }
245                else if( ic->flags & IRC_CHANNEL_TEMP )
246                {
[ab6006c]247                        irc_channel_free_soon( ic );
[06f9548]248                }
[9052bc1]249                else
250                {
251                        /* Flush userlist now. The user won't see it anyway. */
252                        while( ic->users )
253                        {
254                                g_free( ic->users->data );
255                                ic->users = g_slist_remove( ic->users, ic->users->data );
256                        }
257                        irc_channel_add_user( ic, ic->irc->root );
258                }
[1c40aa7]259        }
[4be8239]260       
261        return 1;
262}
263
[e54112f]264irc_channel_user_t *irc_channel_has_user( irc_channel_t *ic, irc_user_t *iu )
[0b5cc72]265{
[e54112f]266        GSList *l;
267       
268        for( l = ic->users; l; l = l->next )
269        {
270                irc_channel_user_t *icu = l->data;
271               
272                if( icu->iu == iu )
273                        return icu;
274        }
275       
276        return NULL;
[0b5cc72]277}
278
[83e92bf]279int irc_channel_set_topic( irc_channel_t *ic, const char *topic, const irc_user_t *iu )
[4be8239]280{
281        g_free( ic->topic );
282        ic->topic = g_strdup( topic );
283       
[83e92bf]284        g_free( ic->topic_who );
285        if( iu )
286                ic->topic_who = g_strdup_printf( "%s!%s@%s", iu->nick, iu->user, iu->host );
287        else
288                ic->topic_who = NULL;
289       
290        ic->topic_time = time( NULL );
291       
[4be8239]292        if( ic->flags & IRC_CHANNEL_JOINED )
[83e92bf]293                irc_send_topic( ic, TRUE );
[4be8239]294       
295        return 1;
296}
[b919363]297
[6a9d068]298void irc_channel_user_set_mode( irc_channel_t *ic, irc_user_t *iu, irc_channel_user_flags_t flags )
299{
300        irc_channel_user_t *icu = irc_channel_has_user( ic, iu );
301       
[c5aefa4]302        if( !icu || icu->flags == flags )
[6a9d068]303                return;
304       
305        if( ic->flags & IRC_CHANNEL_JOINED )
306                irc_send_channel_user_mode_diff( ic, iu, icu->flags, flags );
307       
308        icu->flags = flags;
309}
310
[9893da3]311void irc_channel_printf( irc_channel_t *ic, char *format, ... )
312{
313        va_list params;
314        char *text;
315       
316        va_start( params, format );
317        text = g_strdup_vprintf( format, params );
318        va_end( params );
319       
320        irc_send_msg( ic->irc->root, "PRIVMSG", ic->name, text, NULL );
321        g_free( text );
322}
323
[6b90431]324gboolean irc_channel_name_ok( const char *name_ )
[b919363]325{
[6b90431]326        const unsigned char *name = (unsigned char*) name_;
327        int i;
[4c17d19]328       
[a670aeb]329        if( name_[0] == '\0' )
330                return FALSE;
331       
[4c17d19]332        /* Check if the first character is in CTYPES (#&) */
[6b90431]333        if( strchr( CTYPES, name_[0] ) == NULL )
[4c17d19]334                return FALSE;
335       
[6b90431]336        /* RFC 1459 keeps amazing me: While only a "few" chars are allowed
337           in nicknames, channel names can be pretty much anything as long
338           as they start with # or &. I'll be a little bit more strict and
339           disallow all non-printable characters. */
340        for( i = 1; name[i]; i ++ )
341                if( name[i] <= ' ' || name[i] == ',' )
342                        return FALSE;
343       
344        return TRUE;
345}
346
[134a02c]347void irc_channel_name_strip( char *name )
348{
349        int i, j;
350       
351        for( i = j = 0; name[i]; i ++ )
352                if( name[i] > ' ' && name[i] != ',' )
353                        name[j++] = name[i];
354       
355        name[j] = '\0';
356}
357
[6b90431]358int irc_channel_name_cmp( const char *a_, const char *b_ )
359{
360        static unsigned char case_map[256];
361        const unsigned char *a = (unsigned char*) a_, *b = (unsigned char*) b_;
362        int i;
363       
364        if( case_map['A'] == '\0' )
365        {
366                for( i = 33; i < 256; i ++ )
367                        if( i != ',' )
368                                case_map[i] = i;
369               
370                for( i = 0; i < 26; i ++ )
371                        case_map['A'+i] = 'a' + i;
372               
373                case_map['['] = '{';
374                case_map[']'] = '}';
375                case_map['~'] = '`';
376                case_map['\\'] = '|';
377        }
378       
379        if( !irc_channel_name_ok( a_ ) || !irc_channel_name_ok( b_ ) )
380                return -1;
381       
382        for( i = 0; a[i] && b[i] && case_map[a[i]] && case_map[b[i]]; i ++ )
383        {
384                if( case_map[a[i]] == case_map[b[i]] )
385                        continue;
386                else
387                        return case_map[a[i]] - case_map[b[i]];
388        }
389       
390        return case_map[a[i]] - case_map[b[i]];
[b919363]391}
[280c56a]392
[e54112f]393static gint irc_channel_user_cmp( gconstpointer a_, gconstpointer b_ )
394{
395        const irc_channel_user_t *a = a_, *b = b_;
396       
397        return irc_user_cmp( a->iu, b->iu );
398}
399
[c5aefa4]400void irc_channel_update_ops( irc_channel_t *ic, char *value )
401{
402        irc_channel_user_set_mode( ic, ic->irc->root,
403                ( strcmp( value, "both" ) == 0 ||
404                  strcmp( value, "root" ) == 0 ) ? IRC_CHANNEL_USER_OP : 0 );
405        irc_channel_user_set_mode( ic, ic->irc->user,
406                ( strcmp( value, "both" ) == 0 ||
407                  strcmp( value, "user" ) == 0 ) ? IRC_CHANNEL_USER_OP : 0 );
408}
409
410char *set_eval_irc_channel_ops( set_t *set, char *value )
411{
412        irc_t *irc = set->data;
413        GSList *l;
414       
415        if( strcmp( value, "both" ) != 0 && strcmp( value, "none" ) != 0 && 
416            strcmp( value, "user" ) != 0 && strcmp( value, "root" ) != 0 )
417                return SET_INVALID;
418       
419        for( l = irc->channels; l; l = l->next )
420                irc_channel_update_ops( l->data, value );
421       
422        return value;
423}
424
[280c56a]425/* Channel-type dependent functions, for control channels: */
426static gboolean control_channel_privmsg( irc_channel_t *ic, const char *msg )
427{
[bce78c8]428        irc_t *irc = ic->irc;
429        const char *s;
[fb117aee]430       
[bce78c8]431        /* Scan for non-whitespace chars followed by a colon: */
[b0364dc]432        for( s = msg; *s && !isspace( *s ) && *s != ':' && *s != ','; s ++ ) {}
[74f1cde]433       
[b0364dc]434        if( *s == ':' || *s == ',' )
[bce78c8]435        {
436                char to[s-msg+1];
437                irc_user_t *iu;
438               
[1a3ba05]439                memset( to, 0, sizeof( to ) );
[bce78c8]440                strncpy( to, msg, s - msg );
441                while( *(++s) && isspace( *s ) ) {}
442               
443                iu = irc_user_by_name( irc, to );
444                if( iu && iu->f->privmsg )
445                {
[92c8d41]446                        iu->last_channel = ic;
[bce78c8]447                        iu->f->privmsg( iu, s );
448                }
[1a3ba05]449                else
450                {
[9893da3]451                        irc_channel_printf( ic, "User does not exist: %s", to );
[1a3ba05]452                }
[bce78c8]453        }
454        else
455        {
456                /* TODO: Maybe just use root->privmsg here now? */
457                char cmd[strlen(msg)+1];
458               
459                g_free( ic->irc->last_root_cmd );
460                ic->irc->last_root_cmd = g_strdup( ic->name );
461               
462                strcpy( cmd, msg );
463                root_command_string( ic->irc, cmd );
464        }
[280c56a]465       
466        return TRUE;
467}
468
[46d215d]469static gboolean control_channel_invite( irc_channel_t *ic, irc_user_t *iu )
470{
471        struct irc_control_channel *icc = ic->data;
472        bee_user_t *bu = iu->bu;
473       
474        if( bu == NULL )
475                return FALSE;
476       
477        if( icc->type != IRC_CC_TYPE_GROUP )
478        {
479                irc_send_num( ic->irc, 482, "%s :Invitations are only possible to fill_by=group channels", ic->name );
480                return FALSE;
481        }
482       
483        bu->ic->acc->prpl->add_buddy( bu->ic, bu->handle,
484                                      icc->group ? icc->group->name : NULL );
485       
486        return TRUE;
487}
488
[2b8473c]489static char *set_eval_by_account( set_t *set, char *value );
490static char *set_eval_fill_by( set_t *set, char *value );
491static char *set_eval_by_group( set_t *set, char *value );
[7a6ba50]492static char *set_eval_by_protocol( set_t *set, char *value );
[2b8473c]493
[9ac3ed1]494static gboolean control_channel_init( irc_channel_t *ic )
495{
496        struct irc_control_channel *icc;
497       
[2b8473c]498        set_add( &ic->set, "account", NULL, set_eval_by_account, ic );
499        set_add( &ic->set, "fill_by", "all", set_eval_fill_by, ic );
500        set_add( &ic->set, "group", NULL, set_eval_by_group, ic );
[7a6ba50]501        set_add( &ic->set, "protocol", NULL, set_eval_by_protocol, ic );
[2b8473c]502       
[9ac3ed1]503        ic->data = icc = g_new0( struct irc_control_channel, 1 );
504        icc->type = IRC_CC_TYPE_DEFAULT;
505       
[2b8473c]506        if( bee_group_by_name( ic->irc->b, ic->name + 1, FALSE ) )
507        {
508                set_setstr( &ic->set, "group", ic->name + 1 );
509                set_setstr( &ic->set, "fill_by", "group" );
510        }
[217bf4e]511        else if( set_setstr( &ic->set, "protocol", ic->name + 1 ) )
512        {
513                set_setstr( &ic->set, "fill_by", "protocol" );
514        }
[2b8473c]515        else if( set_setstr( &ic->set, "account", ic->name + 1 ) )
516        {
517                set_setstr( &ic->set, "fill_by", "account" );
518        }
[cf1a979]519        else
520        {
521                bee_irc_channel_update( ic->irc, ic, NULL );
522        }
[2b8473c]523       
524        return TRUE;
525}
526
527static char *set_eval_by_account( set_t *set, char *value )
528{
529        struct irc_channel *ic = set->data;
530        struct irc_control_channel *icc = ic->data;
531        account_t *acc;
532       
533        if( !( acc = account_get( ic->irc->b, value ) ) )
534                return SET_INVALID;
535       
536        icc->account = acc;
[cf1a979]537        if( icc->type == IRC_CC_TYPE_ACCOUNT )
538                bee_irc_channel_update( ic->irc, ic, NULL );
[7a6ba50]539       
[2b8473c]540        return g_strdup_printf( "%s(%s)", acc->prpl->name, acc->user );
541}
542
543static char *set_eval_fill_by( set_t *set, char *value )
544{
545        struct irc_channel *ic = set->data;
546        struct irc_control_channel *icc = ic->data;
547       
548        if( strcmp( value, "all" ) == 0 )
549                icc->type = IRC_CC_TYPE_DEFAULT;
550        else if( strcmp( value, "rest" ) == 0 )
551                icc->type = IRC_CC_TYPE_REST;
552        else if( strcmp( value, "group" ) == 0 )
[13c1a9f]553                icc->type = IRC_CC_TYPE_GROUP;
[2b8473c]554        else if( strcmp( value, "account" ) == 0 )
[a067771]555                icc->type = IRC_CC_TYPE_ACCOUNT;
[7a6ba50]556        else if( strcmp( value, "protocol" ) == 0 )
557                icc->type = IRC_CC_TYPE_PROTOCOL;
[2b8473c]558        else
559                return SET_INVALID;
560       
[cf1a979]561        bee_irc_channel_update( ic->irc, ic, NULL );
[2b8473c]562        return value;
563}
564
565static char *set_eval_by_group( set_t *set, char *value )
566{
567        struct irc_channel *ic = set->data;
568        struct irc_control_channel *icc = ic->data;
[13c1a9f]569       
[2b8473c]570        icc->group = bee_group_by_name( ic->irc->b, value, TRUE );
[cf1a979]571        if( icc->type == IRC_CC_TYPE_GROUP )
572                bee_irc_channel_update( ic->irc, ic, NULL );
[7a6ba50]573       
[2b8473c]574        return g_strdup( icc->group->name );
575}
576
[7a6ba50]577static char *set_eval_by_protocol( set_t *set, char *value )
578{
579        struct irc_channel *ic = set->data;
580        struct irc_control_channel *icc = ic->data;
581        struct prpl *prpl;
582       
583        if( !( prpl = find_protocol( value ) ) )
584                return SET_INVALID;
585       
586        icc->protocol = prpl;
587        if( icc->type == IRC_CC_TYPE_PROTOCOL )
588                bee_irc_channel_update( ic->irc, ic, NULL );
589       
590        return value;
591}
592
[2b8473c]593static gboolean control_channel_free( irc_channel_t *ic )
594{
595        struct irc_control_channel *icc = ic->data;
596       
597        set_del( &ic->set, "account" );
598        set_del( &ic->set, "fill_by" );
599        set_del( &ic->set, "group" );
[9052bc1]600        set_del( &ic->set, "protocol" );
[2b8473c]601       
602        g_free( icc );
603        ic->data = NULL;
[13c1a9f]604       
[9ac3ed1]605        return TRUE;
606}
607
[280c56a]608static const struct irc_channel_funcs control_channel_funcs = {
609        control_channel_privmsg,
[9ac3ed1]610        NULL,
611        NULL,
612        NULL,
[46d215d]613        control_channel_invite,
[9ac3ed1]614       
615        control_channel_init,
[2b8473c]616        control_channel_free,
[280c56a]617};
Note: See TracBrowser for help on using the repository browser.