source: protocols/jabber/message.c @ a8196d6

Last change on this file since a8196d6 was 7e68015, checked in by dequis <dx@…>, at 2016-01-21T06:15:44Z

Revert "hipchat: Implement their own variant of self-messages [...]"

This reverts commit d11ccbf6ea94264bde8b0f525c4bbedf50de0174.

After thinking about this long enough I've decided this is a bad idea,
and better wait for the hipchat server to support carbons.

  • Property mode set to 100644
File size: 7.6 KB
RevLine 
[f06894d]1/***************************************************************************\
2*                                                                           *
3*  BitlBee - An IRC to IM gateway                                           *
[21167d2]4*  Jabber module - Handling of message(s) (tags), etc                       *
[f06894d]5*                                                                           *
[0e788f5]6*  Copyright 2006-2012 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#include "jabber.h"
25
[fa8f57b]26static xt_status jabber_pkt_message_normal(struct xt_node *node, gpointer data, gboolean carbons_sent)
[f06894d]27{
[0da65d5]28        struct im_connection *ic = data;
[fa8f57b]29        char *from = xt_find_attr(node, carbons_sent ? "to" : "from");
[5ebff60]30        char *type = xt_find_attr(node, "type");
31        char *id = xt_find_attr(node, "id");
32        struct xt_node *body = xt_find_node(node->children, "body"), *c;
33        struct xt_node *request = xt_find_node(node->children, "request");
[e35d1a1]34        struct jabber_buddy *bud = NULL;
[1aa74f55]35        char *s, *room = NULL, *reason = NULL;
[5ebff60]36
37        if (!from) {
[e35d1a1]38                return XT_HANDLED; /* Consider this packet corrupted. */
[5ebff60]39        }
[ad8a810]40
[fa8f57b]41        if (request && id && g_strcmp0(type, "groupchat") != 0 && !carbons_sent) {
[1444be5]42                /* Send a message receipt (XEP-0184), looking like this:
[ad8a810]43                 * <message from='...' id='...' to='...'>
[1444be5]44                 *  <received xmlns='urn:xmpp:receipts' id='richard2-4.1.247'/>
[ad8a810]45                 * </message>
46                 *
47                 * MUC messages are excluded, since receipts aren't supposed to be sent over MUCs
48                 * (XEP-0184 section 5.3) and replying to those may result in 'forbidden' errors.
49                 */
[91ae87d]50                struct xt_node *received, *receipt;
[5ebff60]51
52                received = xt_new_node("received", NULL, NULL);
53                xt_add_attr(received, "xmlns", XMLNS_RECEIPTS);
54                xt_add_attr(received, "id", id);
55                receipt = jabber_make_packet("message", NULL, from, received);
56
57                jabber_write_packet(ic, receipt);
58                xt_free_node(receipt);
[1444be5]59        }
[5ebff60]60
61        bud = jabber_buddy_by_jid(ic, from, GET_BUDDY_EXACT);
62
63        if (type && strcmp(type, "error") == 0) {
[f0071b7]64                /* Handle type=error packet. */
[5ebff60]65        } else if (type && from && strcmp(type, "groupchat") == 0) {
66                jabber_chat_pkt_message(ic, bud, node);
67        } else { /* "chat", "normal", "headline", no-type or whatever. Should all be pretty similar. */
68                GString *fullmsg = g_string_new("");
[5ec4129]69
[5ebff60]70                for (c = node->children; (c = xt_find_node(c, "x")); c = c->next) {
71                        char *ns = xt_find_attr(c, "xmlns");
[1aa74f55]72                        struct xt_node *inv;
[5ebff60]73
74                        if (ns && strcmp(ns, XMLNS_MUC_USER) == 0 &&
75                            (inv = xt_find_node(c->children, "invite"))) {
[1aa74f55]76                                /* This is an invitation. Set some vars which
77                                   will be passed to imcb_chat_invite() below. */
[5ec4129]78                                room = from;
[5ebff60]79                                if ((from = xt_find_attr(inv, "from")) == NULL) {
[daae10f]80                                        from = room;
[5ebff60]81                                }
82                                if ((inv = xt_find_node(inv->children, "reason")) && inv->text_len > 0) {
[1aa74f55]83                                        reason = inv->text;
[5ebff60]84                                }
[5ec4129]85                        }
86                }
[5ebff60]87
88                if ((s = strchr(from, '/'))) {
89                        if (bud) {
90                                bud->last_msg = time(NULL);
[daae10f]91                                from = bud->ext_jid ? bud->ext_jid : bud->bare_jid;
[5ebff60]92                        } else {
[f0071b7]93                                *s = 0; /* We need to generate a bare JID now. */
[5ebff60]94                        }
[a21a8ac]95                }
[5ebff60]96
97                if (type && strcmp(type, "headline") == 0) {
98                        if ((c = xt_find_node(node->children, "subject")) && c->text_len > 0) {
99                                g_string_append_printf(fullmsg, "Headline: %s\n", c->text);
100                        }
101
[f0071b7]102                        /* <x xmlns="jabber:x:oob"><url>http://....</url></x> can contain a URL, it seems. */
[5ebff60]103                        for (c = node->children; c; c = c->next) {
[f0071b7]104                                struct xt_node *url;
[5ebff60]105
106                                if ((url = xt_find_node(c->children, "url")) && url->text_len > 0) {
107                                        g_string_append_printf(fullmsg, "URL: %s\n", url->text);
108                                }
[f0071b7]109                        }
[5ebff60]110                } else if ((c = xt_find_node(node->children, "subject")) && c->text_len > 0 &&
111                           (!bud || !(bud->flags & JBFLAG_HIDE_SUBJECT))) {
112                        g_string_append_printf(fullmsg, "<< \002BitlBee\002 - Message with subject: %s >>\n", c->text);
113                        if (bud) {
[31dbb90a]114                                bud->flags |= JBFLAG_HIDE_SUBJECT;
[5ebff60]115                        }
116                } else if (bud && !c) {
[31dbb90a]117                        /* Yeah, possibly we're hiding changes to this field now. But nobody uses
118                           this for anything useful anyway, except GMail when people reply to an
119                           e-mail via chat, repeating the same subject all the time. I don't want
120                           to have to remember full subject strings for everyone. */
121                        bud->flags &= ~JBFLAG_HIDE_SUBJECT;
[f0071b7]122                }
[5ebff60]123
124                if (body && body->text_len > 0) { /* Could be just a typing notification. */
125                        fullmsg = g_string_append(fullmsg, body->text);
126                }
127
128                if (fullmsg->len > 0) {
129                        imcb_buddy_msg(ic, from, fullmsg->str,
[fa8f57b]130                                       carbons_sent ? OPT_SELFMESSAGE : 0, jabber_get_timestamp(node));
[5ebff60]131                }
132                if (room) {
133                        imcb_chat_invite(ic, room, from, reason);
134                }
135
136                g_string_free(fullmsg, TRUE);
137
[788a1af]138                /* Handling of incoming typing notifications. */
[fa8f57b]139                if (bud == NULL || carbons_sent) {
140                        /* Can't handle these for unknown buddies.
141                           And ignore them if it's just carbons */
[5ebff60]142                } else if (xt_find_node(node->children, "composing")) {
[788a1af]143                        bud->flags |= JBFLAG_DOES_XEP85;
[5ebff60]144                        imcb_buddy_typing(ic, from, OPT_TYPING);
[a21a8ac]145                }
[5307e88]146                else if (xt_find_node(node->children, "active")) {
[788a1af]147                        bud->flags |= JBFLAG_DOES_XEP85;
[5307e88]148
149                        /* No need to send a "stopped typing" signal when there's a message. */
150                        if (body == NULL) {
151                                imcb_buddy_typing(ic, from, 0);
152                        }
[5ebff60]153                } else if (xt_find_node(node->children, "paused")) {
[788a1af]154                        bud->flags |= JBFLAG_DOES_XEP85;
[5ebff60]155                        imcb_buddy_typing(ic, from, OPT_THINKING);
[788a1af]156                }
[5ebff60]157
158                if (s) {
[788a1af]159                        *s = '/'; /* And convert it back to a full JID. */
[5ebff60]160                }
[dd788bb]161        }
[5ebff60]162
[f06894d]163        return XT_HANDLED;
164}
[fa8f57b]165
166static xt_status jabber_carbons_message(struct xt_node *node, gpointer data)
167{
168        struct im_connection *ic = data;
169        struct xt_node *wrap, *fwd, *msg;
170        gboolean carbons_sent;
171
172        if ((wrap = xt_find_node(node->children, "received"))) {
173                carbons_sent = FALSE;
174        } else if ((wrap = xt_find_node(node->children, "sent"))) {
175                carbons_sent = TRUE;
176        }
177
178        if (wrap == NULL || g_strcmp0(xt_find_attr(wrap, "xmlns"), XMLNS_CARBONS) != 0) {
179                return XT_NEXT;
180        }
181
182        if (!(fwd = xt_find_node(wrap->children, "forwarded")) ||
183             (g_strcmp0(xt_find_attr(fwd, "xmlns"), XMLNS_FORWARDING) != 0) ||
184            !(msg = xt_find_node(fwd->children, "message"))) {
185                imcb_log(ic, "Error: Invalid carbons message received");
186                return XT_ABORT;
187        }
188
189        return jabber_pkt_message_normal(msg, data, carbons_sent);
190}
191
192xt_status jabber_pkt_message(struct xt_node *node, gpointer data)
193{
194        struct im_connection *ic = data;
195        struct jabber_data *jd = ic->proto_data;
196        char *from = xt_find_attr(node, "from");
197
198        if (jabber_compare_jid(jd->me, from)) {    /* Probably a Carbons message */
199                xt_status st = jabber_carbons_message(node, data);
200                if (st == XT_HANDLED || st == XT_ABORT) {
201                        return st;
202                }
203        }
204        return jabber_pkt_message_normal(node, data, FALSE);
205}
Note: See TracBrowser for help on using the repository browser.