arabica/include/XPath/impl/xpath_function.hpp
2007-09-04 22:55:47 +00:00

598 lines
26 KiB
C++

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