extern crate sxd_document;
#[macro_use]
extern crate sxd_xpath;
use std::borrow::ToOwned;
use sxd_document::{dom, parser};
use sxd_xpath::{Value, Factory, Context, evaluate_xpath};
use sxd_xpath::{context, function};
#[test]
fn functions_accept_arguments() {
with_document("", |doc| {
let result = evaluate_xpath(&doc, "concat('hello', ' ', 'world')");
assert_eq!(Ok(Value::String("hello world".to_owned())), result);
});
}
#[test]
fn functions_implicitly_coerce_argument_types() {
with_document("true", |doc| {
// We are searching a nodeset for a boolean. Both should be
// converted to strings by `contains`.
let result = evaluate_xpath(&doc, "count(//*[contains(., true)])");
assert_eq!(Ok(Value::Number(1.0)), result);
});
}
#[test]
fn axis_predicate_order() {
with_document("", |doc| {
// All the `c` elements that are the first child
let result = evaluate_xpath(&doc, "//c[1]");
let a = doc.root().children()[0].element().expect("No element a");
let b0 = a.children()[0].element().expect("No element b0");
let b1 = a.children()[1].element().expect("No element b1");
let c0 = b0.children()[0].element().expect("No element c0");
let c1 = b1.children()[0].element().expect("No element c1");
assert_eq!(Ok(Value::Nodeset(nodeset![c0, c1])), result);
});
}
#[test]
fn position_function_in_predicate() {
with_document("", |doc| {
let result = evaluate_xpath(&doc, "count(//a/*[position() = 2])");
assert_eq!(Ok(Value::Number(1.0)), result);
});
}
#[test]
fn variables_with_qualified_names() {
with_document("", |doc| {
let mut setup = Setup::new();
setup.context.set_variable(("uri:namespace", "name"), 42.0);
setup.context.set_namespace("prefix", "uri:namespace");
let result = setup.evaluate(&doc, "$prefix:name");
assert_eq!(42.0, result);
});
}
#[test]
fn functions_with_qualified_names() {
with_document("", |doc| {
let mut setup = Setup::new();
setup.context.set_function(("uri:namespace", "constant"), ConstantValueFunction(42.0));
setup.context.set_namespace("prefix", "uri:namespace");
let result = setup.evaluate(&doc, "prefix:constant()");
assert_eq!(42.0, result);
});
}
#[test]
fn nodesets_are_unique() {
with_document("", |doc| {
let result = evaluate_xpath(&doc, "/ | /");
assert_eq!(Ok(Value::Nodeset(nodeset![doc.root()])), result);
});
}
fn with_document(xml: &str, f: F)
where F: FnOnce(dom::Document),
{
let package = parser::parse(xml).expect("Unable to parse test XML");
f(package.as_document());
}
#[derive(Default)]
struct Setup<'d> {
context: Context<'d>,
factory: Factory,
}
impl<'d> Setup<'d> {
fn new() -> Setup<'d> {
Default::default()
}
fn evaluate(&self, doc: &'d dom::Document<'d>, xpath: &str) -> Value<'d> {
let xpath = self.factory.build(xpath)
.expect("Unable to build XPath")
.expect("No XPath was built");
xpath.evaluate(&self.context, doc.root())
.expect("Unable to evaluate XPath")
}
}
struct ConstantValueFunction(f64);
impl function::Function for ConstantValueFunction {
fn evaluate<'c, 'd>(&self,
_context: &context::Evaluation<'c, 'd>,
_args: Vec>) -> Result, function::Error>
{
Ok(Value::Number(self.0))
}
}