summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorTobias Markmann <tfar@soc.pidgin.im>2008-08-03 23:16:24 +0000
committerTobias Markmann <tfar@soc.pidgin.im>2008-08-03 23:16:24 +0000
commit3568b5b558aa31997928d823e2948c34ce9a8d42 (patch)
treec62c988006fbc961f724d6cce31f2a324ffc5f90
parente5a6e27bea918ff571766bf312e64f95d34b7079 (diff)
parent0d3dffa2bb987e05de39d993fa93d414e5eef8b2 (diff)
downloadpidgin-soc.2008.xmpp.tar.gz
propagate from branch 'im.pidgin.pidgin' (head 918c72f0ff7f7e3c545d3baf6368ccbd2ba21427)soc.2008.xmpp
to branch 'im.pidgin.soc.2008.xmpp' (head 6f4b6861c9a4a2b3fa5bed70e6b6d1e70638b63b)
-rw-r--r--libpurple/dnssrv.c349
-rw-r--r--libpurple/dnssrv.h25
-rw-r--r--libpurple/protocols/jabber/bosh.c49
-rw-r--r--libpurple/protocols/jabber/bosh.h75
-rw-r--r--libpurple/protocols/jabber/caps.c460
-rw-r--r--libpurple/protocols/jabber/caps.h59
-rw-r--r--libpurple/protocols/jabber/disco.c111
-rw-r--r--libpurple/protocols/jabber/jabber.c147
-rw-r--r--libpurple/protocols/jabber/jabber.h34
-rw-r--r--libpurple/protocols/jabber/libxmpp.c13
-rw-r--r--libpurple/protocols/jabber/message.c2
-rw-r--r--libpurple/protocols/jabber/message.h2
-rw-r--r--libpurple/protocols/jabber/pep.c6
-rw-r--r--libpurple/protocols/jabber/pep.h10
-rw-r--r--libpurple/protocols/jabber/presence.c20
-rw-r--r--libpurple/protocols/jabber/usermood.c4
-rw-r--r--libpurple/protocols/jabber/usernick.c4
-rw-r--r--libpurple/protocols/jabber/usertune.c4
18 files changed, 1068 insertions, 306 deletions
diff --git a/libpurple/dnssrv.c b/libpurple/dnssrv.c
index d57b20f40a..8db47d9477 100644
--- a/libpurple/dnssrv.c
+++ b/libpurple/dnssrv.c
@@ -33,12 +33,18 @@
#ifndef T_SRV
#define T_SRV 33
#endif
+#ifndef T_TXT
+#define T_TXT 16
+#endif
#else
#include <windns.h>
/* Missing from the mingw headers */
#ifndef DNS_TYPE_SRV
# define DNS_TYPE_SRV 33
#endif
+#ifndef DNS_TYPE_TXT
+# define DNS_TYPE_TXT 16
+#endif
#endif
#include "dnssrv.h"
@@ -60,9 +66,11 @@ static void (WINAPI *MyDnsRecordListFree) (PDNS_RECORD pRecordList,
#endif
struct _PurpleSrvQueryData {
- PurpleSrvCallback cb;
+ PurpleSrvCallback srv_cb;
+ PurpleTxtCallback txt_cb;
gpointer extradata;
guint handle;
+ int type;
#ifdef _WIN32
GThread *resolver;
char *query;
@@ -74,6 +82,11 @@ struct _PurpleSrvQueryData {
#endif
};
+typedef struct _PurpleSrvInternalQuery {
+ int type;
+ char query[256];
+} PurpleSrvInternalQuery;
+
static gint
responsecompare(gconstpointer ar, gconstpointer br)
{
@@ -99,6 +112,7 @@ resolve(int in, int out)
{
GList *ret = NULL;
PurpleSrvResponse *srvres;
+ PurpleTxtResponse *txtres;
queryans answer;
int size;
int qdcount;
@@ -107,23 +121,23 @@ resolve(int in, int out)
guchar *cp;
gchar name[256];
guint16 type, dlen, pref, weight, port;
- gchar query[256];
+ PurpleSrvInternalQuery query;
+ int n;
#ifdef HAVE_SIGNAL_H
purple_restore_default_signal_handlers();
#endif
- if (read(in, query, 256) <= 0) {
+ if (read(in, &query, sizeof(query)) <= 0) {
close(out);
close(in);
_exit(0);
}
- size = res_query( query, C_IN, T_SRV, (u_char*)&answer, sizeof( answer));
-
+ size = res_query( query.query, C_IN, query.type, (u_char*)&answer, sizeof( answer));
+
qdcount = ntohs(answer.hdr.qdcount);
ancount = ntohs(answer.hdr.ancount);
-
cp = (guchar*)&answer + sizeof(HEADER);
end = (guchar*)&answer + size;
@@ -138,17 +152,14 @@ resolve(int in, int out)
size = dn_expand((unsigned char*)&answer, end, cp, name, 256);
if(size < 0)
goto end;
-
cp += size;
-
GETSHORT(type,cp);
/* skip ttl and class since we already know it */
cp += 6;
GETSHORT(dlen,cp);
-
- if (type == T_SRV) {
+ if (query.type == T_SRV) {
GETSHORT(pref,cp);
GETSHORT(weight,cp);
@@ -168,6 +179,11 @@ resolve(int in, int out)
srvres->weight = weight;
ret = g_list_insert_sorted(ret, srvres, responsecompare);
+ } else if (query.type == T_TXT) {
+ txtres = g_new0(PurpleTxtResponse, 1);
+ strncpy(txtres->content, ++cp, dlen-1);
+ ret = g_list_append(ret, txtres);
+ cp += dlen - 1;
} else {
cp += dlen;
}
@@ -175,10 +191,12 @@ resolve(int in, int out)
end:
size = g_list_length(ret);
- write(out, &size, sizeof(int));
+ write(out, &(query.type), sizeof(query.type));
+ write(out, &size, sizeof(size));
while (ret != NULL)
{
- write(out, ret->data, sizeof(PurpleSrvResponse));
+ if (query.type == T_SRV) write(out, ret->data, sizeof(PurpleSrvResponse));
+ if (query.type == T_TXT) write(out, ret->data, sizeof(PurpleTxtResponse));
g_free(ret->data);
ret = g_list_remove(ret, ret->data);
}
@@ -193,39 +211,66 @@ static void
resolved(gpointer data, gint source, PurpleInputCondition cond)
{
int size;
+ int type;
PurpleSrvQueryData *query_data = (PurpleSrvQueryData*)data;
- PurpleSrvResponse *res;
- PurpleSrvResponse *tmp;
int i;
- PurpleSrvCallback cb = query_data->cb;
int status;
-
- if (read(source, &size, sizeof(int)) == sizeof(int))
- {
- ssize_t red;
- purple_debug_info("dnssrv","found %d SRV entries\n", size);
- tmp = res = g_new0(PurpleSrvResponse, size);
- for (i = 0; i < size; i++) {
- red = read(source, tmp++, sizeof(PurpleSrvResponse));
- if (red != sizeof(PurpleSrvResponse)) {
- purple_debug_error("dnssrv","unable to read srv "
- "response: %s\n", g_strerror(errno));
+
+ if (read(source, &type, sizeof(type)) == sizeof(type)) {
+ if (type == T_SRV) {
+ PurpleSrvResponse *res;
+ PurpleSrvResponse *tmp;
+ PurpleSrvCallback cb = query_data->srv_cb;
+ if (read(source, &size, sizeof(int)) == sizeof(int)) {
+ ssize_t red;
+ purple_debug_info("dnssrv","found %d SRV entries\n", size);
+ tmp = res = g_new0(PurpleSrvResponse, size);
+ for (i = 0; i < size; i++) {
+ red = read(source, tmp++, sizeof(PurpleSrvResponse));
+ if (red != sizeof(PurpleSrvResponse)) {
+ purple_debug_error("dnssrv","unable to read srv "
+ "response: %s\n", g_strerror(errno));
+ size = 0;
+ g_free(res);
+ res = NULL;
+ }
+ }
+ } else {
+ purple_debug_info("dnssrv","found 0 SRV entries; errno is %i\n", errno);
size = 0;
- g_free(res);
res = NULL;
}
+ cb(res, size, query_data->extradata);
+ } else if (type == T_TXT) {
+ PurpleTxtResponse *res;
+ PurpleTxtResponse *tmp;
+ PurpleTxtCallback cb = query_data->txt_cb;
+ if (read(source, &size, sizeof(int)) == sizeof(int)) {
+ ssize_t red;
+ purple_debug_info("dnssrv","found %d TXT entries\n", size);
+ tmp = res = g_new0(PurpleTxtResponse, size);
+ for (i = 0; i < size; i++) {
+ red = read(source, tmp++, sizeof(PurpleTxtResponse));
+ if (red != sizeof(PurpleTxtResponse)) {
+ purple_debug_error("dnssrv","unable to read txt "
+ "response: %s\n", g_strerror(errno));
+ size = 0;
+ g_free(res);
+ res = NULL;
+ }
+ }
+ } else {
+ purple_debug_info("dnssrv","found 0 SRV entries; errno is %i\n", errno);
+ size = 0;
+ res = NULL;
+ }
+ cb(res, size, query_data->extradata);
+ } else {
+ purple_debug_info("dnssrv","type unknown of DNS result entry; errno is %i\n", errno);
}
}
- else
- {
- purple_debug_info("dnssrv","found 0 SRV entries; errno is %i\n", errno);
- size = 0;
- res = NULL;
- }
- cb(res, size, query_data->extradata);
waitpid(query_data->pid, &status, 0);
-
purple_srv_cancel(query_data);
}
@@ -237,33 +282,56 @@ static gboolean
res_main_thread_cb(gpointer data)
{
PurpleSrvResponse *srvres = NULL;
+ PurpleTxtResponse *txtres = NULL;
int size = 0;
PurpleSrvQueryData *query_data = data;
-
if(query_data->error_message != NULL)
purple_debug_error("dnssrv", query_data->error_message);
else {
- PurpleSrvResponse *srvres_tmp = NULL;
- GSList *lst = query_data->results;
-
- size = g_slist_length(lst);
-
- if(query_data->cb && size > 0)
- srvres_tmp = srvres = g_new0(PurpleSrvResponse, size);
- while (lst) {
- if(query_data->cb)
- memcpy(srvres_tmp++, lst->data, sizeof(PurpleSrvResponse));
- g_free(lst->data);
- lst = g_slist_remove(lst, lst->data);
- }
+ if (query_data->type == T_SRV) {
+ PurpleSrvResponse *srvres_tmp = NULL;
+ GSList *lst = query_data->results;
+
+ size = g_slist_length(lst);
+
+ if(query_data->srv_cb && size > 0)
+ srvres_tmp = srvres = g_new0(PurpleSrvResponse, size);
+ while (lst) {
+ if(query_data->cb)
+ memcpy(srvres_tmp++, lst->data, sizeof(PurpleSrvResponse));
+ g_free(lst->data);
+ lst = g_slist_remove(lst, lst->data);
+ }
- query_data->results = NULL;
+ query_data->results = NULL;
- purple_debug_info("dnssrv", "found %d SRV entries\n", size);
- }
+ purple_debug_info("dnssrv", "found %d SRV entries\n", size);
+
+ if(query_data->srv_cb) query_data->srv_cb(srvres, size, query_data->extradata);
+ } else if (query_data->type == T_TXT) {
+ PurpleTxtResponse *txtres_tmp = NULL;
+ GSList *lst = query_data->results;
+
+ size = g_slist_length(lst);
- if(query_data->cb)
- query_data->cb(srvres, size, query_data->extradata);
+ if(query_data->txt_cb && size > 0)
+ txtres_tmp = txtres = g_new0(PurpleTxtResponse, size);
+ while (lst) {
+ if(query_data->txt_cb)
+ memcpy(txtres_tmp++, lst->data, sizeof(PurpleTxtResponse));
+ g_free(lst->data);
+ lst = g_slist_remove(lst, lst->data);
+ }
+
+ query_data->results = NULL;
+
+ purple_debug_info("dnssrv", "found %d TXT entries\n", size);
+
+ if(query_data->txt_cb) query_data->txt_cb(txtres, size, query_data->extradata);
+ } else {
+ purple_debug_error("dnssrv", "unknown query type");
+ }
+ }
query_data->resolver = NULL;
query_data->handle = 0;
@@ -277,40 +345,50 @@ static gpointer
res_thread(gpointer data)
{
PDNS_RECORD dr = NULL;
- int type = DNS_TYPE_SRV;
+ int type;
DNS_STATUS ds;
PurpleSrvQueryData *query_data = data;
-
+ type = query_data->type;
ds = MyDnsQuery_UTF8(query_data->query, type, DNS_QUERY_STANDARD, NULL, &dr, NULL);
if (ds != ERROR_SUCCESS) {
gchar *msg = g_win32_error_message(ds);
- query_data->error_message = g_strdup_printf("Couldn't look up SRV record. %s (%lu).\n", msg, ds);
+ if (type == T_SRV) {
+ query_data->error_message = g_strdup_printf("Couldn't look up SRV record. %s (%lu).\n", msg, ds);
+ } else if (type == T_TXT) {
+ query_data->error_message = g_strdup_printf("Couldn't look up TXT record. %s (%lu).\n", msg, ds);
+ }
g_free(msg);
} else {
- PDNS_RECORD dr_tmp;
- GSList *lst = NULL;
- DNS_SRV_DATA *srv_data;
- PurpleSrvResponse *srvres;
-
- for (dr_tmp = dr; dr_tmp != NULL; dr_tmp = dr_tmp->pNext) {
- /* Discard any incorrect entries. I'm not sure if this is necessary */
- if (dr_tmp->wType != type || strcmp(dr_tmp->pName, query_data->query) != 0) {
- continue;
+ if (type == T_SRV) {
+ PDNS_RECORD dr_tmp;
+ GSList *lst = NULL;
+ DNS_SRV_DATA *srv_data;
+ PurpleSrvResponse *srvres;
+
+ for (dr_tmp = dr; dr_tmp != NULL; dr_tmp = dr_tmp->pNext) {
+ /* Discard any incorrect entries. I'm not sure if this is necessary */
+ if (dr_tmp->wType != type || strcmp(dr_tmp->pName, query_data->query) != 0) {
+ continue;
+ }
+
+ srv_data = &dr_tmp->Data.SRV;
+ srvres = g_new0(PurpleSrvResponse, 1);
+ strncpy(srvres->hostname, srv_data->pNameTarget, 255);
+ srvres->hostname[255] = '\0';
+ srvres->pref = srv_data->wPriority;
+ srvres->port = srv_data->wPort;
+ srvres->weight = srv_data->wWeight;
+
+ lst = g_slist_insert_sorted(lst, srvres, responsecompare);
}
- srv_data = &dr_tmp->Data.SRV;
- srvres = g_new0(PurpleSrvResponse, 1);
- strncpy(srvres->hostname, srv_data->pNameTarget, 255);
- srvres->hostname[255] = '\0';
- srvres->pref = srv_data->wPriority;
- srvres->port = srv_data->wPort;
- srvres->weight = srv_data->wWeight;
-
- lst = g_slist_insert_sorted(lst, srvres, responsecompare);
+ MyDnsRecordListFree(dr, DnsFreeRecordList);
+ query_data->results = lst;
+ } else if (type == T_TXT) {
+ #error IMPLEMENTATION MISSING
+ } else {
+
}
-
- MyDnsRecordListFree(dr, DnsFreeRecordList);
- query_data->results = lst;
}
/* back to main thread */
@@ -328,6 +406,7 @@ purple_srv_resolve(const char *protocol, const char *transport, const char *doma
{
char *query;
PurpleSrvQueryData *query_data;
+ PurpleSrvInternalQuery internal_query;
#ifndef _WIN32
int in[2], out[2];
int pid;
@@ -369,11 +448,16 @@ purple_srv_resolve(const char *protocol, const char *transport, const char *doma
close(out[1]);
close(in[0]);
- if (write(in[1], query, strlen(query)+1) < 0)
+ internal_query.type = T_SRV;
+ strncpy(internal_query.query, query, 255);
+
+ if (write(in[1], &internal_query, sizeof(internal_query)) < 0)
purple_debug_error("dnssrv", "Could not write to SRV resolver\n");
+
query_data = g_new0(PurpleSrvQueryData, 1);
- query_data->cb = cb;
+ query_data->type = T_SRV;
+ query_data->srv_cb = cb;
query_data->extradata = extradata;
query_data->pid = pid;
query_data->fd_out = out[0];
@@ -392,7 +476,104 @@ purple_srv_resolve(const char *protocol, const char *transport, const char *doma
}
query_data = g_new0(PurpleSrvQueryData, 1);
- query_data->cb = cb;
+ query_data->srv_cb = cb;
+ query_data->query = query;
+ query_data->extradata = extradata;
+
+ if (!MyDnsQuery_UTF8 || !MyDnsRecordListFree)
+ query_data->error_message = g_strdup("System missing DNS API (Requires W2K+)\n");
+ else {
+ query_data->resolver = g_thread_create(res_thread, query_data, FALSE, &err);
+ if (query_data->resolver == NULL) {
+ query_data->error_message = g_strdup_printf("SRV thread create failure: %s\n", (err && err->message) ? err->message : "");
+ g_error_free(err);
+ }
+ }
+
+ /* The query isn't going to happen, so finish the SRV lookup now.
+ * Asynchronously call the callback since stuff may not expect
+ * the callback to be called before this returns */
+ if (query_data->error_message != NULL)
+ query_data->handle = purple_timeout_add(0, res_main_thread_cb, query_data);
+
+ return query_data;
+#endif
+}
+
+PurpleSrvQueryData *purple_txt_resolve(const char *owner, const char *domain, PurpleTxtCallback cb, gpointer extradata)
+{
+ char *query;
+ PurpleSrvQueryData *query_data;
+ PurpleSrvInternalQuery internal_query;
+#ifndef _WIN32
+ int in[2], out[2];
+ int pid;
+#else
+ GError* err = NULL;
+ static gboolean initialized = FALSE;
+#endif
+
+ query = g_strdup_printf("%s.%s", owner, domain);
+ purple_debug_info("dnssrv","querying TXT record for %s\n", query);
+
+#ifndef _WIN32
+ if(pipe(in) || pipe(out)) {
+ purple_debug_error("dnssrv", "Could not create pipe\n");
+ g_free(query);
+ cb(NULL, 0, extradata);
+ return NULL;
+ }
+
+ pid = fork();
+ if (pid == -1) {
+ purple_debug_error("dnssrv", "Could not create process!\n");
+ cb(NULL, 0, extradata);
+ g_free(query);
+ return NULL;
+ }
+
+ /* Child */
+ if (pid == 0)
+ {
+ g_free(query);
+
+ close(out[0]);
+ close(in[1]);
+ resolve(in[0], out[1]);
+ /* resolve() does not return */
+ }
+
+ close(out[1]);
+ close(in[0]);
+
+ internal_query.type = T_TXT;
+ strncpy(internal_query.query, query, 255);
+
+ if (write(in[1], &internal_query, sizeof(internal_query)) < 0)
+ purple_debug_error("dnssrv", "Could not write to TXT resolver\n");
+
+ query_data = g_new0(PurpleSrvQueryData, 1);
+ query_data->type = T_TXT;
+ query_data->txt_cb = cb;
+ query_data->extradata = extradata;
+ query_data->pid = pid;
+ query_data->fd_out = out[0];
+ query_data->fd_in = in[1];
+ query_data->handle = purple_input_add(out[0], PURPLE_INPUT_READ, resolved, query_data);
+
+ g_free(query);
+
+ return query_data;
+#else
+ if (!initialized) {
+ MyDnsQuery_UTF8 = (void*) wpurple_find_and_loadproc("dnsapi.dll", "DnsQuery_UTF8");
+ MyDnsRecordListFree = (void*) wpurple_find_and_loadproc(
+ "dnsapi.dll", "DnsRecordListFree");
+ initialized = TRUE;
+ }
+
+ query_data = g_new0(PurpleSrvQueryData, 1);
+ query_data->txt_cb = cb;
query_data->query = query;
query_data->extradata = extradata;
@@ -440,3 +621,9 @@ purple_srv_cancel(PurpleSrvQueryData *query_data)
#endif
g_free(query_data);
}
+
+void
+purple_txt_cancel(PurpleSrvQueryData *query_data)
+{
+ purple_srv_cancel(query_data);
+}
diff --git a/libpurple/dnssrv.h b/libpurple/dnssrv.h
index 4e17ae068c..4d452021bd 100644
--- a/libpurple/dnssrv.h
+++ b/libpurple/dnssrv.h
@@ -28,8 +28,9 @@
extern "C" {
#endif
-typedef struct _PurpleSrvResponse PurpleSrvResponse;
typedef struct _PurpleSrvQueryData PurpleSrvQueryData;
+typedef struct _PurpleSrvResponse PurpleSrvResponse;
+typedef struct _PurpleTxtResponse PurpleTxtResponse;
struct _PurpleSrvResponse {
char hostname[256];
@@ -38,7 +39,12 @@ struct _PurpleSrvResponse {
int pref;
};
+struct _PurpleTxtResponse {
+ char content[256];
+};
+
typedef void (*PurpleSrvCallback)(PurpleSrvResponse *resp, int results, gpointer data);
+typedef void (*PurpleTxtCallback)(PurpleTxtResponse *resp, int results, gpointer data);
/**
* Queries an SRV record.
@@ -58,6 +64,23 @@ PurpleSrvQueryData *purple_srv_resolve(const char *protocol, const char *transpo
*/
void purple_srv_cancel(PurpleSrvQueryData *query_data);
+/**
+ * Queries an TXT record.
+ *
+ * @param owner Name of the protocol (e.g. "_xmppconnect")
+ * @param domain Domain name to query (e.g. "blubb.com")
+ * @param cb A callback which will be called with the results
+ * @param extradata Extra data to be passed to the callback
+ */
+PurpleSrvQueryData *purple_txt_resolve(const char *owner, const char *domain, PurpleTxtCallback cb, gpointer extradata);
+
+/**
+ * Cancel an TXT DNS query.
+ *
+ * @param query_data The request to cancel.
+ */
+void purple_txt_cancel(PurpleSrvQueryData *query_data);
+
#ifdef __cplusplus
}
#endif
diff --git a/libpurple/protocols/jabber/bosh.c b/libpurple/protocols/jabber/bosh.c
new file mode 100644
index 0000000000..544bce4a8f
--- /dev/null
+++ b/libpurple/protocols/jabber/bosh.c
@@ -0,0 +1,49 @@
+/*
+ * purple - Jabber Protocol Plugin
+ *
+ * Copyright (C) 2008, Tobias Markmann <tmarkmann@googlemail.com>
+ *
+ * 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, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA
+ *
+ */
+#include "internal.h"
+#include "cipher.h"
+#include "debug.h"
+#include "imgstore.h"
+#include "prpl.h"
+#include "notify.h"
+#include "request.h"
+#include "util.h"
+#include "xmlnode.h"
+
+#include "buddy.h"
+#include "chat.h"
+#include "jabber.h"
+#include "iq.h"
+#include "presence.h"
+#include "xdata.h"
+#include "pep.h"
+#include "adhoccommands.h"
+
+PurpleHTTPHeaderField* jabber_bosh_http_header_field(const char *name, *const char *value) {
+ PurpleHTTPHeaderField *tmp = g_new0(PurpleHTTPHeaderField, 0);
+ tmp->name = g_strdup(name);
+ tmp->value = g_strdup(value);
+ return tmp;
+}
+
+void jabber_bosh_http_connection_connect(PurpleHTTPConnection *conn) {
+
+} \ No newline at end of file
diff --git a/libpurple/protocols/jabber/bosh.h b/libpurple/protocols/jabber/bosh.h
new file mode 100644
index 0000000000..34c0697c46
--- /dev/null
+++ b/libpurple/protocols/jabber/bosh.h
@@ -0,0 +1,75 @@
+/**
+ * @file bosh.h Buddy handlers
+ *
+ * purple
+ *
+ * Copyright (C) 2008, Tobias Markmann <tmarkmann@googlemail.com>
+ *
+ * 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, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA
+ */
+#ifndef _PURPLE_JABBER_BOSH_H_
+#define _PURPLE_JABBER_BOSH_H_
+
+#include <glib.h>
+
+typedef struct _PurpleHTTPRequest PurpleHTTPRequest;
+typedef struct _PurpleHTTPResponse PurpleHTTPResponse;
+typedef struct _PurpleHTTPHeaderField PurpleHTTPHeaderField;
+
+typedef void (*PurpleHTTPRequestCallback)(PurpleHTTPRequest *req, PurpleHTTPResponse *res, void *userdata);
+
+typedef struct {
+ int fd;
+ PurpleConnection *conn;
+ GQueue *requests;
+ void *userdata;
+} PurpleHTTPConnection;
+
+typedef struct {
+ char *url;
+ gboolean pipelining;
+ PurpleHTTPConnection *conn_a;
+ PurpleHTTPConnection *conn_b;
+} PurpleBOSHConnection;
+
+struct _PurpleHTTPRequest {
+ PurpleHTTPRequestCallback cb;
+ char *method;
+ char *url;
+ GList *header;
+ char *data;
+ void *userdata;
+};
+
+struct _PurpleHTTPResponse {
+ int status;
+ GList *header;
+ char *data;
+};
+
+struct _PurpleHTTPHeaderField {
+ char *name;
+ char *value;
+};
+
+PurpleHTTPHeaderField *jabber_bosh_http_header_field(const char *name, const char *value);
+
+void jabber_bosh_http_connection_connect(PurpleHTTPConnection *conn);
+void jabber_bosh_http_send_request(PurpleHTTPConnection *conn, PurpleHTTPRequest *req);
+void jabber_bosh_http_connection_clean(PurpleHTTPConnection *conn);
+
+void jabber_bosh_http_request_init(PurpleHTTPRequest *req, const char *method, const char *url, PurpleHTTPRequestCallback cb, void *userdata);
+void jabber_bosh_http_request_clean(PurpleHTTPRequest *req);
+#endif /* _PURPLE_JABBER_BOSH_H_ */
diff --git a/libpurple/protocols/jabber/caps.c b/libpurple/protocols/jabber/caps.c
index 6fc4c78365..adf805fc33 100644
--- a/libpurple/protocols/jabber/caps.c
+++ b/libpurple/protocols/jabber/caps.c
@@ -21,7 +21,9 @@
#include "internal.h"
+#include <glib.h>
#include "caps.h"
+#include "cipher.h"
#include <string.h>
#include "internal.h"
#include "util.h"
@@ -29,43 +31,38 @@
#define JABBER_CAPS_FILENAME "xmpp-caps.xml"
-static GHashTable *capstable = NULL; /* JabberCapsKey -> JabberCapsValue */
-
-typedef struct _JabberCapsKey {
- char *node;
- char *ver;
-} JabberCapsKey;
-
-typedef struct _JabberCapsValueExt {
- GList *identities; /* JabberCapsIdentity */
- GList *features; /* char * */
-} JabberCapsValueExt;
+GHashTable *capstable = NULL; /* JabberCapsKey -> JabberCapsValue */
+static gchar *caps_hash = NULL;
+#if 0
typedef struct _JabberCapsValue {
GList *identities; /* JabberCapsIdentity */
GList *features; /* char * */
GHashTable *ext; /* char * -> JabberCapsValueExt */
} JabberCapsValue;
+#endif
+typedef struct _JabberCapsClientInfo JabberCapsValue;
static guint jabber_caps_hash(gconstpointer key) {
const JabberCapsKey *name = key;
guint nodehash = g_str_hash(name->node);
guint verhash = g_str_hash(name->ver);
-
- return nodehash ^ verhash;
+ guint hashhash = g_str_hash(name->hash);
+ return nodehash ^ verhash ^ hashhash;
}
static gboolean jabber_caps_compare(gconstpointer v1, gconstpointer v2) {
const JabberCapsKey *name1 = v1;
const JabberCapsKey *name2 = v2;
-
- return strcmp(name1->node,name2->node) == 0 && strcmp(name1->ver,name2->ver) == 0;
+
+ return strcmp(name1->node,name2->node) == 0 && strcmp(name1->ver,name2->ver) == 0 && strcmp(name1->hash,name2->hash) == 0;
}
-static void jabber_caps_destroy_key(gpointer key) {
+void jabber_caps_destroy_key(gpointer key) {
JabberCapsKey *keystruct = key;
g_free(keystruct->node);
g_free(keystruct->ver);
+ g_free(keystruct->hash);
g_free(keystruct);
}
@@ -84,7 +81,12 @@ static void jabber_caps_destroy_value(gpointer value) {
g_free(valuestruct->features->data);
valuestruct->features = g_list_delete_link(valuestruct->features,valuestruct->features);
}
- g_hash_table_destroy(valuestruct->ext);
+
+ while(valuestruct->forms) {
+ g_free(valuestruct->forms->data);
+ valuestruct->forms = g_list_delete_link(valuestruct->forms,valuestruct->forms);
+ }
+ //g_hash_table_destroy(valuestruct->ext);
g_free(valuestruct);
}
@@ -111,6 +113,7 @@ static void jabber_caps_load(void);
void jabber_caps_init(void) {
capstable = g_hash_table_new_full(jabber_caps_hash, jabber_caps_compare, jabber_caps_destroy_key, jabber_caps_destroy_value);
jabber_caps_load();
+ jabber_caps_calculate_own_hash();
}
static void jabber_caps_load(void) {
@@ -134,7 +137,7 @@ static void jabber_caps_load(void) {
xmlnode *child;
key->node = g_strdup(xmlnode_get_attrib(client,"node"));
key->ver = g_strdup(xmlnode_get_attrib(client,"ver"));
- value->ext = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, jabber_caps_ext_destroy_value);
+ key->hash = g_strdup(xmlnode_get_attrib(client,"hash"));
for(child = client->child; child; child = child->next) {
if(child->type != XMLNODE_TYPE_TAG)
continue;
@@ -154,36 +157,8 @@ static void jabber_caps_load(void) {
id->name = g_strdup(name);
value->identities = g_list_append(value->identities,id);
- } else if(!strcmp(child->name,"ext")) {
- const char *identifier = xmlnode_get_attrib(child, "identifier");
- if(identifier) {
- xmlnode *extchild;
-
- JabberCapsValueExt *extvalue = g_new0(JabberCapsValueExt, 1);
-
- for(extchild = child->child; extchild; extchild = extchild->next) {
- if(extchild->type != XMLNODE_TYPE_TAG)
- continue;
- if(!strcmp(extchild->name,"feature")) {
- const char *var = xmlnode_get_attrib(extchild, "var");
- if(!var)
- continue;
- extvalue->features = g_list_append(extvalue->features,g_strdup(var));
- } else if(!strcmp(extchild->name,"identity")) {
- const char *category = xmlnode_get_attrib(extchild, "category");
- const char *type = xmlnode_get_attrib(extchild, "type");
- const char *name = xmlnode_get_attrib(extchild, "name");
-
- JabberCapsIdentity *id = g_new0(JabberCapsIdentity, 1);
- id->category = g_strdup(category);
- id->type = g_strdup(type);
- id->name = g_strdup(name);
-
- extvalue->identities = g_list_append(extvalue->identities,id);
- }
- }
- g_hash_table_replace(value->ext, g_strdup(identifier), extvalue);
- }
+ } else if(!strcmp(child->name,"x")) {
+ value->forms = g_list_append(value->forms, xmlnode_copy(child));
}
}
g_hash_table_replace(capstable, key, value);
@@ -192,6 +167,7 @@ static void jabber_caps_load(void) {
xmlnode_free(capsdata);
}
+#if 0
static void jabber_caps_store_ext(gpointer key, gpointer value, gpointer user_data) {
const char *extname = key;
JabberCapsValueExt *props = value;
@@ -216,17 +192,18 @@ static void jabber_caps_store_ext(gpointer key, gpointer value, gpointer user_da
xmlnode_set_attrib(feature, "var", feat);
}
}
+#endif
static void jabber_caps_store_client(gpointer key, gpointer value, gpointer user_data) {
JabberCapsKey *clientinfo = key;
JabberCapsValue *props = value;
xmlnode *root = user_data;
- xmlnode *client = xmlnode_new_child(root,"client");
+ xmlnode *client = xmlnode_new_child(root, "client");
GList *iter;
- xmlnode_set_attrib(client,"node",clientinfo->node);
- xmlnode_set_attrib(client,"ver",clientinfo->ver);
-
+ xmlnode_set_attrib(client, "node", clientinfo->node);
+ xmlnode_set_attrib(client, "ver", clientinfo->ver);
+ xmlnode_set_attrib(client, "hash", clientinfo->hash);
for(iter = props->identities; iter; iter = g_list_next(iter)) {
JabberCapsIdentity *id = iter->data;
xmlnode *identity = xmlnode_new_child(client, "identity");
@@ -242,16 +219,20 @@ static void jabber_caps_store_client(gpointer key, gpointer value, gpointer user
xmlnode_set_attrib(feature, "var", feat);
}
- g_hash_table_foreach(props->ext,jabber_caps_store_ext,client);
+ for(iter = props->forms; iter; iter = g_list_next(iter)) {
+ xmlnode *xdata = iter->data;
+ xmlnode_insert_child(client, xmlnode_copy(xdata));
+ }
}
static void jabber_caps_store(void) {
char *str;
+ int length = 0;
xmlnode *root = xmlnode_new("capabilities");
g_hash_table_foreach(capstable, jabber_caps_store_client, root);
- str = xmlnode_to_formatted_str(root, NULL);
+ str = xmlnode_to_formatted_str(root, &length);
xmlnode_free(root);
- purple_util_write_data_to_file(JABBER_CAPS_FILENAME, str, -1);
+ purple_util_write_data_to_file(JABBER_CAPS_FILENAME, str, length);
g_free(str);
}
@@ -290,7 +271,7 @@ static JabberCapsClientInfo *jabber_caps_collect_info(const char *node, const ch
result->features = g_list_append(result->features,newfeat);
}
-
+#if 0
for(iter = ext; iter; iter = g_list_next(iter)) {
const char *extname = iter->data;
JabberCapsValueExt *extinfo = g_hash_table_lookup(caps->ext,extname);
@@ -314,6 +295,7 @@ static JabberCapsClientInfo *jabber_caps_collect_info(const char *node, const ch
}
}
}
+#endif
return result;
}
@@ -346,7 +328,7 @@ typedef struct _jabber_caps_cbplususerdata {
char *who;
char *node;
char *ver;
- GList *ext;
+ char *hash;
unsigned extOutstanding;
} jabber_caps_cbplususerdata;
@@ -355,6 +337,7 @@ typedef struct jabber_ext_userdata {
char *node;
} jabber_ext_userdata;
+#if 0
static void jabber_caps_get_info_check_completion(jabber_caps_cbplususerdata *userdata) {
if(userdata->extOutstanding == 0) {
userdata->cb(jabber_caps_collect_info(userdata->node, userdata->ver, userdata->ext), userdata->user_data);
@@ -368,7 +351,8 @@ static void jabber_caps_get_info_check_completion(jabber_caps_cbplususerdata *us
g_free(userdata);
}
}
-
+#endif
+#if 0
static void jabber_caps_ext_iqcb(JabberStream *js, xmlnode *packet, gpointer data) {
/* collect data and fetch all exts */
xmlnode *query = xmlnode_get_child_with_namespace(packet, "query", "http://jabber.org/protocol/disco#info");
@@ -379,7 +363,7 @@ static void jabber_caps_ext_iqcb(JabberStream *js, xmlnode *packet, gpointer dat
--userdata->extOutstanding;
/* TODO: Better error handling */
-
+ printf("\n\tjabber_caps_ext_iqcb for %s", xmlnode_get_attrib(packet, "from"));
if(node && query) {
const char *key;
JabberCapsValue *client;
@@ -423,8 +407,6 @@ static void jabber_caps_ext_iqcb(JabberStream *js, xmlnode *packet, gpointer dat
value->identities = g_list_append(value->identities,id);
}
}
- g_hash_table_replace(client->ext, g_strdup(key), value);
-
jabber_caps_store();
}
@@ -432,26 +414,61 @@ static void jabber_caps_ext_iqcb(JabberStream *js, xmlnode *packet, gpointer dat
g_free(extuserdata);
jabber_caps_get_info_check_completion(userdata);
}
-
+#endif
static void jabber_caps_client_iqcb(JabberStream *js, xmlnode *packet, gpointer data) {
/* collect data and fetch all exts */
xmlnode *query = xmlnode_get_child_with_namespace(packet, "query",
"http://jabber.org/protocol/disco#info");
xmlnode *child;
- GList *iter;
jabber_caps_cbplususerdata *userdata = data;
/* TODO: Better error checking! */
-
+ if (!strcmp(xmlnode_get_attrib(packet, "type"), "error"))return;
if (query) {
+ // check hash
+ JabberCapsClientInfo *info = jabber_caps_parse_client_info(query);
+ gchar *hash = 0;
+ if (!strcmp(userdata->hash, "sha-1")) {
+ hash = jabber_caps_calcualte_hash(info, "sha1");
+ } else if (!strcmp(userdata->hash, "md5")) {
+ hash = jabber_caps_calcualte_hash(info, "md5");
+ } else {
+ // clean up
+ return;
+ }
+
+ printf("\n\tfrom: %s", xmlnode_get_attrib(packet, "from"));
+ printf("\n\tnode: %s", xmlnode_get_attrib(query, "node"));
+ printf("\n\tcalculated key: %s", hash);
+ printf("\n\thash: %s", userdata->hash);
+ printf("\n");
+
+ if (strcmp(hash, userdata->ver)) {
+ g_free(info);
+ g_free(hash);
+ printf("\n! ! ! invalid hash ! ! !");
+ return;
+ }
+
+ g_free(hash);
+
JabberCapsValue *value = g_new0(JabberCapsValue, 1);
JabberCapsKey *key = g_new0(JabberCapsKey, 1);
-
+#if 0
value->ext = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, jabber_caps_ext_destroy_value);
-
+#endif
key->node = g_strdup(userdata->node);
key->ver = g_strdup(userdata->ver);
-
+ key->hash = g_strdup(userdata->hash);
+
+ /* check whether it's stil not in the table */
+ if (g_hash_table_lookup(capstable, key)) {
+ jabber_caps_destroy_key(key);
+ g_free(value);
+ return;
+ }
+
+
for(child = query->child; child; child = child->next) {
if(child->type != XMLNODE_TYPE_TAG)
continue;
@@ -464,19 +481,23 @@ static void jabber_caps_client_iqcb(JabberStream *js, xmlnode *packet, gpointer
const char *category = xmlnode_get_attrib(child, "category");
const char *type = xmlnode_get_attrib(child, "type");
const char *name = xmlnode_get_attrib(child, "name");
-
+
JabberCapsIdentity *id = g_new0(JabberCapsIdentity, 1);
id->category = g_strdup(category);
id->type = g_strdup(type);
id->name = g_strdup(name);
value->identities = g_list_append(value->identities,id);
+ } else if(!strcmp(child->name, "x")) {
+ value->forms = g_list_append(value->forms, xmlnode_copy(child));
}
}
+
g_hash_table_replace(capstable, key, value);
jabber_caps_store();
}
+#if 0
/* fetch all exts */
for(iter = userdata->ext; iter; iter = g_list_next(iter)) {
JabberIq *iq = jabber_iq_new_query(js, JABBER_IQ_GET, "http://jabber.org/protocol/disco#info");
@@ -494,37 +515,28 @@ static void jabber_caps_client_iqcb(JabberStream *js, xmlnode *packet, gpointer
}
jabber_caps_get_info_check_completion(userdata);
+#endif
}
-void jabber_caps_get_info(JabberStream *js, const char *who, const char *node, const char *ver, const char *ext, jabber_caps_get_info_cb cb, gpointer user_data) {
+void jabber_caps_get_info(JabberStream *js, const char *who, const char *node, const char *ver, const char *hash, jabber_caps_get_info_cb cb, gpointer user_data) {
JabberCapsValue *client;
JabberCapsKey *key = g_new0(JabberCapsKey, 1);
- char *originalext = g_strdup(ext);
jabber_caps_cbplususerdata *userdata = g_new0(jabber_caps_cbplususerdata, 1);
userdata->cb = cb;
userdata->user_data = user_data;
userdata->who = g_strdup(who);
userdata->node = g_strdup(node);
userdata->ver = g_strdup(ver);
+ userdata->hash = g_strdup(hash);
- if(originalext) {
- int i;
- gchar **splat = g_strsplit(originalext, " ", 0);
- for(i =0; splat[i]; i++) {
- userdata->ext = g_list_append(userdata->ext, splat[i]);
- ++userdata->extOutstanding;
- }
- g_free(splat);
- }
- g_free(originalext);
-
- key->node = (char *)node;
- key->ver = (char *)ver;
-
+ key->node = g_strdup(node);
+ key->ver = g_strdup(ver);
+ key->hash = g_strdup(hash);
+
client = g_hash_table_lookup(capstable, key);
- g_free(key);
-
+ g_hash_table_replace(jabber_contact_info, g_strdup(who), key);
+
if(!client) {
JabberIq *iq = jabber_iq_new_query(js,JABBER_IQ_GET,"http://jabber.org/protocol/disco#info");
xmlnode *query = xmlnode_get_child_with_namespace(iq->node,"query","http://jabber.org/protocol/disco#info");
@@ -535,6 +547,8 @@ void jabber_caps_get_info(JabberStream *js, const char *who, const char *node, c
jabber_iq_set_callback(iq,jabber_caps_client_iqcb,userdata);
jabber_iq_send(iq);
+ }
+#if 0
} else {
GList *iter;
/* fetch unknown exts only */
@@ -568,5 +582,279 @@ void jabber_caps_get_info(JabberStream *js, const char *who, const char *node, c
/* maybe we have all data available anyways? This is the ideal case where no network traffic is necessary */
jabber_caps_get_info_check_completion(userdata);
}
+#endif
+}
+
+static gint jabber_caps_jabber_identity_compare(gconstpointer a, gconstpointer b) {
+ const JabberIdentity *ac;
+ const JabberIdentity *bc;
+ gint cat_cmp;
+ gint typ_cmp;
+
+ ac = a;
+ bc = b;
+
+ if ((cat_cmp = strcmp(ac->category, bc->category)) == 0) {
+ if ((typ_cmp = strcmp(ac->type, bc->type)) == 0) {
+ return strcmp(ac->lang, bc->lang);
+ } else {
+ return typ_cmp;
+ }
+ } else {
+ return cat_cmp;
+ }
+}
+
+static gint jabber_caps_jabber_feature_compare(gconstpointer a, gconstpointer b) {
+ const JabberFeature *ac;
+ const JabberFeature *bc;
+
+ ac = a;
+ bc = b;
+
+ return strcmp(ac->namespace, bc->namespace);
+}
+
+static gint jabber_caps_string_compare(gconstpointer a, gconstpointer b) {
+ const gchar *ac;
+ const gchar *bc;
+
+ ac = a;
+ bc = b;
+
+ return strcmp(ac, bc);
+}
+
+gchar *jabber_caps_get_formtype(const xmlnode *x) {
+ xmlnode *formtypefield;
+ formtypefield = xmlnode_get_child(x, "field");
+ while (formtypefield && strcmp(xmlnode_get_attrib(formtypefield, "var"), "FORM_TYPE")) formtypefield = xmlnode_get_next_twin(formtypefield);
+ formtypefield = xmlnode_get_child(formtypefield, "value");
+ return xmlnode_get_data(formtypefield);;
+}
+
+static gint jabber_caps_jabber_xdata_compare(gconstpointer a, gconstpointer b) {
+ const xmlnode *aformtypefield = a;
+ const xmlnode *bformtypefield = b;
+ char *aformtype;
+ char *bformtype;
+ int result;
+
+ aformtype = jabber_caps_get_formtype(aformtypefield);
+ bformtype = jabber_caps_get_formtype(bformtypefield);
+
+ result = strcmp(aformtype, bformtype);
+ g_free(aformtype);
+ g_free(bformtype);
+ return result;
+}
+
+JabberCapsClientInfo *jabber_caps_parse_client_info(xmlnode *query) {
+ xmlnode *child;
+
+ if (!query) return 0;
+ if (strcmp(query->xmlns,"http://jabber.org/protocol/disco#info")) return 0;
+
+ JabberCapsClientInfo *info = g_new0(JabberCapsClientInfo, 1);
+
+ for(child = query->child; child; child = child->next) {
+ if (!strcmp(child->name,"identity")) {
+ /* parse identity */
+ const char *category = xmlnode_get_attrib(child, "category");
+ const char *type = xmlnode_get_attrib(child, "type");
+ const char *name = xmlnode_get_attrib(child, "name");
+
+ JabberCapsIdentity *id = g_new0(JabberCapsIdentity, 1);
+ id->category = g_strdup(category);
+ id->type = g_strdup(type);
+ id->name = g_strdup(name);
+
+ info->identities = g_list_append(info->identities, id);
+ } else if (!strcmp(child->name, "feature")) {
+ /* parse feature */
+ const char *var = xmlnode_get_attrib(child, "var");
+ if(!var)
+ continue;
+ info->features = g_list_append(info->features, g_strdup(var));
+ } else if (!strcmp(child->name, "x")) {
+ if (!strcmp(child->xmlns, "jabber:x:data")) {
+ /* x-data form */
+ xmlnode *dataform = xmlnode_copy(child);
+ info->forms = g_list_append(info->forms, dataform);
+ }
+ }
+ }
+ return info;
+}
+
+static gint jabber_caps_xdata_field_compare(gconstpointer a, gconstpointer b) {
+ const JabberDataFormField *ac;
+ const JabberDataFormField *bc;
+
+ ac = a;
+ bc = b;
+
+ return strcmp(ac->var, bc->var);
+}
+
+GList *jabber_caps_xdata_get_fields(const xmlnode *x) {
+ GList *fields = 0;
+ xmlnode *field;
+ xmlnode *value;
+ JabberDataFormField *xdatafield;
+
+ if(!x) return 0;
+
+ for(field = xmlnode_get_child(x, "field"); field != 0; field = xmlnode_get_next_twin(field)) {
+ xdatafield = g_new0(JabberDataFormField, 1);
+ xdatafield->var = g_strdup(xmlnode_get_attrib(field, "var"));
+ for(value = xmlnode_get_child(field, "value"); value != 0; value = xmlnode_get_next_twin(value)) {
+ gchar *val = xmlnode_get_data(value);
+ xdatafield->values = g_list_append(xdatafield->values, val);
+ }
+ xdatafield->values = g_list_sort(xdatafield->values, jabber_caps_string_compare);
+ fields = g_list_append(fields, xdatafield);
+ }
+ fields = g_list_sort(fields, jabber_caps_xdata_field_compare);
+ return fields;
+}
+
+gchar *jabber_caps_verification_append(gchar *verification_string, gchar *string) {
+ gchar *verification;
+ verification = g_strconcat(verification_string, string, "<", NULL);
+ g_free(verification_string);
+ return verification;
+}
+
+gchar *jabber_caps_calcualte_hash(JabberCapsClientInfo *info, const char *hash) {
+ GList *identities;
+ GList *features;
+ GList *xdata;
+ gchar *verification = 0;
+ gchar *feature_string;
+ gchar *free_verification;
+ gchar *identity_string;
+ PurpleCipherContext *context;
+ guint8 checksum[20];
+ gsize checksum_size = 20;
+
+ if (!info) return 0;
+
+ /* sort identities, features and x-data forms */
+ info->identities = g_list_sort(info->identities, jabber_caps_jabber_identity_compare);
+ info->features = g_list_sort(info->features, jabber_caps_string_compare);
+ info->forms = g_list_sort(info->forms, jabber_caps_jabber_xdata_compare);
+
+ /* concat identities to the verification string */
+ for(identities = info->identities; identities; identities = identities->next) {
+ JabberIdentity *ident = (JabberIdentity*)identities->data;
+ identity_string = g_strdup_printf("%s/%s//%s<", ident->category, ident->type, ident->name);
+ free_verification = verification;
+ if(verification == 0) verification = g_strdup(identity_string);
+ else verification = g_strconcat(verification, identity_string, NULL);
+ g_free(identity_string);
+ g_free(free_verification);
+ }
+
+ /* concat features to the verification string */
+ for(features = info->features; features; features = features->next) {
+ feature_string = g_strdup_printf("%s", (gchar*)features->data);
+ verification = jabber_caps_verification_append(verification, feature_string);
+ g_free(feature_string);
+ }
+
+ /* concat x-data forms to the verification string */
+ for(xdata = info->forms; xdata; xdata = xdata->next) {
+ gchar *formtype = 0;
+ GList *fields;
+ /* append FORM_TYPE's field value to the verification string */
+ formtype = jabber_caps_get_formtype((xmlnode*)xdata->data);
+ verification = jabber_caps_verification_append(verification, formtype);
+ g_free(formtype);
+
+ for(fields = jabber_caps_xdata_get_fields((xmlnode*)(xdata->data)); fields != 0; fields = fields->next) {
+ GList *value;
+ JabberDataFormField *field = (JabberDataFormField*)fields->data;
+ if(strcmp(field->var, "FORM_TYPE")) {
+ /* Append the value of the "var" attribute, followed by the '<' character. */
+ verification = jabber_caps_verification_append(verification, field->var);
+ /* For each <value/> element, append the XML character data, followed by the '<' character. */
+ for(value = field->values; value != 0; value = value->next) {
+ verification = jabber_caps_verification_append(verification, value->data);
+ }
+ }
+ for(value = field->values; value != 0; value = value->next) {
+ g_free(value->data);
+ }
+ g_free(field->var);
+ g_list_free(field->values);
+ }
+ g_list_free(fields);
+ }
+
+ /* generate hash */
+ context = purple_cipher_context_new_by_name(hash, NULL);
+ if (context == NULL) {
+ //purple_debug_error("jabber", "Could not find cipher\n");
+ return 0;
+ }
+ purple_cipher_context_append(context, (guchar*)verification, strlen(verification));
+
+ if (!purple_cipher_context_digest(context, strlen(verification), checksum, &checksum_size)) {
+ //purple_debug_error("util", "Failed to get digest.\n");
+ }
+ purple_cipher_context_destroy(context);
+
+ /* apply Base64 on hash */
+
+ g_free(verification);
+ verification = purple_base64_encode(checksum, checksum_size);
+
+ return verification;
+}
+
+void jabber_caps_calculate_own_hash(JabberStream *js) {
+ JabberCapsClientInfo *info;
+ GList *iter = 0;
+ GList *features = 0;
+
+ if (jabber_identities == 0 && jabber_features == 0) return;
+
+ /* sort features */
+ if (jabber_features) {
+ for (iter = jabber_features; iter; iter = iter->next) {
+ JabberFeature *feat = iter->data;
+ if(feat->is_enabled == NULL || feat->is_enabled(js, feat->namespace) == TRUE) {
+ features = g_list_append(features, feat->namespace);
+ }
+ }
+ }
+
+ info = g_new0(JabberCapsClientInfo, 1);
+ info->features = features;
+ info->identities = jabber_identities;
+ info->forms = 0;
+
+ if (caps_hash) g_free(caps_hash);
+ caps_hash = jabber_caps_calcualte_hash(info, "sha1");
+ g_free(info);
+ g_list_free(features);
+}
+
+const gchar* jabber_caps_get_own_hash() {
+ return caps_hash;
+}
+
+void jabber_caps_broadcast_change() {
+ GList *active_accounts = purple_accounts_get_all_active();
+ for (active_accounts = purple_accounts_get_all_active(); active_accounts; active_accounts = active_accounts->next) {
+ PurpleAccount *account = active_accounts->data;
+ if (!strcmp(account->protocol_id, "jabber")) {
+ PurpleConnection *conn = account->gc;
+ JabberStream *js = conn->proto_data;
+ xmlnode *presence = jabber_presence_create_js(js, JABBER_BUDDY_STATE_UNKNOWN, 0, 0);
+ jabber_send(js, presence);
+ }
+ }
}
diff --git a/libpurple/protocols/jabber/caps.h b/libpurple/protocols/jabber/caps.h
index 07cf0baaaf..2fb61d3583 100644
--- a/libpurple/protocols/jabber/caps.h
+++ b/libpurple/protocols/jabber/caps.h
@@ -27,23 +27,70 @@ typedef struct _JabberCapsClientInfo JabberCapsClientInfo;
#include "jabber.h"
/* Implementation of XEP-0115 */
+extern GHashTable *capstable;
-typedef struct _JabberCapsIdentity {
- char *category;
- char *type;
- char *name;
-} JabberCapsIdentity;
+typedef struct _JabberIdentity JabberCapsIdentity;
struct _JabberCapsClientInfo {
GList *identities; /* JabberCapsIdentity */
GList *features; /* char * */
+ GList *forms; /* xmlnode * */
};
+typedef struct _JabberCapsClientInfo JabberCapsValueExt;
+
+typedef struct _JabberDataFormField {
+ gchar *var;
+ GList *values;
+} JabberDataFormField;
+
+typedef struct _JabberCapsKey {
+ char *node;
+ char *ver;
+ char *hash;
+} JabberCapsKey;
+
typedef void (*jabber_caps_get_info_cb)(JabberCapsClientInfo *info, gpointer user_data);
void jabber_caps_init(void);
-void jabber_caps_get_info(JabberStream *js, const char *who, const char *node, const char *ver, const char *ext, jabber_caps_get_info_cb cb, gpointer user_data);
+void jabber_caps_destroy_key(gpointer value);
+
+/**
+ * Main entity capabilites function to get the capabilities of a contact.
+ */
+void jabber_caps_get_info(JabberStream *js, const char *who, const char *node, const char *ver, const char *hash, jabber_caps_get_info_cb cb, gpointer user_data);
void jabber_caps_free_clientinfo(JabberCapsClientInfo *clientinfo);
+/**
+ * Processes a query-node and returns a JabberCapsClientInfo object with all relevant info.
+ *
+ * @param query A query object.
+ * @return A JabberCapsClientInfo object.
+ */
+JabberCapsClientInfo *jabber_caps_parse_client_info(xmlnode *query);
+
+/**
+ * Takes a JabberCapsClientInfo pointer and returns the caps hash according to
+ * XEP-0115 Version 1.5.
+ *
+ * @param info A JabberCapsClientInfo pointer.
+ * @param hash Hash cipher to be used. Either sha-1 or md5.
+ * @return The base64 encoded SHA-1 hash; needs to be freed if not needed
+ * any furthermore.
+ */
+gchar *jabber_caps_calcualte_hash(JabberCapsClientInfo *info, const char *hash);
+
+void jabber_caps_calculate_own_hash();
+
+/** Get the current caps hash.
+ * @ret hash
+**/
+const gchar* jabber_caps_get_own_hash();
+
+/**
+ *
+ */
+void jabber_caps_broadcast_change();
+
#endif /* _PURPLE_JABBER_CAPS_H_ */
diff --git a/libpurple/protocols/jabber/disco.c b/libpurple/protocols/jabber/disco.c
index e0fb8e347e..9ebc96d5f7 100644
--- a/libpurple/protocols/jabber/disco.c
+++ b/libpurple/protocols/jabber/disco.c
@@ -88,7 +88,6 @@ jabber_disco_bytestream_server_cb(JabberStream *js, xmlnode *packet, gpointer da
void jabber_disco_info_parse(JabberStream *js, xmlnode *packet) {
const char *from = xmlnode_get_attrib(packet, "from");
const char *type = xmlnode_get_attrib(packet, "type");
-
if(!from || !type)
return;
@@ -98,6 +97,10 @@ void jabber_disco_info_parse(JabberStream *js, xmlnode *packet) {
xmlnode *in_query;
const char *node = NULL;
+ const char *node_uri = NULL;
+
+ // create custom caps node URI
+ node_uri = g_strconcat(CAPS0115_NODE, "#", jabber_caps_get_own_hash(), NULL);
if((in_query = xmlnode_get_child(packet, "query"))) {
node = xmlnode_get_attrib(in_query, "node");
@@ -115,90 +118,38 @@ void jabber_disco_info_parse(JabberStream *js, xmlnode *packet) {
if(node)
xmlnode_set_attrib(query, "node", node);
- if(!node || !strcmp(node, CAPS0115_NODE "#" VERSION)) {
- identity = xmlnode_new_child(query, "identity");
- xmlnode_set_attrib(identity, "category", "client");
- xmlnode_set_attrib(identity, "type", "pc"); /* XXX: bot, console,
- * handheld, pc, phone,
- * web */
- xmlnode_set_attrib(identity, "name", PACKAGE);
-
- SUPPORT_FEATURE("jabber:iq:last")
- SUPPORT_FEATURE("jabber:iq:oob")
- SUPPORT_FEATURE("jabber:iq:time")
- SUPPORT_FEATURE("xmpp:urn:time")
- SUPPORT_FEATURE("jabber:iq:version")
- SUPPORT_FEATURE("jabber:x:conference")
- SUPPORT_FEATURE("http://jabber.org/protocol/bytestreams")
- SUPPORT_FEATURE("http://jabber.org/protocol/disco#info")
- SUPPORT_FEATURE("http://jabber.org/protocol/disco#items")
-#if 0
- SUPPORT_FEATURE("http://jabber.org/protocol/ibb")
-#endif
- SUPPORT_FEATURE("http://jabber.org/protocol/muc")
- SUPPORT_FEATURE("http://jabber.org/protocol/muc#user")
- SUPPORT_FEATURE("http://jabber.org/protocol/si")
- SUPPORT_FEATURE("http://jabber.org/protocol/si/profile/file-transfer")
- SUPPORT_FEATURE("http://jabber.org/protocol/xhtml-im")
- SUPPORT_FEATURE("urn:xmpp:ping")
- SUPPORT_FEATURE("http://www.xmpp.org/extensions/xep-0199.html#ns")
-
- if(!node) { /* non-caps disco#info, add all enabled extensions */
- GList *features;
- for(features = jabber_features; features; features = features->next) {
- JabberFeature *feat = (JabberFeature*)features->data;
- if(feat->is_enabled == NULL || feat->is_enabled(js, feat->shortname, feat->namespace) == TRUE)
- SUPPORT_FEATURE(feat->namespace);
- }
+
+ if(!node || !strcmp(node, node_uri)) {
+ GList *identities;
+ for(identities = jabber_identities; identities; identities = identities->next) {
+ JabberIdentity *ident = (JabberIdentity*)identities->data;
+ identity = xmlnode_new_child(query, "identity");
+ xmlnode_set_attrib(identity, "category", ident->category);
+ xmlnode_set_attrib(identity, "type", ident->type);
+ if (ident->name != 0) xmlnode_set_attrib(identity, "name", ident->name);
+ }
+ GList *features;
+ for(features = jabber_features; features; features = features->next) {
+ JabberFeature *feat = (JabberFeature*)features->data;
+ if(feat->is_enabled == NULL || feat->is_enabled(js, feat->namespace) == TRUE) {
+ feature = xmlnode_new_child(query, "feature");
+ xmlnode_set_attrib(feature, "var", feat->namespace);
+ }
}
} else {
- const char *ext = NULL;
- unsigned pos;
- unsigned nodelen = strlen(node);
- unsigned capslen = strlen(CAPS0115_NODE);
- /* do a basic plausability check */
- if(nodelen > capslen+1) {
- /* verify that the string is CAPS0115#<ext> and get the pointer to the ext part */
- for(pos = 0; pos < capslen+1; ++pos) {
- if(pos == capslen) {
- if(node[pos] == '#')
- ext = &node[pos+1];
- else
- break;
- } else if(node[pos] != CAPS0115_NODE[pos])
- break;
- }
+ xmlnode *error, *inf;
- if(ext != NULL) {
- /* look for that ext */
- GList *features;
- for(features = jabber_features; features; features = features->next) {
- JabberFeature *feat = (JabberFeature*)features->data;
- if(!strcmp(feat->shortname, ext)) {
- SUPPORT_FEATURE(feat->namespace);
- break;
- }
- }
- if(features == NULL)
- ext = NULL;
- }
- }
+ /* XXX: gross hack, implement jabber_iq_set_type or something */
+ xmlnode_set_attrib(iq->node, "type", "error");
+ iq->type = JABBER_IQ_ERROR;
- if(ext == NULL) {
- xmlnode *error, *inf;
-
- /* XXX: gross hack, implement jabber_iq_set_type or something */
- xmlnode_set_attrib(iq->node, "type", "error");
- iq->type = JABBER_IQ_ERROR;
-
- error = xmlnode_new_child(query, "error");
- xmlnode_set_attrib(error, "code", "404");
- xmlnode_set_attrib(error, "type", "cancel");
- inf = xmlnode_new_child(error, "item-not-found");
- xmlnode_set_namespace(inf, "urn:ietf:params:xml:ns:xmpp-stanzas");
- }
+ error = xmlnode_new_child(query, "error");
+ xmlnode_set_attrib(error, "code", "404");
+ xmlnode_set_attrib(error, "type", "cancel");
+ inf = xmlnode_new_child(error, "item-not-found");
+ xmlnode_set_namespace(inf, "urn:ietf:params:xml:ns:xmpp-stanzas");
}
-
+ g_free(node_uri);
jabber_iq_send(iq);
} else if(!strcmp(type, "result")) {
xmlnode *query = xmlnode_get_child(packet, "query");
diff --git a/libpurple/protocols/jabber/jabber.c b/libpurple/protocols/jabber/jabber.c
index c1d2bb05ef..1c95b85eab 100644
--- a/libpurple/protocols/jabber/jabber.c
+++ b/libpurple/protocols/jabber/jabber.c
@@ -39,6 +39,7 @@
#include "version.h"
#include "xmlnode.h"
+#include "caps.h"
#include "auth.h"
#include "buddy.h"
#include "chat.h"
@@ -60,7 +61,11 @@
#define JABBER_CONNECT_STEPS (js->gsc ? 9 : 5)
static PurplePlugin *my_protocol = NULL;
+
GList *jabber_features = NULL;
+GList *jabber_identities = NULL;
+
+GHashTable *jabber_contact_info = NULL;
static void jabber_unregister_account_cb(JabberStream *js);
static void try_srv_connect(JabberStream *js);
@@ -513,6 +518,33 @@ jabber_login_callback_ssl(gpointer data, PurpleSslConnection *gsc,
jabber_stream_set_state(js, JABBER_STREAM_INITIALIZING_ENCRYPTION);
}
+static void
+txt_resolved_cb(PurpleTxtResponse *resp, int results, gpointer data)
+{
+ PurpleConnection *gc = data;
+ JabberStream *js = gc->proto_data;
+ int n;
+
+ if (results == 0) {
+ gchar *tmp;
+ tmp = g_strdup_printf(_("Could not find alternative XMPP connection methods after failing to connect directly.\n"));
+ purple_connection_error_reason (gc,
+ PURPLE_CONNECTION_ERROR_NETWORK_ERROR, tmp);
+ g_free(tmp);
+ }
+
+ for (n = 0; n < results; n++) {
+ gchar **token;
+ token = g_strsplit(resp[n].content, "=", 2);
+ if (!strcmp(token[0], "_xmpp-client-xbosh")) {
+ purple_debug_info("jabber","Found alternative connection method using %s at %s.\n", token[0], token[1]);
+ js->bosh.url = g_strdup(token[1]);
+ g_strfreev(token);
+ break;
+ }
+ g_strfreev(token);
+ }
+}
static void
jabber_login_callback(gpointer data, gint source, const gchar *error)
@@ -525,12 +557,8 @@ jabber_login_callback(gpointer data, gint source, const gchar *error)
purple_debug_error("jabber", "Unable to connect to server: %s. Trying next SRV record.\n", error);
try_srv_connect(js);
} else {
- gchar *tmp;
- tmp = g_strdup_printf(_("Could not establish a connection with the server:\n%s"),
- error);
- purple_connection_error_reason(gc,
- PURPLE_CONNECTION_ERROR_NETWORK_ERROR, tmp);
- g_free(tmp);
+ purple_debug_info("jabber","Couldn't connect directly to %s. Trying to find alternative connection methods, like BOSH.\n", js->user->domain);
+ purple_txt_resolve("_xmppconnect", js->user->domain, txt_resolved_cb, gc);
}
return;
}
@@ -708,7 +736,7 @@ jabber_login(PurpleAccount *account)
/* no old-ssl, so if they've specified a connect server, we'll use that, otherwise we'll
* invoke the magic of SRV lookups, to figure out host and port */
if(!js->gsc) {
- if(connect_server[0]) {
+ if(connect_server[0]) {
jabber_login_connect(js, js->user->domain, connect_server, purple_account_get_int(account, "port", 5222), TRUE);
} else {
js->srv_query_data = purple_srv_resolve("xmpp-client",
@@ -1427,31 +1455,27 @@ void jabber_idle_set(PurpleConnection *gc, int idle)
js->idle = idle ? time(NULL) - idle : idle;
}
-void jabber_add_feature(const char *shortname, const char *namespace, JabberFeatureEnabled cb) {
+void jabber_add_feature(const char *namespace, JabberFeatureEnabled cb) {
JabberFeature *feat;
- g_return_if_fail(shortname != NULL);
g_return_if_fail(namespace != NULL);
feat = g_new0(JabberFeature,1);
- feat->shortname = g_strdup(shortname);
feat->namespace = g_strdup(namespace);
feat->is_enabled = cb;
/* try to remove just in case it already exists in the list */
- jabber_remove_feature(shortname);
+ jabber_remove_feature(namespace);
jabber_features = g_list_append(jabber_features, feat);
}
-void jabber_remove_feature(const char *shortname) {
+void jabber_remove_feature(const char *namespace) {
GList *feature;
for(feature = jabber_features; feature; feature = feature->next) {
JabberFeature *feat = (JabberFeature*)feature->data;
- if(!strcmp(feat->shortname, shortname)) {
- g_free(feat->shortname);
+ if(!strcmp(feat->namespace, namespace)) {
g_free(feat->namespace);
-
g_free(feature->data);
jabber_features = g_list_delete_link(jabber_features, feature);
break;
@@ -1459,6 +1483,27 @@ void jabber_remove_feature(const char *shortname) {
}
}
+void jabber_add_identity(const gchar *category, const gchar *type, const gchar *name) {
+ GList *identity;
+ JabberIdentity *ident;
+ /* both required according to XEP-0030 */
+ g_return_if_fail(category != NULL);
+ g_return_if_fail(type != NULL);
+
+ for(identity = jabber_identities; identity; identity = identity->next) {
+ JabberIdentity *ident = (JabberIdentity*)identity->data;
+ if(!strcmp(ident->category, category)) {
+ if (!strcmp(ident->type, type)) return;
+ }
+ }
+
+ ident = g_new0(JabberIdentity, 1);
+ ident->category = g_strdup(category);
+ ident->type = g_strdup(type);
+ ident->name = g_strdup(name);
+ jabber_identities = g_list_append(jabber_identities, ident);
+}
+
const char *jabber_list_icon(PurpleAccount *a, PurpleBuddy *b)
{
return "jabber";
@@ -2459,8 +2504,78 @@ void jabber_register_commands(void)
_("buzz: Buzz a user to get their attention"), NULL);
}
+/* IPC fucntions*/
+
+/*
+ * IPC function for checking wheather a client at a full JID supports a certain feature.
+ *
+ * @param fulljid The full JID of the client.
+ * @param featrure The feature's namespace.
+ *
+ * @return TRUE if supports feature; else FALSE.
+ */
+static gboolean
+jabber_ipc_contact_has_feature(gchar *fulljid, gchar *feature)
+{
+ JabberCapsKey *caps_info = NULL;
+ JabberCapsValueExt *capabilities = NULL;
+
+ caps_info = g_hash_table_lookup(jabber_contact_info, fulljid);
+
+ if (!caps_info) return FALSE;
+ capabilities = g_hash_table_lookup(capstable, caps_info);
+
+ if (g_list_find_custom(capabilities->features, feature, strcmp) == NULL) return FALSE ;
+ return TRUE;
+}
+
+static void
+jabber_ipc_add_feature(gchar *feature)
+{
+ if (feature == 0) return;
+ jabber_add_feature(feature, 0);
+
+ // send presence with new caps info for all connected accounts
+ jabber_caps_broadcast_change();
+}
+
void
jabber_init_plugin(PurplePlugin *plugin)
{
- my_protocol = plugin;
+ my_protocol = plugin;
+
+ jabber_add_identity("client", "pc", PACKAGE);
+
+ /* initialize jabber_features list */
+ jabber_add_feature("jabber:iq:last", 0);
+ jabber_add_feature("jabber:iq:oob", 0);
+ jabber_add_feature("jabber:iq:time", 0);
+ jabber_add_feature("xmpp:urn:time", 0);
+ jabber_add_feature("jabber:iq:version", 0);
+ jabber_add_feature("jabber:x:conference", 0);
+ jabber_add_feature("http://jabber.org/protocol/bytestreams", 0);
+ jabber_add_feature("http://jabber.org/protocol/disco#info", 0);
+ jabber_add_feature("http://jabber.org/protocol/disco#items", 0);
+#if 0
+ jabber_add_feature("http://jabber.org/protocol/ibb", 0);
+#endif
+ jabber_add_feature("http://jabber.org/protocol/muc", 0);
+ jabber_add_feature("http://jabber.org/protocol/muc#user", 0);
+ jabber_add_feature("http://jabber.org/protocol/si", 0);
+ jabber_add_feature("http://jabber.org/protocol/si/profile/file-transfer", 0);
+ jabber_add_feature("http://jabber.org/protocol/xhtml-im", 0);
+ jabber_add_feature("urn:xmpp:ping", 0);
+
+ jabber_contact_info = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, jabber_caps_destroy_key);
+
+ /* IPC functions */
+ purple_plugin_ipc_register(plugin, "contact_has_feature", PURPLE_CALLBACK(jabber_ipc_contact_has_feature),
+ purple_marshal_BOOLEAN__POINTER_POINTER,
+ purple_value_new(PURPLE_TYPE_BOOLEAN), 2,
+ purple_value_new(PURPLE_TYPE_STRING),
+ purple_value_new(PURPLE_TYPE_STRING));
+ purple_plugin_ipc_register(plugin, "add_feature", PURPLE_CALLBACK(jabber_ipc_add_feature),
+ purple_marshal_VOID__POINTER,
+ NULL, 1,
+ purple_value_new(PURPLE_TYPE_STRING));
}
diff --git a/libpurple/protocols/jabber/jabber.h b/libpurple/protocols/jabber/jabber.h
index 326de16fc8..ee67405b92 100644
--- a/libpurple/protocols/jabber/jabber.h
+++ b/libpurple/protocols/jabber/jabber.h
@@ -59,12 +59,13 @@ typedef struct _JabberStream JabberStream;
#include "jutil.h"
#include "xmlnode.h"
#include "buddy.h"
+#include "bosh.h"
#ifdef HAVE_CYRUS_SASL
#include <sasl/sasl.h>
#endif
-#define CAPS0115_NODE "http://pidgin.im/caps"
+#define CAPS0115_NODE "http://pidgin.im/"
/* Index into attention_types list */
#define JABBER_BUZZ 0
@@ -201,21 +202,31 @@ struct _JabberStream
/* A purple timeout tag for the keepalive */
int keepalive_timeout;
-
+
PurpleSrvResponse *srv_rec;
guint srv_rec_idx;
guint max_srv_rec_idx;
+
+ /* BOSH stuff*/
+ PurpleBOSHConnection bosh;
};
-typedef gboolean (JabberFeatureEnabled)(JabberStream *js, const gchar *shortname, const gchar *namespace);
+typedef gboolean (JabberFeatureEnabled)(JabberStream *js, const gchar *namespace);
typedef struct _JabberFeature
{
- gchar *shortname;
gchar *namespace;
JabberFeatureEnabled *is_enabled;
} JabberFeature;
+typedef struct _JabberIdentity
+{
+ gchar *category;
+ gchar *type;
+ gchar *name;
+ gchar *lang;
+} JabberIdentity;
+
typedef struct _JabberBytestreamsStreamhost {
char *jid;
char *host;
@@ -225,6 +236,9 @@ typedef struct _JabberBytestreamsStreamhost {
/* what kind of additional features as returned from disco#info are supported? */
extern GList *jabber_features;
+extern GList *jabber_identities;
+
+extern GHashTable *jabber_contact_info; /* char * -> JabberCapsKey */
void jabber_process_packet(JabberStream *js, xmlnode **packet);
void jabber_send(JabberStream *js, xmlnode *data);
@@ -246,8 +260,16 @@ char *jabber_get_next_id(JabberStream *js);
*/
char *jabber_parse_error(JabberStream *js, xmlnode *packet, PurpleConnectionError *reason);
-void jabber_add_feature(const gchar *shortname, const gchar *namespace, JabberFeatureEnabled cb); /* cb may be NULL */
-void jabber_remove_feature(const gchar *shortname);
+void jabber_add_feature(const gchar *namespace, JabberFeatureEnabled cb); /* cb may be NULL */
+void jabber_remove_feature(const gchar *namespace);
+
+/** Adds an identitiy to this jabber library instance. For list of valid values vistit the
+ * webiste of the XMPP Registrar ( http://www.xmpp.org/registrar/disco-categories.html#client ).
+ * @param category the category of the identity.
+ * @param type the type of the identity.
+ * @param name the name of the identity.
+ */
+void jabber_add_identity(const gchar *category, const gchar *type, const gchar *name);
/** PRPL functions */
const char *jabber_list_icon(PurpleAccount *a, PurpleBuddy *b);
diff --git a/libpurple/protocols/jabber/libxmpp.c b/libpurple/protocols/jabber/libxmpp.c
index dd81c2c0a9..a898d9af77 100644
--- a/libpurple/protocols/jabber/libxmpp.c
+++ b/libpurple/protocols/jabber/libxmpp.c
@@ -266,15 +266,16 @@ init_plugin(PurplePlugin *plugin)
jabber_iq_init();
jabber_pep_init();
-
- jabber_tune_init();
jabber_caps_init();
+ jabber_tune_init();
+
+ #warning implement adding and retrieving own features via IPC API
- jabber_add_feature("avatarmeta", AVATARNAMESPACEMETA, jabber_pep_namespace_only_when_pep_enabled_cb);
- jabber_add_feature("avatardata", AVATARNAMESPACEDATA, jabber_pep_namespace_only_when_pep_enabled_cb);
- jabber_add_feature("buzz", "http://www.xmpp.org/extensions/xep-0224.html#ns", jabber_buzz_isenabled);
+ jabber_add_feature(AVATARNAMESPACEMETA, jabber_pep_namespace_only_when_pep_enabled_cb);
+ jabber_add_feature(AVATARNAMESPACEDATA, jabber_pep_namespace_only_when_pep_enabled_cb);
+ jabber_add_feature("http://www.xmpp.org/extensions/xep-0224.html#ns", jabber_buzz_isenabled);
- jabber_pep_register_handler("avatar", AVATARNAMESPACEMETA, jabber_buddy_avatar_update_metadata);
+ jabber_pep_register_handler(AVATARNAMESPACEMETA, jabber_buddy_avatar_update_metadata);
}
diff --git a/libpurple/protocols/jabber/message.c b/libpurple/protocols/jabber/message.c
index 944cbda1cd..b75a8daa4e 100644
--- a/libpurple/protocols/jabber/message.c
+++ b/libpurple/protocols/jabber/message.c
@@ -758,7 +758,7 @@ void jabber_message_conv_closed(JabberStream *js, const char *who)
jabber_message_free(jm);
}
-gboolean jabber_buzz_isenabled(JabberStream *js, const gchar *shortname, const gchar *namespace) {
+gboolean jabber_buzz_isenabled(JabberStream *js, const gchar *namespace) {
return js->allowBuzz;
}
diff --git a/libpurple/protocols/jabber/message.h b/libpurple/protocols/jabber/message.h
index 56c89819b1..d859f94980 100644
--- a/libpurple/protocols/jabber/message.h
+++ b/libpurple/protocols/jabber/message.h
@@ -78,6 +78,6 @@ int jabber_message_send_chat(PurpleConnection *gc, int id, const char *message,
unsigned int jabber_send_typing(PurpleConnection *gc, const char *who, PurpleTypingState state);
void jabber_message_conv_closed(JabberStream *js, const char *who);
-gboolean jabber_buzz_isenabled(JabberStream *js, const gchar *shortname, const gchar *namespace);
+gboolean jabber_buzz_isenabled(JabberStream *js, const gchar *namespace);
#endif /* _PURPLE_JABBER_MESSAGE_H_ */
diff --git a/libpurple/protocols/jabber/pep.c b/libpurple/protocols/jabber/pep.c
index e7d849b184..9b13c66b7a 100644
--- a/libpurple/protocols/jabber/pep.c
+++ b/libpurple/protocols/jabber/pep.c
@@ -45,9 +45,9 @@ void jabber_pep_init_actions(GList **m) {
jabber_nick_init_action(m);
}
-void jabber_pep_register_handler(const char *shortname, const char *xmlns, JabberPEPHandler handlerfunc) {
+void jabber_pep_register_handler(const char *xmlns, JabberPEPHandler handlerfunc) {
gchar *notifyns = g_strdup_printf("%s+notify", xmlns);
- jabber_add_feature(shortname, notifyns, NULL); /* receiving PEPs is always supported */
+ jabber_add_feature(notifyns, NULL); /* receiving PEPs is always supported */
g_free(notifyns);
g_hash_table_replace(pep_handlers, g_strdup(xmlns), handlerfunc);
}
@@ -85,7 +85,7 @@ void jabber_pep_request_item(JabberStream *js, const char *to, const char *node,
jabber_iq_send(iq);
}
-gboolean jabber_pep_namespace_only_when_pep_enabled_cb(JabberStream *js, const gchar *shortname, const gchar *namespace) {
+gboolean jabber_pep_namespace_only_when_pep_enabled_cb(JabberStream *js, const gchar *namespace) {
return js->pep;
}
diff --git a/libpurple/protocols/jabber/pep.h b/libpurple/protocols/jabber/pep.h
index f1e1aca059..80d9875a36 100644
--- a/libpurple/protocols/jabber/pep.h
+++ b/libpurple/protocols/jabber/pep.h
@@ -42,11 +42,10 @@ typedef void (JabberPEPHandler)(JabberStream *js, const char *from, xmlnode *ite
* Registers a callback for PEP events. Also automatically announces this receiving capability via disco#info.
* Don't forget to use jabber_add_feature when supporting the sending of PEP events of this type.
*
- * @parameter shortname A short name for this feature for XEP-0115. It has no semantic meaning, it just has to be unique.
- * @parameter xmlns The namespace for this event
+ * @parameter xmlns The namespace for this event
* @parameter handlerfunc The callback to be used when receiving an event with this namespace
*/
-void jabber_pep_register_handler(const char *shortname, const char *xmlns, JabberPEPHandler handlerfunc);
+void jabber_pep_register_handler(const char *xmlns, JabberPEPHandler handlerfunc);
/*
* Request a specific item from another PEP node.
@@ -64,13 +63,12 @@ void jabber_pep_request_item(JabberStream *js, const char *to, const char *node,
/*
* Default callback that can be used for namespaces which should only be enabled when PEP is supported
*
- * @parameter js The JabberStream struct for this connection
- * @parameter shortname The namespace's shortname (for caps), ignored.
+ * @parameter js The JabberStream struct for this connection
* @parameter namespace The namespace that's queried, ignored.
*
* @returns TRUE when PEP is enabled, FALSE otherwise
*/
-gboolean jabber_pep_namespace_only_when_pep_enabled_cb(JabberStream *js, const gchar *shortname, const gchar *namespace);
+gboolean jabber_pep_namespace_only_when_pep_enabled_cb(JabberStream *js, const gchar *namespace);
void jabber_handle_event(JabberMessage *jm);
diff --git a/libpurple/protocols/jabber/presence.c b/libpurple/protocols/jabber/presence.c
index d5af230b4e..61e812a957 100644
--- a/libpurple/protocols/jabber/presence.c
+++ b/libpurple/protocols/jabber/presence.c
@@ -261,11 +261,16 @@ xmlnode *jabber_presence_create_js(JabberStream *js, JabberBuddyState state, con
}
/* JEP-0115 */
+ /* calculate hash */
+ jabber_caps_calculate_own_hash(js);
+ /* create xml */
c = xmlnode_new_child(presence, "c");
xmlnode_set_namespace(c, "http://jabber.org/protocol/caps");
xmlnode_set_attrib(c, "node", CAPS0115_NODE);
- xmlnode_set_attrib(c, "ver", VERSION);
-
+ xmlnode_set_attrib(c, "hash", "sha-1");
+ xmlnode_set_attrib(c, "ver", jabber_caps_get_own_hash());
+
+#if 0
if(js != NULL) {
/* add the extensions */
char extlist[1024];
@@ -277,7 +282,7 @@ xmlnode *jabber_presence_create_js(JabberStream *js, JabberBuddyState state, con
JabberFeature *feat = (JabberFeature*)feature->data;
unsigned featlen;
- if(feat->is_enabled != NULL && feat->is_enabled(js, feat->shortname, feat->namespace) == FALSE)
+ if(feat->is_enabled != NULL && feat->is_enabled(js, feat->namespace) == FALSE)
continue; /* skip this feature */
featlen = strlen(feat->shortname);
@@ -297,7 +302,7 @@ xmlnode *jabber_presence_create_js(JabberStream *js, JabberBuddyState state, con
if(remaining < 1023)
xmlnode_set_attrib(c, "ext", extlist);
}
-
+#endif
return presence;
}
@@ -748,16 +753,17 @@ void jabber_presence_parse(JabberStream *js, xmlnode *packet)
jbr = jabber_buddy_track_resource(jb, jid->resource, priority,
state, status);
if(caps) {
+ /* handle XEP-0115 */
const char *node = xmlnode_get_attrib(caps,"node");
const char *ver = xmlnode_get_attrib(caps,"ver");
- const char *ext = xmlnode_get_attrib(caps,"ext");
+ const char *hash = xmlnode_get_attrib(caps,"hash");
- if(node && ver) {
+ if(node && ver && hash) {
JabberPresenceCapabilities *userdata = g_new0(JabberPresenceCapabilities, 1);
userdata->js = js;
userdata->jb = jb;
userdata->from = g_strdup(from);
- jabber_caps_get_info(js, from, node, ver, ext, jabber_presence_set_capabilities, userdata);
+ jabber_caps_get_info(js, from, node, ver, hash, jabber_presence_set_capabilities, userdata);
}
}
}
diff --git a/libpurple/protocols/jabber/usermood.c b/libpurple/protocols/jabber/usermood.c
index 3ff09d0402..9c3cfa47a0 100644
--- a/libpurple/protocols/jabber/usermood.c
+++ b/libpurple/protocols/jabber/usermood.c
@@ -141,8 +141,8 @@ static void jabber_mood_cb(JabberStream *js, const char *from, xmlnode *items) {
}
void jabber_mood_init(void) {
- jabber_add_feature("mood", "http://jabber.org/protocol/mood", jabber_pep_namespace_only_when_pep_enabled_cb);
- jabber_pep_register_handler("moodn", "http://jabber.org/protocol/mood", jabber_mood_cb);
+ jabber_add_feature("http://jabber.org/protocol/mood", jabber_pep_namespace_only_when_pep_enabled_cb);
+ jabber_pep_register_handler("http://jabber.org/protocol/mood", jabber_mood_cb);
}
static void do_mood_set_from_fields(PurpleConnection *gc, PurpleRequestFields *fields) {
diff --git a/libpurple/protocols/jabber/usernick.c b/libpurple/protocols/jabber/usernick.c
index 618d493aed..e1503684b7 100644
--- a/libpurple/protocols/jabber/usernick.c
+++ b/libpurple/protocols/jabber/usernick.c
@@ -92,8 +92,8 @@ static void do_nick_set_nick(PurplePluginAction *action) {
}
void jabber_nick_init(void) {
- jabber_add_feature("nick", "http://jabber.org/protocol/nick", jabber_pep_namespace_only_when_pep_enabled_cb);
- jabber_pep_register_handler("nickn", "http://jabber.org/protocol/nick", jabber_nick_cb);
+ jabber_add_feature("http://jabber.org/protocol/nick", jabber_pep_namespace_only_when_pep_enabled_cb);
+ jabber_pep_register_handler("http://jabber.org/protocol/nick", jabber_nick_cb);
}
void jabber_nick_init_action(GList **m) {
diff --git a/libpurple/protocols/jabber/usertune.c b/libpurple/protocols/jabber/usertune.c
index 92d06d3168..e18674b649 100644
--- a/libpurple/protocols/jabber/usertune.c
+++ b/libpurple/protocols/jabber/usertune.c
@@ -109,8 +109,8 @@ static void jabber_tune_cb(JabberStream *js, const char *from, xmlnode *items) {
}
void jabber_tune_init(void) {
- jabber_add_feature("tune", "http://jabber.org/protocol/tune", jabber_pep_namespace_only_when_pep_enabled_cb);
- jabber_pep_register_handler("tunen", "http://jabber.org/protocol/tune", jabber_tune_cb);
+ jabber_add_feature("http://jabber.org/protocol/tune", jabber_pep_namespace_only_when_pep_enabled_cb);
+ jabber_pep_register_handler("http://jabber.org/protocol/tune", jabber_tune_cb);
}
void jabber_tune_set(PurpleConnection *gc, const PurpleJabberTuneInfo *tuneinfo) {