mirror of
https://github.com/MariaDB/server.git
synced 2025-01-22 06:44:16 +01:00
2bb42803df
From the previous branch: commit eda4928ff122a0845baf5ade83b4aa29244a3a89 Author: Olivier Bertrand <bertrandop@gmail.com> Date: Mon Mar 9 22:34:56 2015 +0100 - Add discovery to JSON tables When columns are not defined, CONNECT analyses the json file to find column definitions. This wors only on table that are an array of objects. Pair keys are used to generate the column names and pair values are used for its definition. When the LEVEL option is defined as a not null integer, the eventual JPATH is scanned up to the LEVEL value. From the current one: - Fix MDEV-7521 when column names are utf8 encoded (not a general multi-charset fix) - Adds more to JSON discovery processing and UDF's - Use PlugDup everywhere it replaces PlugSubAlloc + strcpy.
1222 lines
35 KiB
C++
1222 lines
35 KiB
C++
/******************************************************************/
|
|
/* Implementation of XML document processing using libxml2 */
|
|
/* Author: Olivier Bertrand 2007-2015 */
|
|
/******************************************************************/
|
|
#include "my_global.h"
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <libxml/parser.h>
|
|
#include <libxml/tree.h>
|
|
#include <libxml/xpath.h>
|
|
#include <libxml/xpathInternals.h>
|
|
#include <libxml/catalog.h>
|
|
#include <libxml/xmlschemastypes.h>
|
|
#include <libxml/relaxng.h>
|
|
|
|
#if !defined(LIBXML_TREE_ENABLED) || !defined(LIBXML_OUTPUT_ENABLED)
|
|
#error "tree support not compiled in"
|
|
#endif
|
|
|
|
#if !defined(LIBXML_XPATH_ENABLED) || !defined(LIBXML_SAX1_ENABLED)
|
|
#error "XPath not supported"
|
|
#endif
|
|
|
|
#include "global.h"
|
|
#include "plgdbsem.h"
|
|
#include "xobject.h"
|
|
#include "libdoc.h"
|
|
|
|
#include "sql_string.h"
|
|
|
|
/******************************************************************/
|
|
/* Declaration of XML document processing using libxml2 */
|
|
/* Author: Olivier Bertrand 2007-2012 */
|
|
/******************************************************************/
|
|
#include "plgxml.h"
|
|
|
|
typedef class LIBXMLDOC *PXDOC2;
|
|
typedef class XML2NODE *PNODE2;
|
|
typedef class XML2ATTR *PATTR2;
|
|
typedef class XML2NODELIST *PLIST2;
|
|
|
|
/******************************************************************/
|
|
/* XML2 block. Must have the same layout than FBLOCK up to Type. */
|
|
/******************************************************************/
|
|
typedef struct _x2block { /* Loaded XML file block */
|
|
struct _x2block *Next;
|
|
LPCSTR Fname; /* Point on file name */
|
|
size_t Length; /* Used to tell if read mode */
|
|
short Count; /* Nb of times file is used */
|
|
short Type; /* TYPE_FB_XML */
|
|
int Retcode; /* Return code from Load */
|
|
xmlDocPtr Docp; /* Document interface pointer */
|
|
} X2BLOCK, *PX2BLOCK;
|
|
|
|
/******************************************************************/
|
|
/* Declaration of libxml2 document. */
|
|
/******************************************************************/
|
|
class LIBXMLDOC : public XMLDOCUMENT {
|
|
friend class XML2NODE;
|
|
friend class XML2ATTR;
|
|
public:
|
|
// Constructor
|
|
LIBXMLDOC(char *nsl, char *nsdf, char *enc, PFBLOCK fp);
|
|
|
|
// Properties
|
|
virtual short GetDocType(void) {return TYPE_FB_XML2;}
|
|
virtual void *GetDocPtr(void) {return Docp;}
|
|
virtual void SetNofree(bool b) {Nofreelist = b;}
|
|
|
|
// Methods
|
|
virtual bool Initialize(PGLOBAL g);
|
|
virtual bool ParseFile(char *fn);
|
|
virtual bool NewDoc(PGLOBAL g, char *ver);
|
|
virtual void AddComment(PGLOBAL g, char *com);
|
|
virtual PXNODE GetRoot(PGLOBAL g);
|
|
virtual PXNODE NewRoot(PGLOBAL g, char *name);
|
|
virtual PXNODE NewPnode(PGLOBAL g, char *name);
|
|
virtual PXATTR NewPattr(PGLOBAL g);
|
|
virtual PXLIST NewPlist(PGLOBAL g);
|
|
virtual int DumpDoc(PGLOBAL g, char *ofn);
|
|
virtual void CloseDoc(PGLOBAL g, PFBLOCK xp);
|
|
virtual PFBLOCK LinkXblock(PGLOBAL g, MODE m, int rc, char *fn);
|
|
|
|
protected:
|
|
// bool CheckDocument(FILE *of, xmlNodePtr np);
|
|
xmlNodeSetPtr GetNodeList(PGLOBAL g, xmlNodePtr np, char *xp);
|
|
int Decode(xmlChar *cnt, char *buf, int n);
|
|
xmlChar *Encode(PGLOBAL g, char *txt);
|
|
|
|
// Members
|
|
xmlDocPtr Docp;
|
|
xmlNodeSetPtr Nlist;
|
|
xmlXPathContextPtr Ctxp;
|
|
xmlXPathObjectPtr Xop;
|
|
xmlXPathObjectPtr NlXop;
|
|
xmlErrorPtr Xerr;
|
|
char *Buf; // Temporary
|
|
bool Nofreelist;
|
|
}; // end of class LIBXMLDOC
|
|
|
|
/******************************************************************/
|
|
/* Declaration of libxml2 node. */
|
|
/******************************************************************/
|
|
class XML2NODE : public XMLNODE {
|
|
friend class LIBXMLDOC;
|
|
friend class XML2NODELIST;
|
|
public:
|
|
// Properties
|
|
virtual char *GetName(PGLOBAL g) {return (char*)Nodep->name;}
|
|
virtual int GetType(void);
|
|
virtual PXNODE GetNext(PGLOBAL g);
|
|
virtual PXNODE GetChild(PGLOBAL g);
|
|
|
|
// Methods
|
|
virtual RCODE GetContent(PGLOBAL g, char *buf, int len);
|
|
virtual bool SetContent(PGLOBAL g, char *txtp, int len);
|
|
virtual PXNODE Clone(PGLOBAL g, PXNODE np);
|
|
virtual PXLIST GetChildElements(PGLOBAL g, char *xp, PXLIST lp);
|
|
virtual PXLIST SelectNodes(PGLOBAL g, char *xp, PXLIST lp);
|
|
virtual PXNODE SelectSingleNode(PGLOBAL g, char *xp, PXNODE np);
|
|
virtual PXATTR GetAttribute(PGLOBAL g, char *name, PXATTR ap);
|
|
virtual PXNODE AddChildNode(PGLOBAL g, char *name, PXNODE np);
|
|
virtual PXATTR AddProperty(PGLOBAL g, char *name, PXATTR ap);
|
|
virtual void AddText(PGLOBAL g, char *txtp);
|
|
virtual void DeleteChild(PGLOBAL g, PXNODE dnp);
|
|
|
|
protected:
|
|
// Constructor
|
|
XML2NODE(PXDOC dp, xmlNodePtr np);
|
|
|
|
// Members
|
|
xmlDocPtr Docp;
|
|
xmlChar *Content;
|
|
xmlNodePtr Nodep;
|
|
}; // end of class XML2NODE
|
|
|
|
/******************************************************************/
|
|
/* Declaration of libxml2 node list. */
|
|
/******************************************************************/
|
|
class XML2NODELIST : public XMLNODELIST {
|
|
friend class LIBXMLDOC;
|
|
friend class XML2NODE;
|
|
public:
|
|
// Methods
|
|
virtual int GetLength(void);
|
|
virtual PXNODE GetItem(PGLOBAL g, int n, PXNODE np);
|
|
virtual bool DropItem(PGLOBAL g, int n);
|
|
|
|
protected:
|
|
// Constructor
|
|
XML2NODELIST(PXDOC dp, xmlNodeSetPtr lp);
|
|
|
|
// Members
|
|
xmlNodeSetPtr Listp;
|
|
}; // end of class XML2NODELIST
|
|
|
|
/******************************************************************/
|
|
/* Declaration of libxml2 attribute. */
|
|
/******************************************************************/
|
|
class XML2ATTR : public XMLATTRIBUTE {
|
|
friend class LIBXMLDOC;
|
|
friend class XML2NODE;
|
|
public:
|
|
// Properties
|
|
//virtual char *GetText(void);
|
|
|
|
// Methods
|
|
virtual bool SetText(PGLOBAL g, char *txtp, int len);
|
|
|
|
protected:
|
|
// Constructor
|
|
XML2ATTR(PXDOC dp, xmlAttrPtr ap, xmlNodePtr np);
|
|
|
|
// Members
|
|
xmlAttrPtr Atrp;
|
|
xmlNodePtr Parent;
|
|
}; // end of class XML2ATTR
|
|
|
|
|
|
|
|
extern "C" {
|
|
extern char version[];
|
|
} // "C"
|
|
|
|
#if defined(MEMORY_TRACE)
|
|
static int m = 0;
|
|
static char s[500];
|
|
/**************************************************************************/
|
|
/* Tracing output function. */
|
|
/**************************************************************************/
|
|
void xtrc(char const *fmt, ...)
|
|
{
|
|
va_list ap;
|
|
va_start (ap, fmt);
|
|
|
|
//vfprintf(stderr, fmt, ap);
|
|
vsprintf(s, fmt, ap);
|
|
if (s[strlen(s)-1] == '\n')
|
|
s[strlen(s)-1] = 0;
|
|
va_end (ap);
|
|
} // end of htrc
|
|
|
|
static xmlFreeFunc Free;
|
|
static xmlMallocFunc Malloc;
|
|
static xmlMallocFunc MallocA;
|
|
static xmlReallocFunc Realloc;
|
|
static xmlStrdupFunc Strdup;
|
|
|
|
void xmlMyFree(void *mem)
|
|
{
|
|
if (trace) {
|
|
htrc("%.4d Freeing at %p %s\n", ++m, mem, s);
|
|
*s = 0;
|
|
} // endif trace
|
|
Free(mem);
|
|
} // end of xmlMyFree
|
|
|
|
void *xmlMyMalloc(size_t size)
|
|
{
|
|
void *p = Malloc(size);
|
|
if (trace) {
|
|
htrc("%.4d Allocating %.5d at %p %s\n", ++m, size, p, s);
|
|
*s = 0;
|
|
} // endif trace
|
|
return p;
|
|
} // end of xmlMyMalloc
|
|
|
|
void *xmlMyMallocAtomic(size_t size)
|
|
{
|
|
void *p = MallocA(size);
|
|
if (trace) {
|
|
htrc("%.4d Atom alloc %.5d at %p %s\n", ++m, size, p, s);
|
|
*s = 0;
|
|
} // endif trace
|
|
return p;
|
|
} // end of xmlMyMallocAtomic
|
|
|
|
void *xmlMyRealloc(void *mem, size_t size)
|
|
{
|
|
void *p = Realloc(mem, size);
|
|
if (trace) {
|
|
htrc("%.4d ReAlloc %.5d to %p from %p %s\n", ++m, size, p, mem, s);
|
|
*s = 0;
|
|
} // endif trace
|
|
return p;
|
|
} // end of xmlMyRealloc
|
|
|
|
char *xmlMyStrdup(const char *str)
|
|
{
|
|
char *p = Strdup(str);
|
|
if (trace) {
|
|
htrc("%.4d Duplicating to %p from %p %s %s\n", ++m, p, str, str, s);
|
|
*s = 0;
|
|
} // endif trace
|
|
return p;
|
|
} // end of xmlMyStrdup
|
|
#define htrc xtrc
|
|
#endif // MEMORY_TRACE
|
|
|
|
/******************************************************************/
|
|
/* Return a LIBXMLDOC as a XMLDOC. */
|
|
/******************************************************************/
|
|
PXDOC GetLibxmlDoc(PGLOBAL g, char *nsl, char *nsdf,
|
|
char *enc, PFBLOCK fp)
|
|
{
|
|
return (PXDOC) new(g) LIBXMLDOC(nsl, nsdf, enc, fp);
|
|
} // end of GetLibxmlDoc
|
|
|
|
/******************************************************************/
|
|
/* XML library initialization function. */
|
|
/******************************************************************/
|
|
void XmlInitParserLib(void)
|
|
{
|
|
#if defined(MEMORY_TRACE)
|
|
int rc = xmlGcMemGet(&Free, &Malloc, &MallocA, &Realloc, &Strdup);
|
|
|
|
if (!rc)
|
|
rc = xmlGcMemSetup(xmlMyFree,
|
|
xmlMyMalloc,
|
|
xmlMyMallocAtomic,
|
|
xmlMyRealloc,
|
|
xmlMyStrdup);
|
|
|
|
#endif // MEMORY_TRACE
|
|
xmlInitParser();
|
|
} // end of XmlInitParserLib
|
|
|
|
/******************************************************************/
|
|
/* XML library cleanup function. */
|
|
/******************************************************************/
|
|
/*
|
|
This is a copy of xmlCleanupParser() from the libxml2 sources
|
|
with xmlResetLastError() commented.
|
|
|
|
xmlResetLastError() called from the original xmlCleanupParser() causes
|
|
valgrind to report memory leaks. This happens because
|
|
ha_initialize_handlerton() is called from the main thread in mysqld.cc,
|
|
while ha_finalize_handlerton() is called from a non-main thread.
|
|
libxml2 gets confused because of xmlInitParser() and xmlCleanupParser()
|
|
being called from the different threads.
|
|
|
|
Perhaps the code in mysqld.cc should eventually be modified
|
|
to shutdown plugins from the main thread.
|
|
*/
|
|
static void
|
|
xmlCleanupParser_replacement(void)
|
|
{
|
|
xmlCleanupCharEncodingHandlers();
|
|
#ifdef LIBXML_CATALOG_ENABLED
|
|
xmlCatalogCleanup();
|
|
#endif
|
|
xmlDictCleanup();
|
|
xmlCleanupInputCallbacks();
|
|
#ifdef LIBXML_OUTPUT_ENABLED
|
|
xmlCleanupOutputCallbacks();
|
|
#endif
|
|
#ifdef LIBXML_SCHEMAS_ENABLED
|
|
xmlSchemaCleanupTypes();
|
|
xmlRelaxNGCleanupTypes();
|
|
#endif
|
|
//xmlResetLastError();
|
|
xmlCleanupGlobals();
|
|
xmlCleanupThreads(); /* must be last if called not from the main thread */
|
|
xmlCleanupMemory();
|
|
}
|
|
|
|
|
|
void XmlCleanupParserLib(void)
|
|
{
|
|
xmlCleanupParser_replacement();
|
|
} // end of XmlCleanupParserLib
|
|
|
|
/******************************************************************/
|
|
/* Close a loaded libxml2 XML file. */
|
|
/******************************************************************/
|
|
void CloseXML2File(PGLOBAL g, PFBLOCK fp, bool all)
|
|
{
|
|
PX2BLOCK xp = (PX2BLOCK)fp;
|
|
|
|
if (trace)
|
|
htrc("CloseXML2File: xp=%p count=%d\n", xp, (xp) ? xp->Count : 0);
|
|
|
|
if (xp && xp->Count > 1 && !all) {
|
|
xp->Count--;
|
|
} else if (xp && xp->Count > 0) {
|
|
xmlFreeDoc(xp->Docp);
|
|
xp->Count = 0;
|
|
} // endif
|
|
|
|
} // end of CloseXML2File
|
|
|
|
/* ---------------------- class LIBXMLDOC ----------------------- */
|
|
|
|
/******************************************************************/
|
|
/* LIBXMLDOC constructor. */
|
|
/******************************************************************/
|
|
LIBXMLDOC::LIBXMLDOC(char *nsl, char *nsdf, char *enc, PFBLOCK fp)
|
|
: XMLDOCUMENT(nsl, nsdf, enc)
|
|
{
|
|
assert (!fp || fp->Type == TYPE_FB_XML2);
|
|
Docp = (fp) ? ((PX2BLOCK)fp)->Docp : NULL;
|
|
Nlist = NULL;
|
|
Ctxp = NULL;
|
|
Xop = NULL;
|
|
NlXop = NULL;
|
|
Xerr = NULL;
|
|
Buf = NULL;
|
|
Nofreelist = false;
|
|
} // end of LIBXMLDOC constructor
|
|
|
|
/******************************************************************/
|
|
/* Initialize XML parser and check library compatibility. */
|
|
/******************************************************************/
|
|
bool LIBXMLDOC::Initialize(PGLOBAL g)
|
|
{
|
|
int n = xmlKeepBlanksDefault(1);
|
|
return MakeNSlist(g);
|
|
} // end of Initialize
|
|
|
|
/******************************************************************/
|
|
/* Parse the XML file and construct node tree in memory. */
|
|
/******************************************************************/
|
|
bool LIBXMLDOC::ParseFile(char *fn)
|
|
{
|
|
if (trace)
|
|
htrc("ParseFile\n");
|
|
|
|
if ((Docp = xmlParseFile(fn))) {
|
|
if (Docp->encoding)
|
|
Encoding = (char*)Docp->encoding;
|
|
|
|
return false;
|
|
} else if ((Xerr = xmlGetLastError()))
|
|
xmlResetError(Xerr);
|
|
|
|
return true;
|
|
} // end of ParseFile
|
|
|
|
/******************************************************************/
|
|
/* Create or reuse an Xblock for this document. */
|
|
/******************************************************************/
|
|
PFBLOCK LIBXMLDOC::LinkXblock(PGLOBAL g, MODE m, int rc, char *fn)
|
|
{
|
|
PDBUSER dup = (PDBUSER)g->Activityp->Aptr;
|
|
PX2BLOCK xp = (PX2BLOCK)PlugSubAlloc(g, NULL, sizeof(X2BLOCK));
|
|
|
|
memset(xp, 0, sizeof(X2BLOCK));
|
|
xp->Next = (PX2BLOCK)dup->Openlist;
|
|
dup->Openlist = (PFBLOCK)xp;
|
|
xp->Type = TYPE_FB_XML2;
|
|
xp->Fname = (LPCSTR)PlugDup(g, fn);
|
|
xp->Count = 1;
|
|
xp->Length = (m == MODE_READ) ? 1 : 0;
|
|
xp->Retcode = rc;
|
|
xp->Docp = Docp;
|
|
|
|
// Return xp as a fp
|
|
return (PFBLOCK)xp;
|
|
} // end of LinkXblock
|
|
|
|
/******************************************************************/
|
|
/* Construct and add the XML processing instruction node. */
|
|
/******************************************************************/
|
|
bool LIBXMLDOC::NewDoc(PGLOBAL g, char *ver)
|
|
{
|
|
if (trace)
|
|
htrc("NewDoc\n");
|
|
|
|
return ((Docp = xmlNewDoc(BAD_CAST ver)) == NULL);
|
|
} // end of NewDoc
|
|
|
|
/******************************************************************/
|
|
/* Add a new comment node to the document. */
|
|
/******************************************************************/
|
|
void LIBXMLDOC::AddComment(PGLOBAL g, char *txtp)
|
|
{
|
|
if (trace)
|
|
htrc("AddComment: %s\n", txtp);
|
|
|
|
xmlNodePtr cp = xmlNewDocComment(Docp, BAD_CAST txtp);
|
|
xmlAddChild((xmlNodePtr)Docp, cp);
|
|
} // end of AddText
|
|
|
|
/******************************************************************/
|
|
/* Return the node class of the root of the document. */
|
|
/******************************************************************/
|
|
PXNODE LIBXMLDOC::GetRoot(PGLOBAL g)
|
|
{
|
|
if (trace)
|
|
htrc("GetRoot\n");
|
|
|
|
xmlNodePtr root = xmlDocGetRootElement(Docp);
|
|
|
|
if (!root)
|
|
return NULL;
|
|
|
|
return new(g) XML2NODE(this, root);
|
|
} // end of GetRoot
|
|
|
|
/******************************************************************/
|
|
/* Create a new root element and return its class node. */
|
|
/******************************************************************/
|
|
PXNODE LIBXMLDOC::NewRoot(PGLOBAL g, char *name)
|
|
{
|
|
if (trace)
|
|
htrc("NewRoot: %s\n", name);
|
|
|
|
xmlNodePtr root = xmlNewDocNode(Docp, NULL, BAD_CAST name, NULL);
|
|
|
|
if (root) {
|
|
xmlDocSetRootElement(Docp, root);
|
|
return new(g) XML2NODE(this, root);
|
|
} else
|
|
return NULL;
|
|
|
|
} // end of NewRoot
|
|
|
|
/******************************************************************/
|
|
/* Return a void XML2NODE node class. */
|
|
/******************************************************************/
|
|
PXNODE LIBXMLDOC::NewPnode(PGLOBAL g, char *name)
|
|
{
|
|
if (trace)
|
|
htrc("NewNode: %s\n", name);
|
|
|
|
xmlNodePtr nop;
|
|
|
|
if (name) {
|
|
nop = xmlNewDocNode(Docp, NULL, BAD_CAST name, NULL);
|
|
|
|
if (nop == NULL)
|
|
return NULL;
|
|
|
|
} else
|
|
nop = NULL;
|
|
|
|
return new(g) XML2NODE(this, nop);
|
|
} // end of NewPnode
|
|
|
|
/******************************************************************/
|
|
/* Return a void XML2ATTR node class. */
|
|
/******************************************************************/
|
|
PXATTR LIBXMLDOC::NewPattr(PGLOBAL g)
|
|
{
|
|
return new(g) XML2ATTR(this, NULL, NULL);
|
|
} // end of NewPattr
|
|
|
|
/******************************************************************/
|
|
/* Return a void XML2ATTR node class. */
|
|
/******************************************************************/
|
|
PXLIST LIBXMLDOC::NewPlist(PGLOBAL g)
|
|
{
|
|
return new(g) XML2NODELIST(this, NULL);
|
|
} // end of NewPlist
|
|
|
|
/******************************************************************/
|
|
/* Dump the node tree to a new XML file. */
|
|
/******************************************************************/
|
|
int LIBXMLDOC::DumpDoc(PGLOBAL g, char *ofn)
|
|
{
|
|
int rc = 0;
|
|
FILE *of;
|
|
|
|
if (trace)
|
|
htrc("DumpDoc: %s\n", ofn);
|
|
|
|
if (!(of= global_fopen(g, MSGID_CANNOT_OPEN, ofn, "w")))
|
|
return -1;
|
|
|
|
#if 1
|
|
// This function does not crash (
|
|
if (xmlSaveFormatFileEnc((const char *)ofn, Docp, Encoding, 0) < 0) {
|
|
xmlErrorPtr err = xmlGetLastError();
|
|
|
|
strcpy(g->Message, (err) ? err->message : "Error saving XML doc");
|
|
rc = -1;
|
|
} // endif Save
|
|
// rc = xmlDocDump(of, Docp);
|
|
#else // 0
|
|
// Until this function is fixed, do the job ourself
|
|
xmlNodePtr Rootp;
|
|
|
|
// Save the modified document
|
|
fprintf(of, "<?xml version=\"1.0\" encoding=\"%s\"?>\n", Encoding);
|
|
fprintf(of, "<!-- Created by CONNECT %s -->\n", version);
|
|
|
|
if (!(Rootp = xmlDocGetRootElement(Docp)))
|
|
return 1;
|
|
|
|
Buf = (char*)PlugSubAlloc(g, NULL, 1024);
|
|
rc = iconv_close(Cd2);
|
|
Cd2 = iconv_open(Encoding, "UTF-8");
|
|
rc = CheckDocument(of, Rootp);
|
|
#endif // 0
|
|
|
|
fclose(of);
|
|
return rc;
|
|
} // end of Dump
|
|
|
|
/******************************************************************/
|
|
/* Free the document, cleanup the XML library, and */
|
|
/* debug memory for regression tests. */
|
|
/******************************************************************/
|
|
void LIBXMLDOC::CloseDoc(PGLOBAL g, PFBLOCK xp)
|
|
{
|
|
if (trace)
|
|
htrc("CloseDoc: xp=%p count=%d\n", xp, (xp) ? xp->Count : 0);
|
|
|
|
//if (xp && xp->Count == 1) {
|
|
if (Nlist) {
|
|
xmlXPathFreeNodeSet(Nlist);
|
|
|
|
if ((Xerr = xmlGetLastError()))
|
|
xmlResetError(Xerr);
|
|
|
|
Nlist = NULL;
|
|
} // endif Nlist
|
|
|
|
if (Xop) {
|
|
xmlXPathFreeObject(Xop);
|
|
|
|
if ((Xerr = xmlGetLastError()))
|
|
xmlResetError(Xerr);
|
|
|
|
Xop = NULL;
|
|
} // endif Xop
|
|
|
|
if (NlXop) {
|
|
xmlXPathFreeObject(NlXop);
|
|
|
|
if ((Xerr = xmlGetLastError()))
|
|
xmlResetError(Xerr);
|
|
|
|
NlXop = NULL;
|
|
} // endif NlXop
|
|
|
|
if (Ctxp) {
|
|
xmlXPathFreeContext(Ctxp);
|
|
|
|
if ((Xerr = xmlGetLastError()))
|
|
xmlResetError(Xerr);
|
|
|
|
Ctxp = NULL;
|
|
} // endif Ctxp
|
|
|
|
// } // endif Count
|
|
|
|
CloseXML2File(g, xp, false);
|
|
} // end of Close
|
|
|
|
/******************************************************************/
|
|
/* Evaluate the passed Xpath from the passed context node. */
|
|
/******************************************************************/
|
|
xmlNodeSetPtr LIBXMLDOC::GetNodeList(PGLOBAL g, xmlNodePtr np, char *xp)
|
|
{
|
|
xmlNodeSetPtr nl;
|
|
|
|
if (trace)
|
|
htrc("GetNodeList: %s np=%p\n", xp, np);
|
|
|
|
if (!Ctxp) {
|
|
// Init Xpath
|
|
if (trace)
|
|
htrc("Calling xmlPathInit\n");
|
|
|
|
xmlXPathInit();
|
|
|
|
if (trace)
|
|
htrc("Calling xmlXPathNewContext Docp=%p\n", Docp);
|
|
|
|
// Create xpath evaluation context
|
|
if (!(Ctxp = xmlXPathNewContext(Docp))) {
|
|
strcpy(g->Message, MSG(XPATH_CNTX_ERR));
|
|
|
|
if (trace)
|
|
htrc("Context error: %s\n", g->Message);
|
|
|
|
return NULL;
|
|
} // endif xpathCtx
|
|
|
|
// Register namespaces from list (if any)
|
|
for (PNS nsp = Namespaces; nsp; nsp = nsp->Next) {
|
|
if (trace)
|
|
htrc("Calling xmlXPathRegisterNs Prefix=%s Uri=%s\n",
|
|
nsp->Prefix, nsp->Uri);
|
|
|
|
if (xmlXPathRegisterNs(Ctxp, BAD_CAST nsp->Prefix,
|
|
BAD_CAST nsp->Uri)) {
|
|
sprintf(g->Message, MSG(REGISTER_ERR), nsp->Prefix, nsp->Uri);
|
|
|
|
if (trace)
|
|
htrc("Ns error: %s\n", g->Message);
|
|
|
|
return NULL;
|
|
} // endif Registering
|
|
|
|
} // endfor nsp
|
|
|
|
} // endif Ctxp
|
|
|
|
if (Xop) {
|
|
if (trace)
|
|
htrc("Calling xmlXPathFreeNodeSetList Xop=%p NOFREE=%d\n",
|
|
Xop, Nofreelist);
|
|
|
|
if (Nofreelist) {
|
|
// Making Nlist that must not be freed yet
|
|
// xmlXPathFreeNodeSetList(Xop); // Caused memory leak
|
|
assert(!NlXop);
|
|
NlXop = Xop; // Freed on closing
|
|
Nofreelist = false;
|
|
} else
|
|
xmlXPathFreeObject(Xop); // Caused node not found
|
|
|
|
if ((Xerr = xmlGetLastError())) {
|
|
strcpy(g->Message, Xerr->message);
|
|
xmlResetError(Xerr);
|
|
return NULL;
|
|
} // endif Xerr
|
|
|
|
} // endif Xop
|
|
|
|
// Set the context to the calling node
|
|
Ctxp->node = np;
|
|
|
|
if (trace)
|
|
htrc("Calling xmlXPathEval %s Ctxp=%p\n", xp, Ctxp);
|
|
|
|
// Evaluate table xpath
|
|
if (!(Xop = xmlXPathEval(BAD_CAST xp, Ctxp))) {
|
|
sprintf(g->Message, MSG(XPATH_EVAL_ERR), xp);
|
|
|
|
if (trace)
|
|
htrc("Path error: %s\n", g->Message);
|
|
|
|
return NULL;
|
|
} else
|
|
nl = Xop->nodesetval;
|
|
|
|
if (trace)
|
|
htrc("GetNodeList nl=%p n=%p\n", nl, (nl) ? nl->nodeNr : 0);
|
|
|
|
return nl;
|
|
} // end of GetNodeList
|
|
|
|
#if 0 // Not used anymore
|
|
/******************************************************************/
|
|
/* CheckDocument: check if the document is ok to dump. */
|
|
/* Currently this does the dumping of the document. */
|
|
/******************************************************************/
|
|
bool LIBXMLDOC::CheckDocument(FILE *of, xmlNodePtr np)
|
|
{
|
|
int n;
|
|
bool b;
|
|
|
|
if (!np)
|
|
return true;
|
|
|
|
if (np->type == XML_ELEMENT_NODE) {
|
|
n = fprintf(of, "<%s", np->name);
|
|
b = CheckDocument(of, (xmlNodePtr)np->properties);
|
|
|
|
if (np->children)
|
|
n = fprintf(of, ">");
|
|
else
|
|
n = fprintf(of, "/>");
|
|
|
|
} else if (np->type == XML_ATTRIBUTE_NODE)
|
|
n = fprintf(of, " %s=\"", np->name);
|
|
else if (np->type == XML_TEXT_NODE)
|
|
n = fprintf(of, "%s", Encode(NULL, (char*)np->content));
|
|
else if (np->type == XML_COMMENT_NODE)
|
|
n = fprintf(of, "%s", Encode(NULL, (char*)np->content));
|
|
|
|
b = CheckDocument(of, np->children);
|
|
|
|
if (np->type == XML_ATTRIBUTE_NODE)
|
|
n = fprintf(of, "\"");
|
|
else if (!b && np->type == XML_ELEMENT_NODE)
|
|
n = fprintf(of, "</%s>", np->name);
|
|
|
|
b = CheckDocument(of, np->next);
|
|
return false;
|
|
} // end of CheckDocument
|
|
|
|
/******************************************************************/
|
|
/* Convert node or attribute content to latin characters. */
|
|
/******************************************************************/
|
|
int LIBXMLDOC::Decode(xmlChar *cnt, char *buf, int n)
|
|
{
|
|
const char *txt = (const char *)cnt;
|
|
uint dummy_errors;
|
|
uint32 len= copy_and_convert(buf, n, &my_charset_utf8_general_ci, txt,
|
|
strlen(txt), &my_charset_utf8_general_ci,
|
|
&dummy_errors);
|
|
buf[len]= '\0';
|
|
return 0;
|
|
} // end of Decode
|
|
|
|
/******************************************************************/
|
|
/* Convert node or attribute content to latin characters. */
|
|
/******************************************************************/
|
|
xmlChar *LIBXMLDOC::Encode(PGLOBAL g, char *txt)
|
|
{
|
|
const CHARSET_INFO *ics= &my_charset_utf8_general_ci;
|
|
const CHARSET_INFO *ocs= &my_charset_utf8_general_ci;
|
|
size_t i = strlen(txt);
|
|
size_t o = i * ocs->mbmaxlen / ics->mbmaxlen + 1;
|
|
char *buf;
|
|
if (g) {
|
|
buf = (char*)PlugSubAlloc(g, NULL, o);
|
|
} else {
|
|
o = 1024;
|
|
buf = Buf;
|
|
} // endif g
|
|
uint dummy_errors;
|
|
uint32 len= copy_and_convert(buf, o, ocs,
|
|
txt, i, ics,
|
|
&dummy_errors);
|
|
buf[len]= '\0';
|
|
return BAD_CAST buf;
|
|
} // end of Encode
|
|
#endif // 0
|
|
|
|
/* ---------------------- class XML2NODE ------------------------ */
|
|
|
|
/******************************************************************/
|
|
/* XML2NODE constructor. */
|
|
/******************************************************************/
|
|
XML2NODE::XML2NODE(PXDOC dp, xmlNodePtr np) : XMLNODE(dp)
|
|
{
|
|
Docp = ((PXDOC2)dp)->Docp;
|
|
Content = NULL;
|
|
Nodep = np;
|
|
} // end of XML2NODE constructor
|
|
|
|
int XML2NODE::GetType(void)
|
|
{
|
|
if (trace)
|
|
htrc("GetType type=%d\n", Nodep->type);
|
|
|
|
return Nodep->type;
|
|
} // end of GetType
|
|
|
|
/******************************************************************/
|
|
/* Return the node class of next sibling of the node. */
|
|
/******************************************************************/
|
|
PXNODE XML2NODE::GetNext(PGLOBAL g)
|
|
{
|
|
if (trace)
|
|
htrc("GetNext\n");
|
|
|
|
if (!Nodep->next)
|
|
Next = NULL;
|
|
else if (!Next)
|
|
Next = new(g) XML2NODE(Doc, Nodep->next);
|
|
|
|
return Next;
|
|
} // end of GetNext
|
|
|
|
/******************************************************************/
|
|
/* Return the node class of first children of the node. */
|
|
/******************************************************************/
|
|
PXNODE XML2NODE::GetChild(PGLOBAL g)
|
|
{
|
|
if (trace)
|
|
htrc("GetChild\n");
|
|
|
|
if (!Nodep->children)
|
|
Children = NULL;
|
|
else if (!Children)
|
|
Children = new(g) XML2NODE(Doc, Nodep->children);
|
|
|
|
return Children;
|
|
} // end of GetChild
|
|
|
|
/******************************************************************/
|
|
/* Return the content of a node and subnodes. */
|
|
/******************************************************************/
|
|
RCODE XML2NODE::GetContent(PGLOBAL g, char *buf, int len)
|
|
{
|
|
RCODE rc = RC_OK;
|
|
|
|
if (trace)
|
|
htrc("GetContent\n");
|
|
|
|
if (Content)
|
|
xmlFree(Content);
|
|
|
|
if ((Content = xmlNodeGetContent(Nodep))) {
|
|
char *extra = " \t\r\n";
|
|
char *p1 = (char*)Content, *p2 = buf;
|
|
bool b = false;
|
|
|
|
// Copy content eliminating extra characters
|
|
for (; *p1; p1++)
|
|
if ((p2 - buf) < len) {
|
|
if (strchr(extra, *p1)) {
|
|
if (b) {
|
|
// This to have one blank between sub-nodes
|
|
*p2++ = ' ';
|
|
b = false;
|
|
} // endif b
|
|
|
|
} else {
|
|
*p2++ = *p1;
|
|
b = true;
|
|
} // endif p1
|
|
|
|
} else {
|
|
sprintf(g->Message, "Truncated %s content", Nodep->name);
|
|
rc = RC_INFO;
|
|
} // endif len
|
|
|
|
*p2 = 0;
|
|
|
|
if (trace)
|
|
htrc("GetText buf='%s' len=%d\n", buf, len);
|
|
|
|
xmlFree(Content);
|
|
Content = NULL;
|
|
} else
|
|
*buf = '\0';
|
|
|
|
if (trace)
|
|
htrc("GetContent: %s\n", buf);
|
|
|
|
return rc;
|
|
} // end of GetContent
|
|
|
|
/******************************************************************/
|
|
/* Set the content of a node. */
|
|
/******************************************************************/
|
|
bool XML2NODE::SetContent(PGLOBAL g, char *txtp, int len)
|
|
{
|
|
if (trace)
|
|
htrc("SetContent: %s\n", txtp);
|
|
|
|
xmlChar *buf = xmlEncodeEntitiesReentrant(Docp, BAD_CAST txtp);
|
|
|
|
if (trace)
|
|
htrc("SetContent: %s -> %s\n", txtp, buf);
|
|
|
|
xmlNodeSetContent(Nodep, buf);
|
|
xmlFree(buf);
|
|
return false;
|
|
} // end of SetContent
|
|
|
|
/******************************************************************/
|
|
/* Return a clone of this node. */
|
|
/******************************************************************/
|
|
PXNODE XML2NODE::Clone(PGLOBAL g, PXNODE np)
|
|
{
|
|
if (trace)
|
|
htrc("Clone: np=%p\n", np);
|
|
|
|
if (np) {
|
|
((PNODE2)np)->Nodep = Nodep;
|
|
return np;
|
|
} else
|
|
return new(g) XML2NODE(Doc, Nodep);
|
|
|
|
} // end of Clone
|
|
|
|
/******************************************************************/
|
|
/* Return the list of all or matching children that are elements.*/
|
|
/******************************************************************/
|
|
PXLIST XML2NODE::GetChildElements(PGLOBAL g, char *xp, PXLIST lp)
|
|
{
|
|
if (trace)
|
|
htrc("GetChildElements: %s\n", xp);
|
|
|
|
return SelectNodes(g, (xp) ? xp : (char*)"*", lp);
|
|
} // end of GetChildElements
|
|
|
|
/******************************************************************/
|
|
/* Return the list of nodes verifying the passed Xpath. */
|
|
/******************************************************************/
|
|
PXLIST XML2NODE::SelectNodes(PGLOBAL g, char *xp, PXLIST lp)
|
|
{
|
|
if (trace)
|
|
htrc("SelectNodes: %s\n", xp);
|
|
|
|
xmlNodeSetPtr nl = ((PXDOC2)Doc)->GetNodeList(g, Nodep, xp);
|
|
|
|
if (lp) {
|
|
((PLIST2)lp)->Listp = nl;
|
|
return lp;
|
|
} else
|
|
return new(g) XML2NODELIST(Doc, nl);
|
|
|
|
} // end of SelectNodes
|
|
|
|
/******************************************************************/
|
|
/* Return the first node verifying the passed Xapth. */
|
|
/******************************************************************/
|
|
PXNODE XML2NODE::SelectSingleNode(PGLOBAL g, char *xp, PXNODE np)
|
|
{
|
|
if (trace)
|
|
htrc("SelectSingleNode: %s\n", xp);
|
|
|
|
xmlNodeSetPtr nl = ((PXDOC2)Doc)->GetNodeList(g, Nodep, xp);
|
|
|
|
if (nl && nl->nodeNr) {
|
|
if (np) {
|
|
((PNODE2)np)->Nodep = nl->nodeTab[0];
|
|
return np;
|
|
} else
|
|
return new(g) XML2NODE(Doc, nl->nodeTab[0]);
|
|
|
|
} else
|
|
return NULL;
|
|
|
|
} // end of SelectSingleNode
|
|
|
|
/******************************************************************/
|
|
/* Return the node attribute with the specified name. */
|
|
/******************************************************************/
|
|
PXATTR XML2NODE::GetAttribute(PGLOBAL g, char *name, PXATTR ap)
|
|
{
|
|
if (trace)
|
|
htrc("GetAttribute: %s\n", name);
|
|
|
|
xmlAttrPtr atp = xmlHasProp(Nodep, BAD_CAST name);
|
|
|
|
if (atp) {
|
|
if (ap) {
|
|
((PATTR2)ap)->Atrp = atp;
|
|
((PATTR2)ap)->Parent = Nodep;
|
|
return ap;
|
|
} else
|
|
return new(g) XML2ATTR(Doc, atp, Nodep);
|
|
|
|
} else
|
|
return NULL;
|
|
|
|
} // end of GetAttribute
|
|
|
|
/******************************************************************/
|
|
/* Add a new child node to this node and return it. */
|
|
/******************************************************************/
|
|
PXNODE XML2NODE::AddChildNode(PGLOBAL g, char *name, PXNODE np)
|
|
{
|
|
char *p, *pn, *pf = NULL;
|
|
|
|
if (trace)
|
|
htrc("AddChildNode: %s\n", name);
|
|
|
|
// Is a prefix specified
|
|
if ((pn = strchr(name, ':'))) {
|
|
pf = name;
|
|
*pn++ = '\0'; // Separate name from prefix
|
|
} else
|
|
pn = name;
|
|
|
|
// If name has the format m[n] only m is taken as node name
|
|
if ((p = strchr(pn, '[')))
|
|
p = BufAlloc(g, pn, p - pn);
|
|
else
|
|
p = pn;
|
|
|
|
xmlNodePtr nop = xmlNewChild(Nodep, NULL, BAD_CAST p, NULL);
|
|
|
|
if (!nop)
|
|
return NULL;
|
|
|
|
if (pf) {
|
|
// Prefixed name, is it the default NS prefix?
|
|
if (Doc->DefNs && !strcmp(pf, Doc->DefNs))
|
|
pf = NULL; // Default namespace
|
|
|
|
xmlNsPtr nsp = xmlSearchNs(Docp, nop, BAD_CAST pf);
|
|
|
|
if (!nsp)
|
|
nsp = xmlNewNs(nop, NULL, BAD_CAST pf);
|
|
|
|
// Set node namespace
|
|
nop->ns = nsp;
|
|
*(--p) = ':'; // Restore Xname
|
|
} else if (Doc->DefNs && xmlSearchNs(Docp, nop, NULL))
|
|
// Not in default namespace
|
|
nop->ns = xmlNewNs(nop, BAD_CAST "", NULL);
|
|
|
|
if (np)
|
|
((PNODE2)np)->Nodep = nop;
|
|
else
|
|
np = new(g) XML2NODE(Doc, nop);
|
|
|
|
return NewChild(np);
|
|
} // end of AddChildNode
|
|
|
|
/******************************************************************/
|
|
/* Add a new property to this node and return it. */
|
|
/******************************************************************/
|
|
PXATTR XML2NODE::AddProperty(PGLOBAL g, char *name, PXATTR ap)
|
|
{
|
|
if (trace)
|
|
htrc("AddProperty: %s\n", name);
|
|
|
|
xmlAttrPtr atp = xmlNewProp(Nodep, BAD_CAST name, NULL);
|
|
|
|
if (atp) {
|
|
if (ap) {
|
|
((PATTR2)ap)->Atrp = atp;
|
|
((PATTR2)ap)->Parent = Nodep;
|
|
return ap;
|
|
} else
|
|
return new(g) XML2ATTR(Doc, atp, Nodep);
|
|
|
|
} else
|
|
return NULL;
|
|
|
|
} // end of AddProperty
|
|
|
|
/******************************************************************/
|
|
/* Add a new text node to this node. */
|
|
/******************************************************************/
|
|
void XML2NODE::AddText(PGLOBAL g, char *txtp)
|
|
{
|
|
if (trace)
|
|
htrc("AddText: %s\n", txtp);
|
|
|
|
// This is to avoid a blank line when inserting a new line
|
|
xmlNodePtr np = xmlGetLastChild(Nodep);
|
|
|
|
if (np && np->type == XML_TEXT_NODE) {
|
|
xmlUnlinkNode(np);
|
|
xmlFreeNode(np);
|
|
} // endif type
|
|
|
|
// Add the new text
|
|
xmlAddChild(Nodep, xmlNewText(BAD_CAST txtp));
|
|
} // end of AddText
|
|
|
|
/******************************************************************/
|
|
/* Remove a child node from this node. */
|
|
/******************************************************************/
|
|
void XML2NODE::DeleteChild(PGLOBAL g, PXNODE dnp)
|
|
{
|
|
xmlErrorPtr xerr;
|
|
|
|
if (trace)
|
|
htrc("DeleteChild: node=%p\n", dnp);
|
|
|
|
xmlNodePtr np = ((PNODE2)dnp)->Nodep;
|
|
xmlNodePtr text = np->next;
|
|
|
|
// This is specific to row nodes
|
|
if (text && text->type == XML_TEXT_NODE) {
|
|
xmlUnlinkNode(text);
|
|
|
|
if ((xerr = xmlGetLastError()))
|
|
goto err;
|
|
|
|
xmlFreeNode(text);
|
|
|
|
if ((xerr = xmlGetLastError()))
|
|
goto err;
|
|
|
|
} // endif type
|
|
|
|
xmlUnlinkNode(np);
|
|
|
|
if ((xerr = xmlGetLastError()))
|
|
goto err;
|
|
|
|
xmlFreeNode(np);
|
|
|
|
if ((xerr = xmlGetLastError()))
|
|
goto err;
|
|
|
|
Delete(dnp);
|
|
|
|
if ((xerr = xmlGetLastError()))
|
|
goto err;
|
|
|
|
return;
|
|
|
|
err:
|
|
if (trace)
|
|
htrc("DeleteChild: errmsg=%s\n", xerr->message);
|
|
|
|
xmlResetError(xerr);
|
|
} // end of DeleteChild
|
|
|
|
/* -------------------- class XML2NODELIST ---------------------- */
|
|
|
|
/******************************************************************/
|
|
/* XML2NODELIST constructor. */
|
|
/******************************************************************/
|
|
XML2NODELIST::XML2NODELIST(PXDOC dp, xmlNodeSetPtr lp)
|
|
: XMLNODELIST(dp)
|
|
{
|
|
Listp = lp;
|
|
} // end of XML2NODELIST constructor
|
|
|
|
/******************************************************************/
|
|
/* Return the length of the list. */
|
|
/******************************************************************/
|
|
int XML2NODELIST::GetLength(void)
|
|
{
|
|
return (Listp) ? Listp->nodeNr : 0;
|
|
} // end of GetLength
|
|
|
|
/******************************************************************/
|
|
/* Return the nth element of the list. */
|
|
/******************************************************************/
|
|
PXNODE XML2NODELIST::GetItem(PGLOBAL g, int n, PXNODE np)
|
|
{
|
|
if (trace)
|
|
htrc("GetItem: %d\n", n);
|
|
|
|
if (!Listp || Listp->nodeNr <= n)
|
|
return NULL;
|
|
|
|
if (np) {
|
|
((PNODE2)np)->Nodep = Listp->nodeTab[n];
|
|
return np;
|
|
} else
|
|
return new(g) XML2NODE(Doc, Listp->nodeTab[n]);
|
|
|
|
} // end of GetItem
|
|
|
|
/******************************************************************/
|
|
/* Reset the pointer on the deleted item. */
|
|
/******************************************************************/
|
|
bool XML2NODELIST::DropItem(PGLOBAL g, int n)
|
|
{
|
|
if (trace)
|
|
htrc("DropItem: n=%d\n", n);
|
|
|
|
// We should do something here
|
|
if (!Listp || Listp->nodeNr <= n)
|
|
return true;
|
|
|
|
Listp->nodeTab[n] = NULL; // This was causing Valgrind warning
|
|
return false;
|
|
} // end of DropItem
|
|
|
|
/* ---------------------- class XML2ATTR ------------------------ */
|
|
|
|
/******************************************************************/
|
|
/* XML2ATTR constructor. */
|
|
/******************************************************************/
|
|
XML2ATTR::XML2ATTR(PXDOC dp, xmlAttrPtr ap, xmlNodePtr np)
|
|
: XMLATTRIBUTE(dp)
|
|
{
|
|
Atrp = ap;
|
|
Parent = np;
|
|
} // end of XML2ATTR constructor
|
|
|
|
/******************************************************************/
|
|
/* Set the content of an attribute. */
|
|
/******************************************************************/
|
|
bool XML2ATTR::SetText(PGLOBAL g, char *txtp, int len)
|
|
{
|
|
if (trace)
|
|
htrc("SetText: %s %d\n", txtp, len);
|
|
|
|
xmlSetProp(Parent, Atrp->name, BAD_CAST txtp);
|
|
return false;
|
|
} // end of SetText
|