Skip to content
Snippets Groups Projects
xml2json.c 30.5 KiB
Newer Older
/* Icecast
 *
 * This program is distributed under the GNU General Public License, version 2.
 * A copy of this license is included with this source.
 *
 * Copyright 2018-2020, Philipp "ph3-der-loewe" Schafft <lion@lion.leolix.org>,
 */

/**
 * This file contains functions for rendering XML as JSON.
 */

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include "xml2json.h"
#include "json.h"
/* For XMLSTR() */
#include "cfgfile.h"

#include "logging.h"
#define CATMODULE "xml2json"
    const char *default_namespace;
    xmlNsPtr ns;
    void (*render)(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache);
};

struct nodelist {
    xmlNodePtr *nodes;
    size_t len;
    size_t fill;
};

static void render_node(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache);
static void render_node_generic(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache);

static void nodelist_init(struct nodelist *list)
{
    memset(list, 0, sizeof(*list));
}

static void nodelist_free(struct nodelist *list)
{
    free(list->nodes);
    memset(list, 0, sizeof(*list));
}

static void nodelist_push(struct nodelist *list, xmlNodePtr node)
{
    if (list->fill == list->len) {
        xmlNodePtr *n = realloc(list->nodes, sizeof(xmlNodePtr)*(list->len + 16));
        if (!n) {
            ICECAST_LOG_ERROR("Can not allocate memory for node list. BAD.");
            return;
        }

        list->nodes = n;
        list->len += 16;
    }

    list->nodes[list->fill++] = node;
}

static xmlNodePtr nodelist_get(struct nodelist *list, size_t idx)
{
    if (idx >= list->fill)
        return NULL;
    return list->nodes[idx];
}

static void nodelist_unset(struct nodelist *list, size_t idx)
{
    if (idx >= list->fill)
        return;
    list->nodes[idx] = NULL;
}

static size_t nodelist_fill(struct nodelist *list)
{
    return list->fill;
}

static int nodelist_is_empty(struct nodelist *list)
{
    size_t i;

    for (i = 0; i < list->fill; i++)
        if (list->nodes[i])
            return 0;

    return 1;
}

static int has_ns_changed(xmlNodePtr node, xmlNodePtr parent)
{
    xmlChar *xmlns;

    if (parent == NULL)
        return 1;

    if (node->ns != parent->ns) {
        if (node->ns && parent->ns && node->ns->href && parent->ns->href) {
            if (strcmp((const char *)node->ns->href, (const char *)parent->ns->href) != 0)
                return 1;
        } else {
            return 1;
        }
    }

    xmlns = xmlGetProp(node, XMLSTR("xmlns"));
    if (xmlns) {
        xmlFree(xmlns);
        return 1;
    }

    return 0;
}

static void handle_node_identification(json_renderer_t *renderer, const char *name, const char *ns, const char *id, const char *definition, const char *akindof)
{
            json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);
