arabica/include/XSLT/impl/xslt_functions.hpp

442 lines
17 KiB
C++
Raw Normal View History

2007-07-19 19:01:42 +02:00
#ifndef ARABICA_XSLT_FUNCTIONS_HPP
#define ARABICA_XSLT_FUNCTIONS_HPP
2007-08-22 14:22:57 +02:00
2007-09-05 00:55:47 +02:00
#include <DOM/SAX2DOM/SAX2DOM.hpp>
#include <SAX/helpers/CatchErrorHandler.hpp>
2007-08-22 14:22:57 +02:00
namespace Arabica
{
namespace XSLT
{
// node-set document(object, node-set?)
2012-11-08 22:48:35 +01:00
template<class string_type, class string_adaptor>
2012-11-08 17:53:00 +01:00
class DocumentFunction : public Arabica::XPath::NodeSetXPathFunction<string_type, string_adaptor>
2007-08-22 14:22:57 +02:00
{
2012-11-08 17:53:00 +01:00
typedef Arabica::XPath::NodeSetXPathFunction<string_type, string_adaptor> baseT;
2007-08-22 14:22:57 +02:00
2012-11-08 22:48:35 +01:00
typedef Arabica::XPath::XPathExpression<string_type, string_adaptor> XPathExpression;
typedef std::vector<XPathExpression> ArgList;
typedef Arabica::XPath::XPathValue<string_type, string_adaptor> XPathValue;
typedef Arabica::XPath::NodeSet<string_type, string_adaptor> NodeSet;
typedef Arabica::XPath::ExecutionContext<string_type, string_adaptor> XPathExecutionContext;
typedef DOM::Node<string_type, string_adaptor> DOMNode;
2007-08-22 14:22:57 +02:00
public:
static const string_type& name()
{
static const string_type n = string_adaptor::construct_from_utf8("document");
return n;
} // name
2012-11-08 17:53:00 +01:00
DocumentFunction(const string_type& currentBase,
const ArgList& args) :
2012-11-08 22:48:35 +01:00
baseT(1, 2, args),
2007-08-23 15:55:06 +02:00
baseURI_(currentBase)
2007-08-22 14:22:57 +02:00
{ }
protected:
2012-11-08 17:53:00 +01:00
virtual NodeSet doEvaluate(const DOMNode& context,
const XPathExecutionContext& executionContext) const
2007-08-22 14:22:57 +02:00
{
2012-11-08 17:53:00 +01:00
NodeSet nodes;
2007-08-25 13:51:29 +02:00
2013-01-05 22:26:35 +01:00
XPathValue a0 = this->arg(0, context, executionContext);
2012-11-08 18:21:57 +01:00
if(baseT::argCount() != 1)
throw Arabica::XPath::UnsupportedException("two arg version of document()");
if(a0.type() != Arabica::XPath::STRING)
2007-08-25 13:51:29 +02:00
throw Arabica::XPath::UnsupportedException("node-set arg version of document()");
load_document(a0.asString(), nodes);
return nodes;
} // doEvaluate
2007-08-22 14:22:57 +02:00
private:
2012-11-08 17:53:00 +01:00
void load_document(const string_type& location, NodeSet& nodes) const
2007-08-22 14:22:57 +02:00
{
2012-11-08 17:53:00 +01:00
SAX2DOM::Parser<string_type, string_adaptor> domParser;
SAX::CatchErrorHandler<string_type, string_adaptor> eh;
2007-08-22 14:22:57 +02:00
domParser.setErrorHandler(eh);
Arabica::io::URI base(string_adaptor::asStdString(baseURI_));
Arabica::io::URI absolute(base, string_adaptor::asStdString(location));
SAX::InputSource<string_type, string_adaptor> is(string_adaptor::construct_from_utf8(absolute.as_string().c_str()));
2007-08-22 14:22:57 +02:00
domParser.parse(is);
if(!eh.errorsReported())
2012-11-08 17:53:00 +01:00
nodes.push_back(domParser.getDocument());
2007-08-22 14:22:57 +02:00
else
2012-11-08 17:53:00 +01:00
std::cerr << eh.errors() << std::endl;
2007-08-22 14:22:57 +02:00
} // load_document
2007-08-23 15:55:06 +02:00
2012-11-08 17:53:00 +01:00
string_type baseURI_;
2007-08-22 14:22:57 +02:00
}; // DocumentFunction
// node-set key(string, object)
2012-11-08 22:48:35 +01:00
template<class string_type, class string_adaptor>
2012-11-08 17:53:00 +01:00
class KeyFunction : public Arabica::XPath::NodeSetXPathFunction<string_type, string_adaptor>
{
2012-11-08 17:53:00 +01:00
typedef Arabica::XPath::NodeSetXPathFunction<string_type, string_adaptor> baseT;
2012-11-08 22:48:35 +01:00
typedef Arabica::XPath::XPathExpression<string_type, string_adaptor> XPathExpression;
typedef std::vector<XPathExpression> ArgList;
typedef Arabica::XPath::XPathValue<string_type, string_adaptor> XPathValue;
typedef Arabica::XPath::NodeSet<string_type, string_adaptor> NodeSet;
typedef Arabica::XPath::ExecutionContext<string_type, string_adaptor> XPathExecutionContext;
typedef DOM::Node<string_type, string_adaptor> DOMNode;
typedef XML::QualifiedName<string_type, string_adaptor> QualifiedName;
public:
static const string_type& name()
{
static const string_type n = string_adaptor::construct_from_utf8("key");
return n;
} // name
2012-11-08 17:53:00 +01:00
KeyFunction(const DeclaredKeys<string_type, string_adaptor>& keys,
const std::map<string_type, string_type>& inscopeNamespaces,
const ArgList& args) :
2012-11-08 22:48:35 +01:00
baseT(2, 2, args),
keys_(keys),
namespaces_(inscopeNamespaces)
{
} // KeyFunction
protected:
2012-11-08 17:53:00 +01:00
virtual NodeSet doEvaluate(const DOMNode& context,
const XPathExecutionContext& executionContext) const
{
2013-01-05 22:26:35 +01:00
string_type keyname = this->argAsString(0, context, executionContext);
2012-11-08 17:53:00 +01:00
string_type keyClarkName = QualifiedName::parseQName(keyname, true, namespaces_).clarkName();
2009-03-03 08:58:29 +01:00
2012-11-08 17:53:00 +01:00
XPathValue a1 = baseT::arg(1, context, executionContext);
if(a1.type() == Arabica::XPath::NODE_SET)
2009-03-03 08:58:29 +01:00
return nodeSetUnion(keyClarkName, a1.asNodeSet(), executionContext);
return keys_.lookup(keyClarkName, a1.asString(), executionContext);
} // doEvaluate
2012-11-08 17:53:00 +01:00
NodeSet nodeSetUnion(const string_type& keyClarkName,
const NodeSet nodes,
const XPathExecutionContext& executionContext) const
2009-03-03 08:58:29 +01:00
{
2012-11-08 17:53:00 +01:00
NodeSet results;
for(typename NodeSet::const_iterator n = nodes.begin(), ne = nodes.end(); n != ne; ++n)
2009-03-03 08:58:29 +01:00
{
2012-11-08 17:53:00 +01:00
string_type id = Arabica::XPath::impl::nodeStringValue<string_type, string_adaptor>(*n);
2009-03-03 08:58:29 +01:00
results.push_back(keys_.lookup(keyClarkName, id, executionContext));
} // for ...
results.to_document_order();
return results;
2009-03-03 08:58:29 +01:00
} // nodeSetUnion
private:
2012-11-08 17:53:00 +01:00
const DeclaredKeys<string_type, string_adaptor>& keys_;
std::map<string_type, string_type> namespaces_;
}; // class KeyFunction
// string format-number(number, string, string?)
// node-set current()
2012-11-08 22:48:35 +01:00
template<class string_type, class string_adaptor>
2012-11-08 17:53:00 +01:00
class CurrentFunction : public Arabica::XPath::NodeSetXPathFunction<string_type, string_adaptor>
{
2012-11-08 17:53:00 +01:00
typedef Arabica::XPath::NodeSetXPathFunction<string_type, string_adaptor> baseT;
2012-11-08 22:48:35 +01:00
typedef Arabica::XPath::XPathExpression<string_type, string_adaptor> XPathExpression;
typedef std::vector<XPathExpression> ArgList;
typedef Arabica::XPath::NodeSet<string_type, string_adaptor> NodeSet;
typedef Arabica::XPath::ExecutionContext<string_type, string_adaptor> XPathExecutionContext;
typedef DOM::Node<string_type, string_adaptor> DOMNode;
public:
static const string_type& name()
{
static const string_type n = string_adaptor::construct_from_utf8("current");
return n;
} // name
2012-11-08 17:53:00 +01:00
CurrentFunction(const ArgList& args) :
2012-11-08 22:48:35 +01:00
baseT(0, 0, args) { }
protected:
2012-11-08 17:53:00 +01:00
virtual NodeSet doEvaluate(const DOMNode& /* context */,
const XPathExecutionContext& executionContext) const
{
2012-11-08 17:53:00 +01:00
NodeSet set;
set.push_back(executionContext.currentNode());
return set;
} // doEvaluate
}; // CurrentFunction
// string unparsed-entity-uri(string)
2012-11-08 22:48:35 +01:00
template<class string_type, class string_adaptor>
2012-11-08 17:53:00 +01:00
class UnparsedEntityUriFunction : public Arabica::XPath::StringXPathFunction<string_type, string_adaptor>
{
2012-11-08 22:48:35 +01:00
typedef Arabica::XPath::StringXPathFunction<string_type, string_adaptor> baseT;
typedef Arabica::XPath::XPathExpression<string_type, string_adaptor> XPathExpression;
typedef std::vector<XPathExpression> ArgList;
typedef Arabica::XPath::ExecutionContext<string_type, string_adaptor> XPathExecutionContext;
typedef DOM::Node<string_type, string_adaptor> DOMNode;
public:
static const string_type& name()
{
static const string_type n = string_adaptor::construct_from_utf8("unparsed-entity-uri");
return n;
} // name
2012-11-08 17:53:00 +01:00
UnparsedEntityUriFunction(const ArgList& args) :
2012-11-08 22:48:35 +01:00
baseT(1, 1, args) { }
protected:
2012-11-08 17:53:00 +01:00
virtual string_type doEvaluate(const DOMNode& /* context */,
const XPathExecutionContext& /* executionContext */) const
{
// This is a minimal, but I think conformant, implementation
return string_adaptor::empty_string();
} // evaluate
}; // UnparsedEntityUri
// string generate-id(node-set?)
2012-11-08 22:48:35 +01:00
template<class string_type, class string_adaptor>
2012-11-08 17:53:00 +01:00
class GenerateIdFunction : public Arabica::XPath::StringXPathFunction<string_type, string_adaptor>
2008-10-27 20:13:47 +01:00
{
2012-11-08 17:53:00 +01:00
typedef Arabica::XPath::StringXPathFunction<string_type, string_adaptor> baseT;
2012-11-08 22:48:35 +01:00
typedef Arabica::XPath::XPathExpression<string_type, string_adaptor> XPathExpression;
typedef std::vector<XPathExpression> ArgList;
typedef Arabica::XPath::NodeSet<string_type, string_adaptor> NodeSet;
typedef Arabica::XPath::ExecutionContext<string_type, string_adaptor> XPathExecutionContext;
typedef DOM::Node<string_type, string_adaptor> DOMNode;
2008-10-27 20:13:47 +01:00
public:
static const string_type& name()
{
static const string_type n = string_adaptor::construct_from_utf8("generate-id");
return n;
} // name
2012-11-08 17:53:00 +01:00
GenerateIdFunction(const ArgList& args) :
2012-11-08 22:48:35 +01:00
baseT(0, 1, args) { }
2008-10-27 20:13:47 +01:00
protected:
2012-11-08 17:53:00 +01:00
virtual string_type doEvaluate(const DOMNode& context,
const XPathExecutionContext& executionContext) const
2008-10-27 20:13:47 +01:00
{
2012-11-08 17:53:00 +01:00
DOMNode node;
2008-10-27 20:13:47 +01:00
if(baseT::argCount() == 0)
node = context;
else
{
2012-11-08 17:53:00 +01:00
NodeSet ns = baseT::argAsNodeSet(0, context, executionContext);
2008-10-27 20:13:47 +01:00
if(ns.size() == 0)
return string_adaptor::empty_string();
2008-10-27 20:13:47 +01:00
node = ns.top();
} // if ...
std::basic_ostringstream<typename string_adaptor::value_type> os;
2008-10-27 20:13:47 +01:00
os << node.underlying_impl();
2012-11-21 07:59:40 +01:00
return string_adaptor::construct(os.str());
} // doEvaluate
2008-10-27 20:13:47 +01:00
}; // class GenerateIdFunction
2007-08-22 14:22:57 +02:00
// object system-property(string)
2012-11-08 22:48:35 +01:00
template<class string_type, class string_adaptor>
2012-11-08 17:53:00 +01:00
class SystemPropertyFunction : public Arabica::XPath::StringXPathFunction<string_type, string_adaptor>
2007-08-22 14:22:57 +02:00
{
2012-11-08 17:53:00 +01:00
typedef Arabica::XPath::StringXPathFunction<string_type, string_adaptor> baseT;
2012-11-08 22:48:35 +01:00
typedef Arabica::XPath::XPathExpression<string_type, string_adaptor> XPathExpression;
typedef std::vector<XPathExpression> ArgList;
typedef Arabica::XPath::ExecutionContext<string_type, string_adaptor> XPathExecutionContext;
typedef DOM::Node<string_type, string_adaptor> DOMNode;
2007-08-22 14:22:57 +02:00
public:
static const string_type& name()
{
static const string_type n = string_adaptor::construct_from_utf8("system-property");
return n;
} // name
2012-11-08 17:53:00 +01:00
SystemPropertyFunction (const ArgList& args) :
2012-11-08 22:48:35 +01:00
baseT(1, 1, args) { }
protected:
2012-11-08 17:53:00 +01:00
virtual string_type doEvaluate(const DOMNode& context,
const XPathExecutionContext& executionContext) const
2007-08-22 14:22:57 +02:00
{
typedef StylesheetConstant<string_type, string_adaptor> SC;
2012-11-08 17:53:00 +01:00
string_type property = baseT::argAsString(0, context, executionContext);
string_type result;
if(property == SC::version_property)
return SC::Version;
if(property == SC::vendor_property)
return SC::Vendor;
else if(property == SC::vendor_url_property)
return SC::VendorUrl;
return string_adaptor::empty_string();
2007-08-22 14:22:57 +02:00
} // evaluate
}; // SystemPropertyFunction
// boolean element-available(string)
2012-11-08 22:48:35 +01:00
template<class string_type, class string_adaptor>
2012-11-08 17:53:00 +01:00
class ElementAvailableFunction : public Arabica::XPath::BooleanXPathFunction<string_type, string_adaptor>
{
2012-11-08 17:53:00 +01:00
typedef Arabica::XPath::BooleanXPathFunction<string_type, string_adaptor> baseT;
2012-11-08 22:48:35 +01:00
typedef Arabica::XPath::XPathExpression<string_type, string_adaptor> XPathExpression;
typedef std::vector<XPathExpression> ArgList;
typedef Arabica::XPath::ExecutionContext<string_type, string_adaptor> XPathExecutionContext;
typedef DOM::Node<string_type, string_adaptor> DOMNode;
typedef XML::QualifiedName<string_type, string_adaptor> QualifiedName;
public:
static const string_type& name()
{
2012-11-16 10:17:55 +01:00
static const string_type n = string_adaptor::construct_from_utf8("element-available");
return n;
} // name
2012-11-08 17:53:00 +01:00
ElementAvailableFunction(const std::vector<std::pair<string_type, string_type> >& names,
const std::map<string_type, string_type>& inscopeNamespaces,
const ArgList& args) :
2012-11-08 22:48:35 +01:00
baseT(1, 1, args),
namespaces_(inscopeNamespaces),
elementNames_(names)
{
} // ElementAvailableFunction
protected:
2012-11-08 17:53:00 +01:00
virtual bool doEvaluate(const DOMNode& context,
const XPathExecutionContext& executionContext) const
{
typedef StylesheetConstant<string_type, string_adaptor> SC;
2012-11-08 17:53:00 +01:00
const string_type functionName = baseT::argAsString(0, context, executionContext);
const QualifiedName expandedName = QualifiedName::parseQName(functionName, true, namespaces_);
if((expandedName.namespaceUri() != StylesheetConstant<string_type, string_adaptor>::NamespaceURI) &&
2012-11-21 07:59:40 +01:00
(!string_adaptor::empty(expandedName.namespaceUri())))
return false;
static string_type XSLTNames[] = { SC::apply_imports,
SC::apply_templates,
SC::attribute,
SC::call_template,
SC::choose,
SC::comment,
SC::copy,
SC::copy_of,
SC::element,
SC::fallback,
SC::for_each,
SC::if_,
SC::message,
SC::number,
SC::processing_instruction,
SC::text,
SC::value_of,
SC::variable,
string_adaptor::empty_string() };
for(int i = 0; XSLTNames[i] != string_adaptor::empty_string(); ++i)
if(expandedName.localName() == XSLTNames[i])
2012-11-08 17:53:00 +01:00
return true;
return false;
} // doEvaluate
private:
2012-11-08 17:53:00 +01:00
const std::map<string_type, string_type> namespaces_;
std::vector<std::pair<string_type, string_type> > elementNames_;
}; // class ElementAvailableFunction
// boolean function-available(string)
2012-11-08 22:48:35 +01:00
template<class string_type, class string_adaptor>
2012-11-08 17:53:00 +01:00
class FunctionAvailableFunction : public Arabica::XPath::BooleanXPathFunction<string_type, string_adaptor>
{
2012-11-08 17:53:00 +01:00
typedef Arabica::XPath::BooleanXPathFunction<string_type, string_adaptor> baseT;
2012-11-08 22:48:35 +01:00
typedef Arabica::XPath::XPathExpression<string_type, string_adaptor> XPathExpression;
typedef std::vector<XPathExpression> ArgList;
typedef Arabica::XPath::ExecutionContext<string_type, string_adaptor> XPathExecutionContext;
typedef DOM::Node<string_type, string_adaptor> DOMNode;
typedef XML::QualifiedName<string_type, string_adaptor> QualifiedName;
public:
static const string_type& name()
{
static const string_type n = string_adaptor::construct_from_utf8("function-available");
return n;
} // name
2012-11-08 17:53:00 +01:00
FunctionAvailableFunction(const std::vector<std::pair<string_type, string_type> >& names,
const std::map<string_type, string_type>& inscopeNamespaces,
const ArgList& args) :
2012-11-08 22:48:35 +01:00
baseT(1, 1, args),
namespaces_(inscopeNamespaces),
functionNames_(names)
{
2012-11-08 17:53:00 +01:00
Arabica::XPath::StandardXPathFunctionResolver<string_type, string_adaptor> standardResolver;
const std::vector<std::pair<string_type, string_type> > standardNames = standardResolver.validNames();
functionNames_.insert(functionNames_.begin(), standardNames.begin(), standardNames.end());
} // FunctionAvailableFunction
protected:
2012-11-08 17:53:00 +01:00
virtual bool doEvaluate(const DOMNode& context,
2012-11-08 22:48:35 +01:00
const XPathExecutionContext& executionContext) const
{
2012-11-08 17:53:00 +01:00
const string_type functionName = baseT::argAsString(0, context, executionContext);
const QualifiedName expandedName = QualifiedName::parseQName(functionName, true, namespaces_);
2012-11-08 17:53:00 +01:00
const std::pair<string_type, string_type> name_to_check = std::make_pair(expandedName.namespaceUri(), expandedName.localName());
return (std::find(functionNames_.begin(), functionNames_.end(), name_to_check) != functionNames_.end());
} // doEvaluate
private:
2012-11-08 17:53:00 +01:00
const std::map<string_type, string_type> namespaces_;
std::vector<std::pair<string_type, string_type> > functionNames_;
}; // class FunctionAvailableFunction
2007-07-19 19:01:42 +02:00
2012-11-08 22:48:35 +01:00
template<class string_type, class string_adaptor>
2012-11-08 17:53:00 +01:00
class UndefinedFunction : public Arabica::XPath::BooleanXPathFunction<string_type, string_adaptor>
{
2012-11-08 22:48:35 +01:00
typedef Arabica::XPath::BooleanXPathFunction<string_type, string_adaptor> baseT;
typedef Arabica::XPath::XPathExpression<string_type, string_adaptor> XPathExpression;
typedef std::vector<XPathExpression> ArgList;
typedef Arabica::XPath::ExecutionContext<string_type, string_adaptor> XPathExecutionContext;
typedef DOM::Node<string_type, string_adaptor> DOMNode;
public:
2012-11-08 17:53:00 +01:00
UndefinedFunction(const string_type namespace_uri,
2012-11-08 22:48:35 +01:00
const string_type name,
const ArgList& args) :
baseT(-1, -1, args)
{
2012-11-15 23:25:55 +01:00
typedef Arabica::text::Unicode<typename string_adaptor::value_type> UnicodeT;
2012-11-21 07:59:40 +01:00
if(!string_adaptor::empty(namespace_uri))
{
2012-11-21 07:59:40 +01:00
string_adaptor::append(error_, UnicodeT::LEFT_SQUARE_BRACKET);
string_adaptor::append(error_, namespace_uri);
string_adaptor::append(error_, UnicodeT::RIGHT_SQUARE_BRACKET);
} // if ..
2012-11-21 07:59:40 +01:00
string_adaptor::append(error_, name);
} // UndefinedFunction
protected:
2012-11-08 22:48:35 +01:00
virtual bool doEvaluate(const DOMNode&, const XPathExecutionContext&) const
{
throw Arabica::XPath::UndefinedFunctionException(string_adaptor::asStdString(error_));
} // doEvaluate
2012-11-08 17:53:00 +01:00
string_type error_;
}; // class UndefinedFunction
2007-07-19 19:01:42 +02:00
} // namespace XSLT
} // namespace Arabica
#endif // ARABICA_XSLT_FUNCTIONS_HPP
2007-08-22 14:22:57 +02:00