123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303 |
- /*##############################################################################
- HPCC SYSTEMS software Copyright (C) 2018 HPCC Systems®.
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- ############################################################################## */
- #include "jstring.hpp"
- #include "jdebug.hpp"
- #include "jptree.hpp"
- #include "jexcept.hpp"
- #include "jlog.hpp"
- #include <libxml/xmlmemory.h>
- #include <libxml/parserInternals.h>
- #include <libxml/debugXML.h>
- #include <libxml/HTMLtree.h>
- #include <libxml/xmlIO.h>
- #include <libxml/xinclude.h>
- #include <libxml/catalog.h>
- #include <libxml/xpathInternals.h>
- #include <libxml/xpath.h>
- #include <libxml/xmlschemas.h>
- #include <libxml/hash.h>
- #include "xpathprocessor.hpp"
- class CLibCompiledXpath : public CInterface, public ICompiledXpath
- {
- private:
- xmlXPathCompExprPtr m_compiledXpathExpression = nullptr;
- StringBuffer m_xpath;
- ReadWriteLock m_rwlock;
- public:
- IMPLEMENT_IINTERFACE;
- CLibCompiledXpath(const char * xpath)
- {
- m_xpath.set(xpath);
- m_compiledXpathExpression = xmlXPathCompile(BAD_CAST m_xpath.str());
- }
- ~CLibCompiledXpath()
- {
- xmlXPathFreeCompExpr(m_compiledXpathExpression);
- }
- const char * getXpath()
- {
- return m_xpath.str();
- }
- xmlXPathCompExprPtr getCompiledXPathExpression()
- {
- return m_compiledXpathExpression;
- }
- };
- class CLibXpathContext : public CInterface, public IXpathContext
- {
- private:
- xmlDocPtr m_xmlDoc = nullptr;
- xmlXPathContextPtr m_xpathContext = nullptr;
- StringBuffer m_xpath;
- ReadWriteLock m_rwlock;
- public:
- IMPLEMENT_IINTERFACE;
- CLibXpathContext(const char * xmldoc) //not thread safe
- {
- setXmlDoc(xmldoc);
- }
- ~CLibXpathContext()
- {
- xmlXPathFreeContext(m_xpathContext);
- xmlFreeDoc(m_xmlDoc);
- }
- static void tableScanCallback(void *payload, void *data, xmlChar *name)
- {
- DBGLOG("k/v == [%s,%s]\n", (char *) name, (char *) payload);
- }
- virtual const char * getXpath() override
- {
- return m_xpath.str();
- }
- virtual bool addVariable(const char * name, const char * val) override
- {
- WriteLockBlock wblock(m_rwlock);
- if (m_xpathContext && val && *val)
- {
- return xmlXPathRegisterVariable(m_xpathContext, (xmlChar *)name, xmlXPathNewCString(val)) == 0;
- }
- return false;
- }
- virtual const char * getVariable(const char * name) override
- {
- ReadLockBlock rblock(m_rwlock);
- if (m_xpathContext)
- return (const char *)xmlXPathCastToString(xmlXPathVariableLookup(m_xpathContext, (const xmlChar *)name));
- else
- return nullptr;
- }
- virtual bool evaluateAsBoolean(const char * xpath)
- {
- bool bresult = false;
- if (xpath && *xpath)
- {
- xmlXPathObjectPtr evaluatedXpathObj = evaluate(xpath);
- if (evaluatedXpathObj && evaluatedXpathObj->type == XPATH_BOOLEAN)
- {
- bresult = evaluatedXpathObj->boolval;
- }
- else
- throw MakeStringException(-1,"XpathProcessor:evaluateAsBoolean: Error: Could not evaluate XPATH '%s' as boolean", m_xpath.str());
- xmlXPathFreeObject(evaluatedXpathObj);
- }
- else
- throw MakeStringException(-1,"XpathProcessor:evaluateAsBoolean: Error: empty xpath provided");
- return bresult;
- }
- virtual bool evaluateAsString(const char * xpath, StringBuffer & evaluated)
- {
- if (xpath && *xpath)
- {
- xmlXPathObjectPtr evaluatedXpathObj = evaluate(xpath);
- evaluated.clear();
- if (evaluatedXpathObj && evaluatedXpathObj->type == XPATH_NODESET)
- {
- xmlNodeSetPtr nodes = evaluatedXpathObj->nodesetval;
- for (int i = 0; i < nodes->nodeNr; i++)
- {
- xmlNodePtr nodeTab = nodes->nodeTab[i];
- evaluated.append((const char *)xmlNodeGetContent(nodeTab));
- }
- xmlXPathFreeObject(evaluatedXpathObj);
- }
- else
- throw MakeStringException(-1,"XpathProcessor:evaluateAsString: Error: Could not evaluate XPATH as string");
- }
- else
- throw MakeStringException(-1,"XpathProcessor:evaluateAsString: Error: empty xpath provided");
- return evaluated.str();
- }
- virtual bool evaluateAsBoolean(ICompiledXpath * compiledXpath) override
- {
- bool bresult = false;
- CLibCompiledXpath * clibCompiledXpath = static_cast<CLibCompiledXpath *>(compiledXpath);
- xmlXPathObjectPtr evaluatedXpathObj = evaluate(clibCompiledXpath->getCompiledXPathExpression());
- if (evaluatedXpathObj && evaluatedXpathObj->type == XPATH_BOOLEAN)
- {
- bresult = evaluatedXpathObj->boolval;
- }
- else
- throw MakeStringException(-1,"XpathProcessor:evaluateAsBoolean: Error: Could not evaluate XPATH '%s' as boolean", m_xpath.str());
- xmlXPathFreeObject(evaluatedXpathObj);
- return bresult;
- }
- virtual const char * evaluateAsString(ICompiledXpath * compiledXpath, StringBuffer & evaluated) override
- {
- CLibCompiledXpath * clibCompiledXpath = static_cast<CLibCompiledXpath *>(compiledXpath);
- xmlXPathObjectPtr evaluatedXpathObj = evaluate(clibCompiledXpath->getCompiledXPathExpression());
- evaluated.clear();
- if (evaluatedXpathObj)
- {
- switch (evaluatedXpathObj->type)
- {
- case XPATH_NODESET:
- {
- xmlNodeSetPtr nodes = evaluatedXpathObj->nodesetval;
- for (int i = 0; i < nodes->nodeNr; i++)
- {
- xmlNodePtr nodeTab = nodes->nodeTab[i];
- evaluated.append((const char *)xmlNodeGetContent(nodeTab));
- }
- break;
- }
- case XPATH_BOOLEAN:
- case XPATH_NUMBER:
- case XPATH_STRING:
- case XPATH_POINT:
- case XPATH_RANGE:
- case XPATH_LOCATIONSET:
- case XPATH_USERS:
- case XPATH_XSLT_TREE:
- {
- evaluatedXpathObj = xmlXPathConvertString (evaluatedXpathObj); //existing object is freed
- if (!evaluatedXpathObj)
- throw MakeStringException(-1,"XpathProcessor:evaluateAsString: could not convert result to string");
- evaluated.append(evaluatedXpathObj->stringval);
- break;
- }
- default:
- {
- xmlXPathFreeObject(evaluatedXpathObj);
- throw MakeStringException(-1,"XpathProcessor:evaluateAsString: Error: Encountered unsupported XPATH type");
- break;
- }
- }
- }
- else
- {
- xmlXPathFreeObject(evaluatedXpathObj);
- throw MakeStringException(-1,"XpathProcessor:evaluateAsString: Error: Could not evaluate XPATH as string");
- }
- xmlXPathFreeObject(evaluatedXpathObj);
- return evaluated.str();
- }
- private:
- virtual bool setXmlDoc(const char * xmldoc) override
- {
- if (xmldoc && * xmldoc)
- {
- m_xmlDoc = xmlParseDoc((const unsigned char *)xmldoc);
- if (m_xmlDoc == nullptr)
- {
- ERRLOG("XpathProcessor:setxmldoc Error: Unable to parse XMLLib document");
- return false;
- }
- // Create xpath evaluation context
- m_xpathContext = xmlXPathNewContext(m_xmlDoc);
- if(m_xpathContext == nullptr)
- {
- ERRLOG("XpathProcessor:setxmldoc: Error: Unable to create new XMLLib XPath context");
- return false;
- }
- return true;
- }
- return false;
- }
- virtual xmlXPathObjectPtr evaluate(xmlXPathCompExprPtr compiledXpath)
- {
- xmlXPathObjectPtr evaluatedXpathObj = nullptr;
- if (compiledXpath)
- {
- ReadLockBlock rlock(m_rwlock);
- if ( m_xpathContext)
- {
- evaluatedXpathObj = xmlXPathCompiledEval(compiledXpath, m_xpathContext);
- }
- else
- {
- throw MakeStringException(-1,"XpathProcessor:evaluate: Error: Invalid xpathCotext detected. Ensure xmldoc has been set");
- }
- }
- return evaluatedXpathObj;
- }
- virtual xmlXPathObjectPtr evaluate(const char * xpath)
- {
- xmlXPathObjectPtr evaluatedXpathObj = nullptr;
- if (xpath && *xpath)
- {
- ReadLockBlock rlock(m_rwlock);
- if ( m_xpathContext)
- {
- evaluatedXpathObj = xmlXPathEval((const xmlChar *)xpath, m_xpathContext);
- }
- else
- {
- throw MakeStringException(-1,"XpathProcessor:evaluate: Error: Invalid xpathCotext detected. Ensure xmldoc has been set");
- }
- }
- return evaluatedXpathObj;
- }
- };
- extern ICompiledXpath* getCompiledXpath(const char * xpath)
- {
- return new CLibCompiledXpath(xpath);
- }
- extern IXpathContext* getXpathContext(const char * xmldoc)
- {
- return new CLibXpathContext(xmldoc);
- }
|