#ifndef ARABICA_XPATH_FUNCTION_HOLDER_HPP #define ARABICA_XPATH_FUNCTION_HOLDER_HPP #include #include "xpath_expression.hpp" #include "xpath_function.hpp" namespace Arabica { namespace XPath { namespace impl { template XPathFunction* CreateFn(const std::vector >& argExprs) { return new function_type(argExprs); } template class StandardXPathFunctionResolver { public: virtual ~StandardXPathFunctionResolver() { } virtual XPathFunction* resolveFunction(const string_type& namespace_uri, const string_type& name, const std::vector >& argExprs) const { return standardFunction(namespace_uri, name, argExprs); } // resolveFuncton virtual bool hasFunction(const string_type& namespace_uri, const string_type& name) const { return findFunction(namespace_uri, name); } // hasFunction static XPathFunction* standardFunction(const string_type& namespace_uri, const string_type& name, const std::vector >& argExprs) { const NamedFunction* fn = findFunction(namespace_uri, name); return (fn != 0) ? fn->creator(argExprs) : 0; } // standardFunction private: typedef XPathFunction* (*CreateFnPtr)(const std::vector >& argExprs); struct NamedFunction { const char* name; CreateFnPtr creator; }; static const NamedFunction FunctionLookupTable[]; static const NamedFunction* findFunction(const string_type& namespace_uri, const string_type& name) { if(!string_adaptor::empty(namespace_uri)) return 0; for(const NamedFunction* fn = FunctionLookupTable; fn->name != 0; ++fn) if(name == string_adaptor::construct_from_utf8(fn->name)) return fn; return 0; } // findFunction }; // class StandardXPathFunctionResolver template const typename StandardXPathFunctionResolver::NamedFunction StandardXPathFunctionResolver::FunctionLookupTable[] = { // node-set functions { "position", CreateFn, string_type, string_adaptor> }, { "last", CreateFn, string_type, string_adaptor> }, { "count", CreateFn, string_type, string_adaptor> }, { "local-name", CreateFn, string_type, string_adaptor> }, { "namespace-uri", CreateFn, string_type, string_adaptor> }, { "name", CreateFn, string_type, string_adaptor> }, // string functions {"string", CreateFn, string_type, string_adaptor> }, {"concat", CreateFn, string_type, string_adaptor> }, {"starts-with", CreateFn, string_type, string_adaptor> }, {"contains", CreateFn, string_type, string_adaptor> }, {"substring-before", CreateFn, string_type, string_adaptor> }, {"substring-after", CreateFn, string_type, string_adaptor> }, {"substring", CreateFn, string_type, string_adaptor> }, {"string-length", CreateFn, string_type, string_adaptor> }, {"normalize-space", CreateFn, string_type, string_adaptor> }, {"translate", CreateFn, string_type, string_adaptor> }, // boolean functions {"boolean", CreateFn, string_type, string_adaptor> }, {"not", CreateFn, string_type, string_adaptor> }, {"true", CreateFn, string_type, string_adaptor> }, {"false", CreateFn, string_type, string_adaptor> }, // number functions {"number", CreateFn, string_type, string_adaptor> }, {"sum", CreateFn, string_type, string_adaptor> }, {"floor", CreateFn, string_type, string_adaptor> }, {"ceiling", CreateFn, string_type, string_adaptor> }, {"round", CreateFn, string_type, string_adaptor> }, {0, 0} }; template class FunctionHolder : public XPathExpression_impl { public: FunctionHolder(XPathFunction* 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 evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return XPathValue(func_->evaluate(context, executionContext)); } // evaluate static FunctionHolder* createFunction(const string_type& namespace_uri, const string_type& name, const std::vector >& argExprs, const CompilationContext& context) { XPathFunction* func = 0; if(string_adaptor::empty(namespace_uri)) func = StandardXPathFunctionResolver::standardFunction(namespace_uri, 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* func_; string_type namespace_uri_; string_type name_; }; // class FunctionHolder } // namespace impl } // namespace XPath } // namespace Arabica #endif