use nu_plugin::{EvaluatedCall, LabeledError}; use nu_protocol::{Span, Spanned, Value}; use sxd_document::parser; use sxd_xpath::{Context, Factory}; pub fn execute_xpath_query( _name: &str, call: &EvaluatedCall, input: &Value, query: Option>, ) -> Result { let (query_string, span) = match &query { Some(v) => (&v.item, v.span), None => { return Err(LabeledError { msg: "problem with input data".to_string(), label: "problem with input data".to_string(), span: Some(call.head), }) } }; let xpath = build_xpath(query_string, span)?; let input_string = input.as_string()?; let package = parser::parse(&input_string); if package.is_err() { return Err(LabeledError { label: "invalid xml document".to_string(), msg: "invalid xml document".to_string(), span: Some(call.head), }); } let package = package.expect("invalid xml document"); let document = package.as_document(); let context = Context::new(); // leaving this here for augmentation at some point // build_variables(&arguments, &mut context); // build_namespaces(&arguments, &mut context); let res = xpath.evaluate(&context, document.root()); // Some xpath statements can be long, so let's truncate it with ellipsis let mut key = query_string.clone(); if query_string.len() >= 20 { key.truncate(17); key += "..."; } else { key = query_string.to_string(); }; match res { Ok(r) => { let mut cols: Vec = vec![]; let mut vals: Vec = vec![]; let mut records: Vec = vec![]; match r { sxd_xpath::Value::Nodeset(ns) => { for n in ns.into_iter() { cols.push(key.to_string()); vals.push(Value::string(n.string_value(), call.head)); } } sxd_xpath::Value::Boolean(b) => { cols.push(key.to_string()); vals.push(Value::bool(b, call.head)); } sxd_xpath::Value::Number(n) => { cols.push(key.to_string()); vals.push(Value::float(n, call.head)); } sxd_xpath::Value::String(s) => { cols.push(key.to_string()); vals.push(Value::string(s, call.head)); } }; // convert the cols and vecs to a table by creating individual records // for each item so we can then use a list to make a table for (k, v) in cols.iter().zip(vals.iter()) { records.push(Value::Record { cols: vec![k.to_string()], vals: vec![v.clone()], span: call.head, }) } Ok(Value::List { vals: records, span: call.head, }) } Err(_) => Err(LabeledError { label: "xpath query error".to_string(), msg: "xpath query error".to_string(), span: Some(call.head), }), } } fn build_xpath(xpath_str: &str, span: Span) -> Result { let factory = Factory::new(); if let Ok(xpath) = factory.build(xpath_str) { xpath.ok_or_else(|| LabeledError { label: "invalid xpath query".to_string(), msg: "invalid xpath query".to_string(), span: Some(span), }) } else { Err(LabeledError { label: "expected valid xpath query".to_string(), msg: "expected valid xpath query".to_string(), span: Some(span), }) } } #[cfg(test)] mod tests { use super::execute_xpath_query as query; use nu_plugin::EvaluatedCall; use nu_protocol::{Span, Spanned, Value}; #[test] fn position_function_in_predicate() { let call = EvaluatedCall { head: Span::test_data(), positional: vec![], named: vec![], }; let text = Value::string( r#""#, Span::test_data(), ); let spanned_str: Spanned = Spanned { item: "count(//a/*[position() = 2])".to_string(), span: Span::test_data(), }; let actual = query("", &call, &text, Some(spanned_str)).expect("test should not fail"); let expected = Value::List { vals: vec![Value::Record { cols: vec!["count(//a/*[posit...".to_string()], vals: vec![Value::test_float(1.0)], span: Span::test_data(), }], span: Span::test_data(), }; assert_eq!(actual, expected); } #[test] fn functions_implicitly_coerce_argument_types() { let call = EvaluatedCall { head: Span::test_data(), positional: vec![], named: vec![], }; let text = Value::string( r#"true"#, Span::test_data(), ); let spanned_str: Spanned = Spanned { item: "count(//*[contains(., true)])".to_string(), span: Span::test_data(), }; let actual = query("", &call, &text, Some(spanned_str)).expect("test should not fail"); let expected = Value::List { vals: vec![Value::Record { cols: vec!["count(//*[contain...".to_string()], vals: vec![Value::test_float(1.0)], span: Span::test_data(), }], span: Span::test_data(), }; assert_eq!(actual, expected); } }