Commit 19749fb0 authored by Peter Eisentraut's avatar Peter Eisentraut

Replace xmlroot with a properly functioning version that parses the value,

sets the items, and serializes the value back (rather than adding an
arbitrary number of XML preambles as before).

The libxml memory management via palloc had to be disabled because it
crashes when libxml tries to access memory that was helpfully freed
earlier by PostgreSQL.  This needs further thought.
parent 063560bb
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2007, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2007, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/backend/utils/adt/xml.c,v 1.10 2007/01/05 22:19:42 momjian Exp $ * $PostgreSQL: pgsql/src/backend/utils/adt/xml.c,v 1.11 2007/01/06 19:18:36 petere Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -31,6 +31,7 @@ ...@@ -31,6 +31,7 @@
#include <libxml/tree.h> #include <libxml/tree.h>
#include <libxml/uri.h> #include <libxml/uri.h>
#include <libxml/xmlerror.h> #include <libxml/xmlerror.h>
#include <libxml/xmlsave.h>
#endif /* USE_LIBXML */ #endif /* USE_LIBXML */
#include "fmgr.h" #include "fmgr.h"
...@@ -49,10 +50,12 @@ ...@@ -49,10 +50,12 @@
static StringInfo xml_err_buf = NULL; static StringInfo xml_err_buf = NULL;
static void xml_init(void); static void xml_init(void);
#ifdef NOT_USED
static void *xml_palloc(size_t size); static void *xml_palloc(size_t size);
static void *xml_repalloc(void *ptr, size_t size); static void *xml_repalloc(void *ptr, size_t size);
static void xml_pfree(void *ptr); static void xml_pfree(void *ptr);
static char *xml_pstrdup(const char *string); static char *xml_pstrdup(const char *string);
#endif
static void xml_ereport(int level, int sqlcode, static void xml_ereport(int level, int sqlcode,
const char *msg, void *ctxt); const char *msg, void *ctxt);
static void xml_errorHandler(void *ctxt, const char *msg, ...); static void xml_errorHandler(void *ctxt, const char *msg, ...);
...@@ -76,6 +79,7 @@ xml_in(PG_FUNCTION_ARGS) ...@@ -76,6 +79,7 @@ xml_in(PG_FUNCTION_ARGS)
char *s = PG_GETARG_CSTRING(0); char *s = PG_GETARG_CSTRING(0);
size_t len; size_t len;
xmltype *vardata; xmltype *vardata;
xmlDocPtr doc;
len = strlen(s); len = strlen(s);
vardata = palloc(len + VARHDRSZ); vardata = palloc(len + VARHDRSZ);
...@@ -86,7 +90,8 @@ xml_in(PG_FUNCTION_ARGS) ...@@ -86,7 +90,8 @@ xml_in(PG_FUNCTION_ARGS)
* Parse the data to check if it is well-formed XML data. Assume * Parse the data to check if it is well-formed XML data. Assume
* that ERROR occurred if parsing failed. * that ERROR occurred if parsing failed.
*/ */
xml_parse(vardata, false, true); doc = xml_parse(vardata, false, true);
xmlFreeDoc(doc);
PG_RETURN_XML_P(vardata); PG_RETURN_XML_P(vardata);
#else #else
...@@ -120,6 +125,7 @@ xml_recv(PG_FUNCTION_ARGS) ...@@ -120,6 +125,7 @@ xml_recv(PG_FUNCTION_ARGS)
xmltype *result; xmltype *result;
char *str; char *str;
int nbytes; int nbytes;
xmlDocPtr doc;
str = pq_getmsgtext(buf, buf->len - buf->cursor, &nbytes); str = pq_getmsgtext(buf, buf->len - buf->cursor, &nbytes);
...@@ -132,7 +138,8 @@ xml_recv(PG_FUNCTION_ARGS) ...@@ -132,7 +138,8 @@ xml_recv(PG_FUNCTION_ARGS)
* Parse the data to check if it is well-formed XML data. Assume * Parse the data to check if it is well-formed XML data. Assume
* that ERROR occurred if parsing failed. * that ERROR occurred if parsing failed.
*/ */
xml_parse(result, false, true); doc = xml_parse(result, false, true);
xmlFreeDoc(doc);
PG_RETURN_XML_P(result); PG_RETURN_XML_P(result);
#else #else
...@@ -175,6 +182,21 @@ stringinfo_to_xmltype(StringInfo buf) ...@@ -175,6 +182,21 @@ stringinfo_to_xmltype(StringInfo buf)
return result; return result;
} }
static xmltype *
xmlBuffer_to_xmltype(xmlBufferPtr buf)
{
int32 len;
xmltype *result;
len = xmlBufferLength(buf) + VARHDRSZ;
result = palloc(len);
VARATT_SIZEP(result) = len;
memcpy(VARDATA(result), xmlBufferContent(buf), len - VARHDRSZ);
return result;
}
#endif #endif
...@@ -221,7 +243,10 @@ xmltype * ...@@ -221,7 +243,10 @@ xmltype *
xmlparse(text *data, bool is_document, bool preserve_whitespace) xmlparse(text *data, bool is_document, bool preserve_whitespace)
{ {
#ifdef USE_LIBXML #ifdef USE_LIBXML
xml_parse(data, is_document, preserve_whitespace); xmlDocPtr doc;
doc = xml_parse(data, is_document, preserve_whitespace);
xmlFreeDoc(doc);
return (xmltype *) data; return (xmltype *) data;
#else #else
...@@ -280,31 +305,38 @@ xmltype * ...@@ -280,31 +305,38 @@ xmltype *
xmlroot(xmltype *data, text *version, int standalone) xmlroot(xmltype *data, text *version, int standalone)
{ {
#ifdef USE_LIBXML #ifdef USE_LIBXML
xmltype *result; xmlDocPtr doc;
StringInfoData buf; xmlBufferPtr buffer;
xmlSaveCtxtPtr save;
initStringInfo(&buf); doc = xml_parse((text *) data, true, true);
/*
* FIXME: This is probably supposed to be cleverer if there
* already is an XML preamble.
*/
appendStringInfo(&buf,"<?xml");
if (version) if (version)
doc->version = xmlStrdup(xml_text2xmlChar(version));
else
doc->version = NULL;
switch (standalone)
{ {
appendStringInfo(&buf, " version=\""); case 1:
appendStringInfoText(&buf, version); doc->standalone = 1;
appendStringInfo(&buf, "\""); break;
case -1:
doc->standalone = 0;
break;
default:
doc->standalone = -1;
break;
} }
if (standalone)
appendStringInfo(&buf, " standalone=\"%s\"",
(standalone == 1 ? "yes" : "no"));
appendStringInfo(&buf, "?>");
appendStringInfoText(&buf, (text *) data);
result = stringinfo_to_xmltype(&buf); buffer = xmlBufferCreate();
pfree(buf.data); save = xmlSaveToBuffer(buffer, NULL, 0);
return result; xmlSaveDoc(save, doc);
xmlSaveClose(save);
xmlFreeDoc(doc);
return xmlBuffer_to_xmltype(buffer);
#else #else
NO_XML_SUPPORT(); NO_XML_SUPPORT();
return NULL; return NULL;
...@@ -444,7 +476,14 @@ xml_init(void) ...@@ -444,7 +476,14 @@ xml_init(void)
/* Now that xml_err_buf exists, safe to call xml_errorHandler */ /* Now that xml_err_buf exists, safe to call xml_errorHandler */
xmlSetGenericErrorFunc(NULL, xml_errorHandler); xmlSetGenericErrorFunc(NULL, xml_errorHandler);
#ifdef NOT_USED
/*
* FIXME: This doesn't work because libxml assumes that whatever
* libxml allocates, only libxml will free, so we can't just drop
* memory contexts behind it. This needs to be refined.
*/
xmlMemSetup(xml_pfree, xml_palloc, xml_repalloc, xml_pstrdup); xmlMemSetup(xml_pfree, xml_palloc, xml_repalloc, xml_pstrdup);
#endif
xmlInitParser(); xmlInitParser();
LIBXML_TEST_VERSION; LIBXML_TEST_VERSION;
} }
...@@ -528,8 +567,6 @@ xml_parse(text *data, bool is_document, bool preserve_whitespace) ...@@ -528,8 +567,6 @@ xml_parse(text *data, bool is_document, bool preserve_whitespace)
* ) */ * ) */
/* ... */ /* ... */
if (doc)
xmlFreeDoc(doc);
if (ctxt) if (ctxt)
xmlFreeParserCtxt(ctxt); xmlFreeParserCtxt(ctxt);
xmlCleanupParser(); xmlCleanupParser();
...@@ -538,6 +575,7 @@ xml_parse(text *data, bool is_document, bool preserve_whitespace) ...@@ -538,6 +575,7 @@ xml_parse(text *data, bool is_document, bool preserve_whitespace)
{ {
if (doc) if (doc)
xmlFreeDoc(doc); xmlFreeDoc(doc);
doc = NULL;
if (ctxt) if (ctxt)
xmlFreeParserCtxt(ctxt); xmlFreeParserCtxt(ctxt);
xmlCleanupParser(); xmlCleanupParser();
...@@ -567,6 +605,7 @@ xml_text2xmlChar(text *in) ...@@ -567,6 +605,7 @@ xml_text2xmlChar(text *in)
} }
#ifdef NOT_USED
/* /*
* Wrappers for memory management functions * Wrappers for memory management functions
*/ */
...@@ -596,6 +635,7 @@ xml_pstrdup(const char *string) ...@@ -596,6 +635,7 @@ xml_pstrdup(const char *string)
{ {
return pstrdup(string); return pstrdup(string);
} }
#endif /* NOT_USED */
/* /*
......
...@@ -124,6 +124,30 @@ SELECT xmlpi(name foo, 'bar'); ...@@ -124,6 +124,30 @@ SELECT xmlpi(name foo, 'bar');
SELECT xmlpi(name foo, 'in?>valid'); SELECT xmlpi(name foo, 'in?>valid');
ERROR: invalid XML processing instruction ERROR: invalid XML processing instruction
DETAIL: XML processing instruction cannot contain "?>". DETAIL: XML processing instruction cannot contain "?>".
SELECT xmlroot(xml '<foo/>', version no value, standalone no value);
xmlroot
-----------------------
<?xml version="1.0"?>
<foo/>
(1 row)
SELECT xmlroot(xml '<foo/>', version '2.0');
xmlroot
-----------------------
<?xml version="2.0"?>
<foo/>
(1 row)
SELECT xmlroot(xmlroot(xml '<foo/>', version '1.0'), version '1.1', standalone no);
xmlroot
---------------------------------------
<?xml version="1.1" standalone="no"?>
<foo/>
(1 row)
SELECT xmlroot ( SELECT xmlroot (
xmlelement ( xmlelement (
name gazonk, name gazonk,
...@@ -139,9 +163,11 @@ SELECT xmlroot ( ...@@ -139,9 +163,11 @@ SELECT xmlroot (
version '1.0', version '1.0',
standalone yes standalone yes
); );
xmlroot xmlroot
------------------------------------------------------------------------------------------ ----------------------------------------------------
<?xml version="1.0" standalone="yes"?><gazonk name="val" num="2"><qux>foo</qux></gazonk> <?xml version="1.0" standalone="yes"?>
<gazonk name="val" num="2"><qux>foo</qux></gazonk>
(1 row) (1 row)
SELECT xmlserialize(content data as character varying) FROM xmltest; SELECT xmlserialize(content data as character varying) FROM xmltest;
......
...@@ -62,6 +62,12 @@ SELECT xmlpi(name foo, 'bar'); ...@@ -62,6 +62,12 @@ SELECT xmlpi(name foo, 'bar');
ERROR: no XML support in this installation ERROR: no XML support in this installation
SELECT xmlpi(name foo, 'in?>valid'); SELECT xmlpi(name foo, 'in?>valid');
ERROR: no XML support in this installation ERROR: no XML support in this installation
SELECT xmlroot(xml '<foo/>', version no value, standalone no value);
ERROR: no XML support in this installation
SELECT xmlroot(xml '<foo/>', version '2.0');
ERROR: no XML support in this installation
SELECT xmlroot(xmlroot(xml '<foo/>', version '1.0'), version '1.1', standalone no);
ERROR: no XML support in this installation
SELECT xmlroot ( SELECT xmlroot (
xmlelement ( xmlelement (
name gazonk, name gazonk,
......
...@@ -52,6 +52,9 @@ SELECT xmlpi(name xmlstuff); ...@@ -52,6 +52,9 @@ SELECT xmlpi(name xmlstuff);
SELECT xmlpi(name foo, 'bar'); SELECT xmlpi(name foo, 'bar');
SELECT xmlpi(name foo, 'in?>valid'); SELECT xmlpi(name foo, 'in?>valid');
SELECT xmlroot(xml '<foo/>', version no value, standalone no value);
SELECT xmlroot(xml '<foo/>', version '2.0');
SELECT xmlroot(xmlroot(xml '<foo/>', version '1.0'), version '1.1', standalone no);
SELECT xmlroot ( SELECT xmlroot (
xmlelement ( xmlelement (
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment