source: protocols/jabber/jabber.c @ eab2ac4

Last change on this file since eab2ac4 was eab2ac4, checked in by Wilmer van der Gaast <wilmer@…>, at 2006-09-25T07:42:39Z

Saner garbage collection for node cache.

  • Property mode set to 100644
File size: 8.2 KB
Line 
1/***************************************************************************\
2*                                                                           *
3*  BitlBee - An IRC to IM gateway                                           *
4*  Jabber module - Main file                                                *
5*                                                                           *
6*  Copyright 2006 Wilmer van der Gaast <wilmer@gaast.net>                   *
7*                                                                           *
8*  This program is free software; you can redistribute it and/or modify     *
9*  it under the terms of the GNU General Public License as published by     *
10*  the Free Software Foundation; either version 2 of the License, or        *
11*  (at your option) any later version.                                      *
12*                                                                           *
13*  This program is distributed in the hope that it will be useful,          *
14*  but WITHOUT ANY WARRANTY; without even the implied warranty of           *
15*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the            *
16*  GNU General Public License for more details.                             *
17*                                                                           *
18*  You should have received a copy of the GNU General Public License along  *
19*  with this program; if not, write to the Free Software Foundation, Inc.,  *
20*  51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.              *
21*                                                                           *
22\***************************************************************************/
23
24#include <glib.h>
25#include <string.h>
26#include <unistd.h>
27#include <ctype.h>
28#include <stdio.h>
29
30#include "ssl_client.h"
31#include "xmltree.h"
32#include "bitlbee.h"
33#include "jabber.h"
34
35static void jabber_acc_init( account_t *acc )
36{
37        set_t *s;
38       
39        s = set_add( &acc->set, "port", "5222", set_eval_int, acc );
40        s->flags |= ACC_SET_OFFLINE_ONLY;
41       
42        s = set_add( &acc->set, "priority", "0", set_eval_resprio, acc );
43       
44        s = set_add( &acc->set, "resource", "BitlBee", set_eval_resprio, acc );
45       
46        s = set_add( &acc->set, "server", NULL, set_eval_account, acc );
47        s->flags |= ACC_SET_NOSAVE | ACC_SET_OFFLINE_ONLY;
48       
49        s = set_add( &acc->set, "ssl", "false", set_eval_bool, acc );
50        s->flags |= ACC_SET_OFFLINE_ONLY;
51       
52        s = set_add( &acc->set, "tls", "try", set_eval_tls, acc );
53        s->flags |= ACC_SET_OFFLINE_ONLY;
54}
55
56static void jabber_login( account_t *acc )
57{
58        struct gaim_connection *gc = new_gaim_conn( acc );
59        struct jabber_data *jd = g_new0( struct jabber_data, 1 );
60       
61        jd->gc = gc;
62        gc->proto_data = jd;
63       
64        jd->username = g_strdup( acc->user );
65        jd->server = strchr( jd->username, '@' );
66       
67        if( jd->server == NULL )
68        {
69                hide_login_progress( gc, "Incomplete account name (format it like <username@jabberserver.name>)" );
70                signoff( gc );
71                return;
72        }
73       
74        /* So don't think of free()ing jd->server.. :-) */
75        *jd->server = 0;
76        jd->server ++;
77       
78        jd->node_cache = xt_new_node( "cache", NULL, NULL );
79       
80        if( set_getbool( &acc->set, "ssl" ) )
81        {
82                jd->ssl = ssl_connect( jd->server, set_getint( &acc->set, "port" ), jabber_connected_ssl, gc );
83                jd->fd = ssl_getfd( jd->ssl );
84        }
85        else
86        {
87                jd->fd = proxy_connect( jd->server, set_getint( &acc->set, "port" ), jabber_connected_plain, gc );
88        }
89}
90
91static void jabber_close( struct gaim_connection *gc )
92{
93        struct jabber_data *jd = gc->proto_data;
94       
95        jabber_end_stream( gc );
96       
97        if( jd->r_inpa >= 0 )
98                b_event_remove( jd->r_inpa );
99        if( jd->w_inpa >= 0 )
100                b_event_remove( jd->w_inpa );
101       
102        if( jd->ssl )
103                ssl_disconnect( jd->ssl );
104        if( jd->fd >= 0 )
105                closesocket( jd->fd );
106       
107        if( jd->tx_len )
108                g_free( jd->txq );
109       
110        xt_free_node( jd->node_cache );
111        xt_free( jd->xt );
112       
113        g_free( jd->away_message );
114        g_free( jd->username );
115        g_free( jd );
116}
117
118static int jabber_send_im( struct gaim_connection *gc, char *who, char *message, int len, int away )
119{
120        struct xt_node *node, *event;
121        int st;
122       
123        /*
124        event = xt_new_node( "active", NULL, NULL );
125        xt_add_attr( event, "xlmns", "http://jabber.org/protocol/chatstates" );
126       
127        event = xt_new_node( "x", NULL, xt_new_node( "composing", NULL, NULL ) );
128        xt_add_attr( event, "xmlns", "jabber:x:event" );
129        */
130       
131        node = xt_new_node( "body", message, NULL );
132        node = jabber_make_packet( "message", "chat", who, node );
133        xt_add_child( node, event );
134        st = jabber_write_packet( gc, node );
135        xt_free_node( node );
136       
137        return st;
138}
139
140static GList *jabber_away_states( struct gaim_connection *gc )
141{
142        static GList *l = NULL;
143        int i;
144       
145        if( l == NULL )
146                for( i = 0; jabber_away_state_list[i].full_name; i ++ )
147                        l = g_list_append( l, (void*) jabber_away_state_list[i].full_name );
148       
149        return l;
150}
151
152static void jabber_set_away( struct gaim_connection *gc, char *state_txt, char *message )
153{
154        struct jabber_data *jd = gc->proto_data;
155        struct jabber_away_state *state;
156       
157        /* Save all this info. We need it, for example, when changing the priority setting. */
158        state = (void *) jabber_away_state_by_name( state_txt );
159        jd->away_state = state ? state : (void *) jabber_away_state_list; /* Fall back to "Away" if necessary. */
160        g_free( jd->away_message );
161        jd->away_message = ( message && *message ) ? g_strdup( message ) : NULL;
162       
163        presence_send_update( gc );
164}
165
166static void jabber_add_buddy( struct gaim_connection *gc, char *who )
167{
168        if( jabber_add_to_roster( gc, who, NULL ) )
169                presence_send_request( gc, who, "subscribe" );
170}
171
172static void jabber_remove_buddy( struct gaim_connection *gc, char *who, char *group )
173{
174        if( jabber_remove_from_roster( gc, who ) )
175                presence_send_request( gc, who, "unsubscribe" );
176}
177
178static void jabber_keepalive( struct gaim_connection *gc )
179{
180        struct jabber_data *jd = gc->proto_data;
181        struct xt_node *c, *tmp;
182       
183        /* Just any whitespace character is enough as a keepalive for XMPP sessions. */
184        jabber_write( gc, "\n", 1 );
185       
186        /* Let's abuse this keepalive for garbage collection of the node cache too.
187           It runs every minute, so let's mark every node with a special flag the
188           first time we see it, and clean it up the second time (clean up all
189           packets with the flag set).
190           
191           node->flags is normally only used by xmltree itself for parsing/handling,
192           so it should be safe to use the variable for gc. */
193       
194        /* This horrible loop is explained in xmltree.c. Makes me wonder if maybe I
195           didn't choose the perfect data structure... */
196        for( c = jd->node_cache->children; c; c =  c->next )
197                if( !( c->flags & XT_SEEN ) )
198                        break;
199       
200        /* Now c points at the first unflagged node (or at NULL). Clean up
201           everything until that point. */
202        while( jd->node_cache->children != c )
203        {
204                /*
205                printf( "Cleaning up:\n" );
206                xt_print( jd->node_cache->children );
207                */
208               
209                tmp = jd->node_cache->children->next;
210                xt_free_node( jd->node_cache->children );
211                jd->node_cache->children = tmp;
212        }
213       
214        /* Now flag the ones that were still unflagged. */
215        for( c = jd->node_cache->children; c; c = c->next )
216        {
217                /*
218                printf( "Flagged:\n" );
219                xt_print( c );
220                */
221               
222                c->flags |= XT_SEEN;
223        }
224}
225
226static void jabber_add_permit( struct gaim_connection *gc, char *who )
227{
228        presence_send_request( gc, who, "subscribed" );
229}
230
231static void jabber_rem_permit( struct gaim_connection *gc, char *who )
232{
233        presence_send_request( gc, who, "unsubscribed" );
234}
235
236static void jabber_add_deny( struct gaim_connection *gc, char *who )
237{
238}
239
240static void jabber_rem_deny( struct gaim_connection *gc, char *who )
241{
242}
243
244void jabber_init()
245{
246        struct prpl *ret = g_new0( struct prpl, 1 );
247       
248        ret->name = "jabber";
249        ret->login = jabber_login;
250        ret->acc_init = jabber_acc_init;
251        ret->close = jabber_close;
252        ret->send_im = jabber_send_im;
253        ret->away_states = jabber_away_states;
254//      ret->get_status_string = jabber_get_status_string;
255        ret->set_away = jabber_set_away;
256//      ret->set_info = jabber_set_info;
257//      ret->get_info = jabber_get_info;
258        ret->add_buddy = jabber_add_buddy;
259        ret->remove_buddy = jabber_remove_buddy;
260//      ret->chat_send = jabber_chat_send;
261//      ret->chat_invite = jabber_chat_invite;
262//      ret->chat_leave = jabber_chat_leave;
263//      ret->chat_open = jabber_chat_open;
264        ret->keepalive = jabber_keepalive;
265        ret->add_permit = jabber_add_permit;
266        ret->rem_permit = jabber_rem_permit;
267        ret->add_deny = jabber_add_deny;
268        ret->rem_deny = jabber_rem_deny;
269//      ret->send_typing = jabber_send_typing;
270        ret->handle_cmp = g_strcasecmp;
271
272        register_protocol( ret );
273}
Note: See TracBrowser for help on using the repository browser.