#define handle_node_identification__single(name) \
            if ((name)) { \
                json_renderer_write_key(renderer, ( # name ), JSON_RENDERER_FLAGS_NONE); \
                json_renderer_write_string(renderer, (name), JSON_RENDERER_FLAGS_NONE); \

            handle_node_identification__single(name)
            handle_node_identification__single(ns)
            handle_node_identification__single(id)
            handle_node_identification__single(definition)
            handle_node_identification__single(akindof)

static void handle_textchildnode(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache)
{
    xmlChar *value = xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
    if (value) {
        json_renderer_write_key(renderer, (const char *)node->name, JSON_RENDERER_FLAGS_NONE);
        json_renderer_write_string(renderer, (const char*)value, JSON_RENDERER_FLAGS_NONE);
        xmlFree(value);
    }
}

static void handle_booleanchildnode(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache)
{
    xmlChar *value = xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
    if (value) {
        json_renderer_write_key(renderer, (const char *)node->name, JSON_RENDERER_FLAGS_NONE);
        json_renderer_write_boolean(renderer, util_str_to_bool((const char*)value));
        xmlFree(value);
    }
}

static int handle_node_modules(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache)
{
    if (node->type == XML_ELEMENT_NODE && strcmp((const char *)node->name, "modules") == 0) {
        json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);
        if (node->xmlChildrenNode) {
            xmlNodePtr cur = node->xmlChildrenNode;

            do {
                if (cur->type == XML_ELEMENT_NODE && cur->name && strcmp((const char *)cur->name, "module") == 0) {
                    xmlChar *name = xmlGetProp(cur, XMLSTR("name"));
                    if (name) {
                        json_renderer_write_key(renderer, (const char *)name, JSON_RENDERER_FLAGS_NONE);
                        json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);
                        if (node->properties) {
                            xmlAttrPtr prop = node->properties;

                            do {
                                xmlChar *value = xmlNodeListGetString(doc, prop->children, 1);
                                if (value) {
                                    json_renderer_write_key(renderer, (const char*)cur->name, JSON_RENDERER_FLAGS_NONE);
                                    json_renderer_write_string(renderer, (const char*)value, JSON_RENDERER_FLAGS_NONE);
                                    xmlFree(value);
                                }
                            } while ((prop = prop->next));
                        }
                        json_renderer_end(renderer);
                        xmlFree(name);
                    }
                } else {
                    json_renderer_write_key(renderer, "unhandled-child", JSON_RENDERER_FLAGS_NONE);
                    render_node(renderer, doc, cur, node, cache);
                }
                cur = cur->next;
            } while (cur);
        }
        json_renderer_end(renderer);
        return 1;
    }

    return 0;
}

static void render_node_legacyresponse(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache)
{
    int handled = 0;

    if (node->type == XML_ELEMENT_NODE) {
        const char *nodename = (const char *)node->name;
        handled = 1;
        if (strcmp(nodename, "iceresponse") == 0) {
            json_renderer_begin(renderer, JSON_ELEMENT_TYPE_ARRAY);
            handle_node_identification(renderer, "iceresponse", XMLNS_LEGACY_RESPONSE, NULL, NULL, NULL);
            json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);
            if (node->xmlChildrenNode) {
                xmlNodePtr cur = node->xmlChildrenNode;

                do {
                    int handled_child = 1;

                    if (cur->type == XML_ELEMENT_NODE && cur->name) {
                        if (strcmp((const char *)cur->name, "message") == 0) {
                            handle_textchildnode(renderer, doc, cur, node, cache);
                        } else if (strcmp((const char *)cur->name, "return") == 0) {
                            xmlChar *value = xmlNodeListGetString(doc, cur->xmlChildrenNode, 1);
                            if (value) {
                                json_renderer_write_key(renderer, "success", JSON_RENDERER_FLAGS_NONE);
                                json_renderer_write_boolean(renderer, strcmp((const char *)value, "1") == 0);
                                xmlFree(value);
                            }
                        } else if (strcmp((const char *)cur->name, "modules") == 0) {
                            json_renderer_write_key(renderer, "modules", JSON_RENDERER_FLAGS_NONE);
                            render_node(renderer, doc, cur, node, cache);
                        } else {
                            handled_child = 0;
                        }
                    } else {
                        handled_child = 0;
                    }

                    if (!handled_child) {
                        json_renderer_write_key(renderer, "unhandled-child", JSON_RENDERER_FLAGS_NONE);
                        render_node(renderer, doc, cur, node, cache);
                    }
                    cur = cur->next;
                } while (cur);
            }
            json_renderer_end(renderer);
        } else if (strcmp(nodename, "modules") == 0) {
            handled = handle_node_modules(renderer, doc, node, parent, cache);
        } else {
            handled = 0;
        }
    }

    if (!handled)
        render_node_generic(renderer, doc, node, parent, cache);
}
static int handle_simple_child(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache, xmlNodePtr child, const char * number_keys[], const char * boolean_keys[])
{
    if (child->type == XML_ELEMENT_NODE && child->name) {
        const char *childname = (const char *)child->name;
        size_t i;

        for (i = 0; number_keys[i]; i++) {
            if (strcmp(childname, number_keys[i]) == 0) {
                xmlChar *value = xmlNodeListGetString(doc, child->xmlChildrenNode, 1);
                if (value) {
                    json_renderer_write_key(renderer, childname, JSON_RENDERER_FLAGS_NONE);
                    json_renderer_write_int(renderer, strtoll((const char*)value, NULL, 10));
                    xmlFree(value);
                    return 1;
                }
            }
        }

        for (i = 0; boolean_keys[i]; i++) {
            if (strcmp(childname, boolean_keys[i]) == 0) {
                handle_booleanchildnode(renderer, doc, child, node, cache);
                return 1;
            }
        }

        if (strcmp(childname, "max_listeners") == 0) {
            xmlChar *value = xmlNodeListGetString(doc, child->xmlChildrenNode, 1);
            if (value) {
                json_renderer_write_key(renderer, childname, JSON_RENDERER_FLAGS_NONE);
                if (strcmp((const char *)value, "unlimited") == 0) {
                    json_renderer_write_null(renderer);
                } else {
                    json_renderer_write_int(renderer, strtoll((const char*)value, NULL, 10));
                }
                xmlFree(value);
                return 1;
            }
        }

        if (strcmp(childname, "authenticator") == 0) {
            xmlChar *value = xmlNodeListGetString(doc, child->xmlChildrenNode, 1);
            if (value) {
                json_renderer_write_key(renderer, childname, JSON_RENDERER_FLAGS_NONE);
                json_renderer_write_boolean(renderer, strlen((const char *)value));
                xmlFree(value);
                return 1;
            }
        }

        if (child->xmlChildrenNode && !child->xmlChildrenNode->next && child->xmlChildrenNode->type == XML_TEXT_NODE) {
            handle_textchildnode(renderer, doc, child, node, cache);
            return 1;
        }
    }

    return 0;
}

