Blame python/types.c

Packit 423ecb
/*
Packit 423ecb
 * types.c: converter functions between the internal representation
Packit 423ecb
 *          and the Python objects
Packit 423ecb
 *
Packit 423ecb
 * See Copyright for the status of this software.
Packit 423ecb
 *
Packit 423ecb
 * daniel@veillard.com
Packit 423ecb
 */
Packit 423ecb
#include "libxml_wrap.h"
Packit 423ecb
#include <libxml/xpathInternals.h>
Packit 423ecb
Packit 423ecb
#if PY_MAJOR_VERSION >= 3
Packit 423ecb
#define PY_IMPORT_STRING_SIZE PyUnicode_FromStringAndSize
Packit 423ecb
#define PY_IMPORT_STRING PyUnicode_FromString
Packit 423ecb
#define PY_IMPORT_INT PyLong_FromLong
Packit 423ecb
#else
Packit 423ecb
#define PY_IMPORT_STRING_SIZE PyString_FromStringAndSize
Packit 423ecb
#define PY_IMPORT_STRING PyString_FromString
Packit 423ecb
#define PY_IMPORT_INT PyInt_FromLong
Packit 423ecb
#endif
Packit 423ecb
Packit 423ecb
#if PY_MAJOR_VERSION >= 3
Packit 423ecb
#include <stdio.h>
Packit 423ecb
#include <unistd.h>
Packit 423ecb
#include <fcntl.h>
Packit 423ecb
Packit 423ecb
FILE *
Packit 423ecb
libxml_PyFileGet(PyObject *f) {
Packit 423ecb
    int fd, flags;
Packit 423ecb
    FILE *res;
Packit 423ecb
    const char *mode;
Packit 423ecb
Packit 423ecb
    fd = PyObject_AsFileDescriptor(f);
Packit 423ecb
    /*
Packit 423ecb
     * Get the flags on the fd to understand how it was opened
Packit 423ecb
     */
Packit 423ecb
    flags = fcntl(fd, F_GETFL, 0);
Packit 423ecb
    switch (flags & O_ACCMODE) {
Packit 423ecb
        case O_RDWR:
Packit 423ecb
	    if (flags & O_APPEND)
Packit 423ecb
	        mode = "a+";
Packit 423ecb
	    else
Packit 423ecb
	        mode = "rw";
Packit 423ecb
	    break;
Packit 423ecb
        case O_RDONLY:
Packit 423ecb
	    if (flags & O_APPEND)
Packit 423ecb
	        mode = "r+";
Packit 423ecb
	    else
Packit 423ecb
	        mode = "r";
Packit 423ecb
	    break;
Packit 423ecb
	case O_WRONLY:
Packit 423ecb
	    if (flags & O_APPEND)
Packit 423ecb
	        mode = "a";
Packit 423ecb
	    else
Packit 423ecb
	        mode = "w";
Packit 423ecb
	    break;
Packit 423ecb
	default:
Packit 423ecb
	    return(NULL);
Packit 423ecb
    }
Packit 423ecb
Packit 423ecb
    /*
Packit 423ecb
     * the FILE struct gets a new fd, so that it can be closed
Packit 423ecb
     * independently of the file descriptor given. The risk though is
Packit 423ecb
     * lack of sync. So at the python level sync must be implemented
Packit 423ecb
     * before and after a conversion took place. No way around it
Packit 423ecb
     * in the Python3 infrastructure !
Packit 423ecb
     * The duplicated fd and FILE * will be released in the subsequent
Packit 423ecb
     * call to libxml_PyFileRelease() which must be genrated accodingly
Packit 423ecb
     */
Packit 423ecb
    fd = dup(fd);
Packit 423ecb
    if (fd == -1)
Packit 423ecb
        return(NULL);
Packit 423ecb
    res = fdopen(fd, mode);
Packit 423ecb
    if (res == NULL) {
Packit 423ecb
        close(fd);
Packit 423ecb
	return(NULL);
Packit 423ecb
    }
Packit 423ecb
    return(res);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
void libxml_PyFileRelease(FILE *f) {
Packit 423ecb
    if (f != NULL)
Packit 423ecb
        fclose(f);
Packit 423ecb
}
Packit 423ecb
#endif
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_intWrap(int val)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_intWrap: val = %d\n", val);
Packit 423ecb
#endif
Packit 423ecb
    ret = PY_IMPORT_INT((long) val);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_longWrap(long val)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_longWrap: val = %ld\n", val);
Packit 423ecb
#endif
Packit 423ecb
    ret = PyLong_FromLong(val);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_doubleWrap(double val)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_doubleWrap: val = %f\n", val);
Packit 423ecb
#endif
Packit 423ecb
    ret = PyFloat_FromDouble((double) val);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_charPtrWrap(char *str)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlcharPtrWrap: str = %s\n", str);
Packit 423ecb
#endif
Packit 423ecb
    if (str == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PY_IMPORT_STRING(str);
Packit 423ecb
    xmlFree(str);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_charPtrConstWrap(const char *str)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlcharPtrWrap: str = %s\n", str);
Packit 423ecb
#endif
Packit 423ecb
    if (str == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PY_IMPORT_STRING(str);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlCharPtrWrap(xmlChar * str)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlCharPtrWrap: str = %s\n", str);
Packit 423ecb
#endif
Packit 423ecb
    if (str == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PY_IMPORT_STRING((char *) str);
Packit 423ecb
    xmlFree(str);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlCharPtrConstWrap(const xmlChar * str)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlCharPtrWrap: str = %s\n", str);
Packit 423ecb
#endif
Packit 423ecb
    if (str == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PY_IMPORT_STRING((char *) str);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_constcharPtrWrap(const char *str)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlcharPtrWrap: str = %s\n", str);
Packit 423ecb
#endif
Packit 423ecb
    if (str == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PY_IMPORT_STRING(str);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_constxmlCharPtrWrap(const xmlChar * str)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlCharPtrWrap: str = %s\n", str);
Packit 423ecb
#endif
Packit 423ecb
    if (str == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PY_IMPORT_STRING((char *) str);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlDocPtrWrap(xmlDocPtr doc)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlDocPtrWrap: doc = %p\n", doc);
Packit 423ecb
#endif
Packit 423ecb
    if (doc == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    /* TODO: look at deallocation */
Packit 423ecb
    ret = PyCapsule_New((void *) doc, (char *) "xmlDocPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlNodePtrWrap(xmlNodePtr node)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlNodePtrWrap: node = %p\n", node);
Packit 423ecb
#endif
Packit 423ecb
    if (node == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PyCapsule_New((void *) node, (char *) "xmlNodePtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlURIPtrWrap(xmlURIPtr uri)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlURIPtrWrap: uri = %p\n", uri);
Packit 423ecb
#endif
Packit 423ecb
    if (uri == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PyCapsule_New((void *) uri, (char *) "xmlURIPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlNsPtrWrap(xmlNsPtr ns)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlNsPtrWrap: node = %p\n", ns);
Packit 423ecb
#endif
Packit 423ecb
    if (ns == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PyCapsule_New((void *) ns, (char *) "xmlNsPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlAttrPtrWrap(xmlAttrPtr attr)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlAttrNodePtrWrap: attr = %p\n", attr);
Packit 423ecb
#endif
Packit 423ecb
    if (attr == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PyCapsule_New((void *) attr, (char *) "xmlAttrPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlAttributePtrWrap(xmlAttributePtr attr)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlAttributePtrWrap: attr = %p\n", attr);
Packit 423ecb
#endif
Packit 423ecb
    if (attr == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PyCapsule_New((void *) attr, (char *) "xmlAttributePtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlElementPtrWrap(xmlElementPtr elem)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlElementNodePtrWrap: elem = %p\n", elem);
Packit 423ecb
#endif
Packit 423ecb
    if (elem == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PyCapsule_New((void *) elem, (char *) "xmlElementPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlXPathContextPtrWrap(xmlXPathContextPtr ctxt)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlXPathContextPtrWrap: ctxt = %p\n", ctxt);
Packit 423ecb
#endif
Packit 423ecb
    if (ctxt == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PyCapsule_New((void *) ctxt, (char *) "xmlXPathContextPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlXPathParserContextPtrWrap(xmlXPathParserContextPtr ctxt)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlXPathParserContextPtrWrap: ctxt = %p\n", ctxt);
Packit 423ecb
#endif
Packit 423ecb
    if (ctxt == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PyCapsule_New((void *)ctxt, (char *)"xmlXPathParserContextPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlParserCtxtPtrWrap(xmlParserCtxtPtr ctxt)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlParserCtxtPtrWrap: ctxt = %p\n", ctxt);
Packit 423ecb
#endif
Packit 423ecb
    if (ctxt == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
Packit 423ecb
    ret = PyCapsule_New((void *) ctxt, (char *) "xmlParserCtxtPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
/**
Packit 423ecb
 * libxml_xmlXPathDestructNsNode:
Packit 423ecb
 * cap: xmlNsPtr namespace node capsule object
Packit 423ecb
 *
Packit 423ecb
 * This function is called if and when a namespace node returned in
Packit 423ecb
 * an XPath node set is to be destroyed. That's the only kind of
Packit 423ecb
 * object returned in node set not directly linked to the original
Packit 423ecb
 * xmlDoc document, see xmlXPathNodeSetDupNs.
Packit 423ecb
 */
Packit 423ecb
#if PY_VERSION_HEX < 0x02070000
Packit 423ecb
static void
Packit 423ecb
libxml_xmlXPathDestructNsNode(void *cap, void *desc ATTRIBUTE_UNUSED)
Packit 423ecb
#else
Packit 423ecb
static void
Packit 423ecb
libxml_xmlXPathDestructNsNode(PyObject *cap)
Packit 423ecb
#endif
Packit 423ecb
{
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    fprintf(stderr, "libxml_xmlXPathDestructNsNode called %p\n", cap);
Packit 423ecb
#endif
Packit 423ecb
#if PY_VERSION_HEX < 0x02070000
Packit 423ecb
    xmlXPathNodeSetFreeNs((xmlNsPtr) cap);
Packit 423ecb
#else
Packit 423ecb
    xmlXPathNodeSetFreeNs((xmlNsPtr) PyCapsule_GetPointer(cap, "xmlNsPtr"));
Packit 423ecb
#endif
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlXPathObjectPtrWrap(xmlXPathObjectPtr obj)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlXPathObjectPtrWrap: ctxt = %p\n", obj);
Packit 423ecb
#endif
Packit 423ecb
    if (obj == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    switch (obj->type) {
Packit 423ecb
        case XPATH_XSLT_TREE: {
Packit 423ecb
            if ((obj->nodesetval == NULL) ||
Packit 423ecb
		(obj->nodesetval->nodeNr == 0) ||
Packit 423ecb
		(obj->nodesetval->nodeTab == NULL)) {
Packit 423ecb
                ret = PyList_New(0);
Packit 423ecb
	    } else {
Packit 423ecb
		int i, len = 0;
Packit 423ecb
		xmlNodePtr node;
Packit 423ecb
Packit 423ecb
		node = obj->nodesetval->nodeTab[0]->children;
Packit 423ecb
		while (node != NULL) {
Packit 423ecb
		    len++;
Packit 423ecb
		    node = node->next;
Packit 423ecb
		}
Packit 423ecb
		ret = PyList_New(len);
Packit 423ecb
		node = obj->nodesetval->nodeTab[0]->children;
Packit 423ecb
		for (i = 0;i < len;i++) {
Packit 423ecb
                    PyList_SetItem(ret, i, libxml_xmlNodePtrWrap(node));
Packit 423ecb
		    node = node->next;
Packit 423ecb
		}
Packit 423ecb
	    }
Packit 423ecb
	    /*
Packit 423ecb
	     * Return now, do not free the object passed down
Packit 423ecb
	     */
Packit 423ecb
	    return (ret);
Packit 423ecb
	}
Packit 423ecb
        case XPATH_NODESET:
Packit 423ecb
            if ((obj->nodesetval == NULL)
Packit 423ecb
                || (obj->nodesetval->nodeNr == 0)) {
Packit 423ecb
                ret = PyList_New(0);
Packit 423ecb
	    } else {
Packit 423ecb
                int i;
Packit 423ecb
                xmlNodePtr node;
Packit 423ecb
Packit 423ecb
                ret = PyList_New(obj->nodesetval->nodeNr);
Packit 423ecb
                for (i = 0; i < obj->nodesetval->nodeNr; i++) {
Packit 423ecb
                    node = obj->nodesetval->nodeTab[i];
Packit 423ecb
                    if (node->type == XML_NAMESPACE_DECL) {
Packit 423ecb
		        PyObject *ns = PyCapsule_New((void *) node,
Packit 423ecb
                                     (char *) "xmlNsPtr",
Packit 423ecb
				     libxml_xmlXPathDestructNsNode);
Packit 423ecb
			PyList_SetItem(ret, i, ns);
Packit 423ecb
			/* make sure the xmlNsPtr is not destroyed now */
Packit 423ecb
			obj->nodesetval->nodeTab[i] = NULL;
Packit 423ecb
		    } else {
Packit 423ecb
			PyList_SetItem(ret, i, libxml_xmlNodePtrWrap(node));
Packit 423ecb
		    }
Packit 423ecb
                }
Packit 423ecb
            }
Packit 423ecb
            break;
Packit 423ecb
        case XPATH_BOOLEAN:
Packit 423ecb
            ret = PY_IMPORT_INT((long) obj->boolval);
Packit 423ecb
            break;
Packit 423ecb
        case XPATH_NUMBER:
Packit 423ecb
            ret = PyFloat_FromDouble(obj->floatval);
Packit 423ecb
            break;
Packit 423ecb
        case XPATH_STRING:
Packit 423ecb
	    ret = PY_IMPORT_STRING((char *) obj->stringval);
Packit 423ecb
            break;
Packit 423ecb
        case XPATH_POINT:
Packit 423ecb
        {
Packit 423ecb
            PyObject *node;
Packit 423ecb
            PyObject *indexIntoNode;
Packit 423ecb
            PyObject *tuple;
Packit 423ecb
Packit 423ecb
            node = libxml_xmlNodePtrWrap(obj->user);
Packit 423ecb
            indexIntoNode = PY_IMPORT_INT((long) obj->index);
Packit 423ecb
Packit 423ecb
            tuple = PyTuple_New(2);
Packit 423ecb
            PyTuple_SetItem(tuple, 0, node);
Packit 423ecb
            PyTuple_SetItem(tuple, 1, indexIntoNode);
Packit 423ecb
Packit 423ecb
            ret = tuple;
Packit 423ecb
            break;
Packit 423ecb
        }
Packit 423ecb
        case XPATH_RANGE:
Packit 423ecb
        {
Packit 423ecb
            unsigned short bCollapsedRange;
Packit 423ecb
Packit 423ecb
            bCollapsedRange = ( (obj->user2 == NULL) ||
Packit 423ecb
		                ((obj->user2 == obj->user) && (obj->index == obj->index2)) );
Packit 423ecb
            if ( bCollapsedRange ) {
Packit 423ecb
                PyObject *node;
Packit 423ecb
                PyObject *indexIntoNode;
Packit 423ecb
                PyObject *tuple;
Packit 423ecb
                PyObject *list;
Packit 423ecb
Packit 423ecb
                list = PyList_New(1);
Packit 423ecb
Packit 423ecb
                node = libxml_xmlNodePtrWrap(obj->user);
Packit 423ecb
                indexIntoNode = PY_IMPORT_INT((long) obj->index);
Packit 423ecb
Packit 423ecb
                tuple = PyTuple_New(2);
Packit 423ecb
                PyTuple_SetItem(tuple, 0, node);
Packit 423ecb
                PyTuple_SetItem(tuple, 1, indexIntoNode);
Packit 423ecb
Packit 423ecb
                PyList_SetItem(list, 0, tuple);
Packit 423ecb
Packit 423ecb
                ret = list;
Packit 423ecb
            } else {
Packit 423ecb
                PyObject *node;
Packit 423ecb
                PyObject *indexIntoNode;
Packit 423ecb
                PyObject *tuple;
Packit 423ecb
                PyObject *list;
Packit 423ecb
Packit 423ecb
                list = PyList_New(2);
Packit 423ecb
Packit 423ecb
                node = libxml_xmlNodePtrWrap(obj->user);
Packit 423ecb
                indexIntoNode = PY_IMPORT_INT((long) obj->index);
Packit 423ecb
Packit 423ecb
                tuple = PyTuple_New(2);
Packit 423ecb
                PyTuple_SetItem(tuple, 0, node);
Packit 423ecb
                PyTuple_SetItem(tuple, 1, indexIntoNode);
Packit 423ecb
Packit 423ecb
                PyList_SetItem(list, 0, tuple);
Packit 423ecb
Packit 423ecb
                node = libxml_xmlNodePtrWrap(obj->user2);
Packit 423ecb
                indexIntoNode = PY_IMPORT_INT((long) obj->index2);
Packit 423ecb
Packit 423ecb
                tuple = PyTuple_New(2);
Packit 423ecb
                PyTuple_SetItem(tuple, 0, node);
Packit 423ecb
                PyTuple_SetItem(tuple, 1, indexIntoNode);
Packit 423ecb
Packit 423ecb
                PyList_SetItem(list, 1, tuple);
Packit 423ecb
Packit 423ecb
                ret = list;
Packit 423ecb
            }
Packit 423ecb
            break;
Packit 423ecb
        }
Packit 423ecb
        case XPATH_LOCATIONSET:
Packit 423ecb
        {
Packit 423ecb
            xmlLocationSetPtr set;
Packit 423ecb
Packit 423ecb
            set = obj->user;
Packit 423ecb
            if ( set && set->locNr > 0 ) {
Packit 423ecb
                int i;
Packit 423ecb
                PyObject *list;
Packit 423ecb
Packit 423ecb
                list = PyList_New(set->locNr);
Packit 423ecb
Packit 423ecb
                for (i=0; i<set->locNr; i++) {
Packit 423ecb
                    xmlXPathObjectPtr setobj;
Packit 423ecb
                    PyObject *pyobj;
Packit 423ecb
Packit 423ecb
                    setobj = set->locTab[i]; /*xmlXPathObjectPtr setobj*/
Packit 423ecb
Packit 423ecb
                    pyobj = libxml_xmlXPathObjectPtrWrap(setobj);
Packit 423ecb
                    /* xmlXPathFreeObject(setobj) is called */
Packit 423ecb
                    set->locTab[i] = NULL;
Packit 423ecb
Packit 423ecb
                    PyList_SetItem(list, i, pyobj);
Packit 423ecb
                }
Packit 423ecb
                set->locNr = 0;
Packit 423ecb
                ret = list;
Packit 423ecb
            } else {
Packit 423ecb
                Py_INCREF(Py_None);
Packit 423ecb
                ret = Py_None;
Packit 423ecb
            }
Packit 423ecb
            break;
Packit 423ecb
        }
Packit 423ecb
        default:
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
            printf("Unable to convert XPath object type %d\n", obj->type);
Packit 423ecb
#endif
Packit 423ecb
            Py_INCREF(Py_None);
Packit 423ecb
            ret = Py_None;
Packit 423ecb
    }
Packit 423ecb
    xmlXPathFreeObject(obj);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
xmlXPathObjectPtr
Packit 423ecb
libxml_xmlXPathObjectPtrConvert(PyObject *obj)
Packit 423ecb
{
Packit 423ecb
    xmlXPathObjectPtr ret = NULL;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlXPathObjectPtrConvert: obj = %p\n", obj);
Packit 423ecb
#endif
Packit 423ecb
    if (obj == NULL) {
Packit 423ecb
        return (NULL);
Packit 423ecb
    }
Packit 423ecb
    if PyFloat_Check (obj) {
Packit 423ecb
        ret = xmlXPathNewFloat((double) PyFloat_AS_DOUBLE(obj));
Packit 423ecb
    } else if PyLong_Check(obj) {
Packit 423ecb
#ifdef PyLong_AS_LONG
Packit 423ecb
        ret = xmlXPathNewFloat((double) PyLong_AS_LONG(obj));
Packit 423ecb
#else
Packit 423ecb
        ret = xmlXPathNewFloat((double) PyInt_AS_LONG(obj));
Packit 423ecb
#endif
Packit 423ecb
#ifdef PyBool_Check
Packit 423ecb
    } else if PyBool_Check (obj) {
Packit 423ecb
Packit 423ecb
        if (obj == Py_True) {
Packit 423ecb
          ret = xmlXPathNewBoolean(1);
Packit 423ecb
        }
Packit 423ecb
        else {
Packit 423ecb
          ret = xmlXPathNewBoolean(0);
Packit 423ecb
        }
Packit 423ecb
#endif
Packit 423ecb
    } else if PyBytes_Check (obj) {
Packit 423ecb
        xmlChar *str;
Packit 423ecb
Packit 423ecb
        str = xmlStrndup((const xmlChar *) PyBytes_AS_STRING(obj),
Packit 423ecb
                         PyBytes_GET_SIZE(obj));
Packit 423ecb
        ret = xmlXPathWrapString(str);
Packit 423ecb
#ifdef PyUnicode_Check
Packit 423ecb
    } else if PyUnicode_Check (obj) {
Packit 423ecb
#if PY_VERSION_HEX >= 0x03030000
Packit 423ecb
        xmlChar *str;
Packit 423ecb
	const char *tmp;
Packit 423ecb
	Py_ssize_t size;
Packit 423ecb
Packit 423ecb
	/* tmp doesn't need to be deallocated */
Packit 423ecb
        tmp = PyUnicode_AsUTF8AndSize(obj, &size);
Packit 423ecb
        str = xmlStrndup((const xmlChar *) tmp, (int) size);
Packit 423ecb
        ret = xmlXPathWrapString(str);
Packit 423ecb
#else
Packit 423ecb
        xmlChar *str = NULL;
Packit 423ecb
        PyObject *b;
Packit 423ecb
Packit 423ecb
	b = PyUnicode_AsUTF8String(obj);
Packit 423ecb
	if (b != NULL) {
Packit 423ecb
	    str = xmlStrndup((const xmlChar *) PyBytes_AS_STRING(b),
Packit 423ecb
			     PyBytes_GET_SIZE(b));
Packit 423ecb
	    Py_DECREF(b);
Packit 423ecb
	}
Packit 423ecb
	ret = xmlXPathWrapString(str);
Packit 423ecb
#endif
Packit 423ecb
#endif
Packit 423ecb
    } else if PyList_Check (obj) {
Packit 423ecb
        int i;
Packit 423ecb
        PyObject *node;
Packit 423ecb
        xmlNodePtr cur;
Packit 423ecb
        xmlNodeSetPtr set;
Packit 423ecb
Packit 423ecb
        set = xmlXPathNodeSetCreate(NULL);
Packit 423ecb
Packit 423ecb
        for (i = 0; i < PyList_Size(obj); i++) {
Packit 423ecb
            node = PyList_GetItem(obj, i);
Packit 423ecb
            if ((node == NULL) || (node->ob_type == NULL))
Packit 423ecb
                continue;
Packit 423ecb
Packit 423ecb
            cur = NULL;
Packit 423ecb
            if (PyCapsule_CheckExact(node)) {
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
                printf("Got a Capsule\n");
Packit 423ecb
#endif
Packit 423ecb
                cur = PyxmlNode_Get(node);
Packit 423ecb
            } else if ((PyObject_HasAttrString(node, (char *) "_o")) &&
Packit 423ecb
	               (PyObject_HasAttrString(node, (char *) "get_doc"))) {
Packit 423ecb
		PyObject *wrapper;
Packit 423ecb
Packit 423ecb
		wrapper = PyObject_GetAttrString(node, (char *) "_o");
Packit 423ecb
		if (wrapper != NULL)
Packit 423ecb
		    cur = PyxmlNode_Get(wrapper);
Packit 423ecb
            } else {
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
                printf("Unknown object in Python return list\n");
Packit 423ecb
#endif
Packit 423ecb
            }
Packit 423ecb
            if (cur != NULL) {
Packit 423ecb
                xmlXPathNodeSetAdd(set, cur);
Packit 423ecb
            }
Packit 423ecb
        }
Packit 423ecb
        ret = xmlXPathWrapNodeSet(set);
Packit 423ecb
    } else {
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
        printf("Unable to convert Python Object to XPath");
Packit 423ecb
#endif
Packit 423ecb
    }
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlValidCtxtPtrWrap(xmlValidCtxtPtr valid)
Packit 423ecb
{
Packit 423ecb
	PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
	printf("libxml_xmlValidCtxtPtrWrap: valid = %p\n", valid);
Packit 423ecb
#endif
Packit 423ecb
	if (valid == NULL) {
Packit 423ecb
		Py_INCREF(Py_None);
Packit 423ecb
		return (Py_None);
Packit 423ecb
	}
Packit 423ecb
Packit 423ecb
	ret = 
Packit 423ecb
		PyCapsule_New((void *) valid,
Packit 423ecb
									 (char *) "xmlValidCtxtPtr", NULL);
Packit 423ecb
Packit 423ecb
	return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlCatalogPtrWrap(xmlCatalogPtr catal)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlNodePtrWrap: catal = %p\n", catal);
Packit 423ecb
#endif
Packit 423ecb
    if (catal == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret =
Packit 423ecb
        PyCapsule_New((void *) catal,
Packit 423ecb
                                     (char *) "xmlCatalogPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlOutputBufferPtrWrap(xmlOutputBufferPtr buffer)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlOutputBufferPtrWrap: buffer = %p\n", buffer);
Packit 423ecb
#endif
Packit 423ecb
    if (buffer == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret =
Packit 423ecb
        PyCapsule_New((void *) buffer,
Packit 423ecb
                                     (char *) "xmlOutputBufferPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlParserInputBufferPtrWrap(xmlParserInputBufferPtr buffer)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlParserInputBufferPtrWrap: buffer = %p\n", buffer);
Packit 423ecb
#endif
Packit 423ecb
    if (buffer == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret =
Packit 423ecb
        PyCapsule_New((void *) buffer,
Packit 423ecb
                                     (char *) "xmlParserInputBufferPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
#ifdef LIBXML_REGEXP_ENABLED
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlRegexpPtrWrap(xmlRegexpPtr regexp)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlRegexpPtrWrap: regexp = %p\n", regexp);
Packit 423ecb
#endif
Packit 423ecb
    if (regexp == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret =
Packit 423ecb
        PyCapsule_New((void *) regexp,
Packit 423ecb
                                     (char *) "xmlRegexpPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
#endif /* LIBXML_REGEXP_ENABLED */
Packit 423ecb
Packit 423ecb
#ifdef LIBXML_READER_ENABLED
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlTextReaderPtrWrap(xmlTextReaderPtr reader)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlTextReaderPtrWrap: reader = %p\n", reader);
Packit 423ecb
#endif
Packit 423ecb
    if (reader == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret =
Packit 423ecb
        PyCapsule_New((void *) reader,
Packit 423ecb
                                     (char *) "xmlTextReaderPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlTextReaderLocatorPtrWrap(xmlTextReaderLocatorPtr locator)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlTextReaderLocatorPtrWrap: locator = %p\n", locator);
Packit 423ecb
#endif
Packit 423ecb
    if (locator == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret =
Packit 423ecb
        PyCapsule_New((void *) locator,
Packit 423ecb
                                     (char *) "xmlTextReaderLocatorPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
#endif /* LIBXML_READER_ENABLED */
Packit 423ecb
Packit 423ecb
#ifdef LIBXML_SCHEMAS_ENABLED
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlRelaxNGPtrWrap(xmlRelaxNGPtr ctxt)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlRelaxNGPtrWrap: ctxt = %p\n", ctxt);
Packit 423ecb
#endif
Packit 423ecb
    if (ctxt == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret =
Packit 423ecb
        PyCapsule_New((void *) ctxt,
Packit 423ecb
                                     (char *) "xmlRelaxNGPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlRelaxNGParserCtxtPtrWrap(xmlRelaxNGParserCtxtPtr ctxt)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlRelaxNGParserCtxtPtrWrap: ctxt = %p\n", ctxt);
Packit 423ecb
#endif
Packit 423ecb
    if (ctxt == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret =
Packit 423ecb
        PyCapsule_New((void *) ctxt,
Packit 423ecb
                                     (char *) "xmlRelaxNGParserCtxtPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlRelaxNGValidCtxtPtrWrap(xmlRelaxNGValidCtxtPtr valid)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlRelaxNGValidCtxtPtrWrap: valid = %p\n", valid);
Packit 423ecb
#endif
Packit 423ecb
    if (valid == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret =
Packit 423ecb
        PyCapsule_New((void *) valid,
Packit 423ecb
                                     (char *) "xmlRelaxNGValidCtxtPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlSchemaPtrWrap(xmlSchemaPtr ctxt)
Packit 423ecb
{
Packit 423ecb
	PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
	printf("libxml_xmlSchemaPtrWrap: ctxt = %p\n", ctxt);
Packit 423ecb
#endif
Packit 423ecb
	if (ctxt == NULL) {
Packit 423ecb
		Py_INCREF(Py_None);
Packit 423ecb
		return (Py_None);
Packit 423ecb
	}
Packit 423ecb
	ret =
Packit 423ecb
		PyCapsule_New((void *) ctxt,
Packit 423ecb
									 (char *) "xmlSchemaPtr", NULL);
Packit 423ecb
	return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlSchemaParserCtxtPtrWrap(xmlSchemaParserCtxtPtr ctxt)
Packit 423ecb
{
Packit 423ecb
	PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
	printf("libxml_xmlSchemaParserCtxtPtrWrap: ctxt = %p\n", ctxt);
Packit 423ecb
#endif
Packit 423ecb
	if (ctxt == NULL) {
Packit 423ecb
		Py_INCREF(Py_None);
Packit 423ecb
		return (Py_None);
Packit 423ecb
	}
Packit 423ecb
	ret = 
Packit 423ecb
		PyCapsule_New((void *) ctxt,
Packit 423ecb
									 (char *) "xmlSchemaParserCtxtPtr", NULL);
Packit 423ecb
Packit 423ecb
	return (ret);
Packit 423ecb
}
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlSchemaValidCtxtPtrWrap(xmlSchemaValidCtxtPtr valid)
Packit 423ecb
{
Packit 423ecb
	PyObject *ret;
Packit 423ecb
	
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
	printf("libxml_xmlSchemaValidCtxtPtrWrap: valid = %p\n", valid);
Packit 423ecb
#endif
Packit 423ecb
	if (valid == NULL) {
Packit 423ecb
		Py_INCREF(Py_None);
Packit 423ecb
		return (Py_None);
Packit 423ecb
	}
Packit 423ecb
Packit 423ecb
	ret = 
Packit 423ecb
		PyCapsule_New((void *) valid,
Packit 423ecb
									 (char *) "xmlSchemaValidCtxtPtr", NULL);
Packit 423ecb
Packit 423ecb
	return (ret);
Packit 423ecb
}
Packit 423ecb
#endif /* LIBXML_SCHEMAS_ENABLED */
Packit 423ecb
Packit 423ecb
PyObject *
Packit 423ecb
libxml_xmlErrorPtrWrap(xmlErrorPtr error)
Packit 423ecb
{
Packit 423ecb
    PyObject *ret;
Packit 423ecb
Packit 423ecb
#ifdef DEBUG
Packit 423ecb
    printf("libxml_xmlErrorPtrWrap: error = %p\n", error);
Packit 423ecb
#endif
Packit 423ecb
    if (error == NULL) {
Packit 423ecb
        Py_INCREF(Py_None);
Packit 423ecb
        return (Py_None);
Packit 423ecb
    }
Packit 423ecb
    ret = PyCapsule_New((void *) error, (char *) "xmlErrorPtr", NULL);
Packit 423ecb
    return (ret);
Packit 423ecb
}