Standardize all protocol header guard macros.
[pidgin-git.git] / libpurple / xmlnode.c
blob98471539700675cd4b67b7e6c8c212876ba63f01
1 /* purple
3 * Purple is the legal property of its developers, whose names are too numerous
4 * to list here. Please refer to the COPYRIGHT file distributed with this
5 * source distribution.
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA
22 /* A lot of this code at least resembles the code in libxode, but since
23 * libxode uses memory pools that we simply have no need for, I decided to
24 * write my own stuff. Also, re-writing this lets me be as lightweight
25 * as I want to be. Thank you libxode for giving me a good starting point */
27 #include "internal.h"
28 #include "debug.h"
30 #include <libxml/parser.h>
31 #include <string.h>
32 #include <glib.h>
34 #include "util.h"
35 #include "xmlnode.h"
37 #ifdef _WIN32
38 # define NEWLINE_S "\r\n"
39 #else
40 # define NEWLINE_S "\n"
41 #endif
43 static PurpleXmlNode*
44 new_node(const char *name, PurpleXmlNodeType type)
46 PurpleXmlNode *node = g_new0(PurpleXmlNode, 1);
48 node->name = g_strdup(name);
49 node->type = type;
51 return node;
54 PurpleXmlNode*
55 purple_xmlnode_new(const char *name)
57 g_return_val_if_fail(name != NULL && *name != '\0', NULL);
59 return new_node(name, PURPLE_XMLNODE_TYPE_TAG);
62 PurpleXmlNode *
63 purple_xmlnode_new_child(PurpleXmlNode *parent, const char *name)
65 PurpleXmlNode *node;
67 g_return_val_if_fail(parent != NULL, NULL);
68 g_return_val_if_fail(name != NULL && *name != '\0', NULL);
70 node = new_node(name, PURPLE_XMLNODE_TYPE_TAG);
72 purple_xmlnode_insert_child(parent, node);
74 return node;
77 void
78 purple_xmlnode_insert_child(PurpleXmlNode *parent, PurpleXmlNode *child)
80 g_return_if_fail(parent != NULL);
81 g_return_if_fail(child != NULL);
83 child->parent = parent;
85 if(parent->lastchild) {
86 parent->lastchild->next = child;
87 } else {
88 parent->child = child;
91 parent->lastchild = child;
94 void
95 purple_xmlnode_insert_data(PurpleXmlNode *node, const char *data, gssize size)
97 PurpleXmlNode *child;
98 gsize real_size;
100 g_return_if_fail(node != NULL);
101 g_return_if_fail(data != NULL);
102 g_return_if_fail(size != 0);
104 real_size = size == -1 ? strlen(data) : (gsize)size;
106 child = new_node(NULL, PURPLE_XMLNODE_TYPE_DATA);
108 child->data = g_memdup(data, real_size);
109 child->data_sz = real_size;
111 purple_xmlnode_insert_child(node, child);
114 void
115 purple_xmlnode_remove_attrib(PurpleXmlNode *node, const char *attr)
117 PurpleXmlNode *attr_node, *sibling = NULL;
119 g_return_if_fail(node != NULL);
120 g_return_if_fail(attr != NULL);
122 attr_node = node->child;
123 while (attr_node) {
124 if(attr_node->type == PURPLE_XMLNODE_TYPE_ATTRIB &&
125 purple_strequal(attr_node->name, attr))
127 if (node->lastchild == attr_node) {
128 node->lastchild = sibling;
130 if (sibling == NULL) {
131 node->child = attr_node->next;
132 purple_xmlnode_free(attr_node);
133 attr_node = node->child;
134 } else {
135 sibling->next = attr_node->next;
136 sibling = attr_node->next;
137 purple_xmlnode_free(attr_node);
138 attr_node = sibling;
141 else
143 attr_node = attr_node->next;
145 sibling = attr_node;
149 void
150 purple_xmlnode_remove_attrib_with_namespace(PurpleXmlNode *node, const char *attr, const char *xmlns)
152 PurpleXmlNode *attr_node, *sibling = NULL;
154 g_return_if_fail(node != NULL);
155 g_return_if_fail(attr != NULL);
157 for(attr_node = node->child; attr_node; attr_node = attr_node->next)
159 if(attr_node->type == PURPLE_XMLNODE_TYPE_ATTRIB &&
160 purple_strequal(attr, attr_node->name) &&
161 purple_strequal(xmlns, attr_node->xmlns))
163 if(sibling == NULL) {
164 node->child = attr_node->next;
165 } else {
166 sibling->next = attr_node->next;
168 if (node->lastchild == attr_node) {
169 node->lastchild = sibling;
171 purple_xmlnode_free(attr_node);
172 return;
174 sibling = attr_node;
178 void
179 purple_xmlnode_set_attrib(PurpleXmlNode *node, const char *attr, const char *value)
181 purple_xmlnode_remove_attrib(node, attr);
182 purple_xmlnode_set_attrib_full(node, attr, NULL, NULL, value);
185 void
186 purple_xmlnode_set_attrib_full(PurpleXmlNode *node, const char *attr, const char *xmlns, const char *prefix, const char *value)
188 PurpleXmlNode *attrib_node;
190 g_return_if_fail(node != NULL);
191 g_return_if_fail(attr != NULL);
192 g_return_if_fail(value != NULL);
194 purple_xmlnode_remove_attrib_with_namespace(node, attr, xmlns);
195 attrib_node = new_node(attr, PURPLE_XMLNODE_TYPE_ATTRIB);
197 attrib_node->data = g_strdup(value);
198 attrib_node->xmlns = g_strdup(xmlns);
199 attrib_node->prefix = g_strdup(prefix);
201 purple_xmlnode_insert_child(node, attrib_node);
205 const char *
206 purple_xmlnode_get_attrib(const PurpleXmlNode *node, const char *attr)
208 PurpleXmlNode *x;
210 g_return_val_if_fail(node != NULL, NULL);
211 g_return_val_if_fail(attr != NULL, NULL);
213 for(x = node->child; x; x = x->next) {
214 if(x->type == PURPLE_XMLNODE_TYPE_ATTRIB && purple_strequal(attr, x->name)) {
215 return x->data;
219 return NULL;
222 const char *
223 purple_xmlnode_get_attrib_with_namespace(const PurpleXmlNode *node, const char *attr, const char *xmlns)
225 const PurpleXmlNode *x;
227 g_return_val_if_fail(node != NULL, NULL);
228 g_return_val_if_fail(attr != NULL, NULL);
230 for(x = node->child; x; x = x->next) {
231 if(x->type == PURPLE_XMLNODE_TYPE_ATTRIB &&
232 purple_strequal(attr, x->name) &&
233 purple_strequal(xmlns, x->xmlns)) {
234 return x->data;
238 return NULL;
242 void purple_xmlnode_set_namespace(PurpleXmlNode *node, const char *xmlns)
244 char *tmp;
245 g_return_if_fail(node != NULL);
247 tmp = node->xmlns;
248 node->xmlns = g_strdup(xmlns);
250 if (node->namespace_map) {
251 g_hash_table_insert(node->namespace_map,
252 g_strdup(""), g_strdup(xmlns));
255 g_free(tmp);
258 const char *purple_xmlnode_get_namespace(const PurpleXmlNode *node)
260 g_return_val_if_fail(node != NULL, NULL);
262 return node->xmlns;
265 const char *purple_xmlnode_get_default_namespace(const PurpleXmlNode *node)
267 const PurpleXmlNode *current_node;
268 const char *ns = NULL;
270 g_return_val_if_fail(node != NULL, NULL);
272 current_node = node;
273 while (current_node) {
274 /* If this node does *not* have a prefix, node->xmlns is the default
275 * namespace. Otherwise, it's the prefix namespace.
277 if (!current_node->prefix && current_node->xmlns) {
278 return current_node->xmlns;
279 } else if (current_node->namespace_map) {
280 ns = g_hash_table_lookup(current_node->namespace_map, "");
281 if (ns && *ns)
282 return ns;
285 current_node = current_node->parent;
288 return ns;
291 void purple_xmlnode_set_prefix(PurpleXmlNode *node, const char *prefix)
293 g_return_if_fail(node != NULL);
295 g_free(node->prefix);
296 node->prefix = g_strdup(prefix);
299 const char *purple_xmlnode_get_prefix(const PurpleXmlNode *node)
301 g_return_val_if_fail(node != NULL, NULL);
302 return node->prefix;
305 const char *purple_xmlnode_get_prefix_namespace(const PurpleXmlNode *node, const char *prefix)
307 const PurpleXmlNode *current_node;
309 g_return_val_if_fail(node != NULL, NULL);
310 g_return_val_if_fail(prefix != NULL, purple_xmlnode_get_default_namespace(node));
312 current_node = node;
313 while (current_node) {
314 if (current_node->prefix && g_str_equal(prefix, current_node->prefix) &&
315 current_node->xmlns) {
316 return current_node->xmlns;
317 } else if (current_node->namespace_map) {
318 const char *ns = g_hash_table_lookup(current_node->namespace_map, prefix);
319 if (ns && *ns) {
320 return ns;
324 current_node = current_node->parent;
327 return NULL;
330 void purple_xmlnode_strip_prefixes(PurpleXmlNode *node)
332 PurpleXmlNode *child;
333 const char *prefix;
335 g_return_if_fail(node != NULL);
337 for (child = node->child; child; child = child->next) {
338 if (child->type == PURPLE_XMLNODE_TYPE_TAG)
339 purple_xmlnode_strip_prefixes(child);
342 prefix = purple_xmlnode_get_prefix(node);
343 if (prefix) {
344 const char *ns = purple_xmlnode_get_prefix_namespace(node, prefix);
345 purple_xmlnode_set_namespace(node, ns);
346 purple_xmlnode_set_prefix(node, NULL);
347 } else {
348 purple_xmlnode_set_namespace(node, purple_xmlnode_get_default_namespace(node));
352 PurpleXmlNode *purple_xmlnode_get_parent(const PurpleXmlNode *child)
354 g_return_val_if_fail(child != NULL, NULL);
355 return child->parent;
358 void
359 purple_xmlnode_free(PurpleXmlNode *node)
361 PurpleXmlNode *x, *y;
363 g_return_if_fail(node != NULL);
365 /* if we're part of a tree, remove ourselves from the tree first */
366 if(NULL != node->parent) {
367 if(node->parent->child == node) {
368 node->parent->child = node->next;
369 if (node->parent->lastchild == node)
370 node->parent->lastchild = node->next;
371 } else {
372 PurpleXmlNode *prev = node->parent->child;
373 while(prev && prev->next != node) {
374 prev = prev->next;
376 if(prev) {
377 prev->next = node->next;
378 if (node->parent->lastchild == node)
379 node->parent->lastchild = prev;
384 /* now free our children */
385 x = node->child;
386 while(x) {
387 y = x->next;
388 purple_xmlnode_free(x);
389 x = y;
392 /* now dispose of ourselves */
393 g_free(node->name);
394 g_free(node->data);
395 g_free(node->xmlns);
396 g_free(node->prefix);
398 if(node->namespace_map)
399 g_hash_table_destroy(node->namespace_map);
401 g_free(node);
404 PurpleXmlNode*
405 purple_xmlnode_get_child(const PurpleXmlNode *parent, const char *name)
407 return purple_xmlnode_get_child_with_namespace(parent, name, NULL);
410 PurpleXmlNode *
411 purple_xmlnode_get_child_with_namespace(const PurpleXmlNode *parent, const char *name, const char *ns)
413 PurpleXmlNode *x, *ret = NULL;
414 char **names;
415 char *parent_name, *child_name;
417 g_return_val_if_fail(parent != NULL, NULL);
418 g_return_val_if_fail(name != NULL, NULL);
420 names = g_strsplit(name, "/", 2);
421 parent_name = names[0];
422 child_name = names[1];
424 for(x = parent->child; x; x = x->next) {
425 /* XXX: Is it correct to ignore the namespace for the match if none was specified? */
426 const char *xmlns = NULL;
427 if(ns)
428 xmlns = purple_xmlnode_get_namespace(x);
430 if(x->type == PURPLE_XMLNODE_TYPE_TAG && purple_strequal(parent_name, x->name)
431 && purple_strequal(ns, xmlns)) {
432 ret = x;
433 break;
437 if(child_name && ret)
438 ret = purple_xmlnode_get_child(ret, child_name);
440 g_strfreev(names);
441 return ret;
444 char *
445 purple_xmlnode_get_data(const PurpleXmlNode *node)
447 GString *str = NULL;
448 PurpleXmlNode *c;
450 g_return_val_if_fail(node != NULL, NULL);
452 for(c = node->child; c; c = c->next) {
453 if(c->type == PURPLE_XMLNODE_TYPE_DATA) {
454 if(!str)
455 str = g_string_new_len(c->data, c->data_sz);
456 else
457 str = g_string_append_len(str, c->data, c->data_sz);
461 if (str == NULL)
462 return NULL;
464 return g_string_free(str, FALSE);
467 char *
468 purple_xmlnode_get_data_unescaped(const PurpleXmlNode *node)
470 char *escaped = purple_xmlnode_get_data(node);
472 char *unescaped = escaped ? purple_unescape_html(escaped) : NULL;
474 g_free(escaped);
476 return unescaped;
479 static void
480 purple_xmlnode_to_str_foreach_append_ns(const char *key, const char *value,
481 GString *buf)
483 if (*key) {
484 g_string_append_printf(buf, " xmlns:%s='%s'", key, value);
485 } else {
486 g_string_append_printf(buf, " xmlns='%s'", value);
490 static char *
491 purple_xmlnode_to_str_helper(const PurpleXmlNode *node, int *len, gboolean formatting, int depth)
493 GString *text = g_string_new("");
494 const char *prefix;
495 const PurpleXmlNode *c;
496 char *node_name, *esc, *esc2, *tab = NULL;
497 gboolean need_end = FALSE, pretty = formatting;
499 g_return_val_if_fail(node != NULL, NULL);
501 if(pretty && depth) {
502 tab = g_strnfill(depth, '\t');
503 text = g_string_append(text, tab);
506 node_name = g_markup_escape_text(node->name, -1);
507 prefix = purple_xmlnode_get_prefix(node);
509 if (prefix) {
510 g_string_append_printf(text, "<%s:%s", prefix, node_name);
511 } else {
512 g_string_append_printf(text, "<%s", node_name);
515 if (node->namespace_map) {
516 g_hash_table_foreach(node->namespace_map,
517 (GHFunc)purple_xmlnode_to_str_foreach_append_ns, text);
518 } else {
519 /* Figure out if this node has a different default namespace from parent */
520 const char *xmlns = NULL;
521 const char *parent_xmlns = NULL;
522 if (!prefix)
523 xmlns = node->xmlns;
525 if (!xmlns)
526 xmlns = purple_xmlnode_get_default_namespace(node);
527 if (node->parent)
528 parent_xmlns = purple_xmlnode_get_default_namespace(node->parent);
529 if (!purple_strequal(xmlns, parent_xmlns))
531 char *escaped_xmlns = g_markup_escape_text(xmlns, -1);
532 g_string_append_printf(text, " xmlns='%s'", escaped_xmlns);
533 g_free(escaped_xmlns);
536 for(c = node->child; c; c = c->next)
538 if(c->type == PURPLE_XMLNODE_TYPE_ATTRIB) {
539 const char *aprefix = purple_xmlnode_get_prefix(c);
540 esc = g_markup_escape_text(c->name, -1);
541 esc2 = g_markup_escape_text(c->data, -1);
542 if (aprefix) {
543 g_string_append_printf(text, " %s:%s='%s'", aprefix, esc, esc2);
544 } else {
545 g_string_append_printf(text, " %s='%s'", esc, esc2);
547 g_free(esc);
548 g_free(esc2);
549 } else if(c->type == PURPLE_XMLNODE_TYPE_TAG || c->type == PURPLE_XMLNODE_TYPE_DATA) {
550 if(c->type == PURPLE_XMLNODE_TYPE_DATA)
551 pretty = FALSE;
552 need_end = TRUE;
556 if(need_end) {
557 g_string_append_printf(text, ">%s", pretty ? NEWLINE_S : "");
559 for(c = node->child; c; c = c->next)
561 if(c->type == PURPLE_XMLNODE_TYPE_TAG) {
562 int esc_len;
563 esc = purple_xmlnode_to_str_helper(c, &esc_len, pretty, depth+1);
564 text = g_string_append_len(text, esc, esc_len);
565 g_free(esc);
566 } else if(c->type == PURPLE_XMLNODE_TYPE_DATA && c->data_sz > 0) {
567 esc = g_markup_escape_text(c->data, c->data_sz);
568 text = g_string_append(text, esc);
569 g_free(esc);
573 if(tab && pretty)
574 text = g_string_append(text, tab);
575 if (prefix) {
576 g_string_append_printf(text, "</%s:%s>%s", prefix, node_name, formatting ? NEWLINE_S : "");
577 } else {
578 g_string_append_printf(text, "</%s>%s", node_name, formatting ? NEWLINE_S : "");
580 } else {
581 g_string_append_printf(text, "/>%s", formatting ? NEWLINE_S : "");
584 g_free(node_name);
586 g_free(tab);
588 if(len)
589 *len = text->len;
591 return g_string_free(text, FALSE);
594 char *
595 purple_xmlnode_to_str(const PurpleXmlNode *node, int *len)
597 return purple_xmlnode_to_str_helper(node, len, FALSE, 0);
600 char *
601 purple_xmlnode_to_formatted_str(const PurpleXmlNode *node, int *len)
603 char *xml, *xml_with_declaration;
605 g_return_val_if_fail(node != NULL, NULL);
607 xml = purple_xmlnode_to_str_helper(node, len, TRUE, 0);
608 xml_with_declaration =
609 g_strdup_printf("<?xml version='1.0' encoding='UTF-8' ?>" NEWLINE_S NEWLINE_S "%s", xml);
610 g_free(xml);
612 if (len)
613 *len += sizeof("<?xml version='1.0' encoding='UTF-8' ?>" NEWLINE_S NEWLINE_S) - 1;
615 return xml_with_declaration;
618 struct _xmlnode_parser_data {
619 PurpleXmlNode *current;
620 gboolean error;
623 static void
624 purple_xmlnode_parser_element_start_libxml(void *user_data,
625 const xmlChar *element_name, const xmlChar *prefix, const xmlChar *xmlns,
626 int nb_namespaces, const xmlChar **namespaces,
627 int nb_attributes, int nb_defaulted, const xmlChar **attributes)
629 struct _xmlnode_parser_data *xpd = user_data;
630 PurpleXmlNode *node;
631 int i, j;
633 if(!element_name || xpd->error) {
634 return;
635 } else {
636 if(xpd->current)
637 node = purple_xmlnode_new_child(xpd->current, (const char*) element_name);
638 else
639 node = purple_xmlnode_new((const char *) element_name);
641 purple_xmlnode_set_namespace(node, (const char *) xmlns);
642 purple_xmlnode_set_prefix(node, (const char *)prefix);
644 if (nb_namespaces != 0) {
645 node->namespace_map = g_hash_table_new_full(
646 g_str_hash, g_str_equal, g_free, g_free);
648 for (i = 0, j = 0; i < nb_namespaces; i++, j += 2) {
649 const char *key = (const char *)namespaces[j];
650 const char *val = (const char *)namespaces[j + 1];
651 g_hash_table_insert(node->namespace_map,
652 g_strdup(key ? key : ""), g_strdup(val ? val : ""));
656 for(i=0; i < nb_attributes * 5; i+=5) {
657 const char *name = (const char *)attributes[i];
658 const char *prefix = (const char *)attributes[i+1];
659 char *txt;
660 int attrib_len = attributes[i+4] - attributes[i+3];
661 char *attrib = g_strndup((const char *)attributes[i+3], attrib_len);
662 txt = attrib;
663 attrib = purple_unescape_text(txt);
664 g_free(txt);
665 purple_xmlnode_set_attrib_full(node, name, NULL, prefix, attrib);
666 g_free(attrib);
669 xpd->current = node;
673 static void
674 purple_xmlnode_parser_element_end_libxml(void *user_data, const xmlChar *element_name,
675 const xmlChar *prefix, const xmlChar *xmlns)
677 struct _xmlnode_parser_data *xpd = user_data;
679 if(!element_name || !xpd->current || xpd->error)
680 return;
682 if(xpd->current->parent) {
683 if(!xmlStrcmp((xmlChar*) xpd->current->name, element_name))
684 xpd->current = xpd->current->parent;
688 static void
689 purple_xmlnode_parser_element_text_libxml(void *user_data, const xmlChar *text, int text_len)
691 struct _xmlnode_parser_data *xpd = user_data;
693 if(!xpd->current || xpd->error)
694 return;
696 if(!text || !text_len)
697 return;
699 purple_xmlnode_insert_data(xpd->current, (const char*) text, text_len);
702 static void
703 purple_xmlnode_parser_error_libxml(void *user_data, const char *msg, ...)
705 struct _xmlnode_parser_data *xpd = user_data;
706 char errmsg[2048];
707 va_list args;
709 xpd->error = TRUE;
711 va_start(args, msg);
712 vsnprintf(errmsg, sizeof(errmsg), msg, args);
713 va_end(args);
715 purple_debug_error("xmlnode", "Error parsing xml file: %s", errmsg);
718 static void
719 purple_xmlnode_parser_structural_error_libxml(void *user_data, xmlErrorPtr error)
721 struct _xmlnode_parser_data *xpd = user_data;
723 if (error && (error->level == XML_ERR_ERROR ||
724 error->level == XML_ERR_FATAL)) {
725 xpd->error = TRUE;
726 purple_debug_error("xmlnode", "XML parser error for PurpleXmlNode %p: "
727 "Domain %i, code %i, level %i: %s",
728 user_data, error->domain, error->code, error->level,
729 error->message ? error->message : "(null)\n");
730 } else if (error)
731 purple_debug_warning("xmlnode", "XML parser error for PurpleXmlNode %p: "
732 "Domain %i, code %i, level %i: %s",
733 user_data, error->domain, error->code, error->level,
734 error->message ? error->message : "(null)\n");
735 else
736 purple_debug_warning("xmlnode", "XML parser error for PurpleXmlNode %p\n",
737 user_data);
740 static xmlSAXHandler purple_xmlnode_parser_libxml = {
741 NULL, /* internalSubset */
742 NULL, /* isStandalone */
743 NULL, /* hasInternalSubset */
744 NULL, /* hasExternalSubset */
745 NULL, /* resolveEntity */
746 NULL, /* getEntity */
747 NULL, /* entityDecl */
748 NULL, /* notationDecl */
749 NULL, /* attributeDecl */
750 NULL, /* elementDecl */
751 NULL, /* unparsedEntityDecl */
752 NULL, /* setDocumentLocator */
753 NULL, /* startDocument */
754 NULL, /* endDocument */
755 NULL, /* startElement */
756 NULL, /* endElement */
757 NULL, /* reference */
758 purple_xmlnode_parser_element_text_libxml, /* characters */
759 NULL, /* ignorableWhitespace */
760 NULL, /* processingInstruction */
761 NULL, /* comment */
762 NULL, /* warning */
763 purple_xmlnode_parser_error_libxml, /* error */
764 NULL, /* fatalError */
765 NULL, /* getParameterEntity */
766 NULL, /* cdataBlock */
767 NULL, /* externalSubset */
768 XML_SAX2_MAGIC, /* initialized */
769 NULL, /* _private */
770 purple_xmlnode_parser_element_start_libxml, /* startElementNs */
771 purple_xmlnode_parser_element_end_libxml, /* endElementNs */
772 purple_xmlnode_parser_structural_error_libxml, /* serror */
775 PurpleXmlNode *
776 purple_xmlnode_from_str(const char *str, gssize size)
778 struct _xmlnode_parser_data *xpd;
779 PurpleXmlNode *ret;
780 gsize real_size;
782 g_return_val_if_fail(str != NULL, NULL);
784 real_size = size < 0 ? strlen(str) : (gsize)size;
785 xpd = g_new0(struct _xmlnode_parser_data, 1);
787 if (xmlSAXUserParseMemory(&purple_xmlnode_parser_libxml, xpd, str, real_size) < 0) {
788 while(xpd->current && xpd->current->parent)
789 xpd->current = xpd->current->parent;
790 if(xpd->current)
791 purple_xmlnode_free(xpd->current);
792 xpd->current = NULL;
794 ret = xpd->current;
795 if (xpd->error) {
796 ret = NULL;
797 if (xpd->current)
798 purple_xmlnode_free(xpd->current);
801 g_free(xpd);
802 return ret;
805 PurpleXmlNode *
806 purple_xmlnode_from_file(const char *dir, const char *filename, const char *description, const char *process)
808 gchar *filename_full;
809 GError *error = NULL;
810 gchar *contents = NULL;
811 gsize length;
812 PurpleXmlNode *node = NULL;
814 g_return_val_if_fail(dir != NULL, NULL);
816 purple_debug_misc(process, "Reading file %s from directory %s\n",
817 filename, dir);
819 filename_full = g_build_filename(dir, filename, NULL);
821 if (!g_file_test(filename_full, G_FILE_TEST_EXISTS))
823 purple_debug_info(process, "File %s does not exist (this is not "
824 "necessarily an error)\n", filename_full);
825 g_free(filename_full);
826 return NULL;
829 if (!g_file_get_contents(filename_full, &contents, &length, &error))
831 purple_debug_error(process, "Error reading file %s: %s\n",
832 filename_full, error->message);
833 g_error_free(error);
836 if ((contents != NULL) && (length > 0))
838 node = purple_xmlnode_from_str(contents, length);
840 /* If we were unable to parse the file then save its contents to a backup file */
841 if (node == NULL)
843 gchar *filename_temp, *filename_temp_full;
845 filename_temp = g_strdup_printf("%s~", filename);
846 filename_temp_full = g_build_filename(dir, filename_temp, NULL);
848 purple_debug_error("util", "Error parsing file %s. Renaming old "
849 "file to %s\n", filename_full, filename_temp);
850 purple_util_write_data_to_file_absolute(filename_temp_full, contents, length);
852 g_free(filename_temp_full);
853 g_free(filename_temp);
856 g_free(contents);
859 /* If we could not parse the file then show the user an error message */
860 if (node == NULL)
862 gchar *title, *msg;
863 title = g_strdup_printf(_("Error Reading %s"), filename);
864 msg = g_strdup_printf(_("An error was encountered reading your "
865 "%s. The file has not been loaded, and the old file "
866 "has been renamed to %s~."), description, filename_full);
867 purple_notify_error(NULL, NULL, title, msg, NULL);
868 g_free(title);
869 g_free(msg);
872 g_free(filename_full);
874 return node;
877 static void
878 purple_xmlnode_copy_foreach_ns(gpointer key, gpointer value, gpointer user_data)
880 GHashTable *ret = (GHashTable *)user_data;
881 g_hash_table_insert(ret, g_strdup(key), g_strdup(value));
884 PurpleXmlNode *
885 purple_xmlnode_copy(const PurpleXmlNode *src)
887 PurpleXmlNode *ret;
888 PurpleXmlNode *child;
889 PurpleXmlNode *sibling = NULL;
891 g_return_val_if_fail(src != NULL, NULL);
893 ret = new_node(src->name, src->type);
894 ret->xmlns = g_strdup(src->xmlns);
895 if (src->data) {
896 if (src->data_sz) {
897 ret->data = g_memdup(src->data, src->data_sz);
898 ret->data_sz = src->data_sz;
899 } else {
900 ret->data = g_strdup(src->data);
903 ret->prefix = g_strdup(src->prefix);
904 if (src->namespace_map) {
905 ret->namespace_map = g_hash_table_new_full(g_str_hash, g_str_equal,
906 g_free, g_free);
907 g_hash_table_foreach(src->namespace_map, purple_xmlnode_copy_foreach_ns, ret->namespace_map);
910 for (child = src->child; child; child = child->next) {
911 if (sibling) {
912 sibling->next = purple_xmlnode_copy(child);
913 sibling = sibling->next;
914 } else {
915 ret->child = sibling = purple_xmlnode_copy(child);
917 sibling->parent = ret;
920 ret->lastchild = sibling;
922 return ret;
925 PurpleXmlNode *
926 purple_xmlnode_get_next_twin(PurpleXmlNode *node)
928 PurpleXmlNode *sibling;
929 const char *ns = purple_xmlnode_get_namespace(node);
931 g_return_val_if_fail(node != NULL, NULL);
932 g_return_val_if_fail(node->type == PURPLE_XMLNODE_TYPE_TAG, NULL);
934 for(sibling = node->next; sibling; sibling = sibling->next) {
935 /* XXX: Is it correct to ignore the namespace for the match if none was specified? */
936 const char *xmlns = NULL;
937 if(ns)
938 xmlns = purple_xmlnode_get_namespace(sibling);
940 if(sibling->type == PURPLE_XMLNODE_TYPE_TAG && purple_strequal(node->name, sibling->name) &&
941 purple_strequal(ns, xmlns))
942 return sibling;
945 return NULL;
948 GType
949 purple_xmlnode_get_type(void)
951 static GType type = 0;
953 if (type == 0) {
954 type = g_boxed_type_register_static("PurpleXmlNode",
955 (GBoxedCopyFunc)purple_xmlnode_copy,
956 (GBoxedFreeFunc)purple_xmlnode_free);
959 return type;