source: protocols/jabber/jabber.c @ 861c199

Last change on this file since 861c199 was 861c199, checked in by Wilmer van der Gaast <wilmer@…>, at 2006-10-08T18:41:11Z

Moved handling of all IQ packets to event handlers. Cleaned up a LOT of
mess in iq.c!

  • Property mode set to 100644
File size: 7.6 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        struct ns_srv_reply *srv = NULL;
62        char *connect_to;
63       
64        jd->gc = gc;
65        gc->proto_data = jd;
66       
67        jd->username = g_strdup( acc->user );
68        jd->server = strchr( jd->username, '@' );
69       
70        if( jd->server == NULL )
71        {
72                hide_login_progress( gc, "Incomplete account name (format it like <username@jabberserver.name>)" );
73                signoff( gc );
74                return;
75        }
76       
77        /* So don't think of free()ing jd->server.. :-) */
78        *jd->server = 0;
79        jd->server ++;
80       
81        jd->node_cache = g_hash_table_new_full( g_str_hash, g_str_equal, NULL, jabber_cache_entry_free );
82       
83        /* Figure out the hostname to connect to. */
84        if( acc->server )
85                connect_to = acc->server;
86        else if( ( srv = srv_lookup( "xmpp-client", "tcp", jd->server ) ) ||
87                 ( srv = srv_lookup( "jabber-client", "tcp", jd->server ) ) )
88                connect_to = srv->name;
89        else
90                connect_to = jd->server;
91       
92        /* For non-SSL connections we can try to use the port # from the SRV
93           reply, but let's not do that when using SSL, SSL usually runs on
94           non-standard ports... */
95        if( set_getbool( &acc->set, "ssl" ) )
96        {
97                jd->ssl = ssl_connect( connect_to, set_getint( &acc->set, "port" ), jabber_connected_ssl, gc );
98                jd->fd = ssl_getfd( jd->ssl );
99        }
100        else
101        {
102                jd->fd = proxy_connect( connect_to, srv ? srv->port : set_getint( &acc->set, "port" ), jabber_connected_plain, gc );
103        }
104       
105        g_free( srv );
106}
107
108static void jabber_close( struct gaim_connection *gc )
109{
110        struct jabber_data *jd = gc->proto_data;
111       
112        jabber_end_stream( gc );
113       
114        if( jd->r_inpa >= 0 )
115                b_event_remove( jd->r_inpa );
116        if( jd->w_inpa >= 0 )
117                b_event_remove( jd->w_inpa );
118       
119        if( jd->ssl )
120                ssl_disconnect( jd->ssl );
121        if( jd->fd >= 0 )
122                closesocket( jd->fd );
123       
124        if( jd->tx_len )
125                g_free( jd->txq );
126       
127        g_hash_table_destroy( jd->node_cache );
128       
129        xt_free( jd->xt );
130       
131        g_free( jd->away_message );
132        g_free( jd->username );
133        g_free( jd );
134}
135
136static int jabber_send_im( struct gaim_connection *gc, char *who, char *message, int len, int away )
137{
138        struct xt_node *node;
139        int st;
140       
141        /*
142        event = xt_new_node( "active", NULL, NULL );
143        xt_add_attr( event, "xlmns", "http://jabber.org/protocol/chatstates" );
144       
145        event = xt_new_node( "x", NULL, xt_new_node( "composing", NULL, NULL ) );
146        xt_add_attr( event, "xmlns", "jabber:x:event" );
147        */
148       
149        node = xt_new_node( "body", message, NULL );
150        node = jabber_make_packet( "message", "chat", who, node );
151        st = jabber_write_packet( gc, node );
152        xt_free_node( node );
153       
154        return st;
155}
156
157static GList *jabber_away_states( struct gaim_connection *gc )
158{
159        static GList *l = NULL;
160        int i;
161       
162        if( l == NULL )
163                for( i = 0; jabber_away_state_list[i].full_name; i ++ )
164                        l = g_list_append( l, (void*) jabber_away_state_list[i].full_name );
165       
166        return l;
167}
168
169static void jabber_get_info( struct gaim_connection *gc, char *who )
170{
171        struct xt_node *node;
172       
173        node = xt_new_node( "query", NULL, NULL );
174        xt_add_attr( node, "xmlns", "http://jabber.org/protocol/disco#info" );
175        node = jabber_make_packet( "iq", "get", who, node );
176        // jabber_cache_add( gc, node,  );
177       
178        jabber_write_packet( gc, node );
179}
180
181static void jabber_set_away( struct gaim_connection *gc, char *state_txt, char *message )
182{
183        struct jabber_data *jd = gc->proto_data;
184        struct jabber_away_state *state;
185       
186        /* Save all this info. We need it, for example, when changing the priority setting. */
187        state = (void *) jabber_away_state_by_name( state_txt );
188        jd->away_state = state ? state : (void *) jabber_away_state_list; /* Fall back to "Away" if necessary. */
189        g_free( jd->away_message );
190        jd->away_message = ( message && *message ) ? g_strdup( message ) : NULL;
191       
192        presence_send_update( gc );
193}
194
195static void jabber_add_buddy( struct gaim_connection *gc, char *who )
196{
197        if( jabber_add_to_roster( gc, who, NULL ) )
198                presence_send_request( gc, who, "subscribe" );
199}
200
201static void jabber_remove_buddy( struct gaim_connection *gc, char *who, char *group )
202{
203        if( jabber_remove_from_roster( gc, who ) )
204                presence_send_request( gc, who, "unsubscribe" );
205}
206
207static void jabber_keepalive( struct gaim_connection *gc )
208{
209        /* Just any whitespace character is enough as a keepalive for XMPP sessions. */
210        jabber_write( gc, "\n", 1 );
211       
212        /* This runs the garbage collection every minute, which means every packet
213           is in the cache for about a minute (which should be enough AFAIK). */
214        jabber_cache_clean( gc );
215}
216
217void jabber_init()
218{
219        struct prpl *ret = g_new0( struct prpl, 1 );
220       
221        ret->name = "jabber";
222        ret->login = jabber_login;
223        ret->acc_init = jabber_acc_init;
224        ret->close = jabber_close;
225        ret->send_im = jabber_send_im;
226        ret->away_states = jabber_away_states;
227//      ret->get_status_string = jabber_get_status_string;
228        ret->set_away = jabber_set_away;
229//      ret->set_info = jabber_set_info;
230        ret->get_info = jabber_get_info;
231        ret->add_buddy = jabber_add_buddy;
232        ret->remove_buddy = jabber_remove_buddy;
233//      ret->chat_send = jabber_chat_send;
234//      ret->chat_invite = jabber_chat_invite;
235//      ret->chat_leave = jabber_chat_leave;
236//      ret->chat_open = jabber_chat_open;
237        ret->keepalive = jabber_keepalive;
238//      ret->send_typing = jabber_send_typing;
239        ret->handle_cmp = g_strcasecmp;
240
241        register_protocol( ret );
242}
Note: See TracBrowser for help on using the repository browser.