Annotation of embedaddon/libxml2/python/libxml.c, revision 1.1
1.1 ! misho 1: /*
! 2: * libxml.c: this modules implements the main part of the glue of the
! 3: * libxml2 library and the Python interpreter. It provides the
! 4: * entry points where an automatically generated stub is either
! 5: * unpractical or would not match cleanly the Python model.
! 6: *
! 7: * If compiled with MERGED_MODULES, the entry point will be used to
! 8: * initialize both the libxml2 and the libxslt wrappers
! 9: *
! 10: * See Copyright for the status of this software.
! 11: *
! 12: * daniel@veillard.com
! 13: */
! 14: #include <Python.h>
! 15: #include <fileobject.h>
! 16: /* #include "config.h" */
! 17: #include <libxml/xmlmemory.h>
! 18: #include <libxml/parser.h>
! 19: #include <libxml/tree.h>
! 20: #include <libxml/xpath.h>
! 21: #include <libxml/xmlerror.h>
! 22: #include <libxml/xpathInternals.h>
! 23: #include <libxml/xmlmemory.h>
! 24: #include <libxml/xmlIO.h>
! 25: #include <libxml/c14n.h>
! 26: #include <libxml/xmlreader.h>
! 27: #include <libxml/xmlsave.h>
! 28: #include "libxml_wrap.h"
! 29: #include "libxml2-py.h"
! 30:
! 31: #if (defined(_MSC_VER) || defined(__MINGW32__)) && !defined(vsnprintf)
! 32: #define vsnprintf(b,c,f,a) _vsnprintf(b,c,f,a)
! 33: #elif defined(WITH_TRIO) && !defined(vsnprintf)
! 34: #include "trio.h"
! 35: #define vsnprintf trio_vsnprintf
! 36: #endif
! 37:
! 38: /* #define DEBUG */
! 39: /* #define DEBUG_SAX */
! 40: /* #define DEBUG_XPATH */
! 41: /* #define DEBUG_ERROR */
! 42: /* #define DEBUG_MEMORY */
! 43: /* #define DEBUG_FILES */
! 44: /* #define DEBUG_LOADER */
! 45:
! 46: void initlibxml2mod(void);
! 47:
! 48: /**
! 49: * TODO:
! 50: *
! 51: * macro to flag unimplemented blocks
! 52: */
! 53: #define TODO \
! 54: xmlGenericError(xmlGenericErrorContext, \
! 55: "Unimplemented block at %s:%d\n", \
! 56: __FILE__, __LINE__);
! 57: /*
! 58: * the following vars are used for XPath extensions, but
! 59: * are also referenced within the parser cleanup routine.
! 60: */
! 61: static int libxml_xpathCallbacksInitialized = 0;
! 62:
! 63: typedef struct libxml_xpathCallback {
! 64: xmlXPathContextPtr ctx;
! 65: xmlChar *name;
! 66: xmlChar *ns_uri;
! 67: PyObject *function;
! 68: } libxml_xpathCallback, *libxml_xpathCallbackPtr;
! 69: typedef libxml_xpathCallback libxml_xpathCallbackArray[];
! 70: static int libxml_xpathCallbacksAllocd = 10;
! 71: static libxml_xpathCallbackArray *libxml_xpathCallbacks = NULL;
! 72: static int libxml_xpathCallbacksNb = 0;
! 73:
! 74: /************************************************************************
! 75: * *
! 76: * Memory debug interface *
! 77: * *
! 78: ************************************************************************/
! 79:
! 80: #if 0
! 81: extern void xmlMemFree(void *ptr);
! 82: extern void *xmlMemMalloc(size_t size);
! 83: extern void *xmlMemRealloc(void *ptr, size_t size);
! 84: extern char *xmlMemoryStrdup(const char *str);
! 85: #endif
! 86:
! 87: static int libxmlMemoryDebugActivated = 0;
! 88: static long libxmlMemoryAllocatedBase = 0;
! 89:
! 90: static int libxmlMemoryDebug = 0;
! 91: static xmlFreeFunc freeFunc = NULL;
! 92: static xmlMallocFunc mallocFunc = NULL;
! 93: static xmlReallocFunc reallocFunc = NULL;
! 94: static xmlStrdupFunc strdupFunc = NULL;
! 95:
! 96: static void
! 97: libxml_xmlErrorInitialize(void); /* forward declare */
! 98:
! 99: PyObject *
! 100: libxml_xmlMemoryUsed(PyObject * self ATTRIBUTE_UNUSED,
! 101: PyObject * args ATTRIBUTE_UNUSED)
! 102: {
! 103: long ret;
! 104: PyObject *py_retval;
! 105:
! 106: ret = xmlMemUsed();
! 107:
! 108: py_retval = libxml_longWrap(ret);
! 109: return (py_retval);
! 110: }
! 111:
! 112: PyObject *
! 113: libxml_xmlDebugMemory(PyObject * self ATTRIBUTE_UNUSED, PyObject * args)
! 114: {
! 115: int activate;
! 116: PyObject *py_retval;
! 117: long ret;
! 118:
! 119: if (!PyArg_ParseTuple(args, (char *) "i:xmlDebugMemory", &activate))
! 120: return (NULL);
! 121:
! 122: #ifdef DEBUG_MEMORY
! 123: printf("libxml_xmlDebugMemory(%d) called\n", activate);
! 124: #endif
! 125:
! 126: if (activate != 0) {
! 127: if (libxmlMemoryDebug == 0) {
! 128: /*
! 129: * First initialize the library and grab the old memory handlers
! 130: * and switch the library to memory debugging
! 131: */
! 132: xmlMemGet((xmlFreeFunc *) & freeFunc,
! 133: (xmlMallocFunc *) & mallocFunc,
! 134: (xmlReallocFunc *) & reallocFunc,
! 135: (xmlStrdupFunc *) & strdupFunc);
! 136: if ((freeFunc == xmlMemFree) && (mallocFunc == xmlMemMalloc) &&
! 137: (reallocFunc == xmlMemRealloc) &&
! 138: (strdupFunc == xmlMemoryStrdup)) {
! 139: libxmlMemoryAllocatedBase = xmlMemUsed();
! 140: } else {
! 141: /*
! 142: * cleanup first, because some memory has been
! 143: * allocated with the non-debug malloc in xmlInitParser
! 144: * when the python module was imported
! 145: */
! 146: xmlCleanupParser();
! 147: ret = (long) xmlMemSetup(xmlMemFree, xmlMemMalloc,
! 148: xmlMemRealloc, xmlMemoryStrdup);
! 149: if (ret < 0)
! 150: goto error;
! 151: libxmlMemoryAllocatedBase = xmlMemUsed();
! 152: /* reinitialize */
! 153: xmlInitParser();
! 154: libxml_xmlErrorInitialize();
! 155: }
! 156: ret = 0;
! 157: } else if (libxmlMemoryDebugActivated == 0) {
! 158: libxmlMemoryAllocatedBase = xmlMemUsed();
! 159: ret = 0;
! 160: } else {
! 161: ret = xmlMemUsed() - libxmlMemoryAllocatedBase;
! 162: }
! 163: libxmlMemoryDebug = 1;
! 164: libxmlMemoryDebugActivated = 1;
! 165: } else {
! 166: if (libxmlMemoryDebugActivated == 1)
! 167: ret = xmlMemUsed() - libxmlMemoryAllocatedBase;
! 168: else
! 169: ret = 0;
! 170: libxmlMemoryDebugActivated = 0;
! 171: }
! 172: error:
! 173: py_retval = libxml_longWrap(ret);
! 174: return (py_retval);
! 175: }
! 176:
! 177: PyObject *
! 178: libxml_xmlPythonCleanupParser(PyObject *self ATTRIBUTE_UNUSED,
! 179: PyObject *args ATTRIBUTE_UNUSED) {
! 180:
! 181: int ix;
! 182: long freed = -1;
! 183:
! 184: if (libxmlMemoryDebug) {
! 185: freed = xmlMemUsed();
! 186: }
! 187:
! 188: xmlCleanupParser();
! 189: /*
! 190: * Need to confirm whether we really want to do this (required for
! 191: * memcheck) in all cases...
! 192: */
! 193:
! 194: if (libxml_xpathCallbacks != NULL) { /* if ext funcs declared */
! 195: for (ix=0; ix<libxml_xpathCallbacksNb; ix++) {
! 196: if ((*libxml_xpathCallbacks)[ix].name != NULL)
! 197: xmlFree((*libxml_xpathCallbacks)[ix].name);
! 198: if ((*libxml_xpathCallbacks)[ix].ns_uri != NULL)
! 199: xmlFree((*libxml_xpathCallbacks)[ix].ns_uri);
! 200: }
! 201: libxml_xpathCallbacksNb = 0;
! 202: xmlFree(libxml_xpathCallbacks);
! 203: libxml_xpathCallbacks = NULL;
! 204: }
! 205:
! 206: if (libxmlMemoryDebug) {
! 207: freed -= xmlMemUsed();
! 208: libxmlMemoryAllocatedBase -= freed;
! 209: if (libxmlMemoryAllocatedBase < 0)
! 210: libxmlMemoryAllocatedBase = 0;
! 211: }
! 212:
! 213: Py_INCREF(Py_None);
! 214: return(Py_None);
! 215: }
! 216:
! 217: PyObject *
! 218: libxml_xmlDumpMemory(ATTRIBUTE_UNUSED PyObject * self,
! 219: ATTRIBUTE_UNUSED PyObject * args)
! 220: {
! 221:
! 222: if (libxmlMemoryDebug != 0)
! 223: xmlMemoryDump();
! 224: Py_INCREF(Py_None);
! 225: return (Py_None);
! 226: }
! 227:
! 228: /************************************************************************
! 229: * *
! 230: * Handling Python FILE I/O at the C level *
! 231: * The raw I/O attack diectly the File objects, while the *
! 232: * other routines address the ioWrapper instance instead *
! 233: * *
! 234: ************************************************************************/
! 235:
! 236: /**
! 237: * xmlPythonFileCloseUnref:
! 238: * @context: the I/O context
! 239: *
! 240: * Close an I/O channel
! 241: */
! 242: static int
! 243: xmlPythonFileCloseRaw (void * context) {
! 244: PyObject *file, *ret;
! 245:
! 246: #ifdef DEBUG_FILES
! 247: printf("xmlPythonFileCloseUnref\n");
! 248: #endif
! 249: file = (PyObject *) context;
! 250: if (file == NULL) return(-1);
! 251: ret = PyEval_CallMethod(file, (char *) "close", (char *) "()");
! 252: if (ret != NULL) {
! 253: Py_DECREF(ret);
! 254: }
! 255: Py_DECREF(file);
! 256: return(0);
! 257: }
! 258:
! 259: /**
! 260: * xmlPythonFileReadRaw:
! 261: * @context: the I/O context
! 262: * @buffer: where to drop data
! 263: * @len: number of bytes to write
! 264: *
! 265: * Read @len bytes to @buffer from the Python file in the I/O channel
! 266: *
! 267: * Returns the number of bytes read
! 268: */
! 269: static int
! 270: xmlPythonFileReadRaw (void * context, char * buffer, int len) {
! 271: PyObject *file;
! 272: PyObject *ret;
! 273: int lenread = -1;
! 274: char *data;
! 275:
! 276: #ifdef DEBUG_FILES
! 277: printf("xmlPythonFileReadRaw: %d\n", len);
! 278: #endif
! 279: file = (PyObject *) context;
! 280: if (file == NULL) return(-1);
! 281: ret = PyEval_CallMethod(file, (char *) "read", (char *) "(i)", len);
! 282: if (ret == NULL) {
! 283: printf("xmlPythonFileReadRaw: result is NULL\n");
! 284: return(-1);
! 285: } else if (PyString_Check(ret)) {
! 286: lenread = PyString_Size(ret);
! 287: data = PyString_AsString(ret);
! 288: if (lenread > len)
! 289: memcpy(buffer, data, len);
! 290: else
! 291: memcpy(buffer, data, lenread);
! 292: Py_DECREF(ret);
! 293: } else {
! 294: printf("xmlPythonFileReadRaw: result is not a String\n");
! 295: Py_DECREF(ret);
! 296: }
! 297: return(lenread);
! 298: }
! 299:
! 300: /**
! 301: * xmlPythonFileRead:
! 302: * @context: the I/O context
! 303: * @buffer: where to drop data
! 304: * @len: number of bytes to write
! 305: *
! 306: * Read @len bytes to @buffer from the I/O channel.
! 307: *
! 308: * Returns the number of bytes read
! 309: */
! 310: static int
! 311: xmlPythonFileRead (void * context, char * buffer, int len) {
! 312: PyObject *file;
! 313: PyObject *ret;
! 314: int lenread = -1;
! 315: char *data;
! 316:
! 317: #ifdef DEBUG_FILES
! 318: printf("xmlPythonFileRead: %d\n", len);
! 319: #endif
! 320: file = (PyObject *) context;
! 321: if (file == NULL) return(-1);
! 322: ret = PyEval_CallMethod(file, (char *) "io_read", (char *) "(i)", len);
! 323: if (ret == NULL) {
! 324: printf("xmlPythonFileRead: result is NULL\n");
! 325: return(-1);
! 326: } else if (PyString_Check(ret)) {
! 327: lenread = PyString_Size(ret);
! 328: data = PyString_AsString(ret);
! 329: if (lenread > len)
! 330: memcpy(buffer, data, len);
! 331: else
! 332: memcpy(buffer, data, lenread);
! 333: Py_DECREF(ret);
! 334: } else {
! 335: printf("xmlPythonFileRead: result is not a String\n");
! 336: Py_DECREF(ret);
! 337: }
! 338: return(lenread);
! 339: }
! 340:
! 341: /**
! 342: * xmlFileWrite:
! 343: * @context: the I/O context
! 344: * @buffer: where to drop data
! 345: * @len: number of bytes to write
! 346: *
! 347: * Write @len bytes from @buffer to the I/O channel.
! 348: *
! 349: * Returns the number of bytes written
! 350: */
! 351: static int
! 352: xmlPythonFileWrite (void * context, const char * buffer, int len) {
! 353: PyObject *file;
! 354: PyObject *string;
! 355: PyObject *ret = NULL;
! 356: int written = -1;
! 357:
! 358: #ifdef DEBUG_FILES
! 359: printf("xmlPythonFileWrite: %d\n", len);
! 360: #endif
! 361: file = (PyObject *) context;
! 362: if (file == NULL) return(-1);
! 363: string = PyString_FromStringAndSize(buffer, len);
! 364: if (string == NULL) return(-1);
! 365: if (PyObject_HasAttrString(file, (char *) "io_write")) {
! 366: ret = PyEval_CallMethod(file, (char *) "io_write", (char *) "(O)",
! 367: string);
! 368: } else if (PyObject_HasAttrString(file, (char *) "write")) {
! 369: ret = PyEval_CallMethod(file, (char *) "write", (char *) "(O)",
! 370: string);
! 371: }
! 372: Py_DECREF(string);
! 373: if (ret == NULL) {
! 374: printf("xmlPythonFileWrite: result is NULL\n");
! 375: return(-1);
! 376: } else if (PyInt_Check(ret)) {
! 377: written = (int) PyInt_AsLong(ret);
! 378: Py_DECREF(ret);
! 379: } else if (ret == Py_None) {
! 380: written = len;
! 381: Py_DECREF(ret);
! 382: } else {
! 383: printf("xmlPythonFileWrite: result is not an Int nor None\n");
! 384: Py_DECREF(ret);
! 385: }
! 386: return(written);
! 387: }
! 388:
! 389: /**
! 390: * xmlPythonFileClose:
! 391: * @context: the I/O context
! 392: *
! 393: * Close an I/O channel
! 394: */
! 395: static int
! 396: xmlPythonFileClose (void * context) {
! 397: PyObject *file, *ret = NULL;
! 398:
! 399: #ifdef DEBUG_FILES
! 400: printf("xmlPythonFileClose\n");
! 401: #endif
! 402: file = (PyObject *) context;
! 403: if (file == NULL) return(-1);
! 404: if (PyObject_HasAttrString(file, (char *) "io_close")) {
! 405: ret = PyEval_CallMethod(file, (char *) "io_close", (char *) "()");
! 406: } else if (PyObject_HasAttrString(file, (char *) "flush")) {
! 407: ret = PyEval_CallMethod(file, (char *) "flush", (char *) "()");
! 408: }
! 409: if (ret != NULL) {
! 410: Py_DECREF(ret);
! 411: }
! 412: return(0);
! 413: }
! 414:
! 415: #ifdef LIBXML_OUTPUT_ENABLED
! 416: /**
! 417: * xmlOutputBufferCreatePythonFile:
! 418: * @file: a PyFile_Type
! 419: * @encoder: the encoding converter or NULL
! 420: *
! 421: * Create a buffered output for the progressive saving to a PyFile_Type
! 422: * buffered C I/O
! 423: *
! 424: * Returns the new parser output or NULL
! 425: */
! 426: static xmlOutputBufferPtr
! 427: xmlOutputBufferCreatePythonFile(PyObject *file,
! 428: xmlCharEncodingHandlerPtr encoder) {
! 429: xmlOutputBufferPtr ret;
! 430:
! 431: if (file == NULL) return(NULL);
! 432:
! 433: ret = xmlAllocOutputBuffer(encoder);
! 434: if (ret != NULL) {
! 435: ret->context = file;
! 436: /* Py_INCREF(file); */
! 437: ret->writecallback = xmlPythonFileWrite;
! 438: ret->closecallback = xmlPythonFileClose;
! 439: }
! 440:
! 441: return(ret);
! 442: }
! 443:
! 444: PyObject *
! 445: libxml_xmlCreateOutputBuffer(ATTRIBUTE_UNUSED PyObject *self, PyObject *args) {
! 446: PyObject *py_retval;
! 447: PyObject *file;
! 448: xmlChar *encoding;
! 449: xmlCharEncodingHandlerPtr handler = NULL;
! 450: xmlOutputBufferPtr buffer;
! 451:
! 452:
! 453: if (!PyArg_ParseTuple(args, (char *)"Oz:xmlOutputBufferCreate",
! 454: &file, &encoding))
! 455: return(NULL);
! 456: if ((encoding != NULL) && (encoding[0] != 0)) {
! 457: handler = xmlFindCharEncodingHandler((const char *) encoding);
! 458: }
! 459: buffer = xmlOutputBufferCreatePythonFile(file, handler);
! 460: if (buffer == NULL)
! 461: printf("libxml_xmlCreateOutputBuffer: buffer == NULL\n");
! 462: py_retval = libxml_xmlOutputBufferPtrWrap(buffer);
! 463: return(py_retval);
! 464: }
! 465:
! 466: /**
! 467: * libxml_outputBufferGetPythonFile:
! 468: * @buffer: the I/O buffer
! 469: *
! 470: * read the Python I/O from the CObject
! 471: *
! 472: * Returns the new parser output or NULL
! 473: */
! 474: static PyObject *
! 475: libxml_outputBufferGetPythonFile(ATTRIBUTE_UNUSED PyObject *self,
! 476: PyObject *args) {
! 477: PyObject *buffer;
! 478: PyObject *file;
! 479: xmlOutputBufferPtr obj;
! 480:
! 481: if (!PyArg_ParseTuple(args, (char *)"O:outputBufferGetPythonFile",
! 482: &buffer))
! 483: return(NULL);
! 484:
! 485: obj = PyoutputBuffer_Get(buffer);
! 486: if (obj == NULL) {
! 487: fprintf(stderr,
! 488: "outputBufferGetPythonFile: obj == NULL\n");
! 489: Py_INCREF(Py_None);
! 490: return(Py_None);
! 491: }
! 492: if (obj->closecallback != xmlPythonFileClose) {
! 493: fprintf(stderr,
! 494: "outputBufferGetPythonFile: not a python file wrapper\n");
! 495: Py_INCREF(Py_None);
! 496: return(Py_None);
! 497: }
! 498: file = (PyObject *) obj->context;
! 499: if (file == NULL) {
! 500: Py_INCREF(Py_None);
! 501: return(Py_None);
! 502: }
! 503: Py_INCREF(file);
! 504: return(file);
! 505: }
! 506:
! 507: static PyObject *
! 508: libxml_xmlOutputBufferClose(PyObject *self ATTRIBUTE_UNUSED, PyObject *args) {
! 509: PyObject *py_retval;
! 510: int c_retval;
! 511: xmlOutputBufferPtr out;
! 512: PyObject *pyobj_out;
! 513:
! 514: if (!PyArg_ParseTuple(args, (char *)"O:xmlOutputBufferClose", &pyobj_out))
! 515: return(NULL);
! 516: out = (xmlOutputBufferPtr) PyoutputBuffer_Get(pyobj_out);
! 517: /* Buffer may already have been destroyed elsewhere. This is harmless. */
! 518: if (out == NULL) {
! 519: Py_INCREF(Py_None);
! 520: return(Py_None);
! 521: }
! 522:
! 523: c_retval = xmlOutputBufferClose(out);
! 524: py_retval = libxml_intWrap((int) c_retval);
! 525: return(py_retval);
! 526: }
! 527:
! 528: static PyObject *
! 529: libxml_xmlOutputBufferFlush(PyObject *self ATTRIBUTE_UNUSED, PyObject *args) {
! 530: PyObject *py_retval;
! 531: int c_retval;
! 532: xmlOutputBufferPtr out;
! 533: PyObject *pyobj_out;
! 534:
! 535: if (!PyArg_ParseTuple(args, (char *)"O:xmlOutputBufferFlush", &pyobj_out))
! 536: return(NULL);
! 537: out = (xmlOutputBufferPtr) PyoutputBuffer_Get(pyobj_out);
! 538:
! 539: c_retval = xmlOutputBufferFlush(out);
! 540: py_retval = libxml_intWrap((int) c_retval);
! 541: return(py_retval);
! 542: }
! 543:
! 544: static PyObject *
! 545: libxml_xmlSaveFileTo(PyObject *self ATTRIBUTE_UNUSED, PyObject *args) {
! 546: PyObject *py_retval;
! 547: int c_retval;
! 548: xmlOutputBufferPtr buf;
! 549: PyObject *pyobj_buf;
! 550: xmlDocPtr cur;
! 551: PyObject *pyobj_cur;
! 552: char * encoding;
! 553:
! 554: if (!PyArg_ParseTuple(args, (char *)"OOz:xmlSaveFileTo", &pyobj_buf, &pyobj_cur, &encoding))
! 555: return(NULL);
! 556: buf = (xmlOutputBufferPtr) PyoutputBuffer_Get(pyobj_buf);
! 557: cur = (xmlDocPtr) PyxmlNode_Get(pyobj_cur);
! 558:
! 559: c_retval = xmlSaveFileTo(buf, cur, encoding);
! 560: /* xmlSaveTo() freed the memory pointed to by buf, so record that in the
! 561: * Python object. */
! 562: ((PyoutputBuffer_Object *)(pyobj_buf))->obj = NULL;
! 563: py_retval = libxml_intWrap((int) c_retval);
! 564: return(py_retval);
! 565: }
! 566:
! 567: static PyObject *
! 568: libxml_xmlSaveFormatFileTo(PyObject *self ATTRIBUTE_UNUSED, PyObject *args) {
! 569: PyObject *py_retval;
! 570: int c_retval;
! 571: xmlOutputBufferPtr buf;
! 572: PyObject *pyobj_buf;
! 573: xmlDocPtr cur;
! 574: PyObject *pyobj_cur;
! 575: char * encoding;
! 576: int format;
! 577:
! 578: if (!PyArg_ParseTuple(args, (char *)"OOzi:xmlSaveFormatFileTo", &pyobj_buf, &pyobj_cur, &encoding, &format))
! 579: return(NULL);
! 580: buf = (xmlOutputBufferPtr) PyoutputBuffer_Get(pyobj_buf);
! 581: cur = (xmlDocPtr) PyxmlNode_Get(pyobj_cur);
! 582:
! 583: c_retval = xmlSaveFormatFileTo(buf, cur, encoding, format);
! 584: /* xmlSaveFormatFileTo() freed the memory pointed to by buf, so record that
! 585: * in the Python object */
! 586: ((PyoutputBuffer_Object *)(pyobj_buf))->obj = NULL;
! 587: py_retval = libxml_intWrap((int) c_retval);
! 588: return(py_retval);
! 589: }
! 590: #endif /* LIBXML_OUTPUT_ENABLED */
! 591:
! 592:
! 593: /**
! 594: * xmlParserInputBufferCreatePythonFile:
! 595: * @file: a PyFile_Type
! 596: * @encoder: the encoding converter or NULL
! 597: *
! 598: * Create a buffered output for the progressive saving to a PyFile_Type
! 599: * buffered C I/O
! 600: *
! 601: * Returns the new parser output or NULL
! 602: */
! 603: static xmlParserInputBufferPtr
! 604: xmlParserInputBufferCreatePythonFile(PyObject *file,
! 605: xmlCharEncoding encoding) {
! 606: xmlParserInputBufferPtr ret;
! 607:
! 608: if (file == NULL) return(NULL);
! 609:
! 610: ret = xmlAllocParserInputBuffer(encoding);
! 611: if (ret != NULL) {
! 612: ret->context = file;
! 613: /* Py_INCREF(file); */
! 614: ret->readcallback = xmlPythonFileRead;
! 615: ret->closecallback = xmlPythonFileClose;
! 616: }
! 617:
! 618: return(ret);
! 619: }
! 620:
! 621: PyObject *
! 622: libxml_xmlCreateInputBuffer(ATTRIBUTE_UNUSED PyObject *self, PyObject *args) {
! 623: PyObject *py_retval;
! 624: PyObject *file;
! 625: xmlChar *encoding;
! 626: xmlCharEncoding enc = XML_CHAR_ENCODING_NONE;
! 627: xmlParserInputBufferPtr buffer;
! 628:
! 629:
! 630: if (!PyArg_ParseTuple(args, (char *)"Oz:xmlParserInputBufferCreate",
! 631: &file, &encoding))
! 632: return(NULL);
! 633: if ((encoding != NULL) && (encoding[0] != 0)) {
! 634: enc = xmlParseCharEncoding((const char *) encoding);
! 635: }
! 636: buffer = xmlParserInputBufferCreatePythonFile(file, enc);
! 637: if (buffer == NULL)
! 638: printf("libxml_xmlParserInputBufferCreate: buffer == NULL\n");
! 639: py_retval = libxml_xmlParserInputBufferPtrWrap(buffer);
! 640: return(py_retval);
! 641: }
! 642:
! 643: /************************************************************************
! 644: * *
! 645: * Providing the resolver at the Python level *
! 646: * *
! 647: ************************************************************************/
! 648:
! 649: static xmlExternalEntityLoader defaultExternalEntityLoader = NULL;
! 650: static PyObject *pythonExternalEntityLoaderObjext;
! 651:
! 652: static xmlParserInputPtr
! 653: pythonExternalEntityLoader(const char *URL, const char *ID,
! 654: xmlParserCtxtPtr ctxt) {
! 655: xmlParserInputPtr result = NULL;
! 656: if (pythonExternalEntityLoaderObjext != NULL) {
! 657: PyObject *ret;
! 658: PyObject *ctxtobj;
! 659:
! 660: ctxtobj = libxml_xmlParserCtxtPtrWrap(ctxt);
! 661: #ifdef DEBUG_LOADER
! 662: printf("pythonExternalEntityLoader: ready to call\n");
! 663: #endif
! 664:
! 665: ret = PyObject_CallFunction(pythonExternalEntityLoaderObjext,
! 666: (char *) "(ssO)", URL, ID, ctxtobj);
! 667: Py_XDECREF(ctxtobj);
! 668: #ifdef DEBUG_LOADER
! 669: printf("pythonExternalEntityLoader: result ");
! 670: PyObject_Print(ret, stderr, 0);
! 671: printf("\n");
! 672: #endif
! 673:
! 674: if (ret != NULL) {
! 675: if (PyObject_HasAttrString(ret, (char *) "read")) {
! 676: xmlParserInputBufferPtr buf;
! 677:
! 678: buf = xmlAllocParserInputBuffer(XML_CHAR_ENCODING_NONE);
! 679: if (buf != NULL) {
! 680: buf->context = ret;
! 681: buf->readcallback = xmlPythonFileReadRaw;
! 682: buf->closecallback = xmlPythonFileCloseRaw;
! 683: result = xmlNewIOInputStream(ctxt, buf,
! 684: XML_CHAR_ENCODING_NONE);
! 685: }
! 686: #if 0
! 687: } else {
! 688: if (URL != NULL)
! 689: printf("pythonExternalEntityLoader: can't read %s\n",
! 690: URL);
! 691: #endif
! 692: }
! 693: if (result == NULL) {
! 694: Py_DECREF(ret);
! 695: } else if (URL != NULL) {
! 696: result->filename = (char *) xmlStrdup((const xmlChar *)URL);
! 697: result->directory = xmlParserGetDirectory((const char *) URL);
! 698: }
! 699: }
! 700: }
! 701: if ((result == NULL) && (defaultExternalEntityLoader != NULL)) {
! 702: result = defaultExternalEntityLoader(URL, ID, ctxt);
! 703: }
! 704: return(result);
! 705: }
! 706:
! 707: PyObject *
! 708: libxml_xmlSetEntityLoader(ATTRIBUTE_UNUSED PyObject *self, PyObject *args) {
! 709: PyObject *py_retval;
! 710: PyObject *loader;
! 711:
! 712: if (!PyArg_ParseTuple(args, (char *)"O:libxml_xmlSetEntityLoader",
! 713: &loader))
! 714: return(NULL);
! 715:
! 716: #ifdef DEBUG_LOADER
! 717: printf("libxml_xmlSetEntityLoader\n");
! 718: #endif
! 719: if (defaultExternalEntityLoader == NULL)
! 720: defaultExternalEntityLoader = xmlGetExternalEntityLoader();
! 721:
! 722: pythonExternalEntityLoaderObjext = loader;
! 723: xmlSetExternalEntityLoader(pythonExternalEntityLoader);
! 724:
! 725: py_retval = PyInt_FromLong(0);
! 726: return(py_retval);
! 727: }
! 728:
! 729:
! 730: /************************************************************************
! 731: * *
! 732: * Handling SAX/xmllib/sgmlop callback interfaces *
! 733: * *
! 734: ************************************************************************/
! 735:
! 736: static void
! 737: pythonStartElement(void *user_data, const xmlChar * name,
! 738: const xmlChar ** attrs)
! 739: {
! 740: int i;
! 741: PyObject *handler;
! 742: PyObject *dict;
! 743: PyObject *attrname;
! 744: PyObject *attrvalue;
! 745: PyObject *result = NULL;
! 746: int type = 0;
! 747:
! 748: #ifdef DEBUG_SAX
! 749: printf("pythonStartElement(%s) called\n", name);
! 750: #endif
! 751: handler = (PyObject *) user_data;
! 752: if (PyObject_HasAttrString(handler, (char *) "startElement"))
! 753: type = 1;
! 754: else if (PyObject_HasAttrString(handler, (char *) "start"))
! 755: type = 2;
! 756: if (type != 0) {
! 757: /*
! 758: * the xmllib interface always generate a dictionnary,
! 759: * possibly empty
! 760: */
! 761: if ((attrs == NULL) && (type == 1)) {
! 762: Py_XINCREF(Py_None);
! 763: dict = Py_None;
! 764: } else if (attrs == NULL) {
! 765: dict = PyDict_New();
! 766: } else {
! 767: dict = PyDict_New();
! 768: for (i = 0; attrs[i] != NULL; i++) {
! 769: attrname = PyString_FromString((char *) attrs[i]);
! 770: i++;
! 771: if (attrs[i] != NULL) {
! 772: attrvalue = PyString_FromString((char *) attrs[i]);
! 773: } else {
! 774: Py_XINCREF(Py_None);
! 775: attrvalue = Py_None;
! 776: }
! 777: PyDict_SetItem(dict, attrname, attrvalue);
! 778: Py_DECREF(attrname);
! 779: Py_DECREF(attrvalue);
! 780: }
! 781: }
! 782:
! 783: if (type == 1)
! 784: result = PyObject_CallMethod(handler, (char *) "startElement",
! 785: (char *) "sO", name, dict);
! 786: else if (type == 2)
! 787: result = PyObject_CallMethod(handler, (char *) "start",
! 788: (char *) "sO", name, dict);
! 789: if (PyErr_Occurred())
! 790: PyErr_Print();
! 791: Py_XDECREF(dict);
! 792: Py_XDECREF(result);
! 793: }
! 794: }
! 795:
! 796: static void
! 797: pythonStartDocument(void *user_data)
! 798: {
! 799: PyObject *handler;
! 800: PyObject *result;
! 801:
! 802: #ifdef DEBUG_SAX
! 803: printf("pythonStartDocument() called\n");
! 804: #endif
! 805: handler = (PyObject *) user_data;
! 806: if (PyObject_HasAttrString(handler, (char *) "startDocument")) {
! 807: result =
! 808: PyObject_CallMethod(handler, (char *) "startDocument", NULL);
! 809: if (PyErr_Occurred())
! 810: PyErr_Print();
! 811: Py_XDECREF(result);
! 812: }
! 813: }
! 814:
! 815: static void
! 816: pythonEndDocument(void *user_data)
! 817: {
! 818: PyObject *handler;
! 819: PyObject *result;
! 820:
! 821: #ifdef DEBUG_SAX
! 822: printf("pythonEndDocument() called\n");
! 823: #endif
! 824: handler = (PyObject *) user_data;
! 825: if (PyObject_HasAttrString(handler, (char *) "endDocument")) {
! 826: result =
! 827: PyObject_CallMethod(handler, (char *) "endDocument", NULL);
! 828: if (PyErr_Occurred())
! 829: PyErr_Print();
! 830: Py_XDECREF(result);
! 831: }
! 832: /*
! 833: * The reference to the handler is released there
! 834: */
! 835: Py_XDECREF(handler);
! 836: }
! 837:
! 838: static void
! 839: pythonEndElement(void *user_data, const xmlChar * name)
! 840: {
! 841: PyObject *handler;
! 842: PyObject *result;
! 843:
! 844: #ifdef DEBUG_SAX
! 845: printf("pythonEndElement(%s) called\n", name);
! 846: #endif
! 847: handler = (PyObject *) user_data;
! 848: if (PyObject_HasAttrString(handler, (char *) "endElement")) {
! 849: result = PyObject_CallMethod(handler, (char *) "endElement",
! 850: (char *) "s", name);
! 851: if (PyErr_Occurred())
! 852: PyErr_Print();
! 853: Py_XDECREF(result);
! 854: } else if (PyObject_HasAttrString(handler, (char *) "end")) {
! 855: result = PyObject_CallMethod(handler, (char *) "end",
! 856: (char *) "s", name);
! 857: if (PyErr_Occurred())
! 858: PyErr_Print();
! 859: Py_XDECREF(result);
! 860: }
! 861: }
! 862:
! 863: static void
! 864: pythonReference(void *user_data, const xmlChar * name)
! 865: {
! 866: PyObject *handler;
! 867: PyObject *result;
! 868:
! 869: #ifdef DEBUG_SAX
! 870: printf("pythonReference(%s) called\n", name);
! 871: #endif
! 872: handler = (PyObject *) user_data;
! 873: if (PyObject_HasAttrString(handler, (char *) "reference")) {
! 874: result = PyObject_CallMethod(handler, (char *) "reference",
! 875: (char *) "s", name);
! 876: if (PyErr_Occurred())
! 877: PyErr_Print();
! 878: Py_XDECREF(result);
! 879: }
! 880: }
! 881:
! 882: static void
! 883: pythonCharacters(void *user_data, const xmlChar * ch, int len)
! 884: {
! 885: PyObject *handler;
! 886: PyObject *result = NULL;
! 887: int type = 0;
! 888:
! 889: #ifdef DEBUG_SAX
! 890: printf("pythonCharacters(%s, %d) called\n", ch, len);
! 891: #endif
! 892: handler = (PyObject *) user_data;
! 893: if (PyObject_HasAttrString(handler, (char *) "characters"))
! 894: type = 1;
! 895: else if (PyObject_HasAttrString(handler, (char *) "data"))
! 896: type = 2;
! 897: if (type != 0) {
! 898: if (type == 1)
! 899: result = PyObject_CallMethod(handler, (char *) "characters",
! 900: (char *) "s#", ch, len);
! 901: else if (type == 2)
! 902: result = PyObject_CallMethod(handler, (char *) "data",
! 903: (char *) "s#", ch, len);
! 904: if (PyErr_Occurred())
! 905: PyErr_Print();
! 906: Py_XDECREF(result);
! 907: }
! 908: }
! 909:
! 910: static void
! 911: pythonIgnorableWhitespace(void *user_data, const xmlChar * ch, int len)
! 912: {
! 913: PyObject *handler;
! 914: PyObject *result = NULL;
! 915: int type = 0;
! 916:
! 917: #ifdef DEBUG_SAX
! 918: printf("pythonIgnorableWhitespace(%s, %d) called\n", ch, len);
! 919: #endif
! 920: handler = (PyObject *) user_data;
! 921: if (PyObject_HasAttrString(handler, (char *) "ignorableWhitespace"))
! 922: type = 1;
! 923: else if (PyObject_HasAttrString(handler, (char *) "data"))
! 924: type = 2;
! 925: if (type != 0) {
! 926: if (type == 1)
! 927: result =
! 928: PyObject_CallMethod(handler,
! 929: (char *) "ignorableWhitespace",
! 930: (char *) "s#", ch, len);
! 931: else if (type == 2)
! 932: result =
! 933: PyObject_CallMethod(handler, (char *) "data",
! 934: (char *) "s#", ch, len);
! 935: Py_XDECREF(result);
! 936: }
! 937: }
! 938:
! 939: static void
! 940: pythonProcessingInstruction(void *user_data,
! 941: const xmlChar * target, const xmlChar * data)
! 942: {
! 943: PyObject *handler;
! 944: PyObject *result;
! 945:
! 946: #ifdef DEBUG_SAX
! 947: printf("pythonProcessingInstruction(%s, %s) called\n", target, data);
! 948: #endif
! 949: handler = (PyObject *) user_data;
! 950: if (PyObject_HasAttrString(handler, (char *) "processingInstruction")) {
! 951: result = PyObject_CallMethod(handler, (char *)
! 952: "processingInstruction",
! 953: (char *) "ss", target, data);
! 954: Py_XDECREF(result);
! 955: }
! 956: }
! 957:
! 958: static void
! 959: pythonComment(void *user_data, const xmlChar * value)
! 960: {
! 961: PyObject *handler;
! 962: PyObject *result;
! 963:
! 964: #ifdef DEBUG_SAX
! 965: printf("pythonComment(%s) called\n", value);
! 966: #endif
! 967: handler = (PyObject *) user_data;
! 968: if (PyObject_HasAttrString(handler, (char *) "comment")) {
! 969: result =
! 970: PyObject_CallMethod(handler, (char *) "comment", (char *) "s",
! 971: value);
! 972: if (PyErr_Occurred())
! 973: PyErr_Print();
! 974: Py_XDECREF(result);
! 975: }
! 976: }
! 977:
! 978: static void
! 979: pythonWarning(void *user_data, const char *msg, ...)
! 980: {
! 981: PyObject *handler;
! 982: PyObject *result;
! 983: va_list args;
! 984: char buf[1024];
! 985:
! 986: #ifdef DEBUG_SAX
! 987: printf("pythonWarning(%s) called\n", msg);
! 988: #endif
! 989: handler = (PyObject *) user_data;
! 990: if (PyObject_HasAttrString(handler, (char *) "warning")) {
! 991: va_start(args, msg);
! 992: vsnprintf(buf, 1023, msg, args);
! 993: va_end(args);
! 994: buf[1023] = 0;
! 995: result =
! 996: PyObject_CallMethod(handler, (char *) "warning", (char *) "s",
! 997: buf);
! 998: if (PyErr_Occurred())
! 999: PyErr_Print();
! 1000: Py_XDECREF(result);
! 1001: }
! 1002: }
! 1003:
! 1004: static void
! 1005: pythonError(void *user_data, const char *msg, ...)
! 1006: {
! 1007: PyObject *handler;
! 1008: PyObject *result;
! 1009: va_list args;
! 1010: char buf[1024];
! 1011:
! 1012: #ifdef DEBUG_SAX
! 1013: printf("pythonError(%s) called\n", msg);
! 1014: #endif
! 1015: handler = (PyObject *) user_data;
! 1016: if (PyObject_HasAttrString(handler, (char *) "error")) {
! 1017: va_start(args, msg);
! 1018: vsnprintf(buf, 1023, msg, args);
! 1019: va_end(args);
! 1020: buf[1023] = 0;
! 1021: result =
! 1022: PyObject_CallMethod(handler, (char *) "error", (char *) "s",
! 1023: buf);
! 1024: if (PyErr_Occurred())
! 1025: PyErr_Print();
! 1026: Py_XDECREF(result);
! 1027: }
! 1028: }
! 1029:
! 1030: static void
! 1031: pythonFatalError(void *user_data, const char *msg, ...)
! 1032: {
! 1033: PyObject *handler;
! 1034: PyObject *result;
! 1035: va_list args;
! 1036: char buf[1024];
! 1037:
! 1038: #ifdef DEBUG_SAX
! 1039: printf("pythonFatalError(%s) called\n", msg);
! 1040: #endif
! 1041: handler = (PyObject *) user_data;
! 1042: if (PyObject_HasAttrString(handler, (char *) "fatalError")) {
! 1043: va_start(args, msg);
! 1044: vsnprintf(buf, 1023, msg, args);
! 1045: va_end(args);
! 1046: buf[1023] = 0;
! 1047: result =
! 1048: PyObject_CallMethod(handler, (char *) "fatalError",
! 1049: (char *) "s", buf);
! 1050: if (PyErr_Occurred())
! 1051: PyErr_Print();
! 1052: Py_XDECREF(result);
! 1053: }
! 1054: }
! 1055:
! 1056: static void
! 1057: pythonCdataBlock(void *user_data, const xmlChar * ch, int len)
! 1058: {
! 1059: PyObject *handler;
! 1060: PyObject *result = NULL;
! 1061: int type = 0;
! 1062:
! 1063: #ifdef DEBUG_SAX
! 1064: printf("pythonCdataBlock(%s, %d) called\n", ch, len);
! 1065: #endif
! 1066: handler = (PyObject *) user_data;
! 1067: if (PyObject_HasAttrString(handler, (char *) "cdataBlock"))
! 1068: type = 1;
! 1069: else if (PyObject_HasAttrString(handler, (char *) "cdata"))
! 1070: type = 2;
! 1071: if (type != 0) {
! 1072: if (type == 1)
! 1073: result =
! 1074: PyObject_CallMethod(handler, (char *) "cdataBlock",
! 1075: (char *) "s#", ch, len);
! 1076: else if (type == 2)
! 1077: result =
! 1078: PyObject_CallMethod(handler, (char *) "cdata",
! 1079: (char *) "s#", ch, len);
! 1080: if (PyErr_Occurred())
! 1081: PyErr_Print();
! 1082: Py_XDECREF(result);
! 1083: }
! 1084: }
! 1085:
! 1086: static void
! 1087: pythonExternalSubset(void *user_data,
! 1088: const xmlChar * name,
! 1089: const xmlChar * externalID, const xmlChar * systemID)
! 1090: {
! 1091: PyObject *handler;
! 1092: PyObject *result;
! 1093:
! 1094: #ifdef DEBUG_SAX
! 1095: printf("pythonExternalSubset(%s, %s, %s) called\n",
! 1096: name, externalID, systemID);
! 1097: #endif
! 1098: handler = (PyObject *) user_data;
! 1099: if (PyObject_HasAttrString(handler, (char *) "externalSubset")) {
! 1100: result =
! 1101: PyObject_CallMethod(handler, (char *) "externalSubset",
! 1102: (char *) "sss", name, externalID,
! 1103: systemID);
! 1104: Py_XDECREF(result);
! 1105: }
! 1106: }
! 1107:
! 1108: static void
! 1109: pythonEntityDecl(void *user_data,
! 1110: const xmlChar * name,
! 1111: int type,
! 1112: const xmlChar * publicId,
! 1113: const xmlChar * systemId, xmlChar * content)
! 1114: {
! 1115: PyObject *handler;
! 1116: PyObject *result;
! 1117:
! 1118: handler = (PyObject *) user_data;
! 1119: if (PyObject_HasAttrString(handler, (char *) "entityDecl")) {
! 1120: result = PyObject_CallMethod(handler, (char *) "entityDecl",
! 1121: (char *) "sisss", name, type,
! 1122: publicId, systemId, content);
! 1123: if (PyErr_Occurred())
! 1124: PyErr_Print();
! 1125: Py_XDECREF(result);
! 1126: }
! 1127: }
! 1128:
! 1129:
! 1130:
! 1131: static void
! 1132:
! 1133: pythonNotationDecl(void *user_data,
! 1134: const xmlChar * name,
! 1135: const xmlChar * publicId, const xmlChar * systemId)
! 1136: {
! 1137: PyObject *handler;
! 1138: PyObject *result;
! 1139:
! 1140: handler = (PyObject *) user_data;
! 1141: if (PyObject_HasAttrString(handler, (char *) "notationDecl")) {
! 1142: result = PyObject_CallMethod(handler, (char *) "notationDecl",
! 1143: (char *) "sss", name, publicId,
! 1144: systemId);
! 1145: if (PyErr_Occurred())
! 1146: PyErr_Print();
! 1147: Py_XDECREF(result);
! 1148: }
! 1149: }
! 1150:
! 1151: static void
! 1152: pythonAttributeDecl(void *user_data,
! 1153: const xmlChar * elem,
! 1154: const xmlChar * name,
! 1155: int type,
! 1156: int def,
! 1157: const xmlChar * defaultValue, xmlEnumerationPtr tree)
! 1158: {
! 1159: PyObject *handler;
! 1160: PyObject *nameList;
! 1161: PyObject *newName;
! 1162: xmlEnumerationPtr node;
! 1163: PyObject *result;
! 1164: int count;
! 1165:
! 1166: handler = (PyObject *) user_data;
! 1167: if (PyObject_HasAttrString(handler, (char *) "attributeDecl")) {
! 1168: count = 0;
! 1169: for (node = tree; node != NULL; node = node->next) {
! 1170: count++;
! 1171: }
! 1172: nameList = PyList_New(count);
! 1173: count = 0;
! 1174: for (node = tree; node != NULL; node = node->next) {
! 1175: newName = PyString_FromString((char *) node->name);
! 1176: PyList_SetItem(nameList, count, newName);
! 1177: Py_DECREF(newName);
! 1178: count++;
! 1179: }
! 1180: result = PyObject_CallMethod(handler, (char *) "attributeDecl",
! 1181: (char *) "ssiisO", elem, name, type,
! 1182: def, defaultValue, nameList);
! 1183: if (PyErr_Occurred())
! 1184: PyErr_Print();
! 1185: Py_XDECREF(nameList);
! 1186: Py_XDECREF(result);
! 1187: }
! 1188: }
! 1189:
! 1190: static void
! 1191: pythonElementDecl(void *user_data,
! 1192: const xmlChar * name,
! 1193: int type, ATTRIBUTE_UNUSED xmlElementContentPtr content)
! 1194: {
! 1195: PyObject *handler;
! 1196: PyObject *obj;
! 1197: PyObject *result;
! 1198:
! 1199: handler = (PyObject *) user_data;
! 1200: if (PyObject_HasAttrString(handler, (char *) "elementDecl")) {
! 1201: /* TODO: wrap in an elementContent object */
! 1202: printf
! 1203: ("pythonElementDecl: xmlElementContentPtr wrapper missing !\n");
! 1204: obj = Py_None;
! 1205: /* Py_XINCREF(Py_None); isn't the reference just borrowed ??? */
! 1206: result = PyObject_CallMethod(handler, (char *) "elementDecl",
! 1207: (char *) "siO", name, type, obj);
! 1208: if (PyErr_Occurred())
! 1209: PyErr_Print();
! 1210: Py_XDECREF(result);
! 1211: }
! 1212: }
! 1213:
! 1214: static void
! 1215: pythonUnparsedEntityDecl(void *user_data,
! 1216: const xmlChar * name,
! 1217: const xmlChar * publicId,
! 1218: const xmlChar * systemId,
! 1219: const xmlChar * notationName)
! 1220: {
! 1221: PyObject *handler;
! 1222: PyObject *result;
! 1223:
! 1224: handler = (PyObject *) user_data;
! 1225: if (PyObject_HasAttrString(handler, (char *) "unparsedEntityDecl")) {
! 1226: result =
! 1227: PyObject_CallMethod(handler, (char *) "unparsedEntityDecl",
! 1228: (char *) "ssss", name, publicId, systemId,
! 1229: notationName);
! 1230: if (PyErr_Occurred())
! 1231: PyErr_Print();
! 1232: Py_XDECREF(result);
! 1233: }
! 1234: }
! 1235:
! 1236: static void
! 1237: pythonInternalSubset(void *user_data, const xmlChar * name,
! 1238: const xmlChar * ExternalID, const xmlChar * SystemID)
! 1239: {
! 1240: PyObject *handler;
! 1241: PyObject *result;
! 1242:
! 1243: #ifdef DEBUG_SAX
! 1244: printf("pythonInternalSubset(%s, %s, %s) called\n",
! 1245: name, ExternalID, SystemID);
! 1246: #endif
! 1247: handler = (PyObject *) user_data;
! 1248: if (PyObject_HasAttrString(handler, (char *) "internalSubset")) {
! 1249: result = PyObject_CallMethod(handler, (char *) "internalSubset",
! 1250: (char *) "sss", name, ExternalID,
! 1251: SystemID);
! 1252: if (PyErr_Occurred())
! 1253: PyErr_Print();
! 1254: Py_XDECREF(result);
! 1255: }
! 1256: }
! 1257:
! 1258: static xmlSAXHandler pythonSaxHandler = {
! 1259: pythonInternalSubset,
! 1260: NULL, /* TODO pythonIsStandalone, */
! 1261: NULL, /* TODO pythonHasInternalSubset, */
! 1262: NULL, /* TODO pythonHasExternalSubset, */
! 1263: NULL, /* TODO pythonResolveEntity, */
! 1264: NULL, /* TODO pythonGetEntity, */
! 1265: pythonEntityDecl,
! 1266: pythonNotationDecl,
! 1267: pythonAttributeDecl,
! 1268: pythonElementDecl,
! 1269: pythonUnparsedEntityDecl,
! 1270: NULL, /* OBSOLETED pythonSetDocumentLocator, */
! 1271: pythonStartDocument,
! 1272: pythonEndDocument,
! 1273: pythonStartElement,
! 1274: pythonEndElement,
! 1275: pythonReference,
! 1276: pythonCharacters,
! 1277: pythonIgnorableWhitespace,
! 1278: pythonProcessingInstruction,
! 1279: pythonComment,
! 1280: pythonWarning,
! 1281: pythonError,
! 1282: pythonFatalError,
! 1283: NULL, /* TODO pythonGetParameterEntity, */
! 1284: pythonCdataBlock,
! 1285: pythonExternalSubset,
! 1286: 1,
! 1287: NULL, /* TODO mograte to SAX2 */
! 1288: NULL,
! 1289: NULL,
! 1290: NULL
! 1291: };
! 1292:
! 1293: /************************************************************************
! 1294: * *
! 1295: * Handling of specific parser context *
! 1296: * *
! 1297: ************************************************************************/
! 1298:
! 1299: PyObject *
! 1300: libxml_xmlCreatePushParser(ATTRIBUTE_UNUSED PyObject * self,
! 1301: PyObject * args)
! 1302: {
! 1303: const char *chunk;
! 1304: int size;
! 1305: const char *URI;
! 1306: PyObject *pyobj_SAX = NULL;
! 1307: xmlSAXHandlerPtr SAX = NULL;
! 1308: xmlParserCtxtPtr ret;
! 1309: PyObject *pyret;
! 1310:
! 1311: if (!PyArg_ParseTuple
! 1312: (args, (char *) "Oziz:xmlCreatePushParser", &pyobj_SAX, &chunk,
! 1313: &size, &URI))
! 1314: return (NULL);
! 1315:
! 1316: #ifdef DEBUG
! 1317: printf("libxml_xmlCreatePushParser(%p, %s, %d, %s) called\n",
! 1318: pyobj_SAX, chunk, size, URI);
! 1319: #endif
! 1320: if (pyobj_SAX != Py_None) {
! 1321: SAX = &pythonSaxHandler;
! 1322: Py_INCREF(pyobj_SAX);
! 1323: /* The reference is released in pythonEndDocument() */
! 1324: }
! 1325: ret = xmlCreatePushParserCtxt(SAX, pyobj_SAX, chunk, size, URI);
! 1326: pyret = libxml_xmlParserCtxtPtrWrap(ret);
! 1327: return (pyret);
! 1328: }
! 1329:
! 1330: PyObject *
! 1331: libxml_htmlCreatePushParser(ATTRIBUTE_UNUSED PyObject * self,
! 1332: PyObject * args)
! 1333: {
! 1334: #ifdef LIBXML_HTML_ENABLED
! 1335: const char *chunk;
! 1336: int size;
! 1337: const char *URI;
! 1338: PyObject *pyobj_SAX = NULL;
! 1339: xmlSAXHandlerPtr SAX = NULL;
! 1340: xmlParserCtxtPtr ret;
! 1341: PyObject *pyret;
! 1342:
! 1343: if (!PyArg_ParseTuple
! 1344: (args, (char *) "Oziz:htmlCreatePushParser", &pyobj_SAX, &chunk,
! 1345: &size, &URI))
! 1346: return (NULL);
! 1347:
! 1348: #ifdef DEBUG
! 1349: printf("libxml_htmlCreatePushParser(%p, %s, %d, %s) called\n",
! 1350: pyobj_SAX, chunk, size, URI);
! 1351: #endif
! 1352: if (pyobj_SAX != Py_None) {
! 1353: SAX = &pythonSaxHandler;
! 1354: Py_INCREF(pyobj_SAX);
! 1355: /* The reference is released in pythonEndDocument() */
! 1356: }
! 1357: ret = htmlCreatePushParserCtxt(SAX, pyobj_SAX, chunk, size, URI,
! 1358: XML_CHAR_ENCODING_NONE);
! 1359: pyret = libxml_xmlParserCtxtPtrWrap(ret);
! 1360: return (pyret);
! 1361: #else
! 1362: Py_INCREF(Py_None);
! 1363: return (Py_None);
! 1364: #endif /* LIBXML_HTML_ENABLED */
! 1365: }
! 1366:
! 1367: PyObject *
! 1368: libxml_xmlSAXParseFile(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 1369: {
! 1370: int recover;
! 1371: const char *URI;
! 1372: PyObject *pyobj_SAX = NULL;
! 1373: xmlSAXHandlerPtr SAX = NULL;
! 1374:
! 1375: if (!PyArg_ParseTuple(args, (char *) "Osi:xmlSAXParseFile", &pyobj_SAX,
! 1376: &URI, &recover))
! 1377: return (NULL);
! 1378:
! 1379: #ifdef DEBUG
! 1380: printf("libxml_xmlSAXParseFile(%p, %s, %d) called\n",
! 1381: pyobj_SAX, URI, recover);
! 1382: #endif
! 1383: if (pyobj_SAX == Py_None) {
! 1384: Py_INCREF(Py_None);
! 1385: return (Py_None);
! 1386: }
! 1387: SAX = &pythonSaxHandler;
! 1388: Py_INCREF(pyobj_SAX);
! 1389: /* The reference is released in pythonEndDocument() */
! 1390: xmlSAXUserParseFile(SAX, pyobj_SAX, URI);
! 1391: Py_INCREF(Py_None);
! 1392: return (Py_None);
! 1393: }
! 1394:
! 1395: PyObject *
! 1396: libxml_htmlSAXParseFile(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 1397: {
! 1398: #ifdef LIBXML_HTML_ENABLED
! 1399: const char *URI;
! 1400: const char *encoding;
! 1401: PyObject *pyobj_SAX = NULL;
! 1402: xmlSAXHandlerPtr SAX = NULL;
! 1403:
! 1404: if (!PyArg_ParseTuple
! 1405: (args, (char *) "Osz:htmlSAXParseFile", &pyobj_SAX, &URI,
! 1406: &encoding))
! 1407: return (NULL);
! 1408:
! 1409: #ifdef DEBUG
! 1410: printf("libxml_htmlSAXParseFile(%p, %s, %s) called\n",
! 1411: pyobj_SAX, URI, encoding);
! 1412: #endif
! 1413: if (pyobj_SAX == Py_None) {
! 1414: Py_INCREF(Py_None);
! 1415: return (Py_None);
! 1416: }
! 1417: SAX = &pythonSaxHandler;
! 1418: Py_INCREF(pyobj_SAX);
! 1419: /* The reference is released in pythonEndDocument() */
! 1420: htmlSAXParseFile(URI, encoding, SAX, pyobj_SAX);
! 1421: Py_INCREF(Py_None);
! 1422: return (Py_None);
! 1423: #else
! 1424: Py_INCREF(Py_None);
! 1425: return (Py_None);
! 1426: #endif /* LIBXML_HTML_ENABLED */
! 1427: }
! 1428:
! 1429: /************************************************************************
! 1430: * *
! 1431: * Error message callback *
! 1432: * *
! 1433: ************************************************************************/
! 1434:
! 1435: static PyObject *libxml_xmlPythonErrorFuncHandler = NULL;
! 1436: static PyObject *libxml_xmlPythonErrorFuncCtxt = NULL;
! 1437:
! 1438: /* helper to build a xmlMalloc'ed string from a format and va_list */
! 1439: /*
! 1440: * disabled the loop, the repeated call to vsnprintf without reset of ap
! 1441: * in case the initial buffer was too small segfaulted on x86_64
! 1442: * we now directly vsnprintf on a large buffer.
! 1443: */
! 1444: static char *
! 1445: libxml_buildMessage(const char *msg, va_list ap)
! 1446: {
! 1447: int chars;
! 1448: char *str;
! 1449:
! 1450: str = (char *) xmlMalloc(1000);
! 1451: if (str == NULL)
! 1452: return NULL;
! 1453:
! 1454: chars = vsnprintf(str, 999, msg, ap);
! 1455: if (chars >= 998)
! 1456: str[999] = 0;
! 1457:
! 1458: return str;
! 1459: }
! 1460:
! 1461: static void
! 1462: libxml_xmlErrorFuncHandler(ATTRIBUTE_UNUSED void *ctx, const char *msg,
! 1463: ...)
! 1464: {
! 1465: va_list ap;
! 1466: PyObject *list;
! 1467: PyObject *message;
! 1468: PyObject *result;
! 1469: char str[1000];
! 1470:
! 1471: #ifdef DEBUG_ERROR
! 1472: printf("libxml_xmlErrorFuncHandler(%p, %s, ...) called\n", ctx, msg);
! 1473: #endif
! 1474:
! 1475:
! 1476: if (libxml_xmlPythonErrorFuncHandler == NULL) {
! 1477: va_start(ap, msg);
! 1478: vfprintf(stderr, msg, ap);
! 1479: va_end(ap);
! 1480: } else {
! 1481: va_start(ap, msg);
! 1482: if (vsnprintf(str, 999, msg, ap) >= 998)
! 1483: str[999] = 0;
! 1484: va_end(ap);
! 1485:
! 1486: list = PyTuple_New(2);
! 1487: PyTuple_SetItem(list, 0, libxml_xmlPythonErrorFuncCtxt);
! 1488: Py_XINCREF(libxml_xmlPythonErrorFuncCtxt);
! 1489: message = libxml_charPtrConstWrap(str);
! 1490: PyTuple_SetItem(list, 1, message);
! 1491: result = PyEval_CallObject(libxml_xmlPythonErrorFuncHandler, list);
! 1492: Py_XDECREF(list);
! 1493: Py_XDECREF(result);
! 1494: }
! 1495: }
! 1496:
! 1497: static void
! 1498: libxml_xmlErrorInitialize(void)
! 1499: {
! 1500: #ifdef DEBUG_ERROR
! 1501: printf("libxml_xmlErrorInitialize() called\n");
! 1502: #endif
! 1503: xmlSetGenericErrorFunc(NULL, libxml_xmlErrorFuncHandler);
! 1504: xmlThrDefSetGenericErrorFunc(NULL, libxml_xmlErrorFuncHandler);
! 1505: }
! 1506:
! 1507: static PyObject *
! 1508: libxml_xmlRegisterErrorHandler(ATTRIBUTE_UNUSED PyObject * self,
! 1509: PyObject * args)
! 1510: {
! 1511: PyObject *py_retval;
! 1512: PyObject *pyobj_f;
! 1513: PyObject *pyobj_ctx;
! 1514:
! 1515: if (!PyArg_ParseTuple
! 1516: (args, (char *) "OO:xmlRegisterErrorHandler", &pyobj_f,
! 1517: &pyobj_ctx))
! 1518: return (NULL);
! 1519:
! 1520: #ifdef DEBUG_ERROR
! 1521: printf("libxml_xmlRegisterErrorHandler(%p, %p) called\n", pyobj_ctx,
! 1522: pyobj_f);
! 1523: #endif
! 1524:
! 1525: if (libxml_xmlPythonErrorFuncHandler != NULL) {
! 1526: Py_XDECREF(libxml_xmlPythonErrorFuncHandler);
! 1527: }
! 1528: if (libxml_xmlPythonErrorFuncCtxt != NULL) {
! 1529: Py_XDECREF(libxml_xmlPythonErrorFuncCtxt);
! 1530: }
! 1531:
! 1532: Py_XINCREF(pyobj_ctx);
! 1533: Py_XINCREF(pyobj_f);
! 1534:
! 1535: /* TODO: check f is a function ! */
! 1536: libxml_xmlPythonErrorFuncHandler = pyobj_f;
! 1537: libxml_xmlPythonErrorFuncCtxt = pyobj_ctx;
! 1538:
! 1539: py_retval = libxml_intWrap(1);
! 1540: return (py_retval);
! 1541: }
! 1542:
! 1543:
! 1544: /************************************************************************
! 1545: * *
! 1546: * Per parserCtxt error handler *
! 1547: * *
! 1548: ************************************************************************/
! 1549:
! 1550: typedef struct
! 1551: {
! 1552: PyObject *f;
! 1553: PyObject *arg;
! 1554: } xmlParserCtxtPyCtxt;
! 1555: typedef xmlParserCtxtPyCtxt *xmlParserCtxtPyCtxtPtr;
! 1556:
! 1557: static void
! 1558: libxml_xmlParserCtxtGenericErrorFuncHandler(void *ctx, int severity, char *str)
! 1559: {
! 1560: PyObject *list;
! 1561: PyObject *result;
! 1562: xmlParserCtxtPtr ctxt;
! 1563: xmlParserCtxtPyCtxtPtr pyCtxt;
! 1564:
! 1565: #ifdef DEBUG_ERROR
! 1566: printf("libxml_xmlParserCtxtGenericErrorFuncHandler(%p, %s, ...) called\n", ctx, str);
! 1567: #endif
! 1568:
! 1569: ctxt = (xmlParserCtxtPtr)ctx;
! 1570: pyCtxt = (xmlParserCtxtPyCtxtPtr)ctxt->_private;
! 1571:
! 1572: list = PyTuple_New(4);
! 1573: PyTuple_SetItem(list, 0, pyCtxt->arg);
! 1574: Py_XINCREF(pyCtxt->arg);
! 1575: PyTuple_SetItem(list, 1, libxml_charPtrWrap(str));
! 1576: PyTuple_SetItem(list, 2, libxml_intWrap(severity));
! 1577: PyTuple_SetItem(list, 3, Py_None);
! 1578: Py_INCREF(Py_None);
! 1579: result = PyEval_CallObject(pyCtxt->f, list);
! 1580: if (result == NULL)
! 1581: {
! 1582: /* TODO: manage for the exception to be propagated... */
! 1583: PyErr_Print();
! 1584: }
! 1585: Py_XDECREF(list);
! 1586: Py_XDECREF(result);
! 1587: }
! 1588:
! 1589: static void
! 1590: libxml_xmlParserCtxtErrorFuncHandler(void *ctx, const char *msg, ...)
! 1591: {
! 1592: va_list ap;
! 1593:
! 1594: va_start(ap, msg);
! 1595: libxml_xmlParserCtxtGenericErrorFuncHandler(ctx,XML_PARSER_SEVERITY_ERROR,libxml_buildMessage(msg,ap));
! 1596: va_end(ap);
! 1597: }
! 1598:
! 1599: static void
! 1600: libxml_xmlParserCtxtWarningFuncHandler(void *ctx, const char *msg, ...)
! 1601: {
! 1602: va_list ap;
! 1603:
! 1604: va_start(ap, msg);
! 1605: libxml_xmlParserCtxtGenericErrorFuncHandler(ctx,XML_PARSER_SEVERITY_WARNING,libxml_buildMessage(msg,ap));
! 1606: va_end(ap);
! 1607: }
! 1608:
! 1609: static void
! 1610: libxml_xmlParserCtxtValidityErrorFuncHandler(void *ctx, const char *msg, ...)
! 1611: {
! 1612: va_list ap;
! 1613:
! 1614: va_start(ap, msg);
! 1615: libxml_xmlParserCtxtGenericErrorFuncHandler(ctx,XML_PARSER_SEVERITY_VALIDITY_ERROR,libxml_buildMessage(msg,ap));
! 1616: va_end(ap);
! 1617: }
! 1618:
! 1619: static void
! 1620: libxml_xmlParserCtxtValidityWarningFuncHandler(void *ctx, const char *msg, ...)
! 1621: {
! 1622: va_list ap;
! 1623:
! 1624: va_start(ap, msg);
! 1625: libxml_xmlParserCtxtGenericErrorFuncHandler(ctx,XML_PARSER_SEVERITY_VALIDITY_WARNING,libxml_buildMessage(msg,ap));
! 1626: va_end(ap);
! 1627: }
! 1628:
! 1629: static PyObject *
! 1630: libxml_xmlParserCtxtSetErrorHandler(ATTRIBUTE_UNUSED PyObject *self, PyObject *args)
! 1631: {
! 1632: PyObject *py_retval;
! 1633: xmlParserCtxtPtr ctxt;
! 1634: xmlParserCtxtPyCtxtPtr pyCtxt;
! 1635: PyObject *pyobj_ctxt;
! 1636: PyObject *pyobj_f;
! 1637: PyObject *pyobj_arg;
! 1638:
! 1639: if (!PyArg_ParseTuple(args, (char *)"OOO:xmlParserCtxtSetErrorHandler",
! 1640: &pyobj_ctxt, &pyobj_f, &pyobj_arg))
! 1641: return(NULL);
! 1642: ctxt = (xmlParserCtxtPtr) PyparserCtxt_Get(pyobj_ctxt);
! 1643: if (ctxt->_private == NULL) {
! 1644: pyCtxt = xmlMalloc(sizeof(xmlParserCtxtPyCtxt));
! 1645: if (pyCtxt == NULL) {
! 1646: py_retval = libxml_intWrap(-1);
! 1647: return(py_retval);
! 1648: }
! 1649: memset(pyCtxt,0,sizeof(xmlParserCtxtPyCtxt));
! 1650: ctxt->_private = pyCtxt;
! 1651: }
! 1652: else {
! 1653: pyCtxt = (xmlParserCtxtPyCtxtPtr)ctxt->_private;
! 1654: }
! 1655: /* TODO: check f is a function ! */
! 1656: Py_XDECREF(pyCtxt->f);
! 1657: Py_XINCREF(pyobj_f);
! 1658: pyCtxt->f = pyobj_f;
! 1659: Py_XDECREF(pyCtxt->arg);
! 1660: Py_XINCREF(pyobj_arg);
! 1661: pyCtxt->arg = pyobj_arg;
! 1662:
! 1663: if (pyobj_f != Py_None) {
! 1664: ctxt->sax->error = libxml_xmlParserCtxtErrorFuncHandler;
! 1665: ctxt->sax->warning = libxml_xmlParserCtxtWarningFuncHandler;
! 1666: ctxt->vctxt.error = libxml_xmlParserCtxtValidityErrorFuncHandler;
! 1667: ctxt->vctxt.warning = libxml_xmlParserCtxtValidityWarningFuncHandler;
! 1668: }
! 1669: else {
! 1670: ctxt->sax->error = xmlParserError;
! 1671: ctxt->vctxt.error = xmlParserValidityError;
! 1672: ctxt->sax->warning = xmlParserWarning;
! 1673: ctxt->vctxt.warning = xmlParserValidityWarning;
! 1674: }
! 1675:
! 1676: py_retval = libxml_intWrap(1);
! 1677: return(py_retval);
! 1678: }
! 1679:
! 1680: static PyObject *
! 1681: libxml_xmlParserCtxtGetErrorHandler(ATTRIBUTE_UNUSED PyObject *self, PyObject *args)
! 1682: {
! 1683: PyObject *py_retval;
! 1684: xmlParserCtxtPtr ctxt;
! 1685: xmlParserCtxtPyCtxtPtr pyCtxt;
! 1686: PyObject *pyobj_ctxt;
! 1687:
! 1688: if (!PyArg_ParseTuple(args, (char *)"O:xmlParserCtxtGetErrorHandler",
! 1689: &pyobj_ctxt))
! 1690: return(NULL);
! 1691: ctxt = (xmlParserCtxtPtr) PyparserCtxt_Get(pyobj_ctxt);
! 1692: py_retval = PyTuple_New(2);
! 1693: if (ctxt->_private != NULL) {
! 1694: pyCtxt = (xmlParserCtxtPyCtxtPtr)ctxt->_private;
! 1695:
! 1696: PyTuple_SetItem(py_retval, 0, pyCtxt->f);
! 1697: Py_XINCREF(pyCtxt->f);
! 1698: PyTuple_SetItem(py_retval, 1, pyCtxt->arg);
! 1699: Py_XINCREF(pyCtxt->arg);
! 1700: }
! 1701: else {
! 1702: /* no python error handler registered */
! 1703: PyTuple_SetItem(py_retval, 0, Py_None);
! 1704: Py_XINCREF(Py_None);
! 1705: PyTuple_SetItem(py_retval, 1, Py_None);
! 1706: Py_XINCREF(Py_None);
! 1707: }
! 1708: return(py_retval);
! 1709: }
! 1710:
! 1711: static PyObject *
! 1712: libxml_xmlFreeParserCtxt(ATTRIBUTE_UNUSED PyObject *self, PyObject *args) {
! 1713: xmlParserCtxtPtr ctxt;
! 1714: PyObject *pyobj_ctxt;
! 1715: xmlParserCtxtPyCtxtPtr pyCtxt;
! 1716:
! 1717: if (!PyArg_ParseTuple(args, (char *)"O:xmlFreeParserCtxt", &pyobj_ctxt))
! 1718: return(NULL);
! 1719: ctxt = (xmlParserCtxtPtr) PyparserCtxt_Get(pyobj_ctxt);
! 1720:
! 1721: if (ctxt != NULL) {
! 1722: pyCtxt = (xmlParserCtxtPyCtxtPtr)((xmlParserCtxtPtr)ctxt)->_private;
! 1723: if (pyCtxt) {
! 1724: Py_XDECREF(pyCtxt->f);
! 1725: Py_XDECREF(pyCtxt->arg);
! 1726: xmlFree(pyCtxt);
! 1727: }
! 1728: xmlFreeParserCtxt(ctxt);
! 1729: }
! 1730:
! 1731: Py_INCREF(Py_None);
! 1732: return(Py_None);
! 1733: }
! 1734:
! 1735: /***
! 1736: * xmlValidCtxt stuff
! 1737: */
! 1738:
! 1739: typedef struct
! 1740: {
! 1741: PyObject *warn;
! 1742: PyObject *error;
! 1743: PyObject *arg;
! 1744: } xmlValidCtxtPyCtxt;
! 1745: typedef xmlValidCtxtPyCtxt *xmlValidCtxtPyCtxtPtr;
! 1746:
! 1747: static void
! 1748: libxml_xmlValidCtxtGenericErrorFuncHandler(void *ctx, int severity, char *str)
! 1749: {
! 1750: PyObject *list;
! 1751: PyObject *result;
! 1752: xmlValidCtxtPyCtxtPtr pyCtxt;
! 1753:
! 1754: #ifdef DEBUG_ERROR
! 1755: printf("libxml_xmlValidCtxtGenericErrorFuncHandler(%p, %d, %s, ...) called\n", ctx, severity, str);
! 1756: #endif
! 1757:
! 1758: pyCtxt = (xmlValidCtxtPyCtxtPtr)ctx;
! 1759:
! 1760: list = PyTuple_New(2);
! 1761: PyTuple_SetItem(list, 0, libxml_charPtrWrap(str));
! 1762: PyTuple_SetItem(list, 1, pyCtxt->arg);
! 1763: Py_XINCREF(pyCtxt->arg);
! 1764: result = PyEval_CallObject(pyCtxt->error, list);
! 1765: if (result == NULL)
! 1766: {
! 1767: /* TODO: manage for the exception to be propagated... */
! 1768: PyErr_Print();
! 1769: }
! 1770: Py_XDECREF(list);
! 1771: Py_XDECREF(result);
! 1772: }
! 1773:
! 1774: static void
! 1775: libxml_xmlValidCtxtGenericWarningFuncHandler(void *ctx, int severity, char *str)
! 1776: {
! 1777: PyObject *list;
! 1778: PyObject *result;
! 1779: xmlValidCtxtPyCtxtPtr pyCtxt;
! 1780:
! 1781: #ifdef DEBUG_ERROR
! 1782: printf("libxml_xmlValidCtxtGenericWarningFuncHandler(%p, %d, %s, ...) called\n", ctx, severity, str);
! 1783: #endif
! 1784:
! 1785: pyCtxt = (xmlValidCtxtPyCtxtPtr)ctx;
! 1786:
! 1787: list = PyTuple_New(2);
! 1788: PyTuple_SetItem(list, 0, libxml_charPtrWrap(str));
! 1789: PyTuple_SetItem(list, 1, pyCtxt->arg);
! 1790: Py_XINCREF(pyCtxt->arg);
! 1791: result = PyEval_CallObject(pyCtxt->warn, list);
! 1792: if (result == NULL)
! 1793: {
! 1794: /* TODO: manage for the exception to be propagated... */
! 1795: PyErr_Print();
! 1796: }
! 1797: Py_XDECREF(list);
! 1798: Py_XDECREF(result);
! 1799: }
! 1800:
! 1801: static void
! 1802: libxml_xmlValidCtxtErrorFuncHandler(void *ctx, const char *msg, ...)
! 1803: {
! 1804: va_list ap;
! 1805:
! 1806: va_start(ap, msg);
! 1807: libxml_xmlValidCtxtGenericErrorFuncHandler(ctx,XML_PARSER_SEVERITY_VALIDITY_ERROR,libxml_buildMessage(msg,ap));
! 1808: va_end(ap);
! 1809: }
! 1810:
! 1811: static void
! 1812: libxml_xmlValidCtxtWarningFuncHandler(void *ctx, const char *msg, ...)
! 1813: {
! 1814: va_list ap;
! 1815:
! 1816: va_start(ap, msg);
! 1817: libxml_xmlValidCtxtGenericWarningFuncHandler(ctx,XML_PARSER_SEVERITY_VALIDITY_WARNING,libxml_buildMessage(msg,ap));
! 1818: va_end(ap);
! 1819: }
! 1820:
! 1821: static PyObject *
! 1822: libxml_xmlSetValidErrors(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 1823: {
! 1824: PyObject *py_retval;
! 1825: PyObject *pyobj_error;
! 1826: PyObject *pyobj_warn;
! 1827: PyObject *pyobj_ctx;
! 1828: PyObject *pyobj_arg = Py_None;
! 1829: xmlValidCtxtPtr ctxt;
! 1830: xmlValidCtxtPyCtxtPtr pyCtxt;
! 1831:
! 1832: if (!PyArg_ParseTuple
! 1833: (args, (char *) "OOO|O:xmlSetValidErrors", &pyobj_ctx, &pyobj_error, &pyobj_warn, &pyobj_arg))
! 1834: return (NULL);
! 1835:
! 1836: #ifdef DEBUG_ERROR
! 1837: printf("libxml_xmlSetValidErrors(%p, %p, %p) called\n", pyobj_ctx, pyobj_error, pyobj_warn);
! 1838: #endif
! 1839:
! 1840: ctxt = PyValidCtxt_Get(pyobj_ctx);
! 1841: pyCtxt = xmlMalloc(sizeof(xmlValidCtxtPyCtxt));
! 1842: if (pyCtxt == NULL) {
! 1843: py_retval = libxml_intWrap(-1);
! 1844: return(py_retval);
! 1845: }
! 1846: memset(pyCtxt, 0, sizeof(xmlValidCtxtPyCtxt));
! 1847:
! 1848:
! 1849: /* TODO: check warn and error is a function ! */
! 1850: Py_XDECREF(pyCtxt->error);
! 1851: Py_XINCREF(pyobj_error);
! 1852: pyCtxt->error = pyobj_error;
! 1853:
! 1854: Py_XDECREF(pyCtxt->warn);
! 1855: Py_XINCREF(pyobj_warn);
! 1856: pyCtxt->warn = pyobj_warn;
! 1857:
! 1858: Py_XDECREF(pyCtxt->arg);
! 1859: Py_XINCREF(pyobj_arg);
! 1860: pyCtxt->arg = pyobj_arg;
! 1861:
! 1862: ctxt->error = libxml_xmlValidCtxtErrorFuncHandler;
! 1863: ctxt->warning = libxml_xmlValidCtxtWarningFuncHandler;
! 1864: ctxt->userData = pyCtxt;
! 1865:
! 1866: py_retval = libxml_intWrap(1);
! 1867: return (py_retval);
! 1868: }
! 1869:
! 1870:
! 1871: static PyObject *
! 1872: libxml_xmlFreeValidCtxt(PyObject *self ATTRIBUTE_UNUSED, PyObject *args) {
! 1873: xmlValidCtxtPtr cur;
! 1874: xmlValidCtxtPyCtxtPtr pyCtxt;
! 1875: PyObject *pyobj_cur;
! 1876:
! 1877: if (!PyArg_ParseTuple(args, (char *)"O:xmlFreeValidCtxt", &pyobj_cur))
! 1878: return(NULL);
! 1879: cur = (xmlValidCtxtPtr) PyValidCtxt_Get(pyobj_cur);
! 1880:
! 1881: pyCtxt = (xmlValidCtxtPyCtxtPtr)(cur->userData);
! 1882: if (pyCtxt != NULL)
! 1883: {
! 1884: Py_XDECREF(pyCtxt->error);
! 1885: Py_XDECREF(pyCtxt->warn);
! 1886: Py_XDECREF(pyCtxt->arg);
! 1887: xmlFree(pyCtxt);
! 1888: }
! 1889:
! 1890: xmlFreeValidCtxt(cur);
! 1891: Py_INCREF(Py_None);
! 1892: return(Py_None);
! 1893: }
! 1894:
! 1895: #ifdef LIBXML_READER_ENABLED
! 1896: /************************************************************************
! 1897: * *
! 1898: * Per xmlTextReader error handler *
! 1899: * *
! 1900: ************************************************************************/
! 1901:
! 1902: typedef struct
! 1903: {
! 1904: PyObject *f;
! 1905: PyObject *arg;
! 1906: } xmlTextReaderPyCtxt;
! 1907: typedef xmlTextReaderPyCtxt *xmlTextReaderPyCtxtPtr;
! 1908:
! 1909: static void
! 1910: libxml_xmlTextReaderErrorCallback(void *arg,
! 1911: const char *msg,
! 1912: int severity,
! 1913: xmlTextReaderLocatorPtr locator)
! 1914: {
! 1915: xmlTextReaderPyCtxt *pyCtxt = (xmlTextReaderPyCtxt *)arg;
! 1916: PyObject *list;
! 1917: PyObject *result;
! 1918:
! 1919: list = PyTuple_New(4);
! 1920: PyTuple_SetItem(list, 0, pyCtxt->arg);
! 1921: Py_XINCREF(pyCtxt->arg);
! 1922: PyTuple_SetItem(list, 1, libxml_charPtrConstWrap(msg));
! 1923: PyTuple_SetItem(list, 2, libxml_intWrap(severity));
! 1924: PyTuple_SetItem(list, 3, libxml_xmlTextReaderLocatorPtrWrap(locator));
! 1925: result = PyEval_CallObject(pyCtxt->f, list);
! 1926: if (result == NULL)
! 1927: {
! 1928: /* TODO: manage for the exception to be propagated... */
! 1929: PyErr_Print();
! 1930: }
! 1931: Py_XDECREF(list);
! 1932: Py_XDECREF(result);
! 1933: }
! 1934:
! 1935: static PyObject *
! 1936: libxml_xmlTextReaderSetErrorHandler(ATTRIBUTE_UNUSED PyObject *self, PyObject *args)
! 1937: {
! 1938: xmlTextReaderPtr reader;
! 1939: xmlTextReaderPyCtxtPtr pyCtxt;
! 1940: xmlTextReaderErrorFunc f;
! 1941: void *arg;
! 1942: PyObject *pyobj_reader;
! 1943: PyObject *pyobj_f;
! 1944: PyObject *pyobj_arg;
! 1945: PyObject *py_retval;
! 1946:
! 1947: if (!PyArg_ParseTuple(args, (char *)"OOO:xmlTextReaderSetErrorHandler", &pyobj_reader, &pyobj_f, &pyobj_arg))
! 1948: return(NULL);
! 1949: reader = (xmlTextReaderPtr) PyxmlTextReader_Get(pyobj_reader);
! 1950: /* clear previous error handler */
! 1951: xmlTextReaderGetErrorHandler(reader,&f,&arg);
! 1952: if (arg != NULL) {
! 1953: if (f == (xmlTextReaderErrorFunc) libxml_xmlTextReaderErrorCallback) {
! 1954: /* ok, it's our error handler! */
! 1955: pyCtxt = (xmlTextReaderPyCtxtPtr)arg;
! 1956: Py_XDECREF(pyCtxt->f);
! 1957: Py_XDECREF(pyCtxt->arg);
! 1958: xmlFree(pyCtxt);
! 1959: }
! 1960: else {
! 1961: /*
! 1962: * there already an arg, and it's not ours,
! 1963: * there is definitely something wrong going on here...
! 1964: * we don't know how to free it, so we bail out...
! 1965: */
! 1966: py_retval = libxml_intWrap(-1);
! 1967: return(py_retval);
! 1968: }
! 1969: }
! 1970: xmlTextReaderSetErrorHandler(reader,NULL,NULL);
! 1971: /* set new error handler */
! 1972: if (pyobj_f != Py_None)
! 1973: {
! 1974: pyCtxt = (xmlTextReaderPyCtxtPtr)xmlMalloc(sizeof(xmlTextReaderPyCtxt));
! 1975: if (pyCtxt == NULL) {
! 1976: py_retval = libxml_intWrap(-1);
! 1977: return(py_retval);
! 1978: }
! 1979: Py_XINCREF(pyobj_f);
! 1980: pyCtxt->f = pyobj_f;
! 1981: Py_XINCREF(pyobj_arg);
! 1982: pyCtxt->arg = pyobj_arg;
! 1983: xmlTextReaderSetErrorHandler(reader,
! 1984: (xmlTextReaderErrorFunc) libxml_xmlTextReaderErrorCallback,
! 1985: pyCtxt);
! 1986: }
! 1987:
! 1988: py_retval = libxml_intWrap(1);
! 1989: return(py_retval);
! 1990: }
! 1991:
! 1992: static PyObject *
! 1993: libxml_xmlTextReaderGetErrorHandler(ATTRIBUTE_UNUSED PyObject *self, PyObject *args)
! 1994: {
! 1995: xmlTextReaderPtr reader;
! 1996: xmlTextReaderPyCtxtPtr pyCtxt;
! 1997: xmlTextReaderErrorFunc f;
! 1998: void *arg;
! 1999: PyObject *pyobj_reader;
! 2000: PyObject *py_retval;
! 2001:
! 2002: if (!PyArg_ParseTuple(args, (char *)"O:xmlTextReaderSetErrorHandler", &pyobj_reader))
! 2003: return(NULL);
! 2004: reader = (xmlTextReaderPtr) PyxmlTextReader_Get(pyobj_reader);
! 2005: xmlTextReaderGetErrorHandler(reader,&f,&arg);
! 2006: py_retval = PyTuple_New(2);
! 2007: if (f == (xmlTextReaderErrorFunc)libxml_xmlTextReaderErrorCallback) {
! 2008: /* ok, it's our error handler! */
! 2009: pyCtxt = (xmlTextReaderPyCtxtPtr)arg;
! 2010: PyTuple_SetItem(py_retval, 0, pyCtxt->f);
! 2011: Py_XINCREF(pyCtxt->f);
! 2012: PyTuple_SetItem(py_retval, 1, pyCtxt->arg);
! 2013: Py_XINCREF(pyCtxt->arg);
! 2014: }
! 2015: else
! 2016: {
! 2017: /* f is null or it's not our error handler */
! 2018: PyTuple_SetItem(py_retval, 0, Py_None);
! 2019: Py_XINCREF(Py_None);
! 2020: PyTuple_SetItem(py_retval, 1, Py_None);
! 2021: Py_XINCREF(Py_None);
! 2022: }
! 2023: return(py_retval);
! 2024: }
! 2025:
! 2026: static PyObject *
! 2027: libxml_xmlFreeTextReader(ATTRIBUTE_UNUSED PyObject *self, PyObject *args) {
! 2028: xmlTextReaderPtr reader;
! 2029: PyObject *pyobj_reader;
! 2030: xmlTextReaderPyCtxtPtr pyCtxt;
! 2031: xmlTextReaderErrorFunc f;
! 2032: void *arg;
! 2033:
! 2034: if (!PyArg_ParseTuple(args, (char *)"O:xmlFreeTextReader", &pyobj_reader))
! 2035: return(NULL);
! 2036: if (!PyCObject_Check(pyobj_reader)) {
! 2037: Py_INCREF(Py_None);
! 2038: return(Py_None);
! 2039: }
! 2040: reader = (xmlTextReaderPtr) PyxmlTextReader_Get(pyobj_reader);
! 2041: if (reader == NULL) {
! 2042: Py_INCREF(Py_None);
! 2043: return(Py_None);
! 2044: }
! 2045:
! 2046: xmlTextReaderGetErrorHandler(reader,&f,&arg);
! 2047: if (arg != NULL) {
! 2048: if (f == (xmlTextReaderErrorFunc) libxml_xmlTextReaderErrorCallback) {
! 2049: /* ok, it's our error handler! */
! 2050: pyCtxt = (xmlTextReaderPyCtxtPtr)arg;
! 2051: Py_XDECREF(pyCtxt->f);
! 2052: Py_XDECREF(pyCtxt->arg);
! 2053: xmlFree(pyCtxt);
! 2054: }
! 2055: /*
! 2056: * else, something wrong happened, because the error handler is
! 2057: * not owned by the python bindings...
! 2058: */
! 2059: }
! 2060:
! 2061: xmlFreeTextReader(reader);
! 2062: Py_INCREF(Py_None);
! 2063: return(Py_None);
! 2064: }
! 2065: #endif
! 2066:
! 2067: /************************************************************************
! 2068: * *
! 2069: * XPath extensions *
! 2070: * *
! 2071: ************************************************************************/
! 2072:
! 2073: static void
! 2074: libxml_xmlXPathFuncCallback(xmlXPathParserContextPtr ctxt, int nargs)
! 2075: {
! 2076: PyObject *list, *cur, *result;
! 2077: xmlXPathObjectPtr obj;
! 2078: xmlXPathContextPtr rctxt;
! 2079: PyObject *current_function = NULL;
! 2080: const xmlChar *name;
! 2081: const xmlChar *ns_uri;
! 2082: int i;
! 2083:
! 2084: if (ctxt == NULL)
! 2085: return;
! 2086: rctxt = ctxt->context;
! 2087: if (rctxt == NULL)
! 2088: return;
! 2089: name = rctxt->function;
! 2090: ns_uri = rctxt->functionURI;
! 2091: #ifdef DEBUG_XPATH
! 2092: printf("libxml_xmlXPathFuncCallback called name %s URI %s\n", name,
! 2093: ns_uri);
! 2094: #endif
! 2095:
! 2096: /*
! 2097: * Find the function, it should be there it was there at lookup
! 2098: */
! 2099: for (i = 0; i < libxml_xpathCallbacksNb; i++) {
! 2100: if ( /* TODO (ctxt == libxml_xpathCallbacks[i].ctx) && */
! 2101: (xmlStrEqual(name, (*libxml_xpathCallbacks)[i].name)) &&
! 2102: (xmlStrEqual(ns_uri, (*libxml_xpathCallbacks)[i].ns_uri))) {
! 2103: current_function = (*libxml_xpathCallbacks)[i].function;
! 2104: }
! 2105: }
! 2106: if (current_function == NULL) {
! 2107: printf
! 2108: ("libxml_xmlXPathFuncCallback: internal error %s not found !\n",
! 2109: name);
! 2110: return;
! 2111: }
! 2112:
! 2113: list = PyTuple_New(nargs + 1);
! 2114: PyTuple_SetItem(list, 0, libxml_xmlXPathParserContextPtrWrap(ctxt));
! 2115: for (i = nargs - 1; i >= 0; i--) {
! 2116: obj = valuePop(ctxt);
! 2117: cur = libxml_xmlXPathObjectPtrWrap(obj);
! 2118: PyTuple_SetItem(list, i + 1, cur);
! 2119: }
! 2120: result = PyEval_CallObject(current_function, list);
! 2121: Py_DECREF(list);
! 2122:
! 2123: obj = libxml_xmlXPathObjectPtrConvert(result);
! 2124: valuePush(ctxt, obj);
! 2125: }
! 2126:
! 2127: static xmlXPathFunction
! 2128: libxml_xmlXPathFuncLookupFunc(void *ctxt, const xmlChar * name,
! 2129: const xmlChar * ns_uri)
! 2130: {
! 2131: int i;
! 2132:
! 2133: #ifdef DEBUG_XPATH
! 2134: printf("libxml_xmlXPathFuncLookupFunc(%p, %s, %s) called\n",
! 2135: ctxt, name, ns_uri);
! 2136: #endif
! 2137: /*
! 2138: * This is called once only. The address is then stored in the
! 2139: * XPath expression evaluation, the proper object to call can
! 2140: * then still be found using the execution context function
! 2141: * and functionURI fields.
! 2142: */
! 2143: for (i = 0; i < libxml_xpathCallbacksNb; i++) {
! 2144: if ((ctxt == (*libxml_xpathCallbacks)[i].ctx) &&
! 2145: (xmlStrEqual(name, (*libxml_xpathCallbacks)[i].name)) &&
! 2146: (xmlStrEqual(ns_uri, (*libxml_xpathCallbacks)[i].ns_uri))) {
! 2147: return (libxml_xmlXPathFuncCallback);
! 2148: }
! 2149: }
! 2150: return (NULL);
! 2151: }
! 2152:
! 2153: static void
! 2154: libxml_xpathCallbacksInitialize(void)
! 2155: {
! 2156: int i;
! 2157:
! 2158: if (libxml_xpathCallbacksInitialized != 0)
! 2159: return;
! 2160:
! 2161: #ifdef DEBUG_XPATH
! 2162: printf("libxml_xpathCallbacksInitialized called\n");
! 2163: #endif
! 2164: libxml_xpathCallbacks = (libxml_xpathCallbackArray*)xmlMalloc(
! 2165: libxml_xpathCallbacksAllocd*sizeof(libxml_xpathCallback));
! 2166:
! 2167: for (i = 0; i < libxml_xpathCallbacksAllocd; i++) {
! 2168: (*libxml_xpathCallbacks)[i].ctx = NULL;
! 2169: (*libxml_xpathCallbacks)[i].name = NULL;
! 2170: (*libxml_xpathCallbacks)[i].ns_uri = NULL;
! 2171: (*libxml_xpathCallbacks)[i].function = NULL;
! 2172: }
! 2173: libxml_xpathCallbacksInitialized = 1;
! 2174: }
! 2175:
! 2176: PyObject *
! 2177: libxml_xmlRegisterXPathFunction(ATTRIBUTE_UNUSED PyObject * self,
! 2178: PyObject * args)
! 2179: {
! 2180: PyObject *py_retval;
! 2181: int c_retval = 0;
! 2182: xmlChar *name;
! 2183: xmlChar *ns_uri;
! 2184: xmlXPathContextPtr ctx;
! 2185: PyObject *pyobj_ctx;
! 2186: PyObject *pyobj_f;
! 2187: int i;
! 2188:
! 2189: if (!PyArg_ParseTuple
! 2190: (args, (char *) "OszO:registerXPathFunction", &pyobj_ctx, &name,
! 2191: &ns_uri, &pyobj_f))
! 2192: return (NULL);
! 2193:
! 2194: ctx = (xmlXPathContextPtr) PyxmlXPathContext_Get(pyobj_ctx);
! 2195: if (libxml_xpathCallbacksInitialized == 0)
! 2196: libxml_xpathCallbacksInitialize();
! 2197: xmlXPathRegisterFuncLookup(ctx, libxml_xmlXPathFuncLookupFunc, ctx);
! 2198:
! 2199: if ((pyobj_ctx == NULL) || (name == NULL) || (pyobj_f == NULL)) {
! 2200: py_retval = libxml_intWrap(-1);
! 2201: return (py_retval);
! 2202: }
! 2203: #ifdef DEBUG_XPATH
! 2204: printf("libxml_registerXPathFunction(%p, %s, %s) called\n",
! 2205: ctx, name, ns_uri);
! 2206: #endif
! 2207: for (i = 0; i < libxml_xpathCallbacksNb; i++) {
! 2208: if ((ctx == (*libxml_xpathCallbacks)[i].ctx) &&
! 2209: (xmlStrEqual(name, (*libxml_xpathCallbacks)[i].name)) &&
! 2210: (xmlStrEqual(ns_uri, (*libxml_xpathCallbacks)[i].ns_uri))) {
! 2211: Py_XINCREF(pyobj_f);
! 2212: Py_XDECREF((*libxml_xpathCallbacks)[i].function);
! 2213: (*libxml_xpathCallbacks)[i].function = pyobj_f;
! 2214: c_retval = 1;
! 2215: goto done;
! 2216: }
! 2217: }
! 2218: if (libxml_xpathCallbacksNb >= libxml_xpathCallbacksAllocd) {
! 2219: libxml_xpathCallbacksAllocd+=10;
! 2220: libxml_xpathCallbacks = (libxml_xpathCallbackArray*)xmlRealloc(
! 2221: libxml_xpathCallbacks,
! 2222: libxml_xpathCallbacksAllocd*sizeof(libxml_xpathCallback));
! 2223: }
! 2224: i = libxml_xpathCallbacksNb++;
! 2225: Py_XINCREF(pyobj_f);
! 2226: (*libxml_xpathCallbacks)[i].ctx = ctx;
! 2227: (*libxml_xpathCallbacks)[i].name = xmlStrdup(name);
! 2228: (*libxml_xpathCallbacks)[i].ns_uri = xmlStrdup(ns_uri);
! 2229: (*libxml_xpathCallbacks)[i].function = pyobj_f;
! 2230: c_retval = 1;
! 2231:
! 2232: done:
! 2233: py_retval = libxml_intWrap((int) c_retval);
! 2234: return (py_retval);
! 2235: }
! 2236:
! 2237: /************************************************************************
! 2238: * *
! 2239: * Global properties access *
! 2240: * *
! 2241: ************************************************************************/
! 2242: static PyObject *
! 2243: libxml_name(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2244: {
! 2245: PyObject *resultobj, *obj;
! 2246: xmlNodePtr cur;
! 2247: const xmlChar *res;
! 2248:
! 2249: if (!PyArg_ParseTuple(args, (char *) "O:name", &obj))
! 2250: return NULL;
! 2251: cur = PyxmlNode_Get(obj);
! 2252:
! 2253: #ifdef DEBUG
! 2254: printf("libxml_name: cur = %p type %d\n", cur, cur->type);
! 2255: #endif
! 2256:
! 2257: switch (cur->type) {
! 2258: case XML_DOCUMENT_NODE:
! 2259: #ifdef LIBXML_DOCB_ENABLED
! 2260: case XML_DOCB_DOCUMENT_NODE:
! 2261: #endif
! 2262: case XML_HTML_DOCUMENT_NODE:{
! 2263: xmlDocPtr doc = (xmlDocPtr) cur;
! 2264:
! 2265: res = doc->URL;
! 2266: break;
! 2267: }
! 2268: case XML_ATTRIBUTE_NODE:{
! 2269: xmlAttrPtr attr = (xmlAttrPtr) cur;
! 2270:
! 2271: res = attr->name;
! 2272: break;
! 2273: }
! 2274: case XML_NAMESPACE_DECL:{
! 2275: xmlNsPtr ns = (xmlNsPtr) cur;
! 2276:
! 2277: res = ns->prefix;
! 2278: break;
! 2279: }
! 2280: default:
! 2281: res = cur->name;
! 2282: break;
! 2283: }
! 2284: resultobj = libxml_constxmlCharPtrWrap(res);
! 2285:
! 2286: return resultobj;
! 2287: }
! 2288:
! 2289: static PyObject *
! 2290: libxml_doc(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2291: {
! 2292: PyObject *resultobj, *obj;
! 2293: xmlNodePtr cur;
! 2294: xmlDocPtr res;
! 2295:
! 2296: if (!PyArg_ParseTuple(args, (char *) "O:doc", &obj))
! 2297: return NULL;
! 2298: cur = PyxmlNode_Get(obj);
! 2299:
! 2300: #ifdef DEBUG
! 2301: printf("libxml_doc: cur = %p\n", cur);
! 2302: #endif
! 2303:
! 2304: switch (cur->type) {
! 2305: case XML_DOCUMENT_NODE:
! 2306: #ifdef LIBXML_DOCB_ENABLED
! 2307: case XML_DOCB_DOCUMENT_NODE:
! 2308: #endif
! 2309: case XML_HTML_DOCUMENT_NODE:
! 2310: res = NULL;
! 2311: break;
! 2312: case XML_ATTRIBUTE_NODE:{
! 2313: xmlAttrPtr attr = (xmlAttrPtr) cur;
! 2314:
! 2315: res = attr->doc;
! 2316: break;
! 2317: }
! 2318: case XML_NAMESPACE_DECL:
! 2319: res = NULL;
! 2320: break;
! 2321: default:
! 2322: res = cur->doc;
! 2323: break;
! 2324: }
! 2325: resultobj = libxml_xmlDocPtrWrap(res);
! 2326: return resultobj;
! 2327: }
! 2328:
! 2329: static PyObject *
! 2330: libxml_properties(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2331: {
! 2332: PyObject *resultobj, *obj;
! 2333: xmlNodePtr cur;
! 2334: xmlAttrPtr res;
! 2335:
! 2336: if (!PyArg_ParseTuple(args, (char *) "O:properties", &obj))
! 2337: return NULL;
! 2338: cur = PyxmlNode_Get(obj);
! 2339: if ((cur != NULL) && (cur->type == XML_ELEMENT_NODE))
! 2340: res = cur->properties;
! 2341: else
! 2342: res = NULL;
! 2343: resultobj = libxml_xmlAttrPtrWrap(res);
! 2344: return resultobj;
! 2345: }
! 2346:
! 2347: static PyObject *
! 2348: libxml_next(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2349: {
! 2350: PyObject *resultobj, *obj;
! 2351: xmlNodePtr cur;
! 2352: xmlNodePtr res;
! 2353:
! 2354: if (!PyArg_ParseTuple(args, (char *) "O:next", &obj))
! 2355: return NULL;
! 2356: cur = PyxmlNode_Get(obj);
! 2357:
! 2358: #ifdef DEBUG
! 2359: printf("libxml_next: cur = %p\n", cur);
! 2360: #endif
! 2361:
! 2362: switch (cur->type) {
! 2363: case XML_DOCUMENT_NODE:
! 2364: #ifdef LIBXML_DOCB_ENABLED
! 2365: case XML_DOCB_DOCUMENT_NODE:
! 2366: #endif
! 2367: case XML_HTML_DOCUMENT_NODE:
! 2368: res = NULL;
! 2369: break;
! 2370: case XML_ATTRIBUTE_NODE:{
! 2371: xmlAttrPtr attr = (xmlAttrPtr) cur;
! 2372:
! 2373: res = (xmlNodePtr) attr->next;
! 2374: break;
! 2375: }
! 2376: case XML_NAMESPACE_DECL:{
! 2377: xmlNsPtr ns = (xmlNsPtr) cur;
! 2378:
! 2379: res = (xmlNodePtr) ns->next;
! 2380: break;
! 2381: }
! 2382: default:
! 2383: res = cur->next;
! 2384: break;
! 2385:
! 2386: }
! 2387: resultobj = libxml_xmlNodePtrWrap(res);
! 2388: return resultobj;
! 2389: }
! 2390:
! 2391: static PyObject *
! 2392: libxml_prev(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2393: {
! 2394: PyObject *resultobj, *obj;
! 2395: xmlNodePtr cur;
! 2396: xmlNodePtr res;
! 2397:
! 2398: if (!PyArg_ParseTuple(args, (char *) "O:prev", &obj))
! 2399: return NULL;
! 2400: cur = PyxmlNode_Get(obj);
! 2401:
! 2402: #ifdef DEBUG
! 2403: printf("libxml_prev: cur = %p\n", cur);
! 2404: #endif
! 2405:
! 2406: switch (cur->type) {
! 2407: case XML_DOCUMENT_NODE:
! 2408: #ifdef LIBXML_DOCB_ENABLED
! 2409: case XML_DOCB_DOCUMENT_NODE:
! 2410: #endif
! 2411: case XML_HTML_DOCUMENT_NODE:
! 2412: res = NULL;
! 2413: break;
! 2414: case XML_ATTRIBUTE_NODE:{
! 2415: xmlAttrPtr attr = (xmlAttrPtr) cur;
! 2416:
! 2417: res = (xmlNodePtr) attr->prev;
! 2418: }
! 2419: break;
! 2420: case XML_NAMESPACE_DECL:
! 2421: res = NULL;
! 2422: break;
! 2423: default:
! 2424: res = cur->prev;
! 2425: break;
! 2426: }
! 2427: resultobj = libxml_xmlNodePtrWrap(res);
! 2428: return resultobj;
! 2429: }
! 2430:
! 2431: static PyObject *
! 2432: libxml_children(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2433: {
! 2434: PyObject *resultobj, *obj;
! 2435: xmlNodePtr cur;
! 2436: xmlNodePtr res;
! 2437:
! 2438: if (!PyArg_ParseTuple(args, (char *) "O:children", &obj))
! 2439: return NULL;
! 2440: cur = PyxmlNode_Get(obj);
! 2441:
! 2442: #ifdef DEBUG
! 2443: printf("libxml_children: cur = %p\n", cur);
! 2444: #endif
! 2445:
! 2446: switch (cur->type) {
! 2447: case XML_ELEMENT_NODE:
! 2448: case XML_ENTITY_REF_NODE:
! 2449: case XML_ENTITY_NODE:
! 2450: case XML_PI_NODE:
! 2451: case XML_COMMENT_NODE:
! 2452: case XML_DOCUMENT_NODE:
! 2453: #ifdef LIBXML_DOCB_ENABLED
! 2454: case XML_DOCB_DOCUMENT_NODE:
! 2455: #endif
! 2456: case XML_HTML_DOCUMENT_NODE:
! 2457: case XML_DTD_NODE:
! 2458: res = cur->children;
! 2459: break;
! 2460: case XML_ATTRIBUTE_NODE:{
! 2461: xmlAttrPtr attr = (xmlAttrPtr) cur;
! 2462:
! 2463: res = attr->children;
! 2464: break;
! 2465: }
! 2466: default:
! 2467: res = NULL;
! 2468: break;
! 2469: }
! 2470: resultobj = libxml_xmlNodePtrWrap(res);
! 2471: return resultobj;
! 2472: }
! 2473:
! 2474: static PyObject *
! 2475: libxml_last(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2476: {
! 2477: PyObject *resultobj, *obj;
! 2478: xmlNodePtr cur;
! 2479: xmlNodePtr res;
! 2480:
! 2481: if (!PyArg_ParseTuple(args, (char *) "O:last", &obj))
! 2482: return NULL;
! 2483: cur = PyxmlNode_Get(obj);
! 2484:
! 2485: #ifdef DEBUG
! 2486: printf("libxml_last: cur = %p\n", cur);
! 2487: #endif
! 2488:
! 2489: switch (cur->type) {
! 2490: case XML_ELEMENT_NODE:
! 2491: case XML_ENTITY_REF_NODE:
! 2492: case XML_ENTITY_NODE:
! 2493: case XML_PI_NODE:
! 2494: case XML_COMMENT_NODE:
! 2495: case XML_DOCUMENT_NODE:
! 2496: #ifdef LIBXML_DOCB_ENABLED
! 2497: case XML_DOCB_DOCUMENT_NODE:
! 2498: #endif
! 2499: case XML_HTML_DOCUMENT_NODE:
! 2500: case XML_DTD_NODE:
! 2501: res = cur->last;
! 2502: break;
! 2503: case XML_ATTRIBUTE_NODE:{
! 2504: xmlAttrPtr attr = (xmlAttrPtr) cur;
! 2505:
! 2506: res = attr->last;
! 2507: }
! 2508: default:
! 2509: res = NULL;
! 2510: break;
! 2511: }
! 2512: resultobj = libxml_xmlNodePtrWrap(res);
! 2513: return resultobj;
! 2514: }
! 2515:
! 2516: static PyObject *
! 2517: libxml_parent(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2518: {
! 2519: PyObject *resultobj, *obj;
! 2520: xmlNodePtr cur;
! 2521: xmlNodePtr res;
! 2522:
! 2523: if (!PyArg_ParseTuple(args, (char *) "O:parent", &obj))
! 2524: return NULL;
! 2525: cur = PyxmlNode_Get(obj);
! 2526:
! 2527: #ifdef DEBUG
! 2528: printf("libxml_parent: cur = %p\n", cur);
! 2529: #endif
! 2530:
! 2531: switch (cur->type) {
! 2532: case XML_DOCUMENT_NODE:
! 2533: case XML_HTML_DOCUMENT_NODE:
! 2534: #ifdef LIBXML_DOCB_ENABLED
! 2535: case XML_DOCB_DOCUMENT_NODE:
! 2536: #endif
! 2537: res = NULL;
! 2538: break;
! 2539: case XML_ATTRIBUTE_NODE:{
! 2540: xmlAttrPtr attr = (xmlAttrPtr) cur;
! 2541:
! 2542: res = attr->parent;
! 2543: }
! 2544: break;
! 2545: case XML_ENTITY_DECL:
! 2546: case XML_NAMESPACE_DECL:
! 2547: case XML_XINCLUDE_START:
! 2548: case XML_XINCLUDE_END:
! 2549: res = NULL;
! 2550: break;
! 2551: default:
! 2552: res = cur->parent;
! 2553: break;
! 2554: }
! 2555: resultobj = libxml_xmlNodePtrWrap(res);
! 2556: return resultobj;
! 2557: }
! 2558:
! 2559: static PyObject *
! 2560: libxml_type(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2561: {
! 2562: PyObject *resultobj, *obj;
! 2563: xmlNodePtr cur;
! 2564: const xmlChar *res = NULL;
! 2565:
! 2566: if (!PyArg_ParseTuple(args, (char *) "O:last", &obj))
! 2567: return NULL;
! 2568: cur = PyxmlNode_Get(obj);
! 2569:
! 2570: #ifdef DEBUG
! 2571: printf("libxml_type: cur = %p\n", cur);
! 2572: #endif
! 2573:
! 2574: switch (cur->type) {
! 2575: case XML_ELEMENT_NODE:
! 2576: res = (const xmlChar *) "element";
! 2577: break;
! 2578: case XML_ATTRIBUTE_NODE:
! 2579: res = (const xmlChar *) "attribute";
! 2580: break;
! 2581: case XML_TEXT_NODE:
! 2582: res = (const xmlChar *) "text";
! 2583: break;
! 2584: case XML_CDATA_SECTION_NODE:
! 2585: res = (const xmlChar *) "cdata";
! 2586: break;
! 2587: case XML_ENTITY_REF_NODE:
! 2588: res = (const xmlChar *) "entity_ref";
! 2589: break;
! 2590: case XML_ENTITY_NODE:
! 2591: res = (const xmlChar *) "entity";
! 2592: break;
! 2593: case XML_PI_NODE:
! 2594: res = (const xmlChar *) "pi";
! 2595: break;
! 2596: case XML_COMMENT_NODE:
! 2597: res = (const xmlChar *) "comment";
! 2598: break;
! 2599: case XML_DOCUMENT_NODE:
! 2600: res = (const xmlChar *) "document_xml";
! 2601: break;
! 2602: case XML_DOCUMENT_TYPE_NODE:
! 2603: res = (const xmlChar *) "doctype";
! 2604: break;
! 2605: case XML_DOCUMENT_FRAG_NODE:
! 2606: res = (const xmlChar *) "fragment";
! 2607: break;
! 2608: case XML_NOTATION_NODE:
! 2609: res = (const xmlChar *) "notation";
! 2610: break;
! 2611: case XML_HTML_DOCUMENT_NODE:
! 2612: res = (const xmlChar *) "document_html";
! 2613: break;
! 2614: case XML_DTD_NODE:
! 2615: res = (const xmlChar *) "dtd";
! 2616: break;
! 2617: case XML_ELEMENT_DECL:
! 2618: res = (const xmlChar *) "elem_decl";
! 2619: break;
! 2620: case XML_ATTRIBUTE_DECL:
! 2621: res = (const xmlChar *) "attribute_decl";
! 2622: break;
! 2623: case XML_ENTITY_DECL:
! 2624: res = (const xmlChar *) "entity_decl";
! 2625: break;
! 2626: case XML_NAMESPACE_DECL:
! 2627: res = (const xmlChar *) "namespace";
! 2628: break;
! 2629: case XML_XINCLUDE_START:
! 2630: res = (const xmlChar *) "xinclude_start";
! 2631: break;
! 2632: case XML_XINCLUDE_END:
! 2633: res = (const xmlChar *) "xinclude_end";
! 2634: break;
! 2635: #ifdef LIBXML_DOCB_ENABLED
! 2636: case XML_DOCB_DOCUMENT_NODE:
! 2637: res = (const xmlChar *) "document_docbook";
! 2638: break;
! 2639: #endif
! 2640: }
! 2641: #ifdef DEBUG
! 2642: printf("libxml_type: cur = %p: %s\n", cur, res);
! 2643: #endif
! 2644:
! 2645: resultobj = libxml_constxmlCharPtrWrap(res);
! 2646: return resultobj;
! 2647: }
! 2648:
! 2649: /************************************************************************
! 2650: * *
! 2651: * Specific accessor functions *
! 2652: * *
! 2653: ************************************************************************/
! 2654: PyObject *
! 2655: libxml_xmlNodeGetNsDefs(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2656: {
! 2657: PyObject *py_retval;
! 2658: xmlNsPtr c_retval;
! 2659: xmlNodePtr node;
! 2660: PyObject *pyobj_node;
! 2661:
! 2662: if (!PyArg_ParseTuple
! 2663: (args, (char *) "O:xmlNodeGetNsDefs", &pyobj_node))
! 2664: return (NULL);
! 2665: node = (xmlNodePtr) PyxmlNode_Get(pyobj_node);
! 2666:
! 2667: if ((node == NULL) || (node->type != XML_ELEMENT_NODE)) {
! 2668: Py_INCREF(Py_None);
! 2669: return (Py_None);
! 2670: }
! 2671: c_retval = node->nsDef;
! 2672: py_retval = libxml_xmlNsPtrWrap((xmlNsPtr) c_retval);
! 2673: return (py_retval);
! 2674: }
! 2675:
! 2676: PyObject *
! 2677: libxml_xmlNodeRemoveNsDef(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2678: {
! 2679: PyObject *py_retval;
! 2680: xmlNsPtr ns, prev;
! 2681: xmlNodePtr node;
! 2682: PyObject *pyobj_node;
! 2683: xmlChar *href;
! 2684: xmlNsPtr c_retval;
! 2685:
! 2686: if (!PyArg_ParseTuple
! 2687: (args, (char *) "Oz:xmlNodeRemoveNsDef", &pyobj_node, &href))
! 2688: return (NULL);
! 2689: node = (xmlNodePtr) PyxmlNode_Get(pyobj_node);
! 2690: ns = NULL;
! 2691:
! 2692: if ((node == NULL) || (node->type != XML_ELEMENT_NODE)) {
! 2693: Py_INCREF(Py_None);
! 2694: return (Py_None);
! 2695: }
! 2696:
! 2697: if (href == NULL) {
! 2698: ns = node->nsDef;
! 2699: node->nsDef = NULL;
! 2700: c_retval = 0;
! 2701: }
! 2702: else {
! 2703: prev = NULL;
! 2704: ns = node->nsDef;
! 2705: while (ns != NULL) {
! 2706: if (xmlStrEqual(ns->href, href)) {
! 2707: if (prev != NULL)
! 2708: prev->next = ns->next;
! 2709: else
! 2710: node->nsDef = ns->next;
! 2711: ns->next = NULL;
! 2712: c_retval = 0;
! 2713: break;
! 2714: }
! 2715: prev = ns;
! 2716: ns = ns->next;
! 2717: }
! 2718: }
! 2719:
! 2720: c_retval = ns;
! 2721: py_retval = libxml_xmlNsPtrWrap((xmlNsPtr) c_retval);
! 2722: return (py_retval);
! 2723: }
! 2724:
! 2725: PyObject *
! 2726: libxml_xmlNodeGetNs(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2727: {
! 2728: PyObject *py_retval;
! 2729: xmlNsPtr c_retval;
! 2730: xmlNodePtr node;
! 2731: PyObject *pyobj_node;
! 2732:
! 2733: if (!PyArg_ParseTuple(args, (char *) "O:xmlNodeGetNs", &pyobj_node))
! 2734: return (NULL);
! 2735: node = (xmlNodePtr) PyxmlNode_Get(pyobj_node);
! 2736:
! 2737: if ((node == NULL) ||
! 2738: ((node->type != XML_ELEMENT_NODE) &&
! 2739: (node->type != XML_ATTRIBUTE_NODE))) {
! 2740: Py_INCREF(Py_None);
! 2741: return (Py_None);
! 2742: }
! 2743: c_retval = node->ns;
! 2744: py_retval = libxml_xmlNsPtrWrap((xmlNsPtr) c_retval);
! 2745: return (py_retval);
! 2746: }
! 2747:
! 2748: #ifdef LIBXML_OUTPUT_ENABLED
! 2749: /************************************************************************
! 2750: * *
! 2751: * Serialization front-end *
! 2752: * *
! 2753: ************************************************************************/
! 2754:
! 2755: static PyObject *
! 2756: libxml_serializeNode(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2757: {
! 2758: PyObject *py_retval = NULL;
! 2759: xmlChar *c_retval;
! 2760: PyObject *pyobj_node;
! 2761: xmlNodePtr node;
! 2762: xmlDocPtr doc;
! 2763: const char *encoding;
! 2764: int format;
! 2765: int len;
! 2766: xmlSaveCtxtPtr ctxt;
! 2767: xmlBufferPtr buf;
! 2768: int options = 0;
! 2769:
! 2770: if (!PyArg_ParseTuple(args, (char *) "Ozi:serializeNode", &pyobj_node,
! 2771: &encoding, &format))
! 2772: return (NULL);
! 2773: node = (xmlNodePtr) PyxmlNode_Get(pyobj_node);
! 2774:
! 2775: if (node == NULL) {
! 2776: Py_INCREF(Py_None);
! 2777: return (Py_None);
! 2778: }
! 2779: if (node->type == XML_DOCUMENT_NODE) {
! 2780: doc = (xmlDocPtr) node;
! 2781: node = NULL;
! 2782: #ifdef LIBXML_HTML_ENABLED
! 2783: } else if (node->type == XML_HTML_DOCUMENT_NODE) {
! 2784: doc = (xmlDocPtr) node;
! 2785: node = NULL;
! 2786: #endif
! 2787: } else {
! 2788: if (node->type == XML_NAMESPACE_DECL)
! 2789: doc = NULL;
! 2790: else
! 2791: doc = node->doc;
! 2792: if ((doc == NULL) || (doc->type == XML_DOCUMENT_NODE)) {
! 2793: #ifdef LIBXML_HTML_ENABLED
! 2794: } else if (doc->type == XML_HTML_DOCUMENT_NODE) {
! 2795: #endif /* LIBXML_HTML_ENABLED */
! 2796: } else {
! 2797: Py_INCREF(Py_None);
! 2798: return (Py_None);
! 2799: }
! 2800: }
! 2801:
! 2802:
! 2803: buf = xmlBufferCreate();
! 2804: if (buf == NULL) {
! 2805: Py_INCREF(Py_None);
! 2806: return (Py_None);
! 2807: }
! 2808: if (format) options |= XML_SAVE_FORMAT;
! 2809: ctxt = xmlSaveToBuffer(buf, encoding, options);
! 2810: if (ctxt == NULL) {
! 2811: xmlBufferFree(buf);
! 2812: Py_INCREF(Py_None);
! 2813: return (Py_None);
! 2814: }
! 2815: if (node == NULL)
! 2816: xmlSaveDoc(ctxt, doc);
! 2817: else
! 2818: xmlSaveTree(ctxt, node);
! 2819: xmlSaveClose(ctxt);
! 2820:
! 2821: c_retval = buf->content;
! 2822: buf->content = NULL;
! 2823:
! 2824: xmlBufferFree(buf);
! 2825: py_retval = libxml_charPtrWrap((char *) c_retval);
! 2826:
! 2827: return (py_retval);
! 2828: }
! 2829:
! 2830: static PyObject *
! 2831: libxml_saveNodeTo(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2832: {
! 2833: PyObject *py_file = NULL;
! 2834: FILE *output;
! 2835: PyObject *pyobj_node;
! 2836: xmlNodePtr node;
! 2837: xmlDocPtr doc;
! 2838: const char *encoding;
! 2839: int format;
! 2840: int len;
! 2841: xmlOutputBufferPtr buf;
! 2842: xmlCharEncodingHandlerPtr handler = NULL;
! 2843:
! 2844: if (!PyArg_ParseTuple(args, (char *) "OOzi:serializeNode", &pyobj_node,
! 2845: &py_file, &encoding, &format))
! 2846: return (NULL);
! 2847: node = (xmlNodePtr) PyxmlNode_Get(pyobj_node);
! 2848:
! 2849: if (node == NULL) {
! 2850: return (PyInt_FromLong((long) -1));
! 2851: }
! 2852: if ((py_file == NULL) || (!(PyFile_Check(py_file)))) {
! 2853: return (PyInt_FromLong((long) -1));
! 2854: }
! 2855: output = PyFile_AsFile(py_file);
! 2856: if (output == NULL) {
! 2857: return (PyInt_FromLong((long) -1));
! 2858: }
! 2859:
! 2860: if (node->type == XML_DOCUMENT_NODE) {
! 2861: doc = (xmlDocPtr) node;
! 2862: } else if (node->type == XML_HTML_DOCUMENT_NODE) {
! 2863: doc = (xmlDocPtr) node;
! 2864: } else {
! 2865: doc = node->doc;
! 2866: }
! 2867: #ifdef LIBXML_HTML_ENABLED
! 2868: if (doc->type == XML_HTML_DOCUMENT_NODE) {
! 2869: if (encoding == NULL)
! 2870: encoding = (const char *) htmlGetMetaEncoding(doc);
! 2871: }
! 2872: #endif /* LIBXML_HTML_ENABLED */
! 2873: if (encoding != NULL) {
! 2874: handler = xmlFindCharEncodingHandler(encoding);
! 2875: if (handler == NULL) {
! 2876: return (PyInt_FromLong((long) -1));
! 2877: }
! 2878: }
! 2879: if (doc->type == XML_HTML_DOCUMENT_NODE) {
! 2880: if (handler == NULL)
! 2881: handler = xmlFindCharEncodingHandler("HTML");
! 2882: if (handler == NULL)
! 2883: handler = xmlFindCharEncodingHandler("ascii");
! 2884: }
! 2885:
! 2886: buf = xmlOutputBufferCreateFile(output, handler);
! 2887: if (node->type == XML_DOCUMENT_NODE) {
! 2888: len = xmlSaveFormatFileTo(buf, doc, encoding, format);
! 2889: #ifdef LIBXML_HTML_ENABLED
! 2890: } else if (node->type == XML_HTML_DOCUMENT_NODE) {
! 2891: htmlDocContentDumpFormatOutput(buf, doc, encoding, format);
! 2892: len = xmlOutputBufferClose(buf);
! 2893: } else if (doc->type == XML_HTML_DOCUMENT_NODE) {
! 2894: htmlNodeDumpFormatOutput(buf, doc, node, encoding, format);
! 2895: len = xmlOutputBufferClose(buf);
! 2896: #endif /* LIBXML_HTML_ENABLED */
! 2897: } else {
! 2898: xmlNodeDumpOutput(buf, doc, node, 0, format, encoding);
! 2899: len = xmlOutputBufferClose(buf);
! 2900: }
! 2901: return (PyInt_FromLong((long) len));
! 2902: }
! 2903: #endif /* LIBXML_OUTPUT_ENABLED */
! 2904:
! 2905: /************************************************************************
! 2906: * *
! 2907: * Extra stuff *
! 2908: * *
! 2909: ************************************************************************/
! 2910: PyObject *
! 2911: libxml_xmlNewNode(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2912: {
! 2913: PyObject *py_retval;
! 2914: xmlChar *name;
! 2915: xmlNodePtr node;
! 2916:
! 2917: if (!PyArg_ParseTuple(args, (char *) "s:xmlNewNode", &name))
! 2918: return (NULL);
! 2919: node = (xmlNodePtr) xmlNewNode(NULL, name);
! 2920: #ifdef DEBUG
! 2921: printf("NewNode: %s : %p\n", name, (void *) node);
! 2922: #endif
! 2923:
! 2924: if (node == NULL) {
! 2925: Py_INCREF(Py_None);
! 2926: return (Py_None);
! 2927: }
! 2928: py_retval = libxml_xmlNodePtrWrap(node);
! 2929: return (py_retval);
! 2930: }
! 2931:
! 2932:
! 2933: /************************************************************************
! 2934: * *
! 2935: * Local Catalog stuff *
! 2936: * *
! 2937: ************************************************************************/
! 2938: static PyObject *
! 2939: libxml_addLocalCatalog(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 2940: {
! 2941: xmlChar *URL;
! 2942: xmlParserCtxtPtr ctxt;
! 2943: PyObject *pyobj_ctxt;
! 2944:
! 2945: if (!PyArg_ParseTuple(args, (char *)"Os:addLocalCatalog", &pyobj_ctxt, &URL))
! 2946: return(NULL);
! 2947:
! 2948: ctxt = (xmlParserCtxtPtr) PyparserCtxt_Get(pyobj_ctxt);
! 2949:
! 2950: if (URL != NULL) {
! 2951: ctxt->catalogs = xmlCatalogAddLocal(ctxt->catalogs, URL);
! 2952: }
! 2953:
! 2954: #ifdef DEBUG
! 2955: printf("LocalCatalog: %s\n", URL);
! 2956: #endif
! 2957:
! 2958: Py_INCREF(Py_None);
! 2959: return (Py_None);
! 2960: }
! 2961:
! 2962: #ifdef LIBXML_SCHEMAS_ENABLED
! 2963:
! 2964: /************************************************************************
! 2965: * *
! 2966: * RelaxNG error handler registration *
! 2967: * *
! 2968: ************************************************************************/
! 2969:
! 2970: typedef struct
! 2971: {
! 2972: PyObject *warn;
! 2973: PyObject *error;
! 2974: PyObject *arg;
! 2975: } xmlRelaxNGValidCtxtPyCtxt;
! 2976: typedef xmlRelaxNGValidCtxtPyCtxt *xmlRelaxNGValidCtxtPyCtxtPtr;
! 2977:
! 2978: static void
! 2979: libxml_xmlRelaxNGValidityGenericErrorFuncHandler(void *ctx, char *str)
! 2980: {
! 2981: PyObject *list;
! 2982: PyObject *result;
! 2983: xmlRelaxNGValidCtxtPyCtxtPtr pyCtxt;
! 2984:
! 2985: #ifdef DEBUG_ERROR
! 2986: printf("libxml_xmlRelaxNGValidityGenericErrorFuncHandler(%p, %s, ...) called\n", ctx, str);
! 2987: #endif
! 2988:
! 2989: pyCtxt = (xmlRelaxNGValidCtxtPyCtxtPtr)ctx;
! 2990:
! 2991: list = PyTuple_New(2);
! 2992: PyTuple_SetItem(list, 0, libxml_charPtrWrap(str));
! 2993: PyTuple_SetItem(list, 1, pyCtxt->arg);
! 2994: Py_XINCREF(pyCtxt->arg);
! 2995: result = PyEval_CallObject(pyCtxt->error, list);
! 2996: if (result == NULL)
! 2997: {
! 2998: /* TODO: manage for the exception to be propagated... */
! 2999: PyErr_Print();
! 3000: }
! 3001: Py_XDECREF(list);
! 3002: Py_XDECREF(result);
! 3003: }
! 3004:
! 3005: static void
! 3006: libxml_xmlRelaxNGValidityGenericWarningFuncHandler(void *ctx, char *str)
! 3007: {
! 3008: PyObject *list;
! 3009: PyObject *result;
! 3010: xmlRelaxNGValidCtxtPyCtxtPtr pyCtxt;
! 3011:
! 3012: #ifdef DEBUG_ERROR
! 3013: printf("libxml_xmlRelaxNGValidityGenericWarningFuncHandler(%p, %s, ...) called\n", ctx, str);
! 3014: #endif
! 3015:
! 3016: pyCtxt = (xmlRelaxNGValidCtxtPyCtxtPtr)ctx;
! 3017:
! 3018: list = PyTuple_New(2);
! 3019: PyTuple_SetItem(list, 0, libxml_charPtrWrap(str));
! 3020: PyTuple_SetItem(list, 1, pyCtxt->arg);
! 3021: Py_XINCREF(pyCtxt->arg);
! 3022: result = PyEval_CallObject(pyCtxt->warn, list);
! 3023: if (result == NULL)
! 3024: {
! 3025: /* TODO: manage for the exception to be propagated... */
! 3026: PyErr_Print();
! 3027: }
! 3028: Py_XDECREF(list);
! 3029: Py_XDECREF(result);
! 3030: }
! 3031:
! 3032: static void
! 3033: libxml_xmlRelaxNGValidityErrorFunc(void *ctx, const char *msg, ...)
! 3034: {
! 3035: va_list ap;
! 3036:
! 3037: va_start(ap, msg);
! 3038: libxml_xmlRelaxNGValidityGenericErrorFuncHandler(ctx, libxml_buildMessage(msg, ap));
! 3039: va_end(ap);
! 3040: }
! 3041:
! 3042: static void
! 3043: libxml_xmlRelaxNGValidityWarningFunc(void *ctx, const char *msg, ...)
! 3044: {
! 3045: va_list ap;
! 3046:
! 3047: va_start(ap, msg);
! 3048: libxml_xmlRelaxNGValidityGenericWarningFuncHandler(ctx, libxml_buildMessage(msg, ap));
! 3049: va_end(ap);
! 3050: }
! 3051:
! 3052: static PyObject *
! 3053: libxml_xmlRelaxNGSetValidErrors(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 3054: {
! 3055: PyObject *py_retval;
! 3056: PyObject *pyobj_error;
! 3057: PyObject *pyobj_warn;
! 3058: PyObject *pyobj_ctx;
! 3059: PyObject *pyobj_arg = Py_None;
! 3060: xmlRelaxNGValidCtxtPtr ctxt;
! 3061: xmlRelaxNGValidCtxtPyCtxtPtr pyCtxt;
! 3062:
! 3063: if (!PyArg_ParseTuple
! 3064: (args, (char *) "OOO|O:xmlRelaxNGSetValidErrors", &pyobj_ctx, &pyobj_error, &pyobj_warn, &pyobj_arg))
! 3065: return (NULL);
! 3066:
! 3067: #ifdef DEBUG_ERROR
! 3068: printf("libxml_xmlRelaxNGSetValidErrors(%p, %p, %p) called\n", pyobj_ctx, pyobj_error, pyobj_warn);
! 3069: #endif
! 3070:
! 3071: ctxt = PyrelaxNgValidCtxt_Get(pyobj_ctx);
! 3072: if (xmlRelaxNGGetValidErrors(ctxt, NULL, NULL, (void **) &pyCtxt) == -1)
! 3073: {
! 3074: py_retval = libxml_intWrap(-1);
! 3075: return(py_retval);
! 3076: }
! 3077:
! 3078: if (pyCtxt == NULL)
! 3079: {
! 3080: /* first time to set the error handlers */
! 3081: pyCtxt = xmlMalloc(sizeof(xmlRelaxNGValidCtxtPyCtxt));
! 3082: if (pyCtxt == NULL) {
! 3083: py_retval = libxml_intWrap(-1);
! 3084: return(py_retval);
! 3085: }
! 3086: memset(pyCtxt, 0, sizeof(xmlRelaxNGValidCtxtPyCtxt));
! 3087: }
! 3088:
! 3089: /* TODO: check warn and error is a function ! */
! 3090: Py_XDECREF(pyCtxt->error);
! 3091: Py_XINCREF(pyobj_error);
! 3092: pyCtxt->error = pyobj_error;
! 3093:
! 3094: Py_XDECREF(pyCtxt->warn);
! 3095: Py_XINCREF(pyobj_warn);
! 3096: pyCtxt->warn = pyobj_warn;
! 3097:
! 3098: Py_XDECREF(pyCtxt->arg);
! 3099: Py_XINCREF(pyobj_arg);
! 3100: pyCtxt->arg = pyobj_arg;
! 3101:
! 3102: xmlRelaxNGSetValidErrors(ctxt, &libxml_xmlRelaxNGValidityErrorFunc, &libxml_xmlRelaxNGValidityWarningFunc, pyCtxt);
! 3103:
! 3104: py_retval = libxml_intWrap(1);
! 3105: return (py_retval);
! 3106: }
! 3107:
! 3108: static PyObject *
! 3109: libxml_xmlRelaxNGFreeValidCtxt(ATTRIBUTE_UNUSED PyObject *self, PyObject *args) {
! 3110: xmlRelaxNGValidCtxtPtr ctxt;
! 3111: xmlRelaxNGValidCtxtPyCtxtPtr pyCtxt;
! 3112: PyObject *pyobj_ctxt;
! 3113:
! 3114: if (!PyArg_ParseTuple(args, (char *)"O:xmlRelaxNGFreeValidCtxt", &pyobj_ctxt))
! 3115: return(NULL);
! 3116: ctxt = (xmlRelaxNGValidCtxtPtr) PyrelaxNgValidCtxt_Get(pyobj_ctxt);
! 3117:
! 3118: if (xmlRelaxNGGetValidErrors(ctxt, NULL, NULL, (void **) &pyCtxt) == 0)
! 3119: {
! 3120: if (pyCtxt != NULL)
! 3121: {
! 3122: Py_XDECREF(pyCtxt->error);
! 3123: Py_XDECREF(pyCtxt->warn);
! 3124: Py_XDECREF(pyCtxt->arg);
! 3125: xmlFree(pyCtxt);
! 3126: }
! 3127: }
! 3128:
! 3129: xmlRelaxNGFreeValidCtxt(ctxt);
! 3130: Py_INCREF(Py_None);
! 3131: return(Py_None);
! 3132: }
! 3133:
! 3134: typedef struct
! 3135: {
! 3136: PyObject *warn;
! 3137: PyObject *error;
! 3138: PyObject *arg;
! 3139: } xmlSchemaValidCtxtPyCtxt;
! 3140: typedef xmlSchemaValidCtxtPyCtxt *xmlSchemaValidCtxtPyCtxtPtr;
! 3141:
! 3142: static void
! 3143: libxml_xmlSchemaValidityGenericErrorFuncHandler(void *ctx, char *str)
! 3144: {
! 3145: PyObject *list;
! 3146: PyObject *result;
! 3147: xmlSchemaValidCtxtPyCtxtPtr pyCtxt;
! 3148:
! 3149: #ifdef DEBUG_ERROR
! 3150: printf("libxml_xmlSchemaValiditiyGenericErrorFuncHandler(%p, %s, ...) called\n", ctx, str);
! 3151: #endif
! 3152:
! 3153: pyCtxt = (xmlSchemaValidCtxtPyCtxtPtr) ctx;
! 3154:
! 3155: list = PyTuple_New(2);
! 3156: PyTuple_SetItem(list, 0, libxml_charPtrWrap(str));
! 3157: PyTuple_SetItem(list, 1, pyCtxt->arg);
! 3158: Py_XINCREF(pyCtxt->arg);
! 3159: result = PyEval_CallObject(pyCtxt->error, list);
! 3160: if (result == NULL)
! 3161: {
! 3162: /* TODO: manage for the exception to be propagated... */
! 3163: PyErr_Print();
! 3164: }
! 3165: Py_XDECREF(list);
! 3166: Py_XDECREF(result);
! 3167: }
! 3168:
! 3169: static void
! 3170: libxml_xmlSchemaValidityGenericWarningFuncHandler(void *ctx, char *str)
! 3171: {
! 3172: PyObject *list;
! 3173: PyObject *result;
! 3174: xmlSchemaValidCtxtPyCtxtPtr pyCtxt;
! 3175:
! 3176: #ifdef DEBUG_ERROR
! 3177: printf("libxml_xmlSchemaValidityGenericWarningFuncHandler(%p, %s, ...) called\n", ctx, str);
! 3178: #endif
! 3179:
! 3180: pyCtxt = (xmlSchemaValidCtxtPyCtxtPtr) ctx;
! 3181:
! 3182: list = PyTuple_New(2);
! 3183: PyTuple_SetItem(list, 0, libxml_charPtrWrap(str));
! 3184: PyTuple_SetItem(list, 1, pyCtxt->arg);
! 3185: Py_XINCREF(pyCtxt->arg);
! 3186: result = PyEval_CallObject(pyCtxt->warn, list);
! 3187: if (result == NULL)
! 3188: {
! 3189: /* TODO: manage for the exception to be propagated... */
! 3190: PyErr_Print();
! 3191: }
! 3192: Py_XDECREF(list);
! 3193: Py_XDECREF(result);
! 3194: }
! 3195:
! 3196: static void
! 3197: libxml_xmlSchemaValidityErrorFunc(void *ctx, const char *msg, ...)
! 3198: {
! 3199: va_list ap;
! 3200:
! 3201: va_start(ap, msg);
! 3202: libxml_xmlSchemaValidityGenericErrorFuncHandler(ctx, libxml_buildMessage(msg, ap));
! 3203: va_end(ap);
! 3204: }
! 3205:
! 3206: static void
! 3207: libxml_xmlSchemaValidityWarningFunc(void *ctx, const char *msg, ...)
! 3208: {
! 3209: va_list ap;
! 3210:
! 3211: va_start(ap, msg);
! 3212: libxml_xmlSchemaValidityGenericWarningFuncHandler(ctx, libxml_buildMessage(msg, ap));
! 3213: va_end(ap);
! 3214: }
! 3215:
! 3216: PyObject *
! 3217: libxml_xmlSchemaSetValidErrors(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 3218: {
! 3219: PyObject *py_retval;
! 3220: PyObject *pyobj_error;
! 3221: PyObject *pyobj_warn;
! 3222: PyObject *pyobj_ctx;
! 3223: PyObject *pyobj_arg = Py_None;
! 3224: xmlSchemaValidCtxtPtr ctxt;
! 3225: xmlSchemaValidCtxtPyCtxtPtr pyCtxt;
! 3226:
! 3227: if (!PyArg_ParseTuple
! 3228: (args, (char *) "OOO|O:xmlSchemaSetValidErrors", &pyobj_ctx, &pyobj_error, &pyobj_warn, &pyobj_arg))
! 3229: return (NULL);
! 3230:
! 3231: #ifdef DEBUG_ERROR
! 3232: printf("libxml_xmlSchemaSetValidErrors(%p, %p, %p) called\n", pyobj_ctx, pyobj_error, pyobj_warn);
! 3233: #endif
! 3234:
! 3235: ctxt = PySchemaValidCtxt_Get(pyobj_ctx);
! 3236: if (xmlSchemaGetValidErrors(ctxt, NULL, NULL, (void **) &pyCtxt) == -1)
! 3237: {
! 3238: py_retval = libxml_intWrap(-1);
! 3239: return(py_retval);
! 3240: }
! 3241:
! 3242: if (pyCtxt == NULL)
! 3243: {
! 3244: /* first time to set the error handlers */
! 3245: pyCtxt = xmlMalloc(sizeof(xmlSchemaValidCtxtPyCtxt));
! 3246: if (pyCtxt == NULL) {
! 3247: py_retval = libxml_intWrap(-1);
! 3248: return(py_retval);
! 3249: }
! 3250: memset(pyCtxt, 0, sizeof(xmlSchemaValidCtxtPyCtxt));
! 3251: }
! 3252:
! 3253: /* TODO: check warn and error is a function ! */
! 3254: Py_XDECREF(pyCtxt->error);
! 3255: Py_XINCREF(pyobj_error);
! 3256: pyCtxt->error = pyobj_error;
! 3257:
! 3258: Py_XDECREF(pyCtxt->warn);
! 3259: Py_XINCREF(pyobj_warn);
! 3260: pyCtxt->warn = pyobj_warn;
! 3261:
! 3262: Py_XDECREF(pyCtxt->arg);
! 3263: Py_XINCREF(pyobj_arg);
! 3264: pyCtxt->arg = pyobj_arg;
! 3265:
! 3266: xmlSchemaSetValidErrors(ctxt, &libxml_xmlSchemaValidityErrorFunc, &libxml_xmlSchemaValidityWarningFunc, pyCtxt);
! 3267:
! 3268: py_retval = libxml_intWrap(1);
! 3269: return(py_retval);
! 3270: }
! 3271:
! 3272: static PyObject *
! 3273: libxml_xmlSchemaFreeValidCtxt(ATTRIBUTE_UNUSED PyObject * self, PyObject * args)
! 3274: {
! 3275: xmlSchemaValidCtxtPtr ctxt;
! 3276: xmlSchemaValidCtxtPyCtxtPtr pyCtxt;
! 3277: PyObject *pyobj_ctxt;
! 3278:
! 3279: if (!PyArg_ParseTuple(args, (char *)"O:xmlSchemaFreeValidCtxt", &pyobj_ctxt))
! 3280: return(NULL);
! 3281: ctxt = (xmlSchemaValidCtxtPtr) PySchemaValidCtxt_Get(pyobj_ctxt);
! 3282:
! 3283: if (xmlSchemaGetValidErrors(ctxt, NULL, NULL, (void **) &pyCtxt) == 0)
! 3284: {
! 3285: if (pyCtxt != NULL)
! 3286: {
! 3287: Py_XDECREF(pyCtxt->error);
! 3288: Py_XDECREF(pyCtxt->warn);
! 3289: Py_XDECREF(pyCtxt->arg);
! 3290: xmlFree(pyCtxt);
! 3291: }
! 3292: }
! 3293:
! 3294: xmlSchemaFreeValidCtxt(ctxt);
! 3295: Py_INCREF(Py_None);
! 3296: return(Py_None);
! 3297: }
! 3298:
! 3299: #endif
! 3300:
! 3301: #ifdef LIBXML_C14N_ENABLED
! 3302: #ifdef LIBXML_OUTPUT_ENABLED
! 3303:
! 3304: /************************************************************************
! 3305: * *
! 3306: * XML Canonicalization c14n *
! 3307: * *
! 3308: ************************************************************************/
! 3309:
! 3310: static int
! 3311: PyxmlNodeSet_Convert(PyObject *py_nodeset, xmlNodeSetPtr *result)
! 3312: {
! 3313: xmlNodeSetPtr nodeSet;
! 3314: int is_tuple = 0;
! 3315:
! 3316: if (PyTuple_Check(py_nodeset))
! 3317: is_tuple = 1;
! 3318: else if (PyList_Check(py_nodeset))
! 3319: is_tuple = 0;
! 3320: else if (py_nodeset == Py_None) {
! 3321: *result = NULL;
! 3322: return 0;
! 3323: }
! 3324: else {
! 3325: PyErr_SetString(PyExc_TypeError,
! 3326: "must be a tuple or list of nodes.");
! 3327: return -1;
! 3328: }
! 3329:
! 3330: nodeSet = (xmlNodeSetPtr) xmlMalloc(sizeof(xmlNodeSet));
! 3331: if (nodeSet == NULL) {
! 3332: PyErr_SetString(PyExc_MemoryError, "");
! 3333: return -1;
! 3334: }
! 3335:
! 3336: nodeSet->nodeNr = 0;
! 3337: nodeSet->nodeMax = (is_tuple
! 3338: ? PyTuple_GET_SIZE(py_nodeset)
! 3339: : PyList_GET_SIZE(py_nodeset));
! 3340: nodeSet->nodeTab
! 3341: = (xmlNodePtr *) xmlMalloc (nodeSet->nodeMax
! 3342: * sizeof(xmlNodePtr));
! 3343: if (nodeSet->nodeTab == NULL) {
! 3344: xmlFree(nodeSet);
! 3345: PyErr_SetString(PyExc_MemoryError, "");
! 3346: return -1;
! 3347: }
! 3348: memset(nodeSet->nodeTab, 0 ,
! 3349: nodeSet->nodeMax * sizeof(xmlNodePtr));
! 3350:
! 3351: {
! 3352: int idx;
! 3353: for (idx=0; idx < nodeSet->nodeMax; ++idx) {
! 3354: xmlNodePtr pynode =
! 3355: PyxmlNode_Get (is_tuple
! 3356: ? PyTuple_GET_ITEM(py_nodeset, idx)
! 3357: : PyList_GET_ITEM(py_nodeset, idx));
! 3358: if (pynode)
! 3359: nodeSet->nodeTab[nodeSet->nodeNr++] = pynode;
! 3360: }
! 3361: }
! 3362: *result = nodeSet;
! 3363: return 0;
! 3364: }
! 3365:
! 3366: static int
! 3367: PystringSet_Convert(PyObject *py_strings, xmlChar *** result)
! 3368: {
! 3369: /* NOTE: the array should be freed, but the strings are shared
! 3370: with the python strings and so must not be freed. */
! 3371:
! 3372: xmlChar ** strings;
! 3373: int is_tuple = 0;
! 3374: int count;
! 3375: int init_index = 0;
! 3376:
! 3377: if (PyTuple_Check(py_strings))
! 3378: is_tuple = 1;
! 3379: else if (PyList_Check(py_strings))
! 3380: is_tuple = 0;
! 3381: else if (py_strings == Py_None) {
! 3382: *result = NULL;
! 3383: return 0;
! 3384: }
! 3385: else {
! 3386: PyErr_SetString(PyExc_TypeError,
! 3387: "must be a tuple or list of strings.");
! 3388: return -1;
! 3389: }
! 3390:
! 3391: count = (is_tuple
! 3392: ? PyTuple_GET_SIZE(py_strings)
! 3393: : PyList_GET_SIZE(py_strings));
! 3394:
! 3395: strings = (xmlChar **) xmlMalloc(sizeof(xmlChar *) * count);
! 3396:
! 3397: if (strings == NULL) {
! 3398: PyErr_SetString(PyExc_MemoryError, "");
! 3399: return -1;
! 3400: }
! 3401:
! 3402: memset(strings, 0 , sizeof(xmlChar *) * count);
! 3403:
! 3404: {
! 3405: int idx;
! 3406: for (idx=0; idx < count; ++idx) {
! 3407: char* s = PyString_AsString
! 3408: (is_tuple
! 3409: ? PyTuple_GET_ITEM(py_strings, idx)
! 3410: : PyList_GET_ITEM(py_strings, idx));
! 3411: if (s)
! 3412: strings[init_index++] = (xmlChar *)s;
! 3413: else {
! 3414: xmlFree(strings);
! 3415: PyErr_SetString(PyExc_TypeError,
! 3416: "must be a tuple or list of strings.");
! 3417: return -1;
! 3418: }
! 3419: }
! 3420: }
! 3421:
! 3422: *result = strings;
! 3423: return 0;
! 3424: }
! 3425:
! 3426: static PyObject *
! 3427: libxml_C14NDocDumpMemory(ATTRIBUTE_UNUSED PyObject * self,
! 3428: PyObject * args)
! 3429: {
! 3430: PyObject *py_retval = NULL;
! 3431:
! 3432: PyObject *pyobj_doc;
! 3433: PyObject *pyobj_nodes;
! 3434: int exclusive;
! 3435: PyObject *pyobj_prefixes;
! 3436: int with_comments;
! 3437:
! 3438: xmlDocPtr doc;
! 3439: xmlNodeSetPtr nodes;
! 3440: xmlChar **prefixes = NULL;
! 3441: xmlChar *doc_txt;
! 3442:
! 3443: int result;
! 3444:
! 3445: if (!PyArg_ParseTuple(args, (char *) "OOiOi:C14NDocDumpMemory",
! 3446: &pyobj_doc,
! 3447: &pyobj_nodes,
! 3448: &exclusive,
! 3449: &pyobj_prefixes,
! 3450: &with_comments))
! 3451: return (NULL);
! 3452:
! 3453: doc = (xmlDocPtr) PyxmlNode_Get(pyobj_doc);
! 3454: if (!doc) {
! 3455: PyErr_SetString(PyExc_TypeError, "bad document.");
! 3456: return NULL;
! 3457: }
! 3458:
! 3459: result = PyxmlNodeSet_Convert(pyobj_nodes, &nodes);
! 3460: if (result < 0) return NULL;
! 3461:
! 3462: if (exclusive) {
! 3463: result = PystringSet_Convert(pyobj_prefixes, &prefixes);
! 3464: if (result < 0) {
! 3465: if (nodes) {
! 3466: xmlFree(nodes->nodeTab);
! 3467: xmlFree(nodes);
! 3468: }
! 3469: return NULL;
! 3470: }
! 3471: }
! 3472:
! 3473: result = xmlC14NDocDumpMemory(doc,
! 3474: nodes,
! 3475: exclusive,
! 3476: prefixes,
! 3477: with_comments,
! 3478: &doc_txt);
! 3479:
! 3480: if (nodes) {
! 3481: xmlFree(nodes->nodeTab);
! 3482: xmlFree(nodes);
! 3483: }
! 3484: if (prefixes) {
! 3485: xmlChar ** idx = prefixes;
! 3486: while (*idx) xmlFree(*(idx++));
! 3487: xmlFree(prefixes);
! 3488: }
! 3489:
! 3490: if (result < 0) {
! 3491: PyErr_SetString(PyExc_Exception,
! 3492: "libxml2 xmlC14NDocDumpMemory failure.");
! 3493: return NULL;
! 3494: }
! 3495: else {
! 3496: py_retval = PyString_FromStringAndSize((const char *) doc_txt,
! 3497: result);
! 3498: xmlFree(doc_txt);
! 3499: return py_retval;
! 3500: }
! 3501: }
! 3502:
! 3503: static PyObject *
! 3504: libxml_C14NDocSaveTo(ATTRIBUTE_UNUSED PyObject * self,
! 3505: PyObject * args)
! 3506: {
! 3507: PyObject *pyobj_doc;
! 3508: PyObject *py_file;
! 3509: PyObject *pyobj_nodes;
! 3510: int exclusive;
! 3511: PyObject *pyobj_prefixes;
! 3512: int with_comments;
! 3513:
! 3514: xmlDocPtr doc;
! 3515: xmlNodeSetPtr nodes;
! 3516: xmlChar **prefixes = NULL;
! 3517: FILE * output;
! 3518: xmlOutputBufferPtr buf;
! 3519:
! 3520: int result;
! 3521: int len;
! 3522:
! 3523: if (!PyArg_ParseTuple(args, (char *) "OOiOiO:C14NDocSaveTo",
! 3524: &pyobj_doc,
! 3525: &pyobj_nodes,
! 3526: &exclusive,
! 3527: &pyobj_prefixes,
! 3528: &with_comments,
! 3529: &py_file))
! 3530: return (NULL);
! 3531:
! 3532: doc = (xmlDocPtr) PyxmlNode_Get(pyobj_doc);
! 3533: if (!doc) {
! 3534: PyErr_SetString(PyExc_TypeError, "bad document.");
! 3535: return NULL;
! 3536: }
! 3537:
! 3538: if ((py_file == NULL) || (!(PyFile_Check(py_file)))) {
! 3539: PyErr_SetString(PyExc_TypeError, "bad file.");
! 3540: return NULL;
! 3541: }
! 3542: output = PyFile_AsFile(py_file);
! 3543: if (output == NULL) {
! 3544: PyErr_SetString(PyExc_TypeError, "bad file.");
! 3545: return NULL;
! 3546: }
! 3547: buf = xmlOutputBufferCreateFile(output, NULL);
! 3548:
! 3549: result = PyxmlNodeSet_Convert(pyobj_nodes, &nodes);
! 3550: if (result < 0) return NULL;
! 3551:
! 3552: if (exclusive) {
! 3553: result = PystringSet_Convert(pyobj_prefixes, &prefixes);
! 3554: if (result < 0) {
! 3555: if (nodes) {
! 3556: xmlFree(nodes->nodeTab);
! 3557: xmlFree(nodes);
! 3558: }
! 3559: return NULL;
! 3560: }
! 3561: }
! 3562:
! 3563: result = xmlC14NDocSaveTo(doc,
! 3564: nodes,
! 3565: exclusive,
! 3566: prefixes,
! 3567: with_comments,
! 3568: buf);
! 3569:
! 3570: if (nodes) {
! 3571: xmlFree(nodes->nodeTab);
! 3572: xmlFree(nodes);
! 3573: }
! 3574: if (prefixes) {
! 3575: xmlChar ** idx = prefixes;
! 3576: while (*idx) xmlFree(*(idx++));
! 3577: xmlFree(prefixes);
! 3578: }
! 3579:
! 3580: len = xmlOutputBufferClose(buf);
! 3581:
! 3582: if (result < 0) {
! 3583: PyErr_SetString(PyExc_Exception,
! 3584: "libxml2 xmlC14NDocSaveTo failure.");
! 3585: return NULL;
! 3586: }
! 3587: else
! 3588: return PyInt_FromLong((long) len);
! 3589: }
! 3590:
! 3591: #endif
! 3592: #endif
! 3593:
! 3594: static PyObject *
! 3595: libxml_getObjDesc(PyObject *self ATTRIBUTE_UNUSED, PyObject *args) {
! 3596:
! 3597: PyObject *obj;
! 3598: char *str;
! 3599:
! 3600: if (!PyArg_ParseTuple(args, (char *)"O:getObjDesc", &obj))
! 3601: return NULL;
! 3602: str = PyCObject_GetDesc(obj);
! 3603: return Py_BuildValue((char *)"s", str);
! 3604: }
! 3605:
! 3606: static PyObject *
! 3607: libxml_compareNodesEqual(PyObject *self ATTRIBUTE_UNUSED, PyObject *args) {
! 3608:
! 3609: PyObject *py_node1, *py_node2;
! 3610: xmlNodePtr node1, node2;
! 3611:
! 3612: if (!PyArg_ParseTuple(args, (char *)"OO:compareNodesEqual",
! 3613: &py_node1, &py_node2))
! 3614: return NULL;
! 3615: /* To compare two node objects, we compare their pointer addresses */
! 3616: node1 = PyxmlNode_Get(py_node1);
! 3617: node2 = PyxmlNode_Get(py_node2);
! 3618: if ( node1 == node2 )
! 3619: return Py_BuildValue((char *)"i", 1);
! 3620: else
! 3621: return Py_BuildValue((char *)"i", 0);
! 3622:
! 3623: }
! 3624:
! 3625: static PyObject *
! 3626: libxml_nodeHash(PyObject *self ATTRIBUTE_UNUSED, PyObject *args) {
! 3627:
! 3628: PyObject *py_node1;
! 3629: xmlNodePtr node1;
! 3630:
! 3631: if (!PyArg_ParseTuple(args, (char *)"O:nodeHash", &py_node1))
! 3632: return NULL;
! 3633: /* For simplicity, we use the node pointer address as a hash value */
! 3634: node1 = PyxmlNode_Get(py_node1);
! 3635:
! 3636: return PyLong_FromVoidPtr(node1);
! 3637:
! 3638: }
! 3639:
! 3640: /************************************************************************
! 3641: * *
! 3642: * The registration stuff *
! 3643: * *
! 3644: ************************************************************************/
! 3645: static PyMethodDef libxmlMethods[] = {
! 3646: #include "libxml2-export.c"
! 3647: {(char *) "name", libxml_name, METH_VARARGS, NULL},
! 3648: {(char *) "children", libxml_children, METH_VARARGS, NULL},
! 3649: {(char *) "properties", libxml_properties, METH_VARARGS, NULL},
! 3650: {(char *) "last", libxml_last, METH_VARARGS, NULL},
! 3651: {(char *) "prev", libxml_prev, METH_VARARGS, NULL},
! 3652: {(char *) "next", libxml_next, METH_VARARGS, NULL},
! 3653: {(char *) "parent", libxml_parent, METH_VARARGS, NULL},
! 3654: {(char *) "type", libxml_type, METH_VARARGS, NULL},
! 3655: {(char *) "doc", libxml_doc, METH_VARARGS, NULL},
! 3656: {(char *) "xmlNewNode", libxml_xmlNewNode, METH_VARARGS, NULL},
! 3657: {(char *) "xmlNodeRemoveNsDef", libxml_xmlNodeRemoveNsDef, METH_VARARGS, NULL},
! 3658: {(char *)"xmlSetValidErrors", libxml_xmlSetValidErrors, METH_VARARGS, NULL},
! 3659: {(char *)"xmlFreeValidCtxt", libxml_xmlFreeValidCtxt, METH_VARARGS, NULL},
! 3660: #ifdef LIBXML_OUTPUT_ENABLED
! 3661: {(char *) "serializeNode", libxml_serializeNode, METH_VARARGS, NULL},
! 3662: {(char *) "saveNodeTo", libxml_saveNodeTo, METH_VARARGS, NULL},
! 3663: {(char *) "outputBufferCreate", libxml_xmlCreateOutputBuffer, METH_VARARGS, NULL},
! 3664: {(char *) "outputBufferGetPythonFile", libxml_outputBufferGetPythonFile, METH_VARARGS, NULL},
! 3665: {(char *) "xmlOutputBufferClose", libxml_xmlOutputBufferClose, METH_VARARGS, NULL},
! 3666: { (char *)"xmlOutputBufferFlush", libxml_xmlOutputBufferFlush, METH_VARARGS, NULL },
! 3667: { (char *)"xmlSaveFileTo", libxml_xmlSaveFileTo, METH_VARARGS, NULL },
! 3668: { (char *)"xmlSaveFormatFileTo", libxml_xmlSaveFormatFileTo, METH_VARARGS, NULL },
! 3669: #endif /* LIBXML_OUTPUT_ENABLED */
! 3670: {(char *) "inputBufferCreate", libxml_xmlCreateInputBuffer, METH_VARARGS, NULL},
! 3671: {(char *) "setEntityLoader", libxml_xmlSetEntityLoader, METH_VARARGS, NULL},
! 3672: {(char *)"xmlRegisterErrorHandler", libxml_xmlRegisterErrorHandler, METH_VARARGS, NULL },
! 3673: {(char *)"xmlParserCtxtSetErrorHandler", libxml_xmlParserCtxtSetErrorHandler, METH_VARARGS, NULL },
! 3674: {(char *)"xmlParserCtxtGetErrorHandler", libxml_xmlParserCtxtGetErrorHandler, METH_VARARGS, NULL },
! 3675: {(char *)"xmlFreeParserCtxt", libxml_xmlFreeParserCtxt, METH_VARARGS, NULL },
! 3676: #ifdef LIBXML_READER_ENABLED
! 3677: {(char *)"xmlTextReaderSetErrorHandler", libxml_xmlTextReaderSetErrorHandler, METH_VARARGS, NULL },
! 3678: {(char *)"xmlTextReaderGetErrorHandler", libxml_xmlTextReaderGetErrorHandler, METH_VARARGS, NULL },
! 3679: {(char *)"xmlFreeTextReader", libxml_xmlFreeTextReader, METH_VARARGS, NULL },
! 3680: #endif
! 3681: {(char *)"addLocalCatalog", libxml_addLocalCatalog, METH_VARARGS, NULL },
! 3682: #ifdef LIBXML_SCHEMAS_ENABLED
! 3683: {(char *)"xmlRelaxNGSetValidErrors", libxml_xmlRelaxNGSetValidErrors, METH_VARARGS, NULL},
! 3684: {(char *)"xmlRelaxNGFreeValidCtxt", libxml_xmlRelaxNGFreeValidCtxt, METH_VARARGS, NULL},
! 3685: {(char *)"xmlSchemaSetValidErrors", libxml_xmlSchemaSetValidErrors, METH_VARARGS, NULL},
! 3686: {(char *)"xmlSchemaFreeValidCtxt", libxml_xmlSchemaFreeValidCtxt, METH_VARARGS, NULL},
! 3687: #endif
! 3688: #ifdef LIBXML_C14N_ENABLED
! 3689: #ifdef LIBXML_OUTPUT_ENABLED
! 3690: {(char *)"xmlC14NDocDumpMemory", libxml_C14NDocDumpMemory, METH_VARARGS, NULL},
! 3691: {(char *)"xmlC14NDocSaveTo", libxml_C14NDocSaveTo, METH_VARARGS, NULL},
! 3692: #endif
! 3693: #endif
! 3694: {(char *) "getObjDesc", libxml_getObjDesc, METH_VARARGS, NULL},
! 3695: {(char *) "compareNodesEqual", libxml_compareNodesEqual, METH_VARARGS, NULL},
! 3696: {(char *) "nodeHash", libxml_nodeHash, METH_VARARGS, NULL},
! 3697: {NULL, NULL, 0, NULL}
! 3698: };
! 3699:
! 3700: #ifdef MERGED_MODULES
! 3701: extern void initlibxsltmod(void);
! 3702: #endif
! 3703:
! 3704: void
! 3705: initlibxml2mod(void)
! 3706: {
! 3707: static int initialized = 0;
! 3708:
! 3709: if (initialized != 0)
! 3710: return;
! 3711:
! 3712: /* intialize the python extension module */
! 3713: Py_InitModule((char *) "libxml2mod", libxmlMethods);
! 3714:
! 3715: /* initialize libxml2 */
! 3716: xmlInitParser();
! 3717: libxml_xmlErrorInitialize();
! 3718:
! 3719: initialized = 1;
! 3720:
! 3721: #ifdef MERGED_MODULES
! 3722: initlibxsltmod();
! 3723: #endif
! 3724: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>