Personal copy of mcabber
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

369 lines
11 KiB

/*
* xmpp_helper.c -- Jabber protocol helper functions
*
* Copyright (C) 2008-2010 Frank Zschockelt <mcabber@freakysoft.de>
* Copyright (C) 2005-2010 Mikael Berthe <mikael@lilotux.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, see <http://www.gnu.org/licenses/>.
*/
#include <string.h>
#include <stdlib.h>
#include <stdio.h> // snprintf
#include "xmpp_helper.h"
#include "settings.h"
#include "utils.h"
#include "caps.h"
#include "logprint.h"
time_t iqlast; // last message/status change time
extern char *imstatus_showmap[];
struct xmpp_error xmpp_errors[] = {
{XMPP_ERROR_REDIRECT, "302",
"Redirect", "redirect", "modify"},
{XMPP_ERROR_BAD_REQUEST, "400",
"Bad Request", "bad-request", "modify"},
{XMPP_ERROR_NOT_AUTHORIZED, "401",
"Not Authorized", "not-authorized", "auth"},
{XMPP_ERROR_PAYMENT_REQUIRED, "402",
"Payment Required", "payment-required", "auth"},
{XMPP_ERROR_FORBIDDEN, "403",
"Forbidden", "forbidden", "auth"},
{XMPP_ERROR_NOT_FOUND, "404",
"Not Found", "item-not-found", "cancel"},
{XMPP_ERROR_NOT_ALLOWED, "405",
"Not Allowed", "not-allowed", "cancel"},
{XMPP_ERROR_NOT_ACCEPTABLE, "406",
"Not Acceptable", "not-acceptable", "modify"},
{XMPP_ERROR_REGISTRATION_REQUIRED, "407",
"Registration required", "registration-required", "auth"},
{XMPP_ERROR_REQUEST_TIMEOUT, "408",
"Request Timeout", "remote-server-timeout", "wait"},
{XMPP_ERROR_CONFLICT, "409",
"Conflict", "conflict", "cancel"},
{XMPP_ERROR_INTERNAL_SERVER_ERROR, "500",
"Internal Server Error", "internal-server-error", "wait"},
{XMPP_ERROR_NOT_IMPLEMENTED, "501",
"Not Implemented", "feature-not-implemented", "cancel"},
{XMPP_ERROR_REMOTE_SERVER_ERROR, "502",
"Remote Server Error", "service-unavailable", "wait"},
{XMPP_ERROR_SERVICE_UNAVAILABLE, "503",
"Service Unavailable", "service-unavailable", "cancel"},
{XMPP_ERROR_REMOTE_SERVER_TIMEOUT, "504",
"Remote Server Timeout", "remote-server-timeout", "wait"},
{XMPP_ERROR_DISCONNECTED, "510",
"Disconnected", "service-unavailable", "cancel"},
{0, NULL, NULL, NULL, NULL}
};
const gchar* lm_message_node_get_child_value(LmMessageNode *node,
const gchar *child)
{
LmMessageNode *tmp;
if (G_UNLIKELY(!node || !child)) return NULL;
tmp = lm_message_node_get_child(node, child);
if (tmp) {
const gchar *val = lm_message_node_get_value(tmp);
return (val ? val : "");
}
return NULL;
}
static LmMessageNode *hidden = NULL;
void lm_message_node_hide(LmMessageNode *node)
{
LmMessageNode *parent = node->parent, *prev_sibling = node->prev;
if (hidden) {
hidden->children = hidden->next = hidden->prev = hidden->parent = NULL;
lm_message_node_unref(hidden);
}
if (parent->children == node)
parent->children = node->next;
if (prev_sibling)
prev_sibling->next = node->next;
if (node->next)
node->next->prev = prev_sibling;
}
// Maybe not a good idea, because it uses internals of loudmouth...
// It's used for rosternotes/bookmarks
LmMessageNode *lm_message_node_new(const gchar *name, const gchar *xmlns)
{
LmMessageNode *node;
node = g_new0 (LmMessageNode, 1);
if (G_UNLIKELY(!node)) return NULL;
node->name = g_strdup (name);
node->value = NULL;
node->raw_mode = FALSE;
node->attributes = NULL;
node->next = NULL;
node->prev = NULL;
node->parent = NULL;
node->children = NULL;
node->ref_count = 1;
lm_message_node_set_attribute(node, "xmlns", xmlns);
return node;
}
void lm_message_node_insert_childnode(LmMessageNode *node,
LmMessageNode *child)
{
LmMessageNode *x;
if (G_UNLIKELY(!node)) return;
lm_message_node_deep_ref(child);
if (node->children == NULL)
node->children = child;
else {
for (x = node->children; x->next; x = x->next)
;
x->next = child;
}
}
void lm_message_node_deep_ref(LmMessageNode *node)
{
if (G_UNLIKELY(!node)) return;
lm_message_node_ref(node);
lm_message_node_deep_ref(node->next);
lm_message_node_deep_ref(node->children);
}
const gchar* lm_message_get_from(LmMessage *m)
{
return lm_message_node_get_attribute(m->node, "from");
}
const gchar* lm_message_get_id(LmMessage *m)
{
return lm_message_node_get_attribute(m->node, "id");
}
LmMessage *lm_message_new_iq_from_query(LmMessage *m,
LmMessageSubType type)
{
LmMessage *new;
const char *from = lm_message_node_get_attribute(m->node, "from");
const char *id = lm_message_node_get_attribute(m->node, "id");
new = lm_message_new_with_sub_type(from, LM_MESSAGE_TYPE_IQ,
type);
if (id)
lm_message_node_set_attribute(new->node, "id", id);
return new;
}
// entity_version(enum imstatus status)
// Return a static version string for Entity Capabilities.
// It should be specific to the client version, please change the id
// if you alter mcabber's disco support (or add something to the version
// number) so that it doesn't conflict with the official client.
const char *entity_version(enum imstatus status)
{
static char *ver, *ver_notavail;
if (ver && (status != notavail))
return ver;
if (ver_notavail)
return ver_notavail;
caps_add("");
caps_set_identity("", "client", "mcabber", "pc");
caps_add_feature("", NS_DISCO_INFO);
caps_add_feature("", NS_CAPS);
caps_add_feature("", NS_MUC);
// advertise ChatStates only if they aren't disabled
if (!settings_opt_get_int("disable_chatstates"))
caps_add_feature("", NS_CHATSTATES);
caps_add_feature("", NS_TIME);
caps_add_feature("", NS_XMPP_TIME);
caps_add_feature("", NS_VERSION);
caps_add_feature("", NS_PING);
caps_add_feature("", NS_COMMANDS);
caps_add_feature("", NS_RECEIPTS);
caps_add_feature("", NS_X_CONFERENCE);
if (!settings_opt_get_int("iq_last_disable") &&
(!settings_opt_get_int("iq_last_disable_when_notavail") ||
status != notavail))
caps_add_feature("", NS_LAST);
if (status == notavail) {
ver_notavail = caps_generate();
return ver_notavail;
}
ver = caps_generate();
return ver;
}
LmMessageNode *lm_message_node_find_xmlns(LmMessageNode *node,
const char *xmlns)
{
LmMessageNode *x;
const char *p;
if (G_UNLIKELY(!node)) return NULL;
for (x = node->children ; x; x = x->next) {
if ((p = lm_message_node_get_attribute(x, "xmlns")) && !strcmp(p, xmlns))
break;
}
return x;
}
time_t lm_message_node_get_timestamp(LmMessageNode *node)
{
LmMessageNode *x;
const char *p;
x = lm_message_node_find_xmlns(node, NS_XMPP_DELAY);
if (x && (!strcmp(x->name, "delay")) &&
(p = lm_message_node_get_attribute(x, "stamp")) != NULL)
return from_iso8601(p, 1);
x = lm_message_node_find_xmlns(node, NS_DELAY);
if (x && (p = lm_message_node_get_attribute(x, "stamp")) != NULL)
return from_iso8601(p, 1);
return 0;
}
// lm_message_new_presence(status, recipient, message)
// Create an xmlnode with default presence attributes
// Note: the caller must free the node after use
LmMessage *lm_message_new_presence(enum imstatus st,
const char *recipient,
const char *msg)
{
unsigned int prio;
LmMessage *x = lm_message_new(recipient, LM_MESSAGE_TYPE_PRESENCE);
switch(st) {
case away:
case notavail:
case dontdisturb:
case freeforchat:
lm_message_node_add_child(x->node, "show", imstatus_showmap[st]);
break;
#ifdef WITH_DEPRECATED_STATUS_INVISIBLE
case invisible:
lm_message_node_set_attribute(x->node, "type", "invisible");
break;
#endif
case offline:
lm_message_node_set_attribute(x->node, "type", "unavailable");
break;
default:
break;
}
if (st == away || st == notavail)
prio = settings_opt_get_int("priority_away");
else
prio = settings_opt_get_int("priority");
if (prio) {
char strprio[8];
snprintf(strprio, 8, "%d", (int)prio);
lm_message_node_add_child(x->node, "priority", strprio);
}
if (msg)
lm_message_node_add_child(x->node, "status", msg);
return x;
}
static const char *defaulterrormsg(guint code)
{
int i;
for (i = 0; xmpp_errors[i].code; ++i) {
if (xmpp_errors[i].code == code)
return xmpp_errors[i].meaning;
}
return NULL;
}
// display_server_error(x)
// Display the error to the user
// x: error tag xmlnode pointer
void display_server_error(LmMessageNode *x, const char *from)
{
const char *desc = NULL, *errname = NULL, *s;
char *sdesc, *tmp;
if (!x) return;
/* RFC3920:
* The <error/> element:
* o MUST contain a child element corresponding to one of the defined
* stanza error conditions specified below; this element MUST be
* qualified by the 'urn:ietf:params:xml:ns:xmpp-stanzas' namespace.
*/
if (x->children)
errname = x->children->name;
if (from)
scr_LogPrint(LPRINT_LOGNORM, "Received error packet [%s] from <%s>",
(errname ? errname : ""), from);
else
scr_LogPrint(LPRINT_LOGNORM, "Received error packet [%s]",
(errname ? errname : ""));
// For backward compatibility
if (!errname && ((s = lm_message_node_get_attribute(x, "code")) != NULL)) {
// Default message
desc = defaulterrormsg(atoi(s));
}
// Error tag data is better, if available
s = lm_message_node_get_value(x);
if (s && *s) desc = s;
// And sometimes there is a text message
s = lm_message_node_get_child_value(x, "text");
if (s && *s) desc = s;
// If we still have no description, let's give up
if (!desc || !*desc)
return;
// Strip trailing newlines
sdesc = g_strdup(desc);
for (tmp = sdesc; *tmp; tmp++) ;
if (tmp > sdesc)
tmp--;
while (tmp >= sdesc && (*tmp == '\n' || *tmp == '\r'))
*tmp-- = '\0';
if (*sdesc)
scr_LogPrint(LPRINT_LOGNORM, "Error message from server: %s", sdesc);
g_free(sdesc);
}
/* vim: set et cindent cinoptions=>2\:2(0 ts=2 sw=2: For Vim users... */