static void render_node_legacystats(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache)
{
    static const char * number_keys_global[] = {
        "listeners", "clients", "client_connections", "connections", "file_connections", "listener_connections",
        "source_client_connections", "source_relay_connections", "source_total_connections", "sources", "stats", "stats_connections", NULL
    };
    static const char * boolean_keys_global[] = {
        NULL
    };
    static const char * number_keys_source[] = {
        "audio_bitrate", "audio_channels", "audio_samplerate", "ice-bitrate", "listener_peak", "listeners", "slow_listeners",
        "total_bytes_read", "total_bytes_sent", NULL
    };
    static const char * boolean_keys_source[] = {
        "public", NULL
    };

    int handled = 0;

    if (node->type == XML_ELEMENT_NODE) {
        const char *nodename = (const char *)node->name;
        handled = 1;
        if (strcmp(nodename, "icestats") == 0 || strcmp(nodename, "source") == 0) {
            int is_icestats = strcmp(nodename, "icestats") == 0;
            struct nodelist nodelist;
            size_t i;
            size_t len;

            nodelist_init(&nodelist);

            if (is_icestats) {
                json_renderer_begin(renderer, JSON_ELEMENT_TYPE_ARRAY);
                handle_node_identification(renderer, "icestats", XMLNS_LEGACY_STATS, NULL, NULL, NULL);
            json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);
            if (node->xmlChildrenNode) {
                xmlNodePtr cur = node->xmlChildrenNode;
                do {
                    if (!handle_simple_child(renderer, doc, node, parent, cache, cur,
                                is_icestats ? number_keys_global : number_keys_source,
                                is_icestats ? boolean_keys_global : boolean_keys_source
                                )) {
                        nodelist_push(&nodelist, cur);
                    }
                    cur = cur->next;
                } while (cur);
            }

            len = nodelist_fill(&nodelist);
            for (i = 0; i < len; i++) {
                xmlNodePtr cur = nodelist_get(&nodelist, i);
                if (cur == NULL)
                    continue;

                if (cur->type == XML_ELEMENT_NODE && cur->name) {
                    if (strcmp((const char *)cur->name, "modules") == 0) {
                        json_renderer_write_key(renderer, (const char *)cur->name, JSON_RENDERER_FLAGS_NONE);
                        handle_node_modules(renderer, doc, cur, node, cache);
                        nodelist_unset(&nodelist, i);
                    } else if (strcmp((const char *)cur->name, "source") == 0) {
                        size_t j;

                        json_renderer_write_key(renderer, (const char *)cur->name, JSON_RENDERER_FLAGS_NONE);
                        json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);

                        for (j = i; j < len; j++) {
                            xmlNodePtr subcur = nodelist_get(&nodelist, j);
                            if (subcur == NULL)
                                continue;

                            if (subcur->type == XML_ELEMENT_NODE && subcur->name && strcmp((const char *)cur->name, (const char *)subcur->name) == 0) {
                                xmlChar *mount = xmlGetProp(subcur, XMLSTR("mount"));
                                if (mount) {
                                    json_renderer_write_key(renderer, (const char *)mount, JSON_RENDERER_FLAGS_NONE);
                                    xmlFree(mount);
                                    nodelist_unset(&nodelist, j);
                                    render_node_legacystats(renderer, doc, subcur, cur, cache);
                                }
                            }
                        }

                        json_renderer_end(renderer);
                        nodelist_unset(&nodelist, i);
                    } else if (strcmp((const char *)cur->name, "metadata") == 0) {
                        size_t j;

                        json_renderer_write_key(renderer, (const char *)cur->name, JSON_RENDERER_FLAGS_NONE);
                        json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);
                        for (j = i; j < len; j++) {
                            xmlNodePtr subcur = nodelist_get(&nodelist, j);
                            if (subcur == NULL)
                                continue;

                            if (subcur->type == XML_ELEMENT_NODE && subcur->name && strcmp((const char *)cur->name, (const char *)subcur->name) == 0) {
                                xmlNodePtr child = subcur->xmlChildrenNode;
                                while (child) {
                                    handle_textchildnode(renderer, doc, child, subcur, cache);
                                    child = child->next;
                                }
                                nodelist_unset(&nodelist, j);
                            }
                        }
                        json_renderer_end(renderer);
                    } else if (strcmp((const char *)cur->name, "authentication") == 0) {
                        size_t j;

                        json_renderer_write_key(renderer, (const char *)cur->name, JSON_RENDERER_FLAGS_NONE);
                        json_renderer_begin(renderer, JSON_ELEMENT_TYPE_ARRAY);
                        for (j = i; j < len; j++) {
                            xmlNodePtr subcur = nodelist_get(&nodelist, j);
                            if (subcur == NULL)
                                continue;

                            if (subcur->type == XML_ELEMENT_NODE && subcur->name && strcmp((const char *)cur->name, (const char *)subcur->name) == 0) {
                                xmlNodePtr child = subcur->xmlChildrenNode;
                                while (child) {
                                    render_node_legacystats(renderer, doc, child, subcur, cache);
                                    child = child->next;
                                }
                                nodelist_unset(&nodelist, j);
                            }
                        }
                        json_renderer_end(renderer);
                    } else if (strcmp((const char *)cur->name, "playlist") == 0) {
                        json_renderer_write_key(renderer, (const char *)cur->name, JSON_RENDERER_FLAGS_NONE);
                        render_node(renderer, doc, cur, node, cache);
                        nodelist_unset(&nodelist, i);
                    }
                }
                //render_node_generic(renderer, doc, node, parent, cache);
            }

            if (!nodelist_is_empty(&nodelist)) {
                json_renderer_write_key(renderer, "unhandled-child", JSON_RENDERER_FLAGS_NONE);
                json_renderer_begin(renderer, JSON_ELEMENT_TYPE_ARRAY);
                len = nodelist_fill(&nodelist);
                for (i = 0; i < len; i++) {
                    xmlNodePtr cur = nodelist_get(&nodelist, i);
                    if (cur == NULL)
                        continue;

                    render_node(renderer, doc, cur, node, cache);
                }
                json_renderer_end(renderer);
            }

            json_renderer_end(renderer);
            if (is_icestats)
                json_renderer_end(renderer);

            nodelist_free(&nodelist);
        } else if (strcmp(nodename, "role") == 0) {
            json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);
            if (node->properties) {
                xmlAttrPtr cur = node->properties;

                do {
                    const char *name = (const char*)cur->name;
                    xmlChar *value = xmlNodeListGetString(doc, cur->children, 1);

                    if (value) {
                        json_renderer_write_key(renderer, name, JSON_RENDERER_FLAGS_NONE);
                        if (strncmp(name, "can-", 4) == 0) {
                            json_renderer_write_boolean(renderer, util_str_to_bool((const char *)value));
                        } else {
                            json_renderer_write_string(renderer, (const char*)value, JSON_RENDERER_FLAGS_NONE);
                        }
                        xmlFree(value);
                    }
                } while ((cur = cur->next));
            }
            json_renderer_end(renderer);
        } else {
            handled = 0;
        }
    }

    if (!handled)
        render_node_generic(renderer, doc, node, parent, cache);
}

