mirror of
https://github.com/jezhiggins/arabica
synced 2025-01-07 05:24:22 +01:00
156 lines
7.2 KiB
C++
156 lines
7.2 KiB
C++
#ifndef ARABICA_XPATH_FUNCTION_HOLDER_HPP
|
|
#define ARABICA_XPATH_FUNCTION_HOLDER_HPP
|
|
|
|
#include <boost/shared_ptr.hpp>
|
|
#include "xpath_expression.hpp"
|
|
#include "xpath_function.hpp"
|
|
|
|
namespace Arabica
|
|
{
|
|
namespace XPath
|
|
{
|
|
namespace impl
|
|
{
|
|
|
|
template<class function_type, class string_type, class string_adaptor>
|
|
XPathFunction<string_type, string_adaptor>* CreateFn(const std::vector<XPathExpression<string_type, string_adaptor> >& argExprs) { return new function_type(argExprs); }
|
|
|
|
template<class string_type, class string_adaptor>
|
|
class StandardXPathFunctionResolver
|
|
{
|
|
public:
|
|
virtual ~StandardXPathFunctionResolver() { }
|
|
|
|
virtual XPathFunction<string_type, string_adaptor>*
|
|
resolveFunction(const string_type& namespace_uri,
|
|
const string_type& name,
|
|
const std::vector<XPathExpression<string_type, string_adaptor> >& argExprs) const
|
|
{
|
|
if(!string_adaptor::empty(namespace_uri))
|
|
return 0;
|
|
return standardFunction(name, argExprs);
|
|
} // resolveFuncton
|
|
|
|
static XPathFunction<string_type, string_adaptor>*
|
|
standardFunction(const string_type& name,
|
|
const std::vector<XPathExpression<string_type, string_adaptor> >& argExprs)
|
|
{
|
|
for(const NamedFunction* fn = FunctionLookupTable; fn->name != 0; ++fn)
|
|
if(name == string_adaptor::construct_from_utf8(fn->name))
|
|
return fn->creator(argExprs);
|
|
|
|
return 0;
|
|
} // standardFunction
|
|
|
|
private:
|
|
|
|
typedef XPathFunction<string_type, string_adaptor>* (*CreateFnPtr)(const std::vector<XPathExpression<string_type, string_adaptor> >& argExprs);
|
|
|
|
struct NamedFunction { const char* name; CreateFnPtr creator; };
|
|
|
|
static const NamedFunction FunctionLookupTable[];
|
|
}; // class StandardXPathFunctionResolver
|
|
|
|
template<class string_type, class string_adaptor>
|
|
const typename StandardXPathFunctionResolver<string_type, string_adaptor>::NamedFunction
|
|
StandardXPathFunctionResolver<string_type, string_adaptor>::FunctionLookupTable[] =
|
|
{ // node-set functions
|
|
{ "position", CreateFn<PositionFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{ "last", CreateFn<LastFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{ "count", CreateFn<CountFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{ "local-name", CreateFn<LocalNameFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{ "namespace-uri", CreateFn<NamespaceURIFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{ "name", CreateFn<NameFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
// string functions
|
|
{"string", CreateFn<StringFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"concat", CreateFn<ConcatFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"starts-with", CreateFn<StartsWithFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"contains", CreateFn<ContainsFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"substring-before", CreateFn<SubstringBeforeFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"substring-after", CreateFn<SubstringAfterFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"substring", CreateFn<SubstringFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"string-length", CreateFn<StringLengthFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"normalize-space", CreateFn<NormalizeSpaceFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"translate", CreateFn<TranslateFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
// boolean functions
|
|
{"boolean", CreateFn<BooleanFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"not", CreateFn<NotFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"true", CreateFn<TrueFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"false", CreateFn<FalseFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
// number functions
|
|
{"number", CreateFn<NumberFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"sum", CreateFn<SumFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"floor", CreateFn<FloorFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"ceiling", CreateFn<CeilingFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{"round", CreateFn<RoundFn<string_type, string_adaptor>, string_type, string_adaptor> },
|
|
{0, 0}
|
|
};
|
|
|
|
template<class string_type, class string_adaptor>
|
|
class FunctionHolder : public XPathExpression_impl<string_type, string_adaptor>
|
|
{
|
|
public:
|
|
FunctionHolder(XPathFunction<string_type, string_adaptor>* func,
|
|
const string_type& namespace_uri,
|
|
const string_type& name) :
|
|
func_(func),
|
|
namespace_uri_(namespace_uri),
|
|
name_(name)
|
|
{
|
|
} // FunctionHolder
|
|
|
|
virtual ~FunctionHolder()
|
|
{
|
|
delete func_;
|
|
} // ~FunctionHolder
|
|
|
|
const string_type& namespace_uri() const { return namespace_uri_; }
|
|
const string_type& name() const { return name_; }
|
|
|
|
virtual ValueType type() const { return func_->type(); }
|
|
|
|
virtual XPathValue<string_type, string_adaptor> evaluate(const DOM::Node<string_type, string_adaptor>& context,
|
|
const ExecutionContext<string_type, string_adaptor>& executionContext) const
|
|
{
|
|
return XPathValue<string_type, string_adaptor>(func_->evaluate(context, executionContext));
|
|
} // evaluate
|
|
|
|
static FunctionHolder* createFunction(const string_type& namespace_uri,
|
|
const string_type& name,
|
|
const std::vector<XPathExpression<string_type, string_adaptor> >& argExprs,
|
|
const CompilationContext<string_type, string_adaptor>& context)
|
|
{
|
|
XPathFunction<string_type, string_adaptor>* func = 0;
|
|
|
|
if(string_adaptor::empty(namespace_uri))
|
|
func = StandardXPathFunctionResolver<string_type, string_adaptor>::standardFunction(name, argExprs);
|
|
if(func == 0)
|
|
func = context.functionResolver().resolveFunction(namespace_uri, name, argExprs);
|
|
|
|
if(func == 0)
|
|
{
|
|
string_type error;
|
|
if(!string_adaptor::empty(namespace_uri))
|
|
{
|
|
string_adaptor::append(error, string_adaptor::construct_from_utf8("{"));
|
|
string_adaptor::append(error, namespace_uri);
|
|
string_adaptor::append(error, string_adaptor::construct_from_utf8("}"));
|
|
} // if ...
|
|
string_adaptor::append(error, name);
|
|
throw UndefinedFunctionException(string_adaptor().asStdString(error));
|
|
} // if(func == 0)
|
|
|
|
return new FunctionHolder(func, namespace_uri, name);
|
|
} // createFunction
|
|
|
|
private:
|
|
XPathFunction<string_type, string_adaptor>* func_;
|
|
string_type namespace_uri_;
|
|
string_type name_;
|
|
}; // class FunctionHolder
|
|
|
|
} // namespace impl
|
|
} // namespace XPath
|
|
} // namespace Arabica
|
|
|
|
#endif
|