#ifndef ARABICA_XPATH_FUNCTION_HPP #define ARABICA_XPATH_FUNCTION_HPP #include #include #include #include #include "xpath_value.hpp" #include "xpath_execution_context.hpp" namespace Arabica { namespace XPath { template class XPathFunction { protected: XPathFunction(int minArgs, int maxArgs, const std::vector >& args) : args_(args) { if(((minArgs != -1) && (static_cast(args.size()) < minArgs)) || ((maxArgs != -1) && (static_cast(args.size()) > maxArgs))) throw SyntaxException("wrong number of arguments to function"); } // XPathFunction public: virtual ~XPathFunction() { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const = 0; protected: size_t argCount() const { return args_.size(); } bool argAsBool(size_t index, const DOM::Node& context, const ExecutionContext& executionContext) const { return args_[index]->evaluate(context, executionContext)->asBool(); } // argAsBool double argAsNumber(size_t index, const DOM::Node& context, const ExecutionContext& executionContext) const { return args_[index]->evaluate(context, executionContext)->asNumber(); } // argAsNumber string_type argAsString(size_t index, const DOM::Node& context, const ExecutionContext& executionContext) const { return args_[index]->evaluate(context, executionContext)->asString(); } // argAsString NodeSet argAsNodeSet(size_t index, const DOM::Node& context, const ExecutionContext& executionContext) const { return args_[index]->evaluate(context, executionContext)->asNodeSet(); } // argAsNodeSet private: const std::vector > args_; }; // class XPathFunction namespace impl { //////////////////////////////// // node-set functions // number last() template class LastFn : public XPathFunction { public: LastFn(const std::vector >& args) : XPathFunction(0, 0, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new NumericValue(executionContext.last()); } // evaluate }; // class LastFn // number position() template class PositionFn : public XPathFunction { public: PositionFn(const std::vector >& args) : XPathFunction(0, 0, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new NumericValue(executionContext.position()); } // evaluate }; // class PositionFn // number count(node-set) template class CountFn : public XPathFunction { typedef XPathFunction baseT; public: CountFn(const std::vector >& args) : XPathFunction(1, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new NumericValue(baseT::argAsNodeSet(0, context, executionContext).size()); } // evaluate }; // class CountFn // node-set id(object) // string local-name(node-set?) template class LocalNameFn : public XPathFunction { typedef XPathFunction baseT; public: LocalNameFn(const std::vector >& args) : XPathFunction(0, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { DOM::Node node; if(baseT::argCount() == 0) node = context; else { NodeSet ns = baseT::argAsNodeSet(0, context, executionContext); if(ns.size() != 0) node = ns.top(); } // if ... if(node != 0) switch(node.getNodeType()) { case DOM::Node_base::ATTRIBUTE_NODE: case DOM::Node_base::ELEMENT_NODE: case DOM::Node_base::PROCESSING_INSTRUCTION_NODE: return new StringValue(node.hasNamespaceURI() ? node.getLocalName() : node.getNodeName()); default: // put this in to keep gcc quiet ; } // switch ... return new StringValue(""); } // evaluate }; // class LocalNameFn // string namespace-uri(node-set?) template class NamespaceURIFn : public XPathFunction { typedef XPathFunction baseT; public: NamespaceURIFn(const std::vector >& args) : XPathFunction(0, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { DOM::Node node; if(baseT::argCount() == 0) node = context; else { NodeSet ns = baseT::argAsNodeSet(0, context, executionContext); if(ns.size() != 0) node = ns.top(); } // if ... if(node != 0) switch(node.getNodeType()) { case DOM::Node_base::ATTRIBUTE_NODE: case DOM::Node_base::ELEMENT_NODE: return new StringValue(node.getNamespaceURI()); default: // put this in to keep gcc quiet ; } // switch ... return new StringValue(""); } // evaluate }; // class NamespaceURIFn // string name(node-set?) template class NameFn : public XPathFunction { typedef XPathFunction baseT; public: NameFn(const std::vector >& args) : XPathFunction(0, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { DOM::Node node; if(baseT::argCount() == 0) node = context; else { NodeSet ns = baseT::argAsNodeSet(0, context, executionContext); if(ns.size() != 0) node = ns.top(); } // if ... if(node != 0) switch(node.getNodeType()) { case DOM::Node_base::ATTRIBUTE_NODE: case DOM::Node_base::ELEMENT_NODE: case DOM::Node_base::PROCESSING_INSTRUCTION_NODE: return new StringValue(node.getNodeName()); default: // stop gcc generating a warning about unhandled enum values ; } // switch ... return new StringValue(""); } // evaluate }; // class NameFn /////////////////////////////////////////////// // string functions // string string(object?) template class StringFn : public XPathFunction { typedef XPathFunction baseT; public: StringFn(const std::vector >& args) : XPathFunction(0, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new StringValue((baseT::argCount() > 0) ? baseT::argAsString(0, context, executionContext) : nodeStringValue(context)); } // evaluate }; // class StringFn // string concat(string, string, string*) template class ConcatFn : public XPathFunction { typedef XPathFunction baseT; public: ConcatFn(const std::vector >& args) : XPathFunction(2, -1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { string_type s; for(size_t a = 0, ae = baseT::argCount(); a < ae; ++a) s.append(baseT::argAsString(a, context, executionContext)); return new StringValue(s); } // evaluate }; // ConcatFn // boolean starts-with(string, string) template class StartsWithFn : public XPathFunction { typedef XPathFunction baseT; public: StartsWithFn(const std::vector >& args) : XPathFunction(2, 2, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { string_type value = baseT::argAsString(0, context, executionContext); string_type start = baseT::argAsString(1, context, executionContext); if(value.length() < start.length()) return new BoolValue(false); for(size_t i = 0, ie = start.length(); i < ie; ++i) if(value[i] != start[i]) return new BoolValue(false); return new BoolValue(true); } // evaluate }; // StartsWithFn // boolean contains(string, string) template class ContainsFn : public XPathFunction { typedef XPathFunction baseT; public: ContainsFn(const std::vector >& args) : XPathFunction(2, 2, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new BoolValue(baseT::argAsString(0, context, executionContext).find(baseT::argAsString(1, context, executionContext)) != string_type::npos); } // evaluate }; // class ContainsFn // string substring-before(string, string) template class SubstringBeforeFn : public XPathFunction { typedef XPathFunction baseT; public: SubstringBeforeFn(const std::vector >& args) : XPathFunction(2, 2, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { string_type value = baseT::argAsString(0, context, executionContext); size_t splitAt = value.find(baseT::argAsString(1, context, executionContext)); if(splitAt == string_type::npos) return new StringValue(""); return new StringValue(value.substr(0, splitAt)); } // evaluate }; // class SubstringBeforeFn // string substring-after(string, string) template class SubstringAfterFn : public XPathFunction { typedef XPathFunction baseT; public: SubstringAfterFn(const std::vector >& args) : XPathFunction(2, 2, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { string_type value = baseT::argAsString(0, context, executionContext); string_type split = baseT::argAsString(1, context, executionContext); size_t splitAt = value.find(split); if((splitAt == string_type::npos) || ((splitAt + split.length()) >= value.length())) return new StringValue(""); return new StringValue(value.substr(splitAt + split.length())); } // evaluate }; // class SubstringAfterFn // string substring(string, number, number?) template class SubstringFn : public XPathFunction { typedef XPathFunction baseT; public: SubstringFn(const std::vector >& args) : XPathFunction(2, 3, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { string_type value = baseT::argAsString(0, context, executionContext); double startAt = roundNumber(baseT::argAsNumber(1, context, executionContext)) - 1; double endAt = roundNumber((baseT::argCount() == 3 ? baseT::argAsNumber(2, context, executionContext) : Infinity)) + startAt; if((endAt < 0) || (endAt < startAt) || (isNaN(endAt))) return new StringValue(""); if(startAt < 0) startAt = 0; if((isInfinite(endAt)) || (endAt > value.length())) endAt = value.length(); return new StringValue(value.substr(static_cast(startAt), static_cast(endAt - startAt))); } // evaluate }; // SubstringFn // number string-length(string?) template class StringLengthFn : public XPathFunction { typedef XPathFunction baseT; public: StringLengthFn(const std::vector >& args) : XPathFunction(0, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new NumericValue(((baseT::argCount() > 0) ? baseT::argAsString(0, context, executionContext) : nodeStringValue(context)).length()); } // evaluate }; // StringLengthFn // string normalize-space(string?) template class NormalizeSpaceFn : public XPathFunction { typedef XPathFunction baseT; public: NormalizeSpaceFn(const std::vector >& args) : XPathFunction(0, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { string_type value = ((baseT::argCount() > 0) ? baseT::argAsString(0, context, executionContext) : nodeStringValue(context)); size_t i = 0, ie = value.length(); // string leading space while((i != ie) && (XML::is_space(static_cast(value[i])))) ++i; size_t p = 0; while(i != ie) { while((i != ie) && (!XML::is_space(static_cast(value[i])))) value[p++] = value[i++]; while((i != ie) && (XML::is_space(static_cast(value[i])))) ++i; if(i != ie) value[p++] = Unicode::SPACE; } // while ... if(p != ie) value.resize(p); return new StringValue(value); } // evaluate }; // class NormalizeSpaceFn // string translate(string, string, string) template class TranslateFn : public XPathFunction { typedef XPathFunction baseT; public: TranslateFn(const std::vector >& args) : XPathFunction(3, 3, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { string_type str = baseT::argAsString(0, context, executionContext); string_type from = baseT::argAsString(1, context, executionContext); string_type to = baseT::argAsString(2, context, executionContext); size_t p = 0; for(size_t i = 0, ie = str.length(); i != ie; ++i) { size_t r = from.find(str[i]); if(r == string_type::npos) ++p; else if(r < to.length()) str[p++] = to[r]; } // for ... if(p != str.length()) str.resize(p); return new StringValue(str); } // evaluate }; // class TranslateFn /////////////////////////////////////////////////////// // boolean functions // boolean boolean(object) template class BooleanFn : public XPathFunction { typedef XPathFunction baseT; public: BooleanFn(const std::vector >& args) : XPathFunction(1, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new BoolValue(baseT::argAsBool(0, context, executionContext)); } // evaluate }; // class BooleanFn // boolean not(boolean) template class NotFn : public XPathFunction { typedef XPathFunction baseT; public: NotFn(const std::vector >& args) : XPathFunction(1, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new BoolValue(!baseT::argAsBool(0, context, executionContext)); } }; // class NotFn // boolean true() template class TrueFn : public XPathFunction { public: TrueFn(const std::vector >& args) : XPathFunction(0, 0, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new BoolValue(true); } // evaluate }; // TrueFn // boolean false() template class FalseFn : public XPathFunction { public: FalseFn(const std::vector >& args) : XPathFunction(0, 0, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new BoolValue(false); } // evaluate }; // FalseFn // boolean lang(string) ///////////////////////////////////////////////// // number functions // number number(object?) template class NumberFn : public XPathFunction { typedef XPathFunction baseT; public: NumberFn(const std::vector >& args) : XPathFunction(0, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { double result = (baseT::argCount() > 0) ? baseT::argAsNumber(0, context, executionContext) : StringValue(nodeStringValue(context)).asNumber(); return new NumericValue(result); } // evaluate }; // NumberFn // number sum(node-set) template class SumFn : public XPathFunction { typedef XPathFunction baseT; public: SumFn(const std::vector >& args) : XPathFunction(1, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { double sum = 0; NodeSet ns = baseT::argAsNodeSet(0, context, executionContext); for(typename NodeSet::const_iterator n = ns.begin(), end = ns.end(); n != end; ++n) sum += nodeNumberValue(*n); return new NumericValue(sum); } // evaluate }; // class SumFn // number floor(number) template class FloorFn : public XPathFunction { typedef XPathFunction baseT; public: FloorFn(const std::vector >& args) : XPathFunction(1, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new NumericValue(std::floor(baseT::argAsNumber(0, context, executionContext))); } // evaluate }; // class FloorFn // number ceiling(number) template class CeilingFn : public XPathFunction { typedef XPathFunction baseT; public: CeilingFn(const std::vector >& args) : XPathFunction(1, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new NumericValue(std::ceil(baseT::argAsNumber(0, context, executionContext))); } // evaluate }; // class CeilingFn // number round(number) template class RoundFn : public XPathFunction { typedef XPathFunction baseT; public: RoundFn(const std::vector >& args) : XPathFunction(1, 1, args) { } virtual XPathValue* evaluate(const DOM::Node& context, const ExecutionContext& executionContext) const { return new NumericValue(roundNumber(baseT::argAsNumber(0, context, executionContext))); } // evaluate }; // class RoundFn } // namespace impl } // namespace XPath } // namespace Arabica #endif