static void render_node_xspf(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache)
{
    const char * text_keys[] = {"title", "creator", "annotation", "info", "identifier", "image", "date", "license", "album", NULL};
    const char * uint_keys[] = {"trackNum", "duration", NULL};
    int handled = 0;

    if (node->type == XML_ELEMENT_NODE) {
        const char *nodename = (const char *)node->name;
        int handle_childs = 0;
        int close_after_me = 0;
        size_t i;

        handled = 1;

        for (i = 0; text_keys[i]; i++) {
            if (strcmp(nodename, text_keys[i]) == 0) {
                handle_textchildnode(renderer, doc, node, parent, cache);
                return;
            }
        }

        for (i = 0; uint_keys[i]; i++) {
            if (strcmp(nodename, uint_keys[i]) == 0) {
                xmlChar *value = xmlNodeListGetString(doc, node->xmlChildrenNode, 1);
                if (value) {
                    long long int val = strtoll((const char*)value, NULL, 10);
                    if (val < 0)
                        return;

                    json_renderer_write_key(renderer, nodename, JSON_RENDERER_FLAGS_NONE);
                    json_renderer_write_uint(renderer, val);
                    xmlFree(value);
                    return ;
                }
                return;
            }
        }

        for (i = 0; text_keys[i]; i++) {
            if (strcmp(nodename, text_keys[i]) == 0) {
                handle_textchildnode(renderer, doc, node, parent, cache);
                return;
            }
        }

        if (strcmp(nodename, "playlist") == 0) {
            json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);
            json_renderer_write_key(renderer, "playlist", JSON_RENDERER_FLAGS_NONE);
            json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);

            handle_childs = 1;
            close_after_me = 2;
        } else if (strcmp(nodename, "trackList") == 0) {
            json_renderer_write_key(renderer, "track", JSON_RENDERER_FLAGS_NONE);
            json_renderer_begin(renderer, JSON_ELEMENT_TYPE_ARRAY);

            handle_childs = 1;
            close_after_me = 1;
        } else if (strcmp(nodename, "track") == 0) {
            json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);

            handle_childs = 1;
            close_after_me = 1;
        } else {
            handled = 0;
        }

        if (handled) {
            if (handle_childs) {
                xmlNodePtr child = node->xmlChildrenNode;
                while (child) {
                    render_node_xspf(renderer, doc, child, node, cache);
                    child = child->next;
                };
            }

            for (; close_after_me; close_after_me--)
                json_renderer_end(renderer);
        }
    }

    if (!handled)
        render_node_generic(renderer, doc, node, parent, cache);
}

