pidgin/pidgin

closing merged branch
port-changes-from-branch-2.x.y-to-default
2020-02-03, Gary Kramlich
2f836435c33c
closing merged branch
/*
* purple - Handling of XEP-0231: Bits of Binary.
*
* Purple is the legal property of its developers, whose names are too numerous
* to list here. Please refer to the COPYRIGHT file distributed with this
* source distribution.
*
* 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 <stdlib.h>
#include <glib.h>
#include <string.h>
#include "data.h"
#include "debug.h"
#include "xmlnode.h"
#include "util.h"
#include "iq.h"
static GHashTable *local_data_by_alt = NULL;
static GHashTable *local_data_by_cid = NULL;
static GHashTable *remote_data_by_cid = NULL;
JabberData *
jabber_data_create_from_data(gconstpointer rawdata, gsize size, const char *type,
gboolean ephemeral, JabberStream *js)
{
JabberData *data;
gchar *checksum;
g_return_val_if_fail(rawdata != NULL, NULL);
g_return_val_if_fail(size > 0, NULL);
g_return_val_if_fail(type != NULL, NULL);
checksum = g_compute_checksum_for_data(G_CHECKSUM_SHA1, rawdata, size);
data = g_new0(JabberData, 1);
data->cid = g_strdup_printf("sha1+%s@bob.xmpp.org", checksum);
data->type = g_strdup(type);
data->size = size;
data->ephemeral = ephemeral;
data->data = g_memdup(rawdata, size);
g_free(checksum);
return data;
}
static void
jabber_data_delete(gpointer cbdata)
{
JabberData *data = cbdata;
g_free(data->cid);
g_free(data->type);
g_free(data->data);
g_free(data);
}
JabberData *
jabber_data_create_from_xml(PurpleXmlNode *tag)
{
JabberData *data;
gchar *raw_data = NULL;
const gchar *cid, *type;
g_return_val_if_fail(tag != NULL, NULL);
/* check if this is a "data" tag */
if (!purple_strequal(tag->name, "data")) {
purple_debug_error("jabber", "Invalid data element\n");
return NULL;
}
cid = purple_xmlnode_get_attrib(tag, "cid");
type = purple_xmlnode_get_attrib(tag, "type");
if (!cid || !type) {
purple_debug_error("jabber", "cid or type missing\n");
return NULL;
}
raw_data = purple_xmlnode_get_data(tag);
if (raw_data == NULL || *raw_data == '\0') {
purple_debug_error("jabber", "data element was empty");
g_free(raw_data);
return NULL;
}
data = g_new0(JabberData, 1);
data->data = g_base64_decode(raw_data, &data->size);
g_free(raw_data);
if (data->data == NULL) {
purple_debug_error("jabber", "Malformed base64 data\n");
g_free(data);
return NULL;
}
data->cid = g_strdup(cid);
data->type = g_strdup(type);
return data;
}
void
jabber_data_destroy(JabberData *data)
{
g_return_if_fail(data != NULL);
jabber_data_delete(data);
}
const char *
jabber_data_get_cid(const JabberData *data)
{
g_return_val_if_fail(data != NULL, NULL);
return data->cid;
}
const char *
jabber_data_get_type(const JabberData *data)
{
g_return_val_if_fail(data != NULL, NULL);
return data->type;
}
gsize
jabber_data_get_size(const JabberData *data)
{
g_return_val_if_fail(data != NULL, 0);
return data->size;
}
gpointer
jabber_data_get_data(const JabberData *data)
{
g_return_val_if_fail(data != NULL, NULL);
return data->data;
}
PurpleXmlNode *
jabber_data_get_xml_definition(const JabberData *data)
{
PurpleXmlNode *tag;
char *base64data;
g_return_val_if_fail(data != NULL, NULL);
tag = purple_xmlnode_new("data");
base64data = g_base64_encode(data->data, data->size);
purple_xmlnode_set_namespace(tag, NS_BOB);
purple_xmlnode_set_attrib(tag, "cid", data->cid);
purple_xmlnode_set_attrib(tag, "type", data->type);
purple_xmlnode_insert_data(tag, base64data, -1);
g_free(base64data);
return tag;
}
PurpleXmlNode *
jabber_data_get_xhtml_im(const JabberData *data, const gchar *alt)
{
PurpleXmlNode *img;
char *src;
g_return_val_if_fail(data != NULL, NULL);
g_return_val_if_fail(alt != NULL, NULL);
img = purple_xmlnode_new("img");
purple_xmlnode_set_attrib(img, "alt", alt);
src = g_strconcat("cid:", data->cid, NULL);
purple_xmlnode_set_attrib(img, "src", src);
g_free(src);
return img;
}
static PurpleXmlNode *
jabber_data_get_xml_request(const gchar *cid)
{
PurpleXmlNode *tag = purple_xmlnode_new("data");
purple_xmlnode_set_namespace(tag, NS_BOB);
purple_xmlnode_set_attrib(tag, "cid", cid);
return tag;
}
static gboolean
jabber_data_has_valid_hash(const JabberData *data)
{
const gchar *cid = jabber_data_get_cid(data);
gchar **cid_parts = g_strsplit(cid, "@", -1);
guint num_cid_parts = 0;
gboolean ret = FALSE;
if (cid_parts)
num_cid_parts = g_strv_length(cid_parts);
if (num_cid_parts == 2 && purple_strequal(cid_parts[1], "bob.xmpp.org")) {
gchar **sub_parts = g_strsplit(cid_parts[0], "+", -1);
guint num_sub_parts = 0;
if (sub_parts)
num_sub_parts = g_strv_length(sub_parts);
if (num_sub_parts == 2) {
const gchar *hash_algo = sub_parts[0];
const gchar *hash_value = sub_parts[1];
GChecksumType hash_type;
gboolean valid_hash_type = TRUE;
if (purple_strequal(hash_algo, "sha1"))
hash_type = G_CHECKSUM_SHA1;
else if (purple_strequal(hash_algo, "sha256"))
hash_type = G_CHECKSUM_SHA256;
else if (purple_strequal(hash_algo, "sha512"))
hash_type = G_CHECKSUM_SHA512;
else if (purple_strequal(hash_algo, "md5"))
hash_type = G_CHECKSUM_MD5;
else
valid_hash_type = FALSE;
if (valid_hash_type) {
gchar *digest = g_compute_checksum_for_data(
hash_type, jabber_data_get_data(data),
jabber_data_get_size(data));
ret = purple_strequal(digest, hash_value);
if (!ret)
purple_debug_warning("jabber", "Unable to validate BoB "
"hash; expecting %s, got %s\n",
cid, digest);
g_free(digest);
} else {
purple_debug_warning("jabber", "Unable to validate BoB hash; "
"unknown hash algorithm %s\n", hash_algo);
}
} else {
purple_debug_warning("jabber", "Malformed BoB CID\n");
}
g_strfreev(sub_parts);
}
g_strfreev(cid_parts);
return ret;
}
typedef struct {
gpointer userdata;
gchar *alt;
gboolean ephemeral;
JabberDataRequestCallback *cb;
} JabberDataRequestData;
static void
jabber_data_request_cb(JabberStream *js, const char *from,
JabberIqType type, const char *id, PurpleXmlNode *packet, gpointer data)
{
JabberDataRequestData *request_data = (JabberDataRequestData *) data;
gpointer userdata = request_data->userdata;
gchar *alt = request_data->alt;
gboolean ephemeral = request_data->ephemeral;
JabberDataRequestCallback *cb = request_data->cb;
PurpleXmlNode *data_element = purple_xmlnode_get_child(packet, "data");
PurpleXmlNode *item_not_found = purple_xmlnode_get_child(packet, "item-not-found");
/* did we get a data element as result? */
if (data_element && type == JABBER_IQ_RESULT) {
JabberData *data = jabber_data_create_from_xml(data_element);
if (data && !ephemeral) {
jabber_data_associate_remote(js, from, data);
}
cb(data, alt, userdata);
} else if (item_not_found) {
purple_debug_info("jabber",
"Responder didn't recognize requested data\n");
cb(NULL, alt, userdata);
} else {
purple_debug_warning("jabber", "Unknown response to data request\n");
cb(NULL, alt, userdata);
}
g_free(request_data);
}
void
jabber_data_request(JabberStream *js, const gchar *cid, const gchar *who,
gchar *alt, gboolean ephemeral, JabberDataRequestCallback cb,
gpointer userdata)
{
JabberIq *request;
PurpleXmlNode *data_request;
JabberDataRequestData *data;
g_return_if_fail(cid != NULL);
g_return_if_fail(who != NULL);
g_return_if_fail(alt != NULL);
request = jabber_iq_new(js, JABBER_IQ_GET);
data_request = jabber_data_get_xml_request(cid);
data = g_new0(JabberDataRequestData, 1);
data->userdata = userdata;
data->alt = alt;
data->ephemeral = ephemeral;
data->cb = cb;
purple_xmlnode_set_attrib(request->node, "to", who);
jabber_iq_set_callback(request, jabber_data_request_cb, data);
purple_xmlnode_insert_child(request->node, data_request);
jabber_iq_send(request);
}
const JabberData *
jabber_data_find_local_by_alt(const gchar *alt)
{
purple_debug_info("jabber", "looking up local data object with alt = %s\n", alt);
return g_hash_table_lookup(local_data_by_alt, alt);
}
const JabberData *
jabber_data_find_local_by_cid(const gchar *cid)
{
purple_debug_info("jabber", "lookup local data object with cid = %s\n", cid);
return g_hash_table_lookup(local_data_by_cid, cid);
}
const JabberData *
jabber_data_find_remote_by_cid(JabberStream *js, const gchar *who,
const gchar *cid)
{
const JabberData *data = g_hash_table_lookup(remote_data_by_cid, cid);
purple_debug_info("jabber", "lookup remote data object with cid = %s\n", cid);
if (data == NULL) {
gchar *jid_cid =
g_strdup_printf("%s@%s/%s%s%s", js->user->node, js->user->domain,
js->user->resource, who, cid);
purple_debug_info("jabber",
"didn't find BoB object by pure CID, try including JIDs: %s\n",
jid_cid);
data = g_hash_table_lookup(remote_data_by_cid, jid_cid);
g_free(jid_cid);
}
return data;
}
void
jabber_data_associate_local(JabberData *data, const gchar *alt)
{
g_return_if_fail(data != NULL);
purple_debug_info("jabber", "associating local data object\n alt = %s, cid = %s\n",
alt , jabber_data_get_cid(data));
if (alt)
g_hash_table_insert(local_data_by_alt, g_strdup(alt), data);
g_hash_table_insert(local_data_by_cid, g_strdup(jabber_data_get_cid(data)),
data);
}
void
jabber_data_associate_remote(JabberStream *js, const gchar *who, JabberData *data)
{
gchar *cid;
g_return_if_fail(data != NULL);
if (jabber_data_has_valid_hash(data)) {
cid = g_strdup(jabber_data_get_cid(data));
} else {
cid = g_strdup_printf("%s@%s/%s%s%s", js->user->node, js->user->domain,
js->user->resource, who, jabber_data_get_cid(data));
}
purple_debug_info("jabber", "associating remote BoB object with cid = %s\n",
cid);
g_hash_table_insert(remote_data_by_cid, cid, data);
}
void
jabber_data_parse(JabberStream *js, const char *who, JabberIqType type,
const char *id, PurpleXmlNode *data_node)
{
JabberIq *result = NULL;
const char *cid = purple_xmlnode_get_attrib(data_node, "cid");
const JabberData *data = cid ? jabber_data_find_local_by_cid(cid) : NULL;
if (!data) {
PurpleXmlNode *item_not_found = purple_xmlnode_new("item-not-found");
result = jabber_iq_new(js, JABBER_IQ_ERROR);
if (who)
purple_xmlnode_set_attrib(result->node, "to", who);
purple_xmlnode_set_attrib(result->node, "id", id);
purple_xmlnode_insert_child(result->node, item_not_found);
} else {
result = jabber_iq_new(js, JABBER_IQ_RESULT);
if (who)
purple_xmlnode_set_attrib(result->node, "to", who);
purple_xmlnode_set_attrib(result->node, "id", id);
purple_xmlnode_insert_child(result->node,
jabber_data_get_xml_definition(data));
/* if the data object is temporary, destroy it and remove the references
to it */
if (data->ephemeral) {
g_hash_table_remove(local_data_by_cid, cid);
}
}
jabber_iq_send(result);
}
void
jabber_data_init(void)
{
if (purple_debug_is_verbose())
purple_debug_misc("jabber", "creating hash tables for data objects");
local_data_by_alt = g_hash_table_new_full(g_str_hash, g_str_equal,
g_free, NULL);
local_data_by_cid = g_hash_table_new_full(g_str_hash, g_str_equal,
g_free, jabber_data_delete);
remote_data_by_cid = g_hash_table_new_full(g_str_hash, g_str_equal,
g_free, jabber_data_delete);
jabber_iq_register_handler("data", NS_BOB, jabber_data_parse);
}
void
jabber_data_uninit(void)
{
if (purple_debug_is_verbose())
purple_debug_info("jabber", "destroying hash tables for data objects");
g_hash_table_destroy(local_data_by_alt);
g_hash_table_destroy(local_data_by_cid);
g_hash_table_destroy(remote_data_by_cid);
local_data_by_alt = local_data_by_cid = remote_data_by_cid = NULL;
}