source: irc_user.c @ 858ea01

Last change on this file since 858ea01 was 4ffd757, checked in by Wilmer van der Gaast <wilmer@…>, at 2010-08-14T10:21:44Z

Don't send a /QUIT for every Twitter contact when going offline, and show
the twitter_$username /QUIT as a netsplit.

  • Property mode set to 100644
File size: 6.9 KB
Line 
1  /********************************************************************\
2  * BitlBee -- An IRC to other IM-networks gateway                     *
3  *                                                                    *
4  * Copyright 2002-2004 Wilmer van der Gaast and others                *
5  \********************************************************************/
6
7/* Stuff to handle, save and search IRC buddies                         */
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#include "ipc.h"
28
29irc_user_t *irc_user_new( irc_t *irc, const char *nick )
30{
31        irc_user_t *iu = g_new0( irc_user_t, 1 );
32       
33        iu->irc = irc;
34        iu->nick = g_strdup( nick );
35        iu->user = iu->host = iu->fullname = iu->nick;
36       
37        if( set_getbool( &irc->b->set, "private" ) )
38                iu->last_channel = NULL;
39        else
40                iu->last_channel = irc->default_channel;
41       
42        iu->key = g_strdup( nick );
43        nick_lc( iu->key );
44        /* Using the hash table for speed and irc->users for easy iteration
45           through the list (since the GLib API doesn't have anything sane
46           for that.) */
47        g_hash_table_insert( irc->nick_user_hash, iu->key, iu );
48        irc->users = g_slist_insert_sorted( irc->users, iu, irc_user_cmp );
49       
50        return iu;
51}
52
53int irc_user_free( irc_t *irc, irc_user_t *iu )
54{
55        static struct im_connection *last_ic;
56        static char *msg;
57       
58        if( !iu )
59                return 0;
60       
61        if( iu->bu &&
62            ( iu->bu->ic->flags & OPT_LOGGING_OUT ) &&
63            iu->bu->ic != last_ic )
64        {
65                char host_prefix[] = "bitlbee.";
66                char *s;
67               
68                /* Irssi recognises netsplits by quitmsgs with two
69                   hostnames, where a hostname is a "word" with one
70                   of more dots. Mangle no-dot hostnames a bit. */
71                if( strchr( irc->root->host, '.' ) )
72                        *host_prefix = '\0';
73               
74                last_ic = iu->bu->ic;
75                g_free( msg );
76                if( !set_getbool( &irc->b->set, "simulate_netsplit" ) )
77                        msg = g_strdup( "Account off-line" );
78                else if( ( s = strchr( iu->bu->ic->acc->user, '@' ) ) )
79                        msg = g_strdup_printf( "%s%s %s", host_prefix,
80                                irc->root->host, s + 1 );
81                else
82                        msg = g_strdup_printf( "%s%s %s.%s",
83                                host_prefix, irc->root->host,
84                                iu->bu->ic->acc->prpl->name, irc->root->host );
85        }
86        else if( !iu->bu || !( iu->bu->ic->flags & OPT_LOGGING_OUT ) )
87        {
88                g_free( msg );
89                msg = g_strdup( "Removed" );
90                last_ic = NULL;
91        }
92        irc_user_quit( iu, msg );
93       
94        irc->users = g_slist_remove( irc->users, iu );
95        g_hash_table_remove( irc->nick_user_hash, iu->key );
96       
97        g_free( iu->nick );
98        if( iu->nick != iu->user ) g_free( iu->user );
99        if( iu->nick != iu->host ) g_free( iu->host );
100        if( iu->nick != iu->fullname ) g_free( iu->fullname );
101        g_free( iu->pastebuf );
102        if( iu->pastebuf_timer ) b_event_remove( iu->pastebuf_timer );
103        g_free( iu->key );
104        g_free( iu );
105       
106        return 1;
107}
108
109irc_user_t *irc_user_by_name( irc_t *irc, const char *nick )
110{
111        char key[strlen(nick)+1];
112       
113        strcpy( key, nick );
114        if( nick_lc( key ) )
115                return g_hash_table_lookup( irc->nick_user_hash, key );
116        else
117                return NULL;
118}
119
120int irc_user_set_nick( irc_user_t *iu, const char *new )
121{
122        irc_t *irc = iu->irc;
123        irc_user_t *new_iu;
124        char key[strlen(new)+1];
125        GSList *cl;
126       
127        strcpy( key, new );
128        if( iu == NULL || !nick_lc( key ) ||
129            ( ( new_iu = irc_user_by_name( irc, new ) ) && new_iu != iu ) )
130                return 0;
131       
132        for( cl = irc->channels; cl; cl = cl->next )
133        {
134                irc_channel_t *ic = cl->data;
135               
136                /* Send a NICK update if we're renaming our user, or someone
137                   who's in the same channel like our user. */
138                if( iu == irc->user ||
139                    ( ( ic->flags & IRC_CHANNEL_JOINED ) &&
140                      irc_channel_has_user( ic, iu ) ) )
141                {
142                        irc_send_nick( iu, new );
143                        break;
144                }
145        }
146       
147        irc->users = g_slist_remove( irc->users, iu );
148        g_hash_table_remove( irc->nick_user_hash, iu->key );
149       
150        if( iu->nick == iu->user ) iu->user = NULL;
151        if( iu->nick == iu->host ) iu->host = NULL;
152        if( iu->nick == iu->fullname ) iu->fullname = NULL;
153        g_free( iu->nick );
154        iu->nick = g_strdup( new );
155        if( iu->user == NULL ) iu->user = g_strdup( iu->nick );
156        if( iu->host == NULL ) iu->host = g_strdup( iu->nick );
157        if( iu->fullname == NULL ) iu->fullname = g_strdup( iu->nick );
158       
159        iu->key = g_strdup( key );
160        g_hash_table_insert( irc->nick_user_hash, iu->key, iu );
161        irc->users = g_slist_insert_sorted( irc->users, iu, irc_user_cmp );
162       
163        if( iu == irc->user )
164                ipc_to_master_str( "NICK :%s\r\n", new );
165       
166        return 1;
167}
168
169gint irc_user_cmp( gconstpointer a_, gconstpointer b_ )
170{
171        const irc_user_t *a = a_, *b = b_;
172       
173        return strcmp( a->key, b->key );
174}
175
176const char *irc_user_get_away( irc_user_t *iu )
177{
178        irc_t *irc = iu->irc;
179        bee_user_t *bu = iu->bu;
180       
181        if( iu == irc->user )
182                return set_getstr( &irc->b->set, "away" );
183        else if( bu )
184        {
185                if( !bu->flags & BEE_USER_ONLINE )
186                        return "Offline";
187                else if( bu->flags & BEE_USER_AWAY )
188                {
189                        if( bu->status_msg )
190                        {
191                                static char ret[MAX_STRING];
192                                g_snprintf( ret, MAX_STRING - 1, "%s (%s)",
193                                            bu->status ? : "Away", bu->status_msg );
194                                return ret;
195                        }
196                        else
197                                return bu->status ? : "Away";
198                }
199        }
200       
201        return NULL;
202}
203
204void irc_user_quit( irc_user_t *iu, const char *msg )
205{
206        GSList *l;
207        gboolean send_quit = FALSE;
208       
209        if( !iu )
210                return;
211       
212        for( l = iu->irc->channels; l; l = l->next )
213        {
214                irc_channel_t *ic = l->data;
215                send_quit |= irc_channel_del_user( ic, iu, IRC_CDU_SILENT, NULL ) &&
216                             ( ic->flags & IRC_CHANNEL_JOINED );
217        }
218       
219        if( send_quit )
220                irc_send_quit( iu, msg );
221}
222
223/* User-type dependent functions, for root/NickServ: */
224static gboolean root_privmsg( irc_user_t *iu, const char *msg )
225{
226        char cmd[strlen(msg)+1];
227       
228        strcpy( cmd, msg );
229        root_command_string( iu->irc, cmd );
230       
231        return TRUE;
232}
233
234static gboolean root_ctcp( irc_user_t *iu, char * const *ctcp )
235{
236        if( g_strcasecmp( ctcp[0], "VERSION" ) == 0 )
237        {
238                irc_send_msg_f( iu, "NOTICE", iu->irc->user->nick, "\001%s %s\001",
239                                ctcp[0], "BitlBee " BITLBEE_VERSION " " ARCH "/" CPU );
240        }
241        else if( g_strcasecmp( ctcp[0], "PING" ) == 0 )
242        {
243                irc_send_msg_f( iu, "NOTICE", iu->irc->user->nick, "\001%s %s\001",
244                                ctcp[0], ctcp[1] ? : "" );
245        }
246       
247        return TRUE;
248}
249
250const struct irc_user_funcs irc_user_root_funcs = {
251        root_privmsg,
252        root_ctcp,
253};
254
255/* Echo to yourself: */
256static gboolean self_privmsg( irc_user_t *iu, const char *msg )
257{
258        irc_send_msg( iu, "PRIVMSG", iu->nick, msg, NULL );
259       
260        return TRUE;
261}
262
263const struct irc_user_funcs irc_user_self_funcs = {
264        self_privmsg,
265};
Note: See TracBrowser for help on using the repository browser.