#ifndef ARABICA_XPATHIC_XPATH_PARSER_HPP #define ARABICA_XPATHIC_XPATH_PARSER_HPP #include #include #include #include #include #include "xpath_object.hpp" #include "xpath_expression.hpp" #include "xpath_ast.hpp" #include "xpath_grammar.hpp" #include "xpath_namespace_context.hpp" #include "xpath_function_resolver.hpp" #include "xpath_variable_resolver.hpp" #include "xpath_resolver_holder.hpp" namespace Arabica { namespace XPath { class SyntaxException : public std::runtime_error { public: SyntaxException(const std::string& thing) : std::runtime_error("Bad XPath: " + thing) { } }; // class SyntaxException class RuntimeException : public std::runtime_error { public: RuntimeException(const std::string& thing) : std::runtime_error("Cannot evaluate XPath: " + thing) { } }; // class RuntimeException class UnsupportedException : public std::runtime_error { public: UnsupportedException(const std::string& thing) : std::runtime_error("Sorry, haven't implemented '" + thing + "' yet") { } }; // class UnsupportedException namespace impl { template class CompilationContext; template class StepExpression; template class StepList : public std::vector*> { }; } // namespace impl template > class XPath { public: XPath() { resetNamespaceContext(); resetVariableResolver(); resetFunctionResolver(); } // XPath ~XPath() { } // ~XPath XPathExpressionPtr compile(const string_type& xpath) const { return do_compile(xpath, &XPath::parse_xpath); } // compile XPathExpressionPtr compile_expr(const string_type& xpath) const { return do_compile(xpath, &XPath::parse_xpath_expr); } // compile XPathValuePtr evaluate(const string_type& xpath, const DOM::Node& context) const { ExecutionContext executionContext; executionContext.setVariableResolver(getVariableResolver()); return compile(xpath)->evaluate(context, executionContext); } // evaluate XPathValuePtr evaluate_expr(const string_type& xpath, const DOM::Node& context) const { ExecutionContext executionContext; executionContext.setVariableResolver(getVariableResolver()); return compile_expr(xpath)->evaluate(context, executionContext); } // evaluate_expr void setNamespaceContext(const NamespaceContext& namespaceContext) { namespaceContext_.set(namespaceContext); } void setNamespaceContext(NamespaceContextPtr namespaceContext) { namespaceContext_.set(namespaceContext); } const NamespaceContext& getNamespaceContext() const { return namespaceContext_.get(); } void resetNamespaceContext() { namespaceContext_.set(NamespaceContextPtr(new NullNamespaceContext())); } void setVariableResolver(const VariableResolver& variableResolver) { variableResolver_.set(variableResolver); } void setVariableResolver(VariableResolverPtr variableResolver) { variableResolver_.set(variableResolver); } const VariableResolver& getVariableResolver() const { return variableResolver_.get(); } void resetVariableResolver() { variableResolver_.set(VariableResolverPtr(new NullVariableResolver())); } void setFunctionResolver(const FunctionResolver& functionResolver) { functionResolver_.set(functionResolver); } void setFunctionResolver(FunctionResolverPtr functionResolver) { functionResolver_.set(functionResolver); } const FunctionResolver& getFunctionResolver() const { return functionResolver_.get(); } void resetFunctionResolver() { functionResolver_.set(FunctionResolverPtr(new NullFunctionResolver())); } private: XPathExpressionPtr do_compile(const string_type& xpath, typename impl::types::tree_info_t(XPath::*fn)(const string_type& str) const) const { typename impl::types::tree_info_t ast; try { ast = (this->*fn)(xpath); if(!ast.full) throw SyntaxException(string_adaptor().asStdString(xpath)); impl::CompilationContext context(*this, getNamespaceContext(), getFunctionResolver()); return XPathExpressionPtr(compile_expression(ast.trees.begin(), context)); } // try catch(std::exception& ex) { throw ex; } // catch catch(...) { throw SyntaxException(string_adaptor().asStdString(xpath)); } // catch } // do_compile typename impl::types::tree_info_t parse_xpath(const string_type& str) const { typename impl::types::str_iter_t first = string_adaptor::begin(str), last = string_adaptor::end(str); return ast_parse(first, last, xpathg_); } // parse_xpath typename impl::types::tree_info_t parse_xpath_expr(const string_type& str) const { typename impl::types::str_iter_t first = string_adaptor::begin(str), last = string_adaptor::end(str); return ast_parse(first, last, xpathge_); } // parse_xpath impl::xpath_grammar xpathg_; impl::xpath_grammar_expr xpathge_; impl::ResolverHolder > namespaceContext_; impl::ResolverHolder > variableResolver_; impl::ResolverHolder > functionResolver_; ///////////////////////////////////////////////////////////////////////////////// public: static XPathExpression* compile_expression(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { long id = impl::getNodeId(i); if(XPath::factory().find(id) == XPath::factory().end()) { //XPath::dump(i, 0); throw UnsupportedException(string_adaptor().asStdString(XPath::names()[id])); } return XPath::factory()[id](i, context); } // compile_expression private: static XPathExpression* createAbsoluteLocationPath(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createRelativeLocationPath(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createSingleStepRelativeLocationPath(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createExpression(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createFunction(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createBinaryExpression(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createLiteral(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createNumber(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createVariable(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createSingleStepAbsoluteLocationPath(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createUnaryExpression(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static XPathExpression* createUnaryNegativeExpr(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static impl::StepList createStepList(typename impl::types::node_iter_t const& from, typename impl::types::node_iter_t const& to, impl::CompilationContext& context); typedef XPathExpression* (*compileFn)(typename impl::types::node_iter_t const& i, impl::CompilationContext& context); static std::map& factory() { static std::map f = init_createFunctions(); return f; } // factory static std::map& names() { static std::map n = init_debugNames(); return n; } // names static const std::map init_createFunctions() { std::map factory; factory[impl::AbsoluteLocationPath_id] = createAbsoluteLocationPath; factory[impl::RelativeLocationPath_id] = createRelativeLocationPath; factory[impl::AbbreviatedAbsoluteLocationPath_id] = createAbsoluteLocationPath; factory[impl::Step_id] = createRelativeLocationPath; factory[impl::PathExpr_id] = createRelativeLocationPath; factory[impl::FilterExpr_id] = createRelativeLocationPath; factory[impl::PrimaryExpr_id] = createExpression; factory[impl::FunctionCall_id] = createFunction; factory[impl::AdditiveExpr_id] = createBinaryExpression; factory[impl::MultiplicativeExpr_id] = createBinaryExpression; factory[impl::EqualityExpr_id] = createBinaryExpression; factory[impl::RelationalExpr_id] = createBinaryExpression; factory[impl::OrExpr_id] = createBinaryExpression; factory[impl::AndExpr_id] = createBinaryExpression; factory[impl::UnionExpr_id] = createBinaryExpression; factory[impl::Literal_id] = createLiteral; factory[impl::Number_id] = createNumber; factory[impl::Digits_id] = createNumber; factory[impl::VariableReference_id] = createVariable; factory[impl::NodeTest_id] = createSingleStepRelativeLocationPath; factory[impl::QName_id] = createSingleStepRelativeLocationPath; factory[impl::NCName_id] = createSingleStepRelativeLocationPath; factory[impl::AnyName_id] = createSingleStepRelativeLocationPath; factory[impl::Text_id] = createSingleStepRelativeLocationPath; factory[impl::Comment_id] = createSingleStepRelativeLocationPath; factory[impl::ProcessingInstruction_id] = createSingleStepRelativeLocationPath; factory[impl::Slash_id] = createSingleStepAbsoluteLocationPath; factory[impl::SelfSelect_id] = createSingleStepRelativeLocationPath; factory[impl::ParentSelect_id] = createSingleStepRelativeLocationPath; factory[impl::UnaryExpr_id] = createUnaryExpression; factory[impl::UnaryMinusOperator_id] = createUnaryNegativeExpr; return factory; } // init_createFunctions static const std::map init_debugNames() { std::map names; string_adaptor sa; names[impl::LocationPath_id] = sa.makeStringT("LocationPath"); names[impl::AbsoluteLocationPath_id] = sa.makeStringT("AbsoluteLocationPath"); names[impl::RelativeLocationPath_id] = sa.makeStringT("RelativeLocationPath"); names[impl::Step_id] = sa.makeStringT("Step"); names[impl::AxisSpecifier_id] = sa.makeStringT("AxisSpecifier"); names[impl::NodeTest_id] = sa.makeStringT("NodeTest"); names[impl::Predicate_id] = sa.makeStringT("Predicate"); names[impl::PredicateExpr_id] = sa.makeStringT("PredicateExpr"); names[impl::AbbreviatedAbsoluteLocationPath_id] = sa.makeStringT("AbbreviatedAbsoluteLocationPath"); names[impl::AbbreviatedStep_id] = sa.makeStringT("AbbreviatedStep"); names[impl::AbbreviatedAxisSpecifier_id] = sa.makeStringT("AbbreviatedAxisSpecifier"); names[impl::Expr_id] = sa.makeStringT("Expr"); names[impl::PrimaryExpr_id] = sa.makeStringT("PrimaryExpr"); names[impl::FunctionCall_id] = sa.makeStringT("FunctionCall"); names[impl::Argument_id] = sa.makeStringT("Argument"); names[impl::UnionExpr_id] = sa.makeStringT("UnionExpr"); names[impl::PathExpr_id] = sa.makeStringT("PathExpr"); names[impl::FilterExpr_id] = sa.makeStringT("FilterExpr"); names[impl::OrExpr_id] = sa.makeStringT("OrExpr"); names[impl::AndExpr_id] = sa.makeStringT("AndExpr"); names[impl::EqualityExpr_id] = sa.makeStringT("EqualityExpr"); names[impl::RelationalExpr_id] = sa.makeStringT("RelationalExpr"); names[impl::AdditiveExpr_id] = sa.makeStringT("AdditiveExpr"); names[impl::MultiplicativeExpr_id] = sa.makeStringT("MultiplicativeExpr"); names[impl::UnaryExpr_id] = sa.makeStringT("UnaryExpr"); names[impl::Literal_id] = sa.makeStringT("Literal"); names[impl::Number_id] = sa.makeStringT("Number"); names[impl::Digits_id] = sa.makeStringT("Digits"); names[impl::MultiplyOperator_id] = sa.makeStringT("MultiplyOperator"); names[impl::FunctionName_id] = sa.makeStringT("FunctionName"); names[impl::VariableReference_id] = sa.makeStringT("VariableReference"); names[impl::NameTest_id] = sa.makeStringT("NameTest"); names[impl::S_id] = sa.makeStringT("S"); names[impl::NodeType_id] = sa.makeStringT("NodeType"); names[impl::AxisName_id] = sa.makeStringT("AxisName"); names[impl::QName_id] = sa.makeStringT("QName"); names[impl::Prefix_id] = sa.makeStringT("Prefix"); names[impl::LocalPart_id] = sa.makeStringT("LocalPart"); names[impl::NCName_id] = sa.makeStringT("NCName"); names[impl::NCNameChar_id] = sa.makeStringT("NCNameChar"); names[impl::Slash_id] = sa.makeStringT("/"); names[impl::SlashSlash_id] = sa.makeStringT("//"); names[impl::AncestorOrSelf_id] = sa.makeStringT("ancestor-or-self::"); names[impl::Ancestor_id] = sa.makeStringT("ancestor::"); names[impl::Attribute_id] = sa.makeStringT("attribute::"); names[impl::Child_id] = sa.makeStringT("child::"); names[impl::DescendantOrSelf_id] = sa.makeStringT("descendant-or-self::"); names[impl::Descendant_id] = sa.makeStringT("descendant::"); names[impl::FollowingSibling_id] = sa.makeStringT("following-sibling::"); names[impl::Following_id] = sa.makeStringT("following::"); names[impl::Namespace_id] = sa.makeStringT("namespace::"); names[impl::Parent_id] = sa.makeStringT("parent::"); names[impl::PrecedingSibling_id] = sa.makeStringT("preceding-sibling::"); names[impl::Preceding_id] = sa.makeStringT("preceding::"); names[impl::Self_id] = sa.makeStringT("self::"); names[impl::Comment_id] = sa.makeStringT("comment()"); names[impl::Text_id] = sa.makeStringT("text()"); names[impl::ProcessingInstruction_id] = sa.makeStringT("processing-instruction()"); names[impl::Node_id] = sa.makeStringT("node()"); names[impl::AnyName_id] = sa.makeStringT("AnyName"); names[impl::SelfSelect_id] = sa.makeStringT("SelfSelect"); names[impl::ParentSelect_id] = sa.makeStringT("ParentSelect"); names[impl::LeftSquare_id] = sa.makeStringT("["); names[impl::RightSquare_id] = sa.makeStringT("]"); names[impl::LeftBracket_id] = sa.makeStringT("("); names[impl::RightBracket_id] = sa.makeStringT(")"); names[impl::PlusOperator_id] = sa.makeStringT("+"); names[impl::MinusOperator_id] = sa.makeStringT("-"); names[impl::ModOperator_id] = sa.makeStringT("mod"); names[impl::DivOperator_id] = sa.makeStringT("div"); names[impl::EqualsOperator_id] = sa.makeStringT("="); names[impl::NotEqualsOperator_id] = sa.makeStringT("!="); names[impl::LessThanOperator_id] = sa.makeStringT("<"); names[impl::LessThanEqualsOperator_id] = sa.makeStringT("<="); names[impl::GreaterThanOperator_id] = sa.makeStringT(">"); names[impl::GreaterThanEqualsOperator_id] = sa.makeStringT(">="); names[impl::OrOperator_id] = sa.makeStringT("or"); names[impl::AndOperator_id] = sa.makeStringT("and"); names[impl::UnionOperator_id] = sa.makeStringT("union"); names[impl::UnaryMinusOperator_id] = sa.makeStringT("minus"); return names; } // init_debugNames /* static void dump(typename impl::types::node_iter_t const& i, int depth) { long id = static_cast(i->value.id().to_long()); for(int d = 0; d < depth; ++d) std::cerr << ' '; std::cerr << names()[id] << " - " << std::string(i->value.begin(), i->value.end()) << std::endl; for(typename impl::types::node_iter_t c = i->children.begin(); c != i->children.end(); ++c) dump(c, depth+2); } // dump */ XPath(const XPath&); XPath& operator=(const XPath&); bool operator==(const XPath&) const; }; // class XPath } // namespace XPath } // namespace Arabica #include #include "xpath_value.hpp" #include "xpath_arithmetic.hpp" #include "xpath_relational.hpp" #include "xpath_logical.hpp" #include "xpath_step.hpp" #include "xpath_compile_context.hpp" #include "xpath_variable.hpp" #include "xpath_function_holder.hpp" #include "xpath_union.hpp" namespace Arabica { namespace XPath { template XPathExpression* XPath::createAbsoluteLocationPath(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { return new impl::AbsoluteLocationPath(createStepList(i->children.begin(), i->children.end(), context)); } // createAbsoluteLocationPath template XPathExpression* XPath::createRelativeLocationPath(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { return new impl::RelativeLocationPath(createStepList(i->children.begin(), i->children.end(), context)); } // createRelativeLocationPath template XPathExpression* XPath::createSingleStepRelativeLocationPath(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { typename impl::types::node_iter_t n = i; return new impl::RelativeLocationPath(impl::StepFactory::createStep(n, context)); } // createSingleStepRelativeLocationPath template XPathExpression* XPath::createExpression(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { typename impl::types::node_iter_t c = i->children.begin(); impl::skipWhitespace(c); return XPath::compile_expression(c, context); } // createExpression template XPathExpression* XPath::createFunction(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { typename impl::types::node_iter_t c = i->children.begin(); // string_type name = string_adaptor().makeStringT(c->value.begin(), c->value.end()); string_type name(c->value.begin(), c->value.end()); ++c; impl::skipWhitespace(c); assert(impl::getNodeId(c) == impl::LeftBracket_id); ++c; impl::skipWhitespace(c); std::vector > args; while(impl::getNodeId(c) != impl::RightBracket_id) { XPathExpressionPtr arg(XPath::compile_expression(c++, context)); args.push_back(arg); impl::skipWhitespace(c); } // while ... // maybe trailing whitespace ... return impl::FunctionHolder::createFunction(name, args, context); } // createFunction template XPathExpression* XPath::createBinaryExpression(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { typename impl::types::node_iter_t c = i->children.begin(); XPathExpression* p1 = XPath::compile_expression(c, context); ++c; do { long op = impl::getNodeId(c); ++c; XPathExpression* p2 = XPath::compile_expression(c, context); switch(op) { case impl::PlusOperator_id: p1 = new impl::PlusOperator(p1, p2); break; case impl::MinusOperator_id: p1 = new impl::MinusOperator(p1, p2); break; case impl::MultiplyOperator_id: p1 = new impl::MultiplyOperator(p1, p2); break; case impl::DivOperator_id: p1 = new impl::DivideOperator(p1, p2); break; case impl::ModOperator_id: p1 = new impl::ModOperator(p1, p2); break; case impl::EqualsOperator_id: p1 = new impl::EqualsOperator(p1, p2); break; case impl::NotEqualsOperator_id: p1 = new impl::NotEqualsOperator(p1, p2); break; case impl::LessThanOperator_id: p1 = new impl::LessThanOperator(p1, p2); break; case impl::LessThanEqualsOperator_id: p1 = new impl::LessThanEqualsOperator(p1, p2); break; case impl::GreaterThanOperator_id: p1 = new impl::GreaterThanOperator(p1, p2); break; case impl::GreaterThanEqualsOperator_id: p1 = new impl::GreaterThanEqualsOperator(p1, p2); break; case impl::OrOperator_id: p1 = new impl::OrOperator(p1, p2); break; case impl::AndOperator_id: p1 = new impl::AndOperator(p1, p2); break; case impl::UnionOperator_id: p1 = new impl::UnionExpression(p1, p2); break; default: throw UnsupportedException(boost::lexical_cast(op)); } // switch } while(++c != i->children.end()); return p1; } // createBinaryExpression template XPathExpression* XPath::createLiteral(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { //string_type str = string_adaptor().makeStringT(i->value.begin(), i->value.end()); string_type str(i->value.begin(), i->value.end()); return new StringValue(str); } // createLiteral template XPathExpression* XPath::createNumber(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { //string_type str = string_adaptor().makeStringT(i->value.begin(), i->value.end()); string_type str(i->value.begin(), i->value.end()); return new NumericValue(boost::lexical_cast(str)); } // createNumber template XPathExpression* XPath::createVariable(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { //string_type str = string_adaptor().makeStringT(i->value.begin()+1, i->value.end()); // skip $ string_type str(i->value.begin()+1, i->value.end()); return new Variable(str); } // createVariable template XPathExpression* XPath::createSingleStepAbsoluteLocationPath(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { typename impl::types::node_iter_t n = i; return new impl::AbsoluteLocationPath(impl::StepFactory::createStep(n, context)); } // createSingleStepAbsoluteLocationPath template XPathExpression* XPath::createUnaryExpression(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { return XPath::compile_expression(i->children.begin(), context); } // createUnaryExpression template XPathExpression* XPath::createUnaryNegativeExpr(typename impl::types::node_iter_t const& i, impl::CompilationContext& context) { return new impl::UnaryNegative(XPath::compile_expression(i+1, context)); } // createUnaryNegativeExpr template impl::StepList XPath::createStepList(typename impl::types::node_iter_t const& from, typename impl::types::node_iter_t const& to, impl::CompilationContext& context) { impl::StepList steps; typename impl::types::node_iter_t c = from; typename impl::types::node_iter_t end = to; while(c != end) switch(impl::getNodeId(c)) { case impl::S_id: case impl::Slash_id: ++c; // just drop it break; case impl::RelativeLocationPath_id: // might get here when handling an absolute path end = c->children.end(); c = c->children.begin(); break; case impl::Step_id: { typename impl::types::node_iter_t step = c->children.begin(); steps.push_back(impl::StepFactory::createStep(step, c->children.end(), context)); ++c; } break; default: steps.push_back(impl::StepFactory::createStep(c, end, context)); } // switch(impl::getNodeId(c)) return steps; } // createStepList } // namespace XPath } // namespace Arabica #endif