source: protocols/jabber/jabber.h @ 7733b8c

Last change on this file since 7733b8c was 7733b8c, checked in by dequis <dx@…>, at 2015-02-21T06:10:54Z

Add the concept of jabber sub-protocols

Currently including: jabber (none), gtalk, facebook, hipchat.

They provide a default server field and an oauth service definition.

Also the "account tag guesses" become subprotocol guesses

  • Property mode set to 100644
File size: 15.5 KB
RevLine 
[f06894d]1/***************************************************************************\
2*                                                                           *
3*  BitlBee - An IRC to IM gateway                                           *
4*  Jabber module - Main file                                                *
5*                                                                           *
[0e788f5]6*  Copyright 2006-2013 Wilmer van der Gaast <wilmer@gaast.net>              *
[f06894d]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#ifndef _JABBER_H
25#define _JABBER_H
26
27#include <glib.h>
28
29#include "bitlbee.h"
[4cf80bb]30#include "md5.h"
31#include "xmltree.h"
[f06894d]32
[b5c8a34]33extern GSList *jabber_connections;
34
[5ebff60]35typedef enum {
[bb95d43]36        JFLAG_STREAM_STARTED = 1,       /* Set when we detected the beginning of the stream
[a21a8ac]37                                           and want to do auth. */
[bb95d43]38        JFLAG_AUTHENTICATED = 2,        /* Set when we're successfully authenticatd. */
39        JFLAG_STREAM_RESTART = 4,       /* Set when we want to restart the stream (after
[a21a8ac]40                                           SASL or TLS). */
[5ebff60]41        JFLAG_WANT_SESSION = 8,         /* Set if the server wants a <session/> tag
[a21a8ac]42                                           before we continue. */
[8fb1263]43        JFLAG_WANT_BIND = 16,           /* ... for <bind> tag. */
[bb95d43]44        JFLAG_WANT_TYPING = 32,         /* Set if we ever sent a typing notification, this
[788a1af]45                                           activates all XEP-85 related code. */
[bb95d43]46        JFLAG_XMLCONSOLE = 64,          /* If the user added an xmlconsole buddy. */
[af7f046]47        JFLAG_STARTTLS_DONE = 128,      /* If a plaintext session was converted to TLS. */
[bb2d198]48
49        JFLAG_GTALK =  0x100000,        /* Is Google Talk, as confirmed by iq discovery */
50
[e1c926f]51        JFLAG_SASL_FB = 0x10000,        /* Trying Facebook authentication. */
[21167d2]52} jabber_flags_t;
53
[5ebff60]54typedef enum {
[63075d7]55        JBFLAG_PROBED_XEP85 = 1,        /* Set this when we sent our probe packet to make
[a21a8ac]56                                           sure it gets sent only once. */
[63075d7]57        JBFLAG_DOES_XEP85 = 2,          /* Set this when the resource seems to support
[788a1af]58                                           XEP85 (typing notification shite). */
[63075d7]59        JBFLAG_IS_CHATROOM = 4,         /* It's convenient to use this JID thingy for
[e35d1a1]60                                           groupchat state info too. */
[63075d7]61        JBFLAG_IS_ANONYMOUS = 8,        /* For anonymous chatrooms, when we don't have
[6286f80]62                                           have a real JID. */
[31dbb90a]63        JBFLAG_HIDE_SUBJECT = 16,       /* Hide the subject field since we probably
64                                           showed it already. */
[0d3f30f]65} jabber_buddy_flags_t;
[a21a8ac]66
[1c3008a]67/* Stores a streamhost's (a.k.a. proxy) data */
[5ebff60]68typedef struct {
[dc0ba9c]69        char *jid;
70        char *host;
71        char port[6];
72} jabber_streamhost_t;
73
[5ebff60]74typedef enum {
[63075d7]75        JCFLAG_MESSAGE_SENT = 1,        /* Set this after sending the first message, so
76                                           we can detect echoes/backlogs. */
77} jabber_chat_flags_t;
78
[7733b8c]79typedef enum {
80        JSUB_NONE = 0,
81        JSUB_GTALK,
82        JSUB_FACEBOOK,
83        JSUB_HIPCHAT,
84} jabber_subproto_t;
85
86typedef struct {
87        const char *name;
88        jabber_subproto_t id;
89        const struct oauth2_service *oauth2_service;
90        const char *server;
91} jabber_subproto_desc_t;
92
[5ebff60]93struct jabber_data {
[0da65d5]94        struct im_connection *ic;
[5ebff60]95
[5e202b0]96        int fd;
97        void *ssl;
98        char *txq;
99        int tx_len;
100        int r_inpa, w_inpa;
[5ebff60]101
[5e202b0]102        struct xt_parser *xt;
103        jabber_flags_t flags;
[7733b8c]104        jabber_subproto_t subproto;
[5ebff60]105
106        char *username;         /* USERNAME@server */
107        char *server;           /* username@SERVER -=> server/domain, not hostname */
108        char *me;               /* bare jid */
[be1efa3]109        char *internal_jid;
[5ebff60]110
[18c6d36]111        const struct oauth2_service *oauth2_service;
[4a5d885]112        char *oauth2_access_token;
[5ebff60]113
[172a73f1]114        /* After changing one of these two (or the priority setting), call
115           presence_send_update() to inform the server about the changes. */
[840bba8]116        const struct jabber_away_state *away_state;
[5e202b0]117        char *away_message;
[5ebff60]118
[89d736a]119        md5_state_t cached_id_prefix;
[038d17f]120        GHashTable *node_cache;
[6a1128d]121        GHashTable *buddies;
[2c2df7d]122
123        GSList *filetransfers;
[dc0ba9c]124        GSList *streamhosts;
125        int have_streamhosts;
[5e202b0]126};
127
[5ebff60]128struct jabber_away_state {
[5e202b0]129        char code[5];
130        char *full_name;
131};
132
[5ebff60]133typedef xt_status (*jabber_cache_event) (struct im_connection *ic, struct xt_node *node, struct xt_node *orig);
[038d17f]134
[5ebff60]135struct jabber_cache_entry {
[979cfb4]136        time_t saved_at;
[038d17f]137        struct xt_node *node;
138        jabber_cache_event func;
139};
140
[76c85b4c]141/* Somewhat messy data structure: We have a hash table with the bare JID as
142   the key and the head of a struct jabber_buddy list as the value. The head
143   is always a bare JID. If the JID has other resources (often the case,
144   except for some transports that don't support multiple resources), those
145   follow. In that case, the bare JID at the beginning doesn't actually
146   refer to a real session and should only be used for operations that
147   support incomplete JIDs. */
[5ebff60]148struct jabber_buddy {
[0d3f30f]149        char *bare_jid;
[a21a8ac]150        char *full_jid;
[6a1128d]151        char *resource;
[5ebff60]152
[6286f80]153        char *ext_jid; /* The JID to use in BitlBee. The real JID if possible, */
154                       /* otherwise something similar to the conference JID. */
[5ebff60]155
[6a1128d]156        int priority;
157        struct jabber_away_state *away_state;
158        char *away_message;
[dc0ba9c]159        GSList *features;
[5ebff60]160
[76c85b4c]161        time_t last_msg;
[0d3f30f]162        jabber_buddy_flags_t flags;
[5ebff60]163
[6a1128d]164        struct jabber_buddy *next;
165};
166
[5ebff60]167struct jabber_chat {
[e35d1a1]168        int flags;
169        char *name;
[9c9b37c]170        char *my_full_jid; /* Separate copy because of case sensitivity. */
[e35d1a1]171        struct jabber_buddy *me;
[fc0640e]172        char *invite;
[e35d1a1]173};
174
[5ebff60]175struct jabber_transfer {
[2c2df7d]176        /* bitlbee's handle for this transfer */
177        file_transfer_t *ft;
178
179        /* the stream's private handle */
180        gpointer streamhandle;
181
[b5cfc2b]182        /* timeout for discover queries */
183        gint disco_timeout;
184        gint disco_timeout_fired;
185
[2c2df7d]186        struct im_connection *ic;
187
[b5cfc2b]188        struct jabber_buddy *bud;
189
[2c2df7d]190        int watch_in;
191        int watch_out;
192
193        char *ini_jid;
194        char *tgt_jid;
195        char *iq_id;
196        char *sid;
197        int accepted;
198
199        size_t bytesread, byteswritten;
200        int fd;
[2ff2076]201        struct sockaddr_storage saddr;
[2c2df7d]202};
203
[06eef80]204#define JABBER_XMLCONSOLE_HANDLE "_xmlconsole"
[34ded90]205#define JABBER_OAUTH_HANDLE "jabber_oauth"
[bb95d43]206
[dfa41a4]207/* Prefixes to use for packet IDs (mainly for IQ packets ATM). Usually the
208   first one should be used, but when storing a packet in the cache, a
209   "special" kind of ID is assigned to make it easier later to figure out
[608f8cf]210   if we have to do call an event handler for the response packet. Also
211   we'll append a hash to make sure we won't trigger on cached packets from
212   other BitlBee users. :-) */
[dfa41a4]213#define JABBER_PACKET_ID "BeeP"
214#define JABBER_CACHED_ID "BeeC"
215
[979cfb4]216/* The number of seconds to keep cached packets before garbage collecting
217   them. This gc is done on every keepalive (every minute). */
218#define JABBER_CACHE_MAX_AGE 600
219
[47d3ac4]220/* RFC 392[01] stuff */
221#define XMLNS_TLS          "urn:ietf:params:xml:ns:xmpp-tls"
222#define XMLNS_SASL         "urn:ietf:params:xml:ns:xmpp-sasl"
223#define XMLNS_BIND         "urn:ietf:params:xml:ns:xmpp-bind"
224#define XMLNS_SESSION      "urn:ietf:params:xml:ns:xmpp-session"
225#define XMLNS_STANZA_ERROR "urn:ietf:params:xml:ns:xmpp-stanzas"
226#define XMLNS_STREAM_ERROR "urn:ietf:params:xml:ns:xmpp-streams"
227#define XMLNS_ROSTER       "jabber:iq:roster"
228
229/* Some supported extensions/legacy stuff */
[2c2df7d]230#define XMLNS_AUTH         "jabber:iq:auth"                                      /* XEP-0078 */
231#define XMLNS_VERSION      "jabber:iq:version"                                   /* XEP-0092 */
[d76e12f]232#define XMLNS_TIME_OLD     "jabber:iq:time"                                      /* XEP-0090 */
233#define XMLNS_TIME         "urn:xmpp:time"                                       /* XEP-0202 */
[eded1f7]234#define XMLNS_PING         "urn:xmpp:ping"                                       /* XEP-0199 */
[1444be5]235#define XMLNS_RECEIPTS     "urn:xmpp:receipts"                                   /* XEP-0184 */
[2c2df7d]236#define XMLNS_VCARD        "vcard-temp"                                          /* XEP-0054 */
[7b40f17]237#define XMLNS_DELAY_OLD    "jabber:x:delay"                                      /* XEP-0091 */
238#define XMLNS_DELAY        "urn:xmpp:delay"                                      /* XEP-0203 */
[2c2df7d]239#define XMLNS_XDATA        "jabber:x:data"                                       /* XEP-0004 */
240#define XMLNS_CHATSTATES   "http://jabber.org/protocol/chatstates"               /* XEP-0085 */
[dc0ba9c]241#define XMLNS_DISCO_INFO   "http://jabber.org/protocol/disco#info"               /* XEP-0030 */
242#define XMLNS_DISCO_ITEMS  "http://jabber.org/protocol/disco#items"              /* XEP-0030 */
[2c2df7d]243#define XMLNS_MUC          "http://jabber.org/protocol/muc"                      /* XEP-0045 */
244#define XMLNS_MUC_USER     "http://jabber.org/protocol/muc#user"                 /* XEP-0045 */
[1ba7e8f]245#define XMLNS_CAPS         "http://jabber.org/protocol/caps"                     /* XEP-0115 */
[2c2df7d]246#define XMLNS_FEATURE      "http://jabber.org/protocol/feature-neg"              /* XEP-0020 */
247#define XMLNS_SI           "http://jabber.org/protocol/si"                       /* XEP-0095 */
248#define XMLNS_FILETRANSFER "http://jabber.org/protocol/si/profile/file-transfer" /* XEP-0096 */
249#define XMLNS_BYTESTREAMS  "http://jabber.org/protocol/bytestreams"              /* XEP-0065 */
250#define XMLNS_IBB          "http://jabber.org/protocol/ibb"                      /* XEP-0047 */
[47d3ac4]251
[4a5d885]252/* jabber.c */
[5ebff60]253void jabber_connect(struct im_connection *ic);
[4a5d885]254
[21167d2]255/* iq.c */
[5ebff60]256xt_status jabber_pkt_iq(struct xt_node *node, gpointer data);
257int jabber_init_iq_auth(struct im_connection *ic);
258xt_status jabber_pkt_bind_sess(struct im_connection *ic, struct xt_node *node, struct xt_node *orig);
259int jabber_get_roster(struct im_connection *ic);
260int jabber_get_vcard(struct im_connection *ic, char *bare_jid);
261int jabber_add_to_roster(struct im_connection *ic, const char *handle, const char *name, const char *group);
262int jabber_remove_from_roster(struct im_connection *ic, char *handle);
263xt_status jabber_iq_query_features(struct im_connection *ic, char *bare_jid);
264xt_status jabber_iq_query_server(struct im_connection *ic, char *jid, char *xmlns);
265void jabber_iq_version_send(struct im_connection *ic, struct jabber_buddy *bud, void *data);
[21167d2]266
[2c2df7d]267/* si.c */
[5ebff60]268int jabber_si_handle_request(struct im_connection *ic, struct xt_node *node, struct xt_node *sinode);
269void jabber_si_transfer_request(struct im_connection *ic, file_transfer_t *ft, char *who);
270void jabber_si_free_transfer(file_transfer_t *ft);
[2ff2076]271
272/* s5bytestream.c */
[5ebff60]273int jabber_bs_recv_request(struct im_connection *ic, struct xt_node *node, struct xt_node *qnode);
274gboolean jabber_bs_send_start(struct jabber_transfer *tf);
275gboolean jabber_bs_send_write(file_transfer_t *ft, char *buffer, unsigned int len);
[21167d2]276
[cfbb3a6]277/* message.c */
[5ebff60]278xt_status jabber_pkt_message(struct xt_node *node, gpointer data);
[0b4a0db]279
280/* presence.c */
[5ebff60]281xt_status jabber_pkt_presence(struct xt_node *node, gpointer data);
282int presence_send_update(struct im_connection *ic);
283int presence_send_request(struct im_connection *ic, char *handle, char *request);
[f06894d]284
[21167d2]285/* jabber_util.c */
[5ebff60]286char *set_eval_priority(set_t *set, char *value);
287char *set_eval_tls(set_t *set, char *value);
288struct xt_node *jabber_make_packet(char *name, char *type, char *to, struct xt_node *children);
289struct xt_node *jabber_make_error_packet(struct xt_node *orig, char *err_cond, char *err_type, char *err_code);
290void jabber_cache_add(struct im_connection *ic, struct xt_node *node, jabber_cache_event func);
291struct xt_node *jabber_cache_get(struct im_connection *ic, char *id);
292void jabber_cache_entry_free(gpointer entry);
293void jabber_cache_clean(struct im_connection *ic);
294xt_status jabber_cache_handle_packet(struct im_connection *ic, struct xt_node *node);
295const struct jabber_away_state *jabber_away_state_by_code(char *code);
296const struct jabber_away_state *jabber_away_state_by_name(char *name);
297void jabber_buddy_ask(struct im_connection *ic, char *handle);
298int jabber_compare_jid(const char *jid1, const char *jid2);
299char *jabber_normalize(const char *orig);
300
301typedef enum {
302        GET_BUDDY_CREAT = 1,    /* Try to create it, if necessary. */
303        GET_BUDDY_EXACT = 2,    /* Get an exact match (only makes sense with bare JIDs). */
304        GET_BUDDY_FIRST = 4,    /* No selection, simply get the first resource for this JID. */
305        GET_BUDDY_BARE = 8,     /* Get the bare version of the JID (possibly inexistent). */
306        GET_BUDDY_BARE_OK = 16, /* Allow returning a bare JID if that seems better. */
[0d3f30f]307} get_buddy_flags_t;
308
[5ebff60]309struct jabber_error {
[1baaef8]310        char *code, *text, *type;
311};
312
[5ebff60]313struct jabber_buddy *jabber_buddy_add(struct im_connection *ic, char *full_jid);
314struct jabber_buddy *jabber_buddy_by_jid(struct im_connection *ic, char *jid, get_buddy_flags_t flags);
315struct jabber_buddy *jabber_buddy_by_ext_jid(struct im_connection *ic, char *jid, get_buddy_flags_t flags);
316int jabber_buddy_remove(struct im_connection *ic, char *full_jid);
317int jabber_buddy_remove_bare(struct im_connection *ic, char *bare_jid);
318void jabber_buddy_remove_all(struct im_connection *ic);
319time_t jabber_get_timestamp(struct xt_node *xt);
320struct jabber_error *jabber_error_parse(struct xt_node *node, char *xmlns);
321void jabber_error_free(struct jabber_error *err);
322gboolean jabber_set_me(struct im_connection *ic, const char *me);
[5e202b0]323
324extern const struct jabber_away_state jabber_away_state_list[];
[21167d2]325
326/* io.c */
[5ebff60]327int jabber_write_packet(struct im_connection *ic, struct xt_node *node);
328int jabber_write(struct im_connection *ic, char *buf, int len);
329gboolean jabber_connected_plain(gpointer data, gint source, b_input_condition cond);
330gboolean jabber_connected_ssl(gpointer data, int returncode, void *source, b_input_condition cond);
331gboolean jabber_start_stream(struct im_connection *ic);
332void jabber_end_stream(struct im_connection *ic);
[21167d2]333
[5997488]334/* sasl.c */
[5ebff60]335xt_status sasl_pkt_mechanisms(struct xt_node *node, gpointer data);
336xt_status sasl_pkt_challenge(struct xt_node *node, gpointer data);
337xt_status sasl_pkt_result(struct xt_node *node, gpointer data);
338gboolean sasl_supported(struct im_connection *ic);
339void sasl_oauth2_init(struct im_connection *ic);
340int sasl_oauth2_get_refresh_token(struct im_connection *ic, const char *msg);
341int sasl_oauth2_refresh(struct im_connection *ic, const char *refresh_token);
[5997488]342
[18c6d36]343extern const struct oauth2_service oauth2_service_google;
344extern const struct oauth2_service oauth2_service_facebook;
345
[e35d1a1]346/* conference.c */
[5ebff60]347struct groupchat *jabber_chat_join(struct im_connection *ic, const char *room, const char *nick, const char *password);
348struct groupchat *jabber_chat_with(struct im_connection *ic, char *who);
349struct groupchat *jabber_chat_by_jid(struct im_connection *ic, const char *name);
350void jabber_chat_free(struct groupchat *c);
351int jabber_chat_msg(struct groupchat *ic, char *message, int flags);
352int jabber_chat_topic(struct groupchat *c, char *topic);
353int jabber_chat_leave(struct groupchat *c, const char *reason);
354void jabber_chat_pkt_presence(struct im_connection *ic, struct jabber_buddy *bud, struct xt_node *node);
355void jabber_chat_pkt_message(struct im_connection *ic, struct jabber_buddy *bud, struct xt_node *node);
356void jabber_chat_invite(struct groupchat *c, char *who, char *message);
[e35d1a1]357
[f06894d]358#endif
Note: See TracBrowser for help on using the repository browser.