static void render_node_reportxml(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache)
{
    static const char * valid_tags[] = {
        "report", "definition", "incident", "state", "backtrace", "position", "more", "fix",
        "action", "reason", "text", "timestamp", "resource", "value", "reference", "extension", NULL
    };
    static const char * skip_props[] = {
        "id", "definition", "akindof", NULL
    };

    if (node->type == XML_ELEMENT_NODE) {
        const char *nodename = (const char *)node->name;
        size_t i;

        for (i = 0; valid_tags[i]; i++) {
            if (strcmp(nodename, valid_tags[i]) == 0) {
                xmlChar *id = xmlGetProp(node, XMLSTR("id"));
                xmlChar *definition = xmlGetProp(node, XMLSTR("definition"));
                xmlChar *akindof = xmlGetProp(node, XMLSTR("akindof"));

                json_renderer_begin(renderer, JSON_ELEMENT_TYPE_ARRAY);
                handle_node_identification(renderer, nodename, has_ns_changed(node, parent) ? XMLNS_REPORTXML : NULL, (const char *)id, (const char *)definition, (const char *)akindof);

                if (id)
                    xmlFree(id);
                if (definition)
                    xmlFree(definition);
                if (akindof)
                    xmlFree(akindof);

                if (node->properties || node->xmlChildrenNode) {
                    json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);
                    if (node->properties) {
                        xmlAttrPtr cur = node->properties;

                        do {
                            int found = 0;
                            size_t j;

                            for (j = 0; skip_props[j]; j++) {
                                if (strcmp((const char*)cur->name, skip_props[j]) == 0) {
                                    found = 1;
                                    break;
                                }
                            }

                            if (!found) {
                                xmlChar *value = xmlNodeListGetString(doc, cur->children, 1);
                                if (value) {
                                    json_renderer_write_key(renderer, (const char*)cur->name, JSON_RENDERER_FLAGS_NONE);
                                    json_renderer_write_string(renderer, (const char*)value, JSON_RENDERER_FLAGS_NONE);
                                    xmlFree(value);
                                }
                            }
                        } while ((cur = cur->next));
                    }
                    json_renderer_end(renderer);
                }

                if (node->xmlChildrenNode) {
                    json_renderer_begin(renderer, JSON_ELEMENT_TYPE_ARRAY);
                    xmlNodePtr child = node->xmlChildrenNode;
                    while (child) {
                        render_node(renderer, doc, child, node, cache);
                        child = child->next;
                    };
                    json_renderer_end(renderer);
                }

                json_renderer_end(renderer);

                return;
            }
        }
    }

    render_node_generic(renderer, doc, node, parent, cache);
}

static void render_node_generic(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache)
    json_renderer_begin(renderer, JSON_ELEMENT_TYPE_ARRAY);
    json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);

    json_renderer_write_key(renderer, "type", JSON_RENDERER_FLAGS_NONE);
    switch (node->type) {
        case XML_ELEMENT_NODE:
            json_renderer_write_string(renderer, "element", JSON_RENDERER_FLAGS_NONE);
            if (node->name) {
                json_renderer_write_key(renderer, "name", JSON_RENDERER_FLAGS_NONE);
                json_renderer_write_string(renderer, (const char *)node->name, JSON_RENDERER_FLAGS_NONE);
            }
        break;
        case XML_TEXT_NODE:
            json_renderer_write_string(renderer, "text", JSON_RENDERER_FLAGS_NONE);
        break;
        case XML_COMMENT_NODE:
            json_renderer_write_string(renderer, "comment", JSON_RENDERER_FLAGS_NONE);
        break;
        default:
            json_renderer_write_null(renderer);
        break;
    }

    if (node->content) {
        json_renderer_write_key(renderer, "text", JSON_RENDERER_FLAGS_NONE);
        json_renderer_write_string(renderer, (const char *)node->content, JSON_RENDERER_FLAGS_NONE);
    }

    json_renderer_write_key(renderer, "ns", JSON_RENDERER_FLAGS_NONE);
    if (node->ns && node->ns->href) {
        json_renderer_write_string(renderer, (const char *)node->ns->href, JSON_RENDERER_FLAGS_NONE);
        if (node->ns->prefix) {
            json_renderer_write_key(renderer, "nsprefix", JSON_RENDERER_FLAGS_NONE);
            json_renderer_write_string(renderer, (const char *)node->ns->prefix, JSON_RENDERER_FLAGS_NONE);
        }
    } else {
        json_renderer_write_null(renderer);
    }
    if (node->properties || node->xmlChildrenNode) {
        xmlAttrPtr cur = node->properties;
        json_renderer_begin(renderer, JSON_ELEMENT_TYPE_OBJECT);
            xmlChar *value = xmlNodeListGetString(doc, cur->children, 1);
            if (value) {
                json_renderer_write_key(renderer, (const char*)cur->name, JSON_RENDERER_FLAGS_NONE);
                json_renderer_write_string(renderer, (const char*)value, JSON_RENDERER_FLAGS_NONE);
                xmlFree(value);
            }
        json_renderer_end(renderer);
    }

    if (node->xmlChildrenNode) {
        xmlNodePtr cur = node->xmlChildrenNode;

        json_renderer_begin(renderer, JSON_ELEMENT_TYPE_ARRAY);
        do {
            render_node(renderer, doc, cur, node, cache);
            cur = cur->next;
        } while (cur);
        json_renderer_end(renderer);
    }

    json_renderer_end(renderer);
}


static void render_node(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache)
{
    void (*render)(json_renderer_t *renderer, xmlDocPtr doc, xmlNodePtr node, xmlNodePtr parent, struct xml2json_cache *cache) = NULL;
    xmlChar * workaroundProp = xmlGetProp(node, XMLSTR("xmlns"));
    if (node->ns == cache->ns && !workaroundProp)
        render = cache->render;

    if (render == NULL) {
        const char *href = cache->default_namespace;

        if (node->ns && node->ns->href)
            href = (const char *)node->ns->href;
            href = (const char *)workaroundProp;

            if (strcmp(href, XMLNS_LEGACY_RESPONSE) == 0) {
                render = render_node_legacyresponse;
            } else if (strcmp(href, XMLNS_LEGACY_STATS) == 0) {
                render = render_node_legacystats;
            } else if (strcmp(href, XMLNS_XSPF) == 0) {
            } else if (strcmp(href, XMLNS_REPORTXML) == 0) {
                render = render_node_reportxml;
        if (render == NULL)
            render = render_node_generic;

        cache->ns = node->ns;
        cache->render = render;
    }
    if (workaroundProp)
        xmlFree(workaroundProp);

    render(renderer, doc, node, parent, cache);
}

char * xml2json_render_doc_simple(xmlDocPtr doc, const char *default_namespace)
    json_renderer_t *renderer;
    xmlNodePtr xmlroot;

    if (!doc)
        return NULL;

    renderer = json_renderer_create(JSON_RENDERER_FLAGS_NONE);
    if (!renderer)
        return NULL;

    xmlroot = xmlDocGetRootElement(doc);
    if (!xmlroot)
        return NULL;

    cache.default_namespace = default_namespace;

    render_node(renderer, doc, xmlroot, NULL, &cache);