2011-11-12 10:51:01 +00:00
|
|
|
/* xmpp-utils.c
|
|
|
|
* Wireshark's XMPP dissector.
|
|
|
|
*
|
|
|
|
* Copyright 2011, Mariusz Okroj <okrojmariusz[]gmail.com>
|
|
|
|
*
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* Wireshark - Network traffic analyzer
|
|
|
|
* By Gerald Combs <gerald@wireshark.org>
|
|
|
|
* Copyright 1998 Gerald Combs
|
|
|
|
*
|
|
|
|
* 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
|
2012-06-28 22:56:06 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
2011-11-12 10:51:01 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <glib.h>
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
#include <epan/packet.h>
|
2011-11-12 10:51:01 +00:00
|
|
|
#include <epan/expert.h>
|
|
|
|
#include <epan/tvbparse.h>
|
2011-11-13 23:33:47 +00:00
|
|
|
#include <epan/strutil.h>
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
#include <epan/dissectors/packet-xml.h>
|
|
|
|
|
|
|
|
#include <packet-xmpp.h>
|
2012-07-22 21:48:51 +00:00
|
|
|
#include <packet-xmpp-core.h>
|
2011-11-12 10:51:01 +00:00
|
|
|
#include <packet-xmpp-utils.h>
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_iq_reqresp_track(packet_info *pinfo, xmpp_element_t *packet, xmpp_conv_info_t *xmpp_info)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
xmpp_transaction_t *xmpp_trans = NULL;
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *attr_id;
|
2011-11-12 10:51:01 +00:00
|
|
|
char *id;
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
attr_id = xmpp_get_attr(packet, "id");
|
2012-07-21 16:29:48 +00:00
|
|
|
|
|
|
|
if (!attr_id) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-11-12 10:51:01 +00:00
|
|
|
id = ep_strdup(attr_id->value);
|
|
|
|
|
|
|
|
if (!pinfo->fd->flags.visited) {
|
2013-03-02 19:21:38 +00:00
|
|
|
xmpp_trans = (xmpp_transaction_t *)se_tree_lookup_string(xmpp_info->req_resp, id, EMEM_TREE_STRING_NOCASE);
|
2011-11-12 10:51:01 +00:00
|
|
|
if (xmpp_trans) {
|
|
|
|
xmpp_trans->resp_frame = pinfo->fd->num;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
char *se_id = se_strdup(id);
|
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
xmpp_trans = (xmpp_transaction_t *)se_alloc(sizeof (xmpp_transaction_t));
|
2011-11-12 10:51:01 +00:00
|
|
|
xmpp_trans->req_frame = pinfo->fd->num;
|
|
|
|
xmpp_trans->resp_frame = 0;
|
|
|
|
|
|
|
|
se_tree_insert_string(xmpp_info->req_resp, se_id, (void *) xmpp_trans, EMEM_TREE_STRING_NOCASE);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
se_tree_lookup_string(xmpp_info->req_resp, id, EMEM_TREE_STRING_NOCASE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_jingle_session_track(packet_info *pinfo, xmpp_element_t *packet, xmpp_conv_info_t *xmpp_info)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t *jingle_packet;
|
2011-11-12 10:51:01 +00:00
|
|
|
GList *jingle_packet_l;
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
jingle_packet_l = xmpp_find_element_by_name(packet,"jingle");
|
2013-03-02 19:21:38 +00:00
|
|
|
jingle_packet = (xmpp_element_t *)(jingle_packet_l?jingle_packet_l->data:NULL);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if (jingle_packet && !pinfo->fd->flags.visited) {
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *attr_id;
|
|
|
|
xmpp_attr_t *attr_sid;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
char *se_id;
|
|
|
|
char *se_sid;
|
|
|
|
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
attr_id = xmpp_get_attr(packet, "id");
|
2012-07-21 16:29:48 +00:00
|
|
|
if (!attr_id) {
|
|
|
|
return;
|
|
|
|
}
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
attr_sid = xmpp_get_attr(jingle_packet, "sid");
|
2012-07-21 16:29:48 +00:00
|
|
|
if (!attr_sid) {
|
|
|
|
return;
|
|
|
|
}
|
2011-11-13 23:33:47 +00:00
|
|
|
|
2012-07-21 16:29:48 +00:00
|
|
|
se_id = se_strdup(attr_id->value);
|
2011-11-12 10:51:01 +00:00
|
|
|
se_sid = se_strdup(attr_sid->value);
|
|
|
|
|
|
|
|
se_tree_insert_string(xmpp_info->jingle_sessions, se_id, (void*) se_sid, EMEM_TREE_STRING_NOCASE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_gtalk_session_track(packet_info *pinfo, xmpp_element_t *packet, xmpp_conv_info_t *xmpp_info)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t *gtalk_packet;
|
2011-11-12 10:51:01 +00:00
|
|
|
GList *gtalk_packet_l;
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
gtalk_packet_l = xmpp_find_element_by_name(packet,"session");
|
2013-03-02 19:21:38 +00:00
|
|
|
gtalk_packet = (xmpp_element_t *)(gtalk_packet_l?gtalk_packet_l->data:NULL);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
if (gtalk_packet && !pinfo->fd->flags.visited) {
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *attr_id;
|
|
|
|
xmpp_attr_t *attr_sid;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
char *se_id;
|
|
|
|
char *se_sid;
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *xmlns = xmpp_get_attr(gtalk_packet, "xmlns");
|
2011-11-12 10:51:01 +00:00
|
|
|
if(xmlns && strcmp(xmlns->value,"http://www.google.com/session") != 0)
|
|
|
|
return;
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
attr_id = xmpp_get_attr(packet, "id");
|
2012-07-21 16:29:48 +00:00
|
|
|
if (!attr_id) {
|
|
|
|
return;
|
|
|
|
}
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
attr_sid = xmpp_get_attr(gtalk_packet, "id");
|
2012-07-21 16:29:48 +00:00
|
|
|
if (!attr_sid) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
se_id = se_strdup(attr_id->value);
|
2011-11-12 10:51:01 +00:00
|
|
|
se_sid = se_strdup(attr_sid->value);
|
|
|
|
|
|
|
|
se_tree_insert_string(xmpp_info->gtalk_sessions, se_id, (void*) se_sid, EMEM_TREE_STRING_NOCASE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_ibb_session_track(packet_info *pinfo, xmpp_element_t *packet, xmpp_conv_info_t *xmpp_info)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t *ibb_packet = NULL;
|
2011-11-12 10:51:01 +00:00
|
|
|
GList *ibb_packet_l;
|
|
|
|
|
|
|
|
if(strcmp(packet->name, "message") == 0)
|
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
ibb_packet_l = xmpp_find_element_by_name(packet,"data");
|
2013-03-02 19:21:38 +00:00
|
|
|
ibb_packet = (xmpp_element_t *)(ibb_packet_l?ibb_packet_l->data:NULL);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
} else if(strcmp(packet->name, "iq") == 0)
|
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
ibb_packet_l = xmpp_find_element_by_name(packet,"open");
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if(!ibb_packet_l)
|
2011-11-13 23:33:47 +00:00
|
|
|
ibb_packet_l = xmpp_find_element_by_name(packet,"close");
|
2011-11-12 10:51:01 +00:00
|
|
|
if(!ibb_packet_l)
|
2011-11-13 23:33:47 +00:00
|
|
|
ibb_packet_l = xmpp_find_element_by_name(packet,"data");
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
ibb_packet = (xmpp_element_t *)(ibb_packet_l?ibb_packet_l->data:NULL);
|
2011-11-12 10:51:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ibb_packet && !pinfo->fd->flags.visited) {
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *attr_id;
|
|
|
|
xmpp_attr_t *attr_sid;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
char *se_id;
|
|
|
|
char *se_sid;
|
|
|
|
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
attr_id = xmpp_get_attr(packet, "id");
|
|
|
|
attr_sid = xmpp_get_attr(ibb_packet, "sid");
|
2011-11-12 10:51:01 +00:00
|
|
|
if(attr_id && attr_sid)
|
|
|
|
{
|
|
|
|
se_id = se_strdup(attr_id->value);
|
|
|
|
se_sid = se_strdup(attr_sid->value);
|
|
|
|
se_tree_insert_string(xmpp_info->ibb_sessions, se_id, (void*) se_sid, EMEM_TREE_STRING_NOCASE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_unknown_items(proto_tree *tree, tvbuff_t *tvb, packet_info *pinfo, xmpp_element_t *element, guint level)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
GList *childs = element->elements;
|
|
|
|
|
|
|
|
DISSECTOR_ASSERT( level < ETT_UNKNOWN_LEN );
|
|
|
|
|
|
|
|
xmpp_unknown_attrs(tree, tvb, pinfo, element, TRUE);
|
|
|
|
|
|
|
|
if(element->data)
|
|
|
|
{
|
|
|
|
proto_tree_add_text(tree, tvb, element->data->offset, element->data->length, "CDATA: %s",element->data->value);
|
|
|
|
}
|
|
|
|
|
|
|
|
while(childs)
|
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
xmpp_element_t *child = (xmpp_element_t *)childs->data;
|
2011-11-13 23:33:47 +00:00
|
|
|
proto_item *child_item = proto_tree_add_text(tree, tvb, child->offset, child->length, "%s", xmpp_ep_string_upcase(child->name));
|
2011-11-12 10:51:01 +00:00
|
|
|
proto_tree *child_tree = proto_item_add_subtree(child_item, ett_unknown[level]);
|
|
|
|
|
|
|
|
if(child->default_ns_abbrev)
|
|
|
|
proto_item_append_text(child_item, "(%s)", child->default_ns_abbrev);
|
|
|
|
|
|
|
|
xmpp_unknown_items(child_tree, tvb, pinfo, child, level +1);
|
|
|
|
|
|
|
|
childs = childs->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_unknown(proto_tree *tree, tvbuff_t *tvb, packet_info *pinfo, xmpp_element_t *element)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
GList *childs = element->elements;
|
|
|
|
|
|
|
|
/*element has unrecognized elements*/
|
|
|
|
while(childs)
|
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
xmpp_element_t *child = (xmpp_element_t *)childs->data;
|
2011-11-12 10:51:01 +00:00
|
|
|
if(!child->was_read)
|
|
|
|
{
|
|
|
|
proto_item *unknown_item;
|
|
|
|
proto_tree *unknown_tree;
|
|
|
|
|
|
|
|
#ifdef XMPP_DEBUG
|
|
|
|
unknown_item = proto_tree_add_string_format(tree,
|
|
|
|
hf_xmpp_unknown, tvb, child->offset, child->length, child->name,
|
2011-11-13 23:33:47 +00:00
|
|
|
"%s", xmpp_ep_string_upcase(child->name));
|
2011-11-12 10:51:01 +00:00
|
|
|
#else
|
|
|
|
unknown_item = proto_tree_add_text(tree, tvb, child->offset, child->length,
|
2011-11-13 23:33:47 +00:00
|
|
|
"%s", xmpp_ep_string_upcase(child->name));
|
2011-11-12 10:51:01 +00:00
|
|
|
#endif
|
|
|
|
unknown_tree = proto_item_add_subtree(unknown_item, ett_unknown[0]);
|
|
|
|
|
|
|
|
/*Add COL_INFO only if root element is IQ*/
|
|
|
|
if(strcmp(element->name,"iq")==0)
|
2011-11-13 23:33:47 +00:00
|
|
|
col_append_fstr(pinfo->cinfo, COL_INFO, "%s ", xmpp_ep_string_upcase(child->name));
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if(child->default_ns_abbrev)
|
|
|
|
proto_item_append_text(unknown_item,"(%s)",child->default_ns_abbrev);
|
|
|
|
|
|
|
|
xmpp_unknown_items(unknown_tree, tvb, pinfo, child, 1);
|
|
|
|
#ifdef XMPP_DEBUG
|
|
|
|
proto_item_append_text(unknown_item, " [UNKNOWN]");
|
|
|
|
|
|
|
|
expert_add_info_format(pinfo, unknown_item, PI_UNDECODED, PI_NOTE,"Unknown element: %s", child->name);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
childs = childs->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_unknown_attrs(proto_tree *tree, tvbuff_t *tvb, packet_info *pinfo _U_, xmpp_element_t *element, gboolean displ_short_list)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
proto_item *item = proto_tree_get_parent(tree);
|
|
|
|
|
|
|
|
GList *keys = g_hash_table_get_keys(element->attrs);
|
|
|
|
GList *values = g_hash_table_get_values(element->attrs);
|
|
|
|
|
|
|
|
GList *keys_head = keys, *values_head = values;
|
|
|
|
|
|
|
|
gboolean short_list_started = FALSE;
|
|
|
|
|
|
|
|
while(keys && values)
|
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *attr = (xmpp_attr_t*) values->data;
|
2011-11-12 10:51:01 +00:00
|
|
|
if (!attr->was_read) {
|
|
|
|
if (displ_short_list) {
|
|
|
|
if (!short_list_started)
|
|
|
|
proto_item_append_text(item, " [");
|
|
|
|
else
|
|
|
|
proto_item_append_text(item, " ");
|
|
|
|
proto_item_append_text(item, "%s=\"%s\"", (gchar*) keys->data, attr->value);
|
|
|
|
|
|
|
|
short_list_started = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*If unknown element has xmlns attrib then header field hf_xmpp_xmlns is added to the tree.
|
|
|
|
In other case only text.*/
|
2013-03-02 19:21:38 +00:00
|
|
|
if (strcmp((const char *)keys->data, "xmlns") == 0)
|
2011-11-12 10:51:01 +00:00
|
|
|
proto_tree_add_string(tree, hf_xmpp_xmlns, tvb, attr->offset, attr->length, attr->value);
|
|
|
|
else {
|
|
|
|
/*xmlns may looks like xmlns:abbrev="sth"*/
|
2013-03-02 19:21:38 +00:00
|
|
|
gchar* xmlns_needle = epan_strcasestr((const char *)keys->data, "xmlns:");
|
2011-11-12 10:51:01 +00:00
|
|
|
if (xmlns_needle && xmlns_needle == keys->data) {
|
|
|
|
proto_tree_add_string_format(tree, hf_xmpp_xmlns, tvb, attr->offset, attr->length, attr->value,"%s: %s", (gchar*)keys->data, attr->value);
|
|
|
|
} else {
|
|
|
|
|
|
|
|
#ifdef XMPP_DEBUG
|
|
|
|
proto_item* unknown_attr_item;
|
|
|
|
unknown_attr_item = proto_tree_add_string_format(tree,
|
|
|
|
hf_xmpp_unknown_attr, tvb, attr->offset, attr->length,
|
|
|
|
attr->name, "%s: %s", attr->name, attr->value);
|
|
|
|
proto_item_append_text(unknown_attr_item, " [UNKNOWN ATTR]");
|
|
|
|
expert_add_info_format(pinfo, unknown_attr_item, PI_UNDECODED, PI_NOTE, "Unknown attribute %s.", attr->name);
|
|
|
|
#else
|
|
|
|
proto_tree_add_text(tree, tvb, attr->offset, attr->length,
|
|
|
|
"%s: %s", attr->name, attr->value);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
keys = keys->next;
|
|
|
|
values = values->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(short_list_started && displ_short_list)
|
|
|
|
proto_item_append_text(item, "]");
|
|
|
|
|
|
|
|
g_list_free(keys_head);
|
|
|
|
g_list_free(values_head);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_cdata(proto_tree *tree, tvbuff_t *tvb, xmpp_element_t *element, gint hf)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
if(element->data)
|
|
|
|
{
|
|
|
|
if (hf == -1) {
|
|
|
|
proto_tree_add_text(tree, tvb, element->data->offset, element->data->length, "CDATA: %s", element->data->value);
|
|
|
|
} else {
|
|
|
|
proto_tree_add_string(tree, hf, tvb, element->data->offset, element->data->length, element->data->value);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
{
|
|
|
|
if (hf == -1) {
|
|
|
|
proto_tree_add_text(tree, tvb, 0, 0, "CDATA: (empty)");
|
|
|
|
} else {
|
|
|
|
proto_tree_add_string(tree, hf, tvb, 0, 0, "");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* displays element that looks like <element_name>element_value</element_name>
|
|
|
|
* ELEMENT_NAME: element_value as TEXT(proto_tree_add_text) int PROTO_TREE
|
|
|
|
*/
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_simple_cdata_elem(proto_tree *tree, tvbuff_t *tvb, packet_info *pinfo _U_, xmpp_element_t *element)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
proto_tree_add_text(tree, tvb, element->offset, element->length, "%s: %s", xmpp_ep_string_upcase(element->name), xmpp_elem_cdata(element));
|
2011-11-12 10:51:01 +00:00
|
|
|
}
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_array_t*
|
|
|
|
xmpp_ep_init_array_t(const gchar** array, gint len)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_array_t *result;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
result = ep_new(xmpp_array_t);
|
2011-11-12 10:51:01 +00:00
|
|
|
result->data = (gpointer) array;
|
|
|
|
result->length = len;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t*
|
2012-12-26 05:57:06 +00:00
|
|
|
xmpp_ep_init_attr_t(const gchar *value, gint offset, gint length)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *result;
|
2013-03-02 19:21:38 +00:00
|
|
|
result = ep_new(xmpp_attr_t);
|
2011-11-12 10:51:01 +00:00
|
|
|
result->value = value;
|
|
|
|
result->offset = offset;
|
|
|
|
result->length = length;
|
|
|
|
result->name = NULL;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
gchar*
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_ep_string_upcase(const gchar* string)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-12 11:27:41 +00:00
|
|
|
gint len = (int)strlen(string);
|
2011-11-12 10:51:01 +00:00
|
|
|
gint i;
|
2013-03-02 19:21:38 +00:00
|
|
|
gchar* result = (gchar *)ep_alloc0(len+1);
|
2011-11-12 10:51:01 +00:00
|
|
|
for(i=0; i<len; i++)
|
|
|
|
{
|
|
|
|
result[i] = string[i];
|
|
|
|
|
|
|
|
if(string[i]>='a' && string[i]<='z')
|
|
|
|
result[i]-='a'-'A';
|
|
|
|
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
gint
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t_cmp(gconstpointer a, gconstpointer b)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
gint result = strcmp(((xmpp_element_t*)a)->name,((xmpp_element_t*)b)->name);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
if(result == 0 && ((xmpp_element_t*)a)->was_read)
|
2011-11-12 10:51:01 +00:00
|
|
|
result = -1;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
GList*
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_find_element_by_name(xmpp_element_t *packet,const gchar *name)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
GList *found_elements;
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t *search_element;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
/*create fake element only with name*/
|
2013-03-02 19:21:38 +00:00
|
|
|
search_element = ep_new(xmpp_element_t);
|
2011-11-12 10:51:01 +00:00
|
|
|
search_element->name = ep_strdup(name);
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
found_elements = g_list_find_custom(packet->elements, search_element, xmpp_element_t_cmp);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if(found_elements)
|
|
|
|
return found_elements;
|
|
|
|
else
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* steal_*
|
|
|
|
* function searches element in packet and sets it as read.
|
|
|
|
* if element doesn't exist, NULL is returned.
|
|
|
|
* If element is set as read, it is invisible for these functions.*/
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t*
|
|
|
|
xmpp_steal_element_by_name(xmpp_element_t *packet,const gchar *name)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
GList *element_l;
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t *element = NULL;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
element_l = xmpp_find_element_by_name(packet, name);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if(element_l)
|
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
element = (xmpp_element_t *)element_l->data;
|
2011-11-12 10:51:01 +00:00
|
|
|
element->was_read = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return element;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t*
|
|
|
|
xmpp_steal_element_by_names(xmpp_element_t *packet, const gchar **names, gint names_len)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
gint i;
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t *el = NULL;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
for(i = 0; i<names_len; i++)
|
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
if((el = xmpp_steal_element_by_name(packet, names[i])))
|
2011-11-12 10:51:01 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return el;
|
|
|
|
}
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t*
|
|
|
|
xmpp_steal_element_by_attr(xmpp_element_t *packet, const gchar *attr_name, const gchar *attr_value)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
GList *childs = packet->elements;
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t *result = NULL;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
while (childs) {
|
2013-03-02 19:21:38 +00:00
|
|
|
xmpp_element_t *child_elem = (xmpp_element_t *)childs->data;
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *attr = xmpp_get_attr(child_elem, attr_name);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if(attr)
|
|
|
|
attr->was_read = FALSE;
|
|
|
|
|
|
|
|
if (!child_elem->was_read && attr && strcmp(attr->value, attr_value) == 0) {
|
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
result = (xmpp_element_t *)childs->data;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
result->was_read = TRUE;
|
|
|
|
|
|
|
|
break;
|
|
|
|
} else
|
|
|
|
childs = childs->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t*
|
|
|
|
xmpp_steal_element_by_name_and_attr(xmpp_element_t *packet, const gchar *name, const gchar *attr_name, const gchar *attr_value)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
GList *childs = packet->elements;
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t *result = NULL;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
while (childs) {
|
2013-03-02 19:21:38 +00:00
|
|
|
xmpp_element_t *child_elem = (xmpp_element_t *)childs->data;
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *attr = xmpp_get_attr(child_elem, attr_name);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if(attr)
|
|
|
|
attr->was_read = FALSE;
|
|
|
|
|
|
|
|
if (!child_elem->was_read && attr && strcmp(child_elem->name, name) == 0 && strcmp(attr->value, attr_value) == 0) {
|
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
result = (xmpp_element_t *)childs->data;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
result->was_read = TRUE;
|
|
|
|
|
|
|
|
break;
|
|
|
|
} else
|
|
|
|
childs = childs->next;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t*
|
|
|
|
xmpp_get_first_element(xmpp_element_t *packet)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
if(packet->elements && packet->elements->data)
|
2013-03-02 19:21:38 +00:00
|
|
|
return (xmpp_element_t *)packet->elements->data;
|
2011-11-12 10:51:01 +00:00
|
|
|
else
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2011-11-13 23:33:47 +00:00
|
|
|
Function converts xml_frame_t structure to xmpp_element_t (simpler representation)
|
2011-11-12 10:51:01 +00:00
|
|
|
*/
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t*
|
|
|
|
xmpp_xml_frame_to_element_t(xml_frame_t *xml_frame, xmpp_element_t *parent, tvbuff_t *tvb)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
xml_frame_t *child;
|
2013-03-02 19:21:38 +00:00
|
|
|
xmpp_element_t *node = ep_new0(xmpp_element_t);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
tvbparse_t* tt;
|
|
|
|
tvbparse_elem_t* elem;
|
|
|
|
|
|
|
|
node->attrs = g_hash_table_new(g_str_hash, g_str_equal);
|
|
|
|
node->elements = NULL;
|
|
|
|
node->data = NULL;
|
|
|
|
node->was_read = FALSE;
|
|
|
|
node->default_ns_abbrev = NULL;
|
|
|
|
|
|
|
|
node->name = ep_strdup(xml_frame->name_orig_case);
|
|
|
|
node->offset = 0;
|
|
|
|
node->length = 0;
|
|
|
|
|
|
|
|
node->namespaces = g_hash_table_new(g_str_hash, g_str_equal);
|
|
|
|
if(parent)
|
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_copy_hash_table(parent->namespaces, node->namespaces);
|
2011-11-12 10:51:01 +00:00
|
|
|
} else
|
|
|
|
{
|
2012-12-26 05:57:06 +00:00
|
|
|
g_hash_table_insert(node->namespaces, (gpointer)"", (gpointer)"jabber:client");
|
2011-11-12 10:51:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(xml_frame->item != NULL)
|
|
|
|
{
|
|
|
|
node->length = xml_frame->item->finfo->length;
|
|
|
|
}
|
|
|
|
|
|
|
|
node->offset = xml_frame->start_offset;
|
|
|
|
|
|
|
|
tt = tvbparse_init(tvb,node->offset,-1,NULL,want_ignore);
|
|
|
|
|
2012-07-22 21:48:51 +00:00
|
|
|
if((elem = tvbparse_get(tt,want_stream_end_with_ns))!=NULL)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
node->default_ns_abbrev = tvb_get_ephemeral_string(elem->sub->tvb, elem->sub->offset, elem->sub->len);
|
|
|
|
}
|
|
|
|
|
|
|
|
child = xml_frame->first_child;
|
|
|
|
|
|
|
|
while(child)
|
|
|
|
{
|
|
|
|
if(child->type != XML_FRAME_TAG)
|
|
|
|
{
|
|
|
|
if(child->type == XML_FRAME_ATTRIB)
|
|
|
|
{
|
|
|
|
gint l;
|
|
|
|
gchar *value = NULL;
|
|
|
|
gchar *xmlns_needle = NULL;
|
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
xmpp_attr_t *attr = ep_new(xmpp_attr_t);
|
2011-11-12 10:51:01 +00:00
|
|
|
attr->length = 0;
|
|
|
|
attr->offset = 0;
|
|
|
|
attr->was_read = FALSE;
|
2011-11-13 23:33:47 +00:00
|
|
|
|
2011-11-12 10:51:01 +00:00
|
|
|
if (child->value != NULL) {
|
|
|
|
l = tvb_reported_length(child->value);
|
2013-03-02 19:21:38 +00:00
|
|
|
value = (gchar *)ep_alloc0(l + 1);
|
2011-11-12 10:51:01 +00:00
|
|
|
tvb_memcpy(child->value, value, 0, l);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(child->item)
|
|
|
|
{
|
|
|
|
attr->length = child->item->finfo->length;
|
|
|
|
}
|
2011-11-13 23:33:47 +00:00
|
|
|
|
2011-11-12 10:51:01 +00:00
|
|
|
attr->offset = child->start_offset;
|
|
|
|
attr->value = value;
|
|
|
|
attr->name = ep_strdup(child->name_orig_case);
|
|
|
|
|
|
|
|
g_hash_table_insert(node->attrs,(gpointer)attr->name,(gpointer)attr);
|
|
|
|
|
|
|
|
/*checking that attr->name looks like xmlns:ns*/
|
|
|
|
xmlns_needle = epan_strcasestr(attr->name, "xmlns");
|
|
|
|
|
|
|
|
if(xmlns_needle == attr->name)
|
|
|
|
{
|
|
|
|
if(attr->name[5] == ':' && strlen(attr->name) > 6)
|
|
|
|
{
|
|
|
|
g_hash_table_insert(node->namespaces, (gpointer)ep_strdup(&attr->name[6]), (gpointer)ep_strdup(attr->value));
|
|
|
|
} else if(attr->name[5] == '\0')
|
|
|
|
{
|
2012-12-26 05:57:06 +00:00
|
|
|
g_hash_table_insert(node->namespaces, (gpointer)"", (gpointer)ep_strdup(attr->value));
|
2011-11-12 10:51:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
else if( child->type == XML_FRAME_CDATA)
|
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_data_t *data = NULL;
|
2011-11-12 10:51:01 +00:00
|
|
|
gint l;
|
|
|
|
gchar* value = NULL;
|
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
data = ep_new(xmpp_data_t);
|
2011-11-12 10:51:01 +00:00
|
|
|
data->length = 0;
|
|
|
|
data->offset = 0;
|
|
|
|
|
|
|
|
if (child->value != NULL) {
|
|
|
|
l = tvb_reported_length(child->value);
|
2013-03-02 19:21:38 +00:00
|
|
|
value = (gchar *)ep_alloc0(l + 1);
|
2011-11-12 10:51:01 +00:00
|
|
|
tvb_memcpy(child->value, value, 0, l);
|
|
|
|
}
|
|
|
|
|
|
|
|
data->value = value;
|
|
|
|
|
|
|
|
if(child->item)
|
|
|
|
{
|
|
|
|
data->length = child->item->finfo->length;
|
|
|
|
}
|
|
|
|
data->offset = child->start_offset;
|
|
|
|
node->data = data;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
node->elements = g_list_append(node->elements,(gpointer)xmpp_xml_frame_to_element_t(child, node,tvb));
|
2011-11-12 10:51:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
child = child->next_sibling;
|
|
|
|
}
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t_tree_free(xmpp_element_t *root)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
GList *childs = root->elements;
|
|
|
|
|
|
|
|
g_hash_table_destroy(root->attrs);
|
|
|
|
g_hash_table_destroy(root->namespaces);
|
|
|
|
|
|
|
|
while(childs)
|
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
xmpp_element_t *child = (xmpp_element_t *)childs->data;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t_tree_free(child);
|
2011-11-12 10:51:01 +00:00
|
|
|
childs = childs->next;
|
|
|
|
}
|
|
|
|
g_list_free(root->elements);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*Function recognize attribute names if they looks like xmlns:ns*/
|
|
|
|
static gboolean
|
|
|
|
attr_find_pred(gpointer key, gpointer value _U_, gpointer user_data)
|
|
|
|
{
|
|
|
|
gchar *attr_name = (gchar*) user_data;
|
|
|
|
|
|
|
|
if( strcmp(attr_name, "xmlns") == 0 )
|
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
gchar *first_occur = epan_strcasestr((const char *)key, "xmlns:");
|
2011-11-12 10:51:01 +00:00
|
|
|
if(first_occur && first_occur == key)
|
|
|
|
return TRUE;
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*Functions returns element's attibute by name and set as read*/
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t*
|
|
|
|
xmpp_get_attr(xmpp_element_t *element, const gchar* attr_name)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
xmpp_attr_t *result = (xmpp_attr_t *)g_hash_table_lookup(element->attrs, attr_name);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if(!result)
|
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
result = (xmpp_attr_t *)g_hash_table_find(element->attrs, attr_find_pred, (gpointer)attr_name);
|
2011-11-12 10:51:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(result)
|
|
|
|
result->was_read = TRUE;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*Functions returns element's attibute by name and namespace abbrev*/
|
2011-11-13 23:33:47 +00:00
|
|
|
static xmpp_attr_t*
|
|
|
|
xmpp_get_attr_ext(xmpp_element_t *element, const gchar* attr_name, const gchar* ns_abbrev)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
gchar* search_phrase;
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *result;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if(strcmp(ns_abbrev,"")==0)
|
|
|
|
search_phrase = ep_strdup(attr_name);
|
|
|
|
else if(strcmp(attr_name, "xmlns") == 0)
|
|
|
|
search_phrase = ep_strdup_printf("%s:%s",attr_name, ns_abbrev);
|
|
|
|
else
|
|
|
|
search_phrase = ep_strdup_printf("%s:%s", ns_abbrev, attr_name);
|
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
result = (xmpp_attr_t *)g_hash_table_lookup(element->attrs, search_phrase);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if(!result)
|
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
result = (xmpp_attr_t *)g_hash_table_find(element->attrs, attr_find_pred, (gpointer)attr_name);
|
2011-11-12 10:51:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(result)
|
|
|
|
result->was_read = TRUE;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
gchar*
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_to_string(tvbuff_t *tvb, xmpp_element_t *element)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
gchar *buff = NULL;
|
|
|
|
|
|
|
|
if(tvb_offset_exists(tvb, element->offset+element->length-1))
|
|
|
|
{
|
|
|
|
buff = tvb_get_ephemeral_string(tvb, element->offset, element->length);
|
|
|
|
}
|
|
|
|
return buff;
|
|
|
|
}
|
|
|
|
|
|
|
|
gchar*
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_to_string(tvbuff_t *tvb, xmpp_attr_t *attr)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
gchar *buff = NULL;
|
|
|
|
|
|
|
|
if(tvb_offset_exists(tvb, attr->offset + attr->length-1))
|
|
|
|
{
|
|
|
|
buff = tvb_get_ephemeral_string(tvb, attr->offset, attr->length);
|
|
|
|
}
|
|
|
|
return buff;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
children_foreach_hide_func(proto_node *node, gpointer data)
|
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
int *i = (int *)data;
|
2011-11-12 10:51:01 +00:00
|
|
|
if((*i) == 0)
|
|
|
|
PROTO_ITEM_SET_HIDDEN(node);
|
|
|
|
(*i)++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
children_foreach_show_func(proto_node *node, gpointer data)
|
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
int *i = (int *)data;
|
2011-11-12 10:51:01 +00:00
|
|
|
if((*i) == 0)
|
|
|
|
PROTO_ITEM_SET_VISIBLE(node);
|
|
|
|
(*i)++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_proto_tree_hide_first_child(proto_tree *tree)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
int i = 0;
|
|
|
|
proto_tree_children_foreach(tree, children_foreach_hide_func, &i);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_proto_tree_show_first_child(proto_tree *tree)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
int i = 0;
|
|
|
|
proto_tree_children_foreach(tree, children_foreach_show_func, &i);
|
|
|
|
}
|
|
|
|
|
|
|
|
gchar*
|
|
|
|
proto_item_get_text(proto_item *item)
|
|
|
|
{
|
|
|
|
field_info *fi = NULL;
|
|
|
|
gchar *result;
|
|
|
|
|
|
|
|
if(item == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
fi = PITEM_FINFO(item);
|
|
|
|
|
|
|
|
if(fi==NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (fi->rep == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
|
|
result = ep_strdup(fi->rep->representation);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_display_attrs(proto_tree *tree, xmpp_element_t *element, packet_info *pinfo, tvbuff_t *tvb, xmpp_attr_info *attrs, guint n)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
proto_item *item = proto_tree_get_parent(tree);
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *attr;
|
2011-11-12 10:51:01 +00:00
|
|
|
guint i;
|
|
|
|
gboolean short_list_started = FALSE;
|
|
|
|
|
|
|
|
if(element->default_ns_abbrev)
|
|
|
|
proto_item_append_text(item, "(%s)",element->default_ns_abbrev);
|
|
|
|
|
|
|
|
proto_item_append_text(item," [");
|
|
|
|
for(i = 0; i < n && attrs!=NULL; i++)
|
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
attr = xmpp_get_attr(element, attrs[i].name);
|
2011-11-12 10:51:01 +00:00
|
|
|
if(attr)
|
|
|
|
{
|
|
|
|
if(attrs[i].hf != -1)
|
|
|
|
{
|
|
|
|
if(attr->name)
|
|
|
|
proto_tree_add_string_format(tree, attrs[i].hf, tvb, attr->offset, attr->length, attr->value,"%s: %s", attr->name, attr->value);
|
|
|
|
else
|
|
|
|
proto_tree_add_string(tree, attrs[i].hf, tvb, attr->offset, attr->length, attr->value);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
proto_tree_add_text(tree, tvb, attr->offset, attr->length, "%s: %s", attr->name?attr->name:attrs[i].name, attr->value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(attrs[i].in_short_list)
|
|
|
|
{
|
|
|
|
if(short_list_started)
|
|
|
|
{
|
|
|
|
proto_item_append_text(item," ");
|
|
|
|
}
|
|
|
|
proto_item_append_text(item,"%s=\"%s\"",attr->name?attr->name:attrs[i].name, attr->value);
|
|
|
|
short_list_started = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else if(attrs[i].is_required)
|
|
|
|
{
|
|
|
|
expert_add_info_format(pinfo, item, PI_PROTOCOL, PI_WARN,
|
|
|
|
"Required attribute \"%s\" doesn't appear in \"%s\".",attrs[i].name,
|
|
|
|
element->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(attrs[i].val_func)
|
|
|
|
{
|
|
|
|
if(attr)
|
|
|
|
attrs[i].val_func(pinfo, item, attrs[i].name, attr->value, attrs[i].data);
|
|
|
|
else
|
|
|
|
attrs[i].val_func(pinfo, item, attrs[i].name, NULL, attrs[i].data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
proto_item_append_text(item,"]");
|
|
|
|
|
|
|
|
/*displays attributes that weren't recognized*/
|
|
|
|
xmpp_unknown_attrs(tree, tvb, pinfo, element, FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_display_attrs_ext(proto_tree *tree, xmpp_element_t *element, packet_info *pinfo, tvbuff_t *tvb, xmpp_attr_info_ext *attrs, guint n)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
proto_item *item = proto_tree_get_parent(tree);
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *attr;
|
2011-11-12 10:51:01 +00:00
|
|
|
guint i;
|
|
|
|
gboolean short_list_started = FALSE;
|
|
|
|
|
|
|
|
GList *ns_abbrevs_head, *ns_abbrevs = g_hash_table_get_keys(element->namespaces);
|
|
|
|
GList *ns_fullnames_head, *ns_fullnames = g_hash_table_get_values(element->namespaces);
|
|
|
|
ns_abbrevs_head = ns_abbrevs;
|
|
|
|
ns_fullnames_head = ns_fullnames;
|
|
|
|
|
|
|
|
if(element->default_ns_abbrev)
|
|
|
|
proto_item_append_text(item, "(%s)",element->default_ns_abbrev);
|
|
|
|
|
|
|
|
proto_item_append_text(item," [");
|
|
|
|
while(ns_abbrevs && ns_fullnames)
|
|
|
|
{
|
|
|
|
for (i = 0; i < n && attrs != NULL; i++) {
|
2013-03-02 19:21:38 +00:00
|
|
|
if(strcmp((const char *)(ns_fullnames->data), attrs[i].ns) == 0)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
attr = xmpp_get_attr_ext(element, attrs[i].info.name, (const gchar *)(ns_abbrevs->data));
|
|
|
|
if(!attr && element->default_ns_abbrev && strcmp((const char *)ns_abbrevs->data, element->default_ns_abbrev)==0)
|
2011-11-13 23:33:47 +00:00
|
|
|
attr = xmpp_get_attr_ext(element, attrs[i].info.name, "");
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if (attr) {
|
|
|
|
if (attrs[i].info.hf != -1) {
|
|
|
|
if (attr->name)
|
|
|
|
proto_tree_add_string_format(tree, attrs[i].info.hf, tvb, attr->offset, attr->length, attr->value, "%s: %s", attr->name, attr->value);
|
|
|
|
else
|
|
|
|
proto_tree_add_string(tree, attrs[i].info.hf, tvb, attr->offset, attr->length, attr->value);
|
|
|
|
} else {
|
|
|
|
proto_tree_add_text(tree, tvb, attr->offset, attr->length, "%s: %s", attr->name ? attr->name : attrs[i].info.name, attr->value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (attrs[i].info.in_short_list) {
|
|
|
|
if (short_list_started) {
|
|
|
|
proto_item_append_text(item, " ");
|
|
|
|
}
|
|
|
|
proto_item_append_text(item, "%s=\"%s\"", attr->name ? attr->name : attrs[i].info.name, attr->value);
|
|
|
|
short_list_started = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else if (attrs[i].info.is_required) {
|
|
|
|
expert_add_info_format(pinfo, item, PI_PROTOCOL, PI_WARN,
|
|
|
|
"Required attribute \"%s\" doesn't appear in \"%s\".", attrs[i].info.name,
|
|
|
|
element->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (attrs[i].info.val_func) {
|
|
|
|
if (attr)
|
|
|
|
attrs[i].info.val_func(pinfo, item, attrs[i].info.name, attr->value, attrs[i].info.data);
|
|
|
|
else
|
|
|
|
attrs[i].info.val_func(pinfo, item, attrs[i].info.name, NULL, attrs[i].info.data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ns_abbrevs = ns_abbrevs->next;
|
|
|
|
ns_fullnames = ns_fullnames->next;
|
|
|
|
}
|
|
|
|
proto_item_append_text(item,"]");
|
|
|
|
|
|
|
|
/*displays attributes that weren't recognized*/
|
|
|
|
xmpp_unknown_attrs(tree, tvb, pinfo, element, FALSE);
|
|
|
|
|
|
|
|
g_list_free(ns_abbrevs_head);
|
|
|
|
g_list_free(ns_fullnames_head);
|
|
|
|
}
|
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
typedef struct _name_attr_t
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2012-12-26 05:57:06 +00:00
|
|
|
const gchar *name;
|
|
|
|
const gchar *attr_name;
|
|
|
|
const gchar *attr_value;
|
2013-03-02 19:21:38 +00:00
|
|
|
} name_attr_t;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
returns pointer to the struct that contains 3 strings(element name, attribute name, attribute value)
|
|
|
|
*/
|
|
|
|
gpointer
|
2012-12-26 05:57:06 +00:00
|
|
|
xmpp_name_attr_struct(const gchar *name, const gchar *attr_name, const gchar *attr_value)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
name_attr_t *result;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
result = ep_new(name_attr_t);
|
2011-11-12 10:51:01 +00:00
|
|
|
result->name = name;
|
|
|
|
result->attr_name = attr_name;
|
|
|
|
result->attr_value = attr_value;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_display_elems(proto_tree *tree, xmpp_element_t *parent, packet_info *pinfo, tvbuff_t *tvb, xmpp_elem_info *elems, guint n)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
guint i;
|
|
|
|
|
|
|
|
for(i = 0; i < n && elems!=NULL; i++)
|
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t *elem = NULL;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
if(elems[i].type == NAME_AND_ATTR)
|
|
|
|
{
|
|
|
|
gboolean loop = TRUE;
|
|
|
|
|
2013-03-02 19:21:38 +00:00
|
|
|
name_attr_t *a = (name_attr_t *)(elems[i].data);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
while(loop && (elem = xmpp_steal_element_by_name_and_attr(parent, a->name, a->attr_name, a->attr_value))!=NULL)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
elems[i].elem_func(tree, tvb, pinfo, elem);
|
|
|
|
if(elems[i].occurrence == ONE)
|
|
|
|
loop = FALSE;
|
|
|
|
}
|
|
|
|
} else if(elems[i].type == NAME)
|
|
|
|
{
|
|
|
|
gboolean loop = TRUE;
|
2013-03-02 19:21:38 +00:00
|
|
|
const gchar *name = (const gchar *)(elems[i].data);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
while(loop && (elem = xmpp_steal_element_by_name(parent, name))!=NULL)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
elems[i].elem_func(tree, tvb, pinfo, elem);
|
|
|
|
if(elems[i].occurrence == ONE)
|
|
|
|
loop = FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(elems[i].type == ATTR)
|
|
|
|
{
|
|
|
|
gboolean loop = TRUE;
|
2013-03-02 19:21:38 +00:00
|
|
|
name_attr_t *attr = (name_attr_t *)(elems[i].data);
|
2011-11-13 23:33:47 +00:00
|
|
|
|
|
|
|
while(loop && (elem = xmpp_steal_element_by_attr(parent, attr->attr_name, attr->attr_value))!=NULL)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
elems[i].elem_func(tree, tvb, pinfo, elem);
|
|
|
|
if(elems[i].occurrence == ONE)
|
|
|
|
loop = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else if(elems[i].type == NAMES)
|
|
|
|
{
|
|
|
|
gboolean loop = TRUE;
|
2013-03-02 19:21:38 +00:00
|
|
|
const xmpp_array_t *names = (const xmpp_array_t *)(elems[i].data);
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
while(loop && (elem = xmpp_steal_element_by_names(parent, (const gchar**)names->data, names->length))!=NULL)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
elems[i].elem_func(tree, tvb, pinfo, elem);
|
|
|
|
if(elems[i].occurrence == ONE)
|
|
|
|
loop = FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
xmpp_unknown(tree, tvb, pinfo, parent);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2011-11-13 23:33:47 +00:00
|
|
|
function checks that variable value is in array ((xmpp_array_t)data)->data
|
2011-11-12 10:51:01 +00:00
|
|
|
*/
|
|
|
|
void
|
2012-12-26 05:57:06 +00:00
|
|
|
xmpp_val_enum_list(packet_info *pinfo, proto_item *item, const gchar *name, const gchar *value, gconstpointer data)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
const xmpp_array_t *enums_array = (const xmpp_array_t *)data;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
|
|
|
gint i;
|
|
|
|
gboolean value_in_enums = FALSE;
|
|
|
|
|
|
|
|
gchar **enums = (char**)enums_array->data;
|
|
|
|
|
|
|
|
if (value != NULL) {
|
|
|
|
for (i = 0; i < enums_array->length; i++) {
|
|
|
|
if (strcmp(value, enums[i]) == 0) {
|
|
|
|
value_in_enums = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!value_in_enums) {
|
|
|
|
expert_add_info_format(pinfo, item, PI_PROTOCOL, PI_WARN,
|
|
|
|
"Field \"%s\" has unexpected value \"%s\"",
|
|
|
|
name, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_change_elem_to_attrib(const gchar *elem_name, const gchar *attr_name, xmpp_element_t *parent, xmpp_attr_t* (*transform_func)(xmpp_element_t *element))
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_element_t *element = NULL;
|
|
|
|
xmpp_attr_t *fake_attr = NULL;
|
2011-11-12 10:51:01 +00:00
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
element = xmpp_steal_element_by_name(parent, elem_name);
|
2011-11-12 10:51:01 +00:00
|
|
|
if(element)
|
|
|
|
fake_attr = transform_func(element);
|
|
|
|
|
|
|
|
if(fake_attr)
|
|
|
|
g_hash_table_insert(parent->attrs, (gpointer)attr_name, fake_attr);
|
|
|
|
}
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t*
|
|
|
|
xmpp_transform_func_cdata(xmpp_element_t *elem)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_attr_t *result = xmpp_ep_init_attr_t(elem->data?elem->data->value:"", elem->offset, elem->length);
|
2011-11-12 10:51:01 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-11-13 23:33:47 +00:00
|
|
|
xmpp_copy_hash_table_func(gpointer key, gpointer value, gpointer user_data)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2013-03-02 19:21:38 +00:00
|
|
|
GHashTable *dst = (GHashTable *)user_data;
|
2011-11-12 10:51:01 +00:00
|
|
|
g_hash_table_insert(dst, key, value);
|
|
|
|
}
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
void xmpp_copy_hash_table(GHashTable *src, GHashTable *dst)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
2011-11-13 23:33:47 +00:00
|
|
|
g_hash_table_foreach(src, xmpp_copy_hash_table_func, dst);
|
2011-11-12 10:51:01 +00:00
|
|
|
}
|
|
|
|
|
2011-11-13 23:33:47 +00:00
|
|
|
#if 0
|
2011-11-12 10:51:01 +00:00
|
|
|
static void
|
|
|
|
printf_hash_table_func(gpointer key, gpointer value, gpointer user_data _U_)
|
|
|
|
{
|
|
|
|
printf("'%s' '%s'\n", (gchar*)key, (gchar*)value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-11-13 23:33:47 +00:00
|
|
|
printf_elements(xmpp_element_t *root)
|
2011-11-12 10:51:01 +00:00
|
|
|
{
|
|
|
|
GList *elems = root->elements;
|
|
|
|
|
|
|
|
printf("%s\n", root->name);
|
|
|
|
g_hash_table_foreach(root->namespaces, printf_hash_table_func, NULL);
|
|
|
|
while(elems)
|
|
|
|
{
|
|
|
|
printf_elements(elems->data);
|
|
|
|
elems = elems->next;
|
|
|
|
}
|
|
|
|
}
|
2011-11-13 23:33:47 +00:00
|
|
|
#endif
|
|
|
|
|
2011-11-12 10:51:01 +00:00
|
|
|
/*
|
|
|
|
* Editor modelines - http://www.wireshark.org/tools/modelines.html
|
|
|
|
*
|
|
|
|
* Local variables:
|
|
|
|
* c-basic-offset: 4
|
|
|
|
* tab-width: 8
|
|
|
|
* indent-tabs-mode: nil
|
|
|
|
* End:
|
|
|
|
*
|
|
|
|
* ex: set shiftwidth=4 tabstop=8 expandtab:
|
|
|
|
* :indentSize=4:tabSize=8:noTabs=true:
|
|
|
|
*/
|