source: protocols/jabber/jabber.c @ ebe7b36

Last change on this file since ebe7b36 was ebe7b36, checked in by Wilmer van der Gaast <wilmer@…>, at 2006-09-25T10:10:14Z

Changing the resource string while online probably doesn't work.

  • Property mode set to 100644
File size: 8.4 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_priority, acc );
43       
44        s = set_add( &acc->set, "resource", "BitlBee", NULL, acc );
45        s->flags |= ACC_SET_OFFLINE_ONLY;
46       
47        s = set_add( &acc->set, "server", NULL, set_eval_account, acc );
48        s->flags |= ACC_SET_NOSAVE | ACC_SET_OFFLINE_ONLY;
49       
50        s = set_add( &acc->set, "ssl", "false", set_eval_bool, acc );
51        s->flags |= ACC_SET_OFFLINE_ONLY;
52       
53        s = set_add( &acc->set, "tls", "try", set_eval_tls, acc );
54        s->flags |= ACC_SET_OFFLINE_ONLY;
55}
56
57static void jabber_login( account_t *acc )
58{
59        struct gaim_connection *gc = new_gaim_conn( acc );
60        struct jabber_data *jd = g_new0( struct jabber_data, 1 );
61       
62        jd->gc = gc;
63        gc->proto_data = jd;
64       
65        jd->username = g_strdup( acc->user );
66        jd->server = strchr( jd->username, '@' );
67       
68        if( jd->server == NULL )
69        {
70                hide_login_progress( gc, "Incomplete account name (format it like <username@jabberserver.name>)" );
71                signoff( gc );
72                return;
73        }
74       
75        /* So don't think of free()ing jd->server.. :-) */
76        *jd->server = 0;
77        jd->server ++;
78       
79        jd->node_cache = xt_new_node( "cache", NULL, NULL );
80       
81        if( set_getbool( &acc->set, "ssl" ) )
82        {
83                jd->ssl = ssl_connect( jd->server, set_getint( &acc->set, "port" ), jabber_connected_ssl, gc );
84                jd->fd = ssl_getfd( jd->ssl );
85        }
86        else
87        {
88                jd->fd = proxy_connect( jd->server, set_getint( &acc->set, "port" ), jabber_connected_plain, gc );
89        }
90}
91
92static void jabber_close( struct gaim_connection *gc )
93{
94        struct jabber_data *jd = gc->proto_data;
95       
96        jabber_end_stream( gc );
97       
98        if( jd->r_inpa >= 0 )
99                b_event_remove( jd->r_inpa );
100        if( jd->w_inpa >= 0 )
101                b_event_remove( jd->w_inpa );
102       
103        if( jd->ssl )
104                ssl_disconnect( jd->ssl );
105        if( jd->fd >= 0 )
106                closesocket( jd->fd );
107       
108        if( jd->tx_len )
109                g_free( jd->txq );
110       
111        xt_free_node( jd->node_cache );
112        xt_free( jd->xt );
113       
114        g_free( jd->away_message );
115        g_free( jd->username );
116        g_free( jd );
117}
118
119static int jabber_send_im( struct gaim_connection *gc, char *who, char *message, int len, int away )
120{
121        struct xt_node *node, *event;
122        int st;
123       
124        /*
125        event = xt_new_node( "active", NULL, NULL );
126        xt_add_attr( event, "xlmns", "http://jabber.org/protocol/chatstates" );
127       
128        event = xt_new_node( "x", NULL, xt_new_node( "composing", NULL, NULL ) );
129        xt_add_attr( event, "xmlns", "jabber:x:event" );
130        */
131       
132        node = xt_new_node( "body", message, NULL );
133        node = jabber_make_packet( "message", "chat", who, node );
134        xt_add_child( node, event );
135        st = jabber_write_packet( gc, node );
136        xt_free_node( node );
137       
138        return st;
139}
140
141static GList *jabber_away_states( struct gaim_connection *gc )
142{
143        static GList *l = NULL;
144        int i;
145       
146        if( l == NULL )
147                for( i = 0; jabber_away_state_list[i].full_name; i ++ )
148                        l = g_list_append( l, (void*) jabber_away_state_list[i].full_name );
149       
150        return l;
151}
152
153static void jabber_set_away( struct gaim_connection *gc, char *state_txt, char *message )
154{
155        struct jabber_data *jd = gc->proto_data;
156        struct jabber_away_state *state;
157       
158        /* Save all this info. We need it, for example, when changing the priority setting. */
159        state = (void *) jabber_away_state_by_name( state_txt );
160        jd->away_state = state ? state : (void *) jabber_away_state_list; /* Fall back to "Away" if necessary. */
161        g_free( jd->away_message );
162        jd->away_message = ( message && *message ) ? g_strdup( message ) : NULL;
163       
164        presence_send_update( gc );
165}
166
167static void jabber_add_buddy( struct gaim_connection *gc, char *who )
168{
169        if( jabber_add_to_roster( gc, who, NULL ) )
170                presence_send_request( gc, who, "subscribe" );
171}
172
173static void jabber_remove_buddy( struct gaim_connection *gc, char *who, char *group )
174{
175        if( jabber_remove_from_roster( gc, who ) )
176                presence_send_request( gc, who, "unsubscribe" );
177}
178
179static void jabber_keepalive( struct gaim_connection *gc )
180{
181        struct jabber_data *jd = gc->proto_data;
182        struct xt_node *c, *tmp;
183       
184        /* Just any whitespace character is enough as a keepalive for XMPP sessions. */
185        jabber_write( gc, "\n", 1 );
186       
187        /* Let's abuse this keepalive for garbage collection of the node cache too.
188           It runs every minute, so let's mark every node with a special flag the
189           first time we see it, and clean it up the second time (clean up all
190           packets with the flag set).
191           
192           node->flags is normally only used by xmltree itself for parsing/handling,
193           so it should be safe to use the variable for gc. */
194       
195        /* This horrible loop is explained in xmltree.c. Makes me wonder if maybe I
196           didn't choose the perfect data structure... */
197        for( c = jd->node_cache->children; c; c =  c->next )
198                if( !( c->flags & XT_SEEN ) )
199                        break;
200       
201        /* Now c points at the first unflagged node (or at NULL). Clean up
202           everything until that point. */
203        while( jd->node_cache->children != c )
204        {
205                /*
206                printf( "Cleaning up:\n" );
207                xt_print( jd->node_cache->children );
208                */
209               
210                tmp = jd->node_cache->children->next;
211                xt_free_node( jd->node_cache->children );
212                jd->node_cache->children = tmp;
213        }
214       
215        /* Now flag the ones that were still unflagged. */
216        for( c = jd->node_cache->children; c; c = c->next )
217        {
218                /*
219                printf( "Flagged:\n" );
220                xt_print( c );
221                */
222               
223                c->flags |= XT_SEEN;
224        }
225}
226
227static void jabber_add_permit( struct gaim_connection *gc, char *who )
228{
229        presence_send_request( gc, who, "subscribed" );
230}
231
232static void jabber_rem_permit( struct gaim_connection *gc, char *who )
233{
234        presence_send_request( gc, who, "unsubscribed" );
235}
236
237/* XMPP doesn't have both a block- and and allow-list, so these two functions
238   will be no-ops: */
239static void jabber_add_deny( struct gaim_connection *gc, char *who )
240{
241}
242
243static void jabber_rem_deny( struct gaim_connection *gc, char *who )
244{
245}
246
247void jabber_init()
248{
249        struct prpl *ret = g_new0( struct prpl, 1 );
250       
251        ret->name = "jabber";
252        ret->login = jabber_login;
253        ret->acc_init = jabber_acc_init;
254        ret->close = jabber_close;
255        ret->send_im = jabber_send_im;
256        ret->away_states = jabber_away_states;
257//      ret->get_status_string = jabber_get_status_string;
258        ret->set_away = jabber_set_away;
259//      ret->set_info = jabber_set_info;
260//      ret->get_info = jabber_get_info;
261        ret->add_buddy = jabber_add_buddy;
262        ret->remove_buddy = jabber_remove_buddy;
263//      ret->chat_send = jabber_chat_send;
264//      ret->chat_invite = jabber_chat_invite;
265//      ret->chat_leave = jabber_chat_leave;
266//      ret->chat_open = jabber_chat_open;
267        ret->keepalive = jabber_keepalive;
268        ret->add_permit = jabber_add_permit;
269        ret->rem_permit = jabber_rem_permit;
270        ret->add_deny = jabber_add_deny;
271        ret->rem_deny = jabber_rem_deny;
272//      ret->send_typing = jabber_send_typing;
273        ret->handle_cmp = g_strcasecmp;
274
275        register_protocol( ret );
276}
Note: See TracBrowser for help on using the repository browser.