use c_signatures::*;
use libc::{c_void, size_t};
use tree::{Document, Node};
use std::str;
use std::ffi::{CStr, CString};
#[derive(Clone)]
pub struct Context<'a> {
pub context_ptr: *mut c_void,
pub document: &'a Document,
}
impl<'a> Drop for Context<'a> {
fn drop(&mut self) {
unsafe {
xmlXPathFreeContext(self.context_ptr);
}
}
}
#[derive(Clone)]
pub struct Object {
pub ptr: *mut c_void,
}
impl<'a> Context<'a> {
pub fn new(doc: &Document) -> Result<Context, ()> {
let ctxtptr: *mut c_void = unsafe { xmlXPathNewContext(doc.doc_ptr) };
if ctxtptr.is_null() {
Err(())
} else {
Ok(Context {
context_ptr: ctxtptr,
document: doc,
})
}
}
pub fn evaluate(&self, xpath: &str) -> Result<Object, ()> {
let c_xpath = CString::new(xpath).unwrap();
let result = unsafe { xmlXPathEvalExpression(c_xpath.as_ptr(), self.context_ptr) };
if result.is_null() {
Err(())
} else {
Ok(Object { ptr: result })
}
}
}
impl Drop for Object {
fn drop(&mut self) {
unsafe {
xmlFreeXPathObject(self.ptr);
}
}
}
impl Object {
pub fn get_number_of_nodes(&self) -> usize {
let v = unsafe { xmlXPathObjectNumberOfNodes(self.ptr) };
if v == -1 {
panic!("rust-libxml: xpath: Passed in null pointer!");
}
if v == -2 {
return 0;
}
if v < -2 {
panic!("rust-libxml: xpath: expected non-negative number of result nodes");
}
v as usize
}
pub fn get_nodes_as_vec(&self) -> Vec<Node> {
let n = self.get_number_of_nodes();
let mut vec: Vec<Node> = Vec::with_capacity(n);
for i in 0..n {
let ptr: *mut c_void = unsafe { xmlXPathObjectGetNode(self.ptr, i as size_t) };
if ptr.is_null() {
panic!("rust-libxml: xpath: found null pointer result set");
}
vec.push(Node { node_ptr: ptr }); }
vec
}
pub fn to_string(&self) -> String {
unsafe {
let v = xmlXPathCastToString(self.ptr);
let c_string = CStr::from_ptr(v);
let utf_string = str::from_utf8(c_string.to_bytes()).unwrap().to_owned();
return utf_string;
}
}
}