summaryrefslogtreecommitdiff
path: root/src/xpath.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/xpath.cc')
-rw-r--r--src/xpath.cc138
1 files changed, 112 insertions, 26 deletions
diff --git a/src/xpath.cc b/src/xpath.cc
index 702aaeae..6efa214a 100644
--- a/src/xpath.cc
+++ b/src/xpath.cc
@@ -699,7 +699,7 @@ xpath_t::parse_path_expr(std::istream& in, flags_t tflags) const
while (tok.kind == token_t::SLASH) {
ptr_op_t prev(node);
- tok = next_token(in, tflags);
+ tok = next_token(in, tflags);
node = new op_t(tok.kind == token_t::SLASH ?
op_t::O_RFIND : op_t::O_FIND);
if (tok.kind != token_t::SLASH)
@@ -1143,8 +1143,8 @@ void xpath_t::op_t::append_value(value_t& val,
result_seq.push_back(val);
}
-xpath_t::ptr_op_t xpath_t::op_t::compile(value_t& context, scope_t * scope,
- bool resolve)
+xpath_t::ptr_op_t
+xpath_t::op_t::compile(value_t& context, scope_t * scope, bool resolve)
{
#if 0
try {
@@ -1661,7 +1661,6 @@ xpath_t::ptr_op_t xpath_t::op_t::compile(value_t& context, scope_t * scope,
throw_(compile_error, "Attempting to apply path selection "
"to non-node(s)");
}
-
return wrap_value(result_seq);
}
@@ -2103,37 +2102,37 @@ void xpath_t::op_t::dump(std::ostream& out, const int depth) const
out << "FUNCTION - " << as_function();
break;
- case O_NOT: out << "O_NOT"; break;
- case O_NEG: out << "O_NEG"; break;
+ case O_NOT: out << "O_NOT"; break;
+ case O_NEG: out << "O_NEG"; break;
- case O_UNION: out << "O_UNION"; break;
+ case O_UNION: out << "O_UNION"; break;
- case O_ADD: out << "O_ADD"; break;
- case O_SUB: out << "O_SUB"; break;
- case O_MUL: out << "O_MUL"; break;
- case O_DIV: out << "O_DIV"; break;
+ case O_ADD: out << "O_ADD"; break;
+ case O_SUB: out << "O_SUB"; break;
+ case O_MUL: out << "O_MUL"; break;
+ case O_DIV: out << "O_DIV"; break;
- case O_NEQ: out << "O_NEQ"; break;
- case O_EQ: out << "O_EQ"; break;
- case O_LT: out << "O_LT"; break;
- case O_LTE: out << "O_LTE"; break;
- case O_GT: out << "O_GT"; break;
- case O_GTE: out << "O_GTE"; break;
+ case O_NEQ: out << "O_NEQ"; break;
+ case O_EQ: out << "O_EQ"; break;
+ case O_LT: out << "O_LT"; break;
+ case O_LTE: out << "O_LTE"; break;
+ case O_GT: out << "O_GT"; break;
+ case O_GTE: out << "O_GTE"; break;
- case O_AND: out << "O_AND"; break;
- case O_OR: out << "O_OR"; break;
+ case O_AND: out << "O_AND"; break;
+ case O_OR: out << "O_OR"; break;
- case O_QUES: out << "O_QUES"; break;
- case O_COLON: out << "O_COLON"; break;
+ case O_QUES: out << "O_QUES"; break;
+ case O_COLON: out << "O_COLON"; break;
- case O_COMMA: out << "O_COMMA"; break;
+ case O_COMMA: out << "O_COMMA"; break;
case O_DEFINE: out << "O_DEFINE"; break;
- case O_EVAL: out << "O_EVAL"; break;
+ case O_EVAL: out << "O_EVAL"; break;
- case O_FIND: out << "O_FIND"; break;
- case O_RFIND: out << "O_RFIND"; break;
- case O_PRED: out << "O_PRED"; break;
+ case O_FIND: out << "O_FIND"; break;
+ case O_RFIND: out << "O_RFIND"; break;
+ case O_PRED: out << "O_PRED"; break;
case LAST:
default:
@@ -2156,5 +2155,92 @@ void xpath_t::op_t::dump(std::ostream& out, const int depth) const
}
}
+xpath_t::path_t::path_t(const xpath_t& path_expr)
+{
+ ptr_op_t op = path_expr.ptr;
+
+ while (true) {
+ switch (op->kind) {
+ case op_t::O_FIND:
+ case op_t::O_RFIND:
+ case op_t::O_PRED:
+ break;
+
+ case op_t::NODE_ID:
+ case op_t::NODE_NAME:
+ case op_t::ATTR_ID:
+ case op_t::ATTR_NAME:
+ break;
+
+ default:
+ throw_(std::logic_error, "XPath expression is not strictly a path selection");
+ break;
+ }
+ break;
+ }
+}
+
+void xpath_t::path_t::check_element(node_t& start,
+ const element_iterator& element,
+ scope_t * scope,
+ const visitor_t& func)
+{
+ if (! element->predicate || element->predicate(start, scope)) {
+ element_iterator next_element = next(element);
+ if (next_element == elements.end())
+ func(start);
+ else
+ walk_elements(start, next_element, scope, func);
+ }
+}
+
+void xpath_t::path_t::walk_elements(node_t& start,
+ const element_iterator& element,
+ scope_t * scope,
+ const visitor_t& func)
+{
+ if (element->ident.type() == typeid(document_t::special_names_t)) {
+ switch (boost::get<document_t::special_names_t>(element->ident)) {
+ case document_t::CURRENT:
+ check_element(start, element, scope, func);
+ break;
+
+ case document_t::PARENT:
+ if (optional<parent_node_t&> parent = start.parent())
+ check_element(*parent, element, scope, func);
+ else
+ throw_(std::logic_error, "Attempt to access parent of root node");
+ break;
+
+ case document_t::ROOT:
+ check_element(start.document(), element, scope, func);
+ break;
+
+ case document_t::ALL:
+ if (! start.is_parent_node())
+ throw_(compile_error, "Referencing child nodes from a non-parent value");
+
+ foreach (node_t * node, start.as_parent_node())
+ check_element(*node, element, scope, func);
+ break;
+ }
+ }
+ else if (start.is_parent_node()) {
+ bool have_name_id = element->ident.type() == typeid(node_t::nameid_t);
+
+ foreach (node_t * child, start.as_parent_node()) {
+ if ((have_name_id &&
+ boost::get<node_t::nameid_t>(element->ident) == child->name_id()) ||
+ (! have_name_id &&
+ boost::get<string>(element->ident) == child->name()))
+ check_element(*child, element, scope, func);
+ }
+ }
+
+ if (element->recurse && start.is_parent_node())
+ foreach (node_t * child, start.as_parent_node())
+ walk_elements(*child, element, scope, func);
+}
+
} // namespace xml
} // namespace ledger