use crate::io::map_io_err; use crate::model::*; use oxigraph::model::Term; use oxigraph::sparql::*; use pyo3::exceptions::{PyRuntimeError, PySyntaxError, PyTypeError, PyValueError}; use pyo3::prelude::{ pyclass, pymethods, pyproto, FromPyObject, IntoPy, Py, PyAny, PyCell, PyErr, PyObject, PyRef, PyRefMut, PyResult, Python, }; use pyo3::{PyIterProtocol, PyMappingProtocol, PyObjectProtocol}; use std::vec::IntoIter; pub fn parse_query( query: &str, use_default_graph_as_union: bool, default_graph: Option<&PyAny>, named_graphs: Option<&PyAny>, ) -> PyResult { let mut query = Query::parse(query, None).map_err(|e| map_evaluation_error(e.into()))?; if use_default_graph_as_union && default_graph.is_some() { return Err(PyValueError::new_err( "The query() method use_default_graph_as_union and default_graph arguments should not be set at the same time", )); } if use_default_graph_as_union { query.dataset_mut().set_default_graph_as_union(); } if let Some(default_graph) = default_graph { if let Ok(default_graphs) = default_graph.iter() { query.dataset_mut().set_default_graph( default_graphs .map(|graph| Ok(graph?.extract::()?.into())) .collect::>()?, ) } else if let Ok(default_graph) = default_graph.extract::() { query .dataset_mut() .set_default_graph(vec![default_graph.into()]); } else { return Err(PyValueError::new_err( format!("The query() method default_graph argument should be a NamedNode, a BlankNode, the DefaultGraph or a not empty list of them. {} found", default_graph.get_type() ))); } } if let Some(named_graphs) = named_graphs { query.dataset_mut().set_available_named_graphs( named_graphs .iter()? .map(|graph| Ok(graph?.extract::()?.into())) .collect::>()?, ) } Ok(query) } pub fn query_results_to_python(py: Python<'_>, results: QueryResults) -> PyResult { Ok(match results { QueryResults::Solutions(inner) => PyQuerySolutions { inner }.into_py(py), QueryResults::Graph(inner) => PyQueryTriples { inner }.into_py(py), QueryResults::Boolean(b) => b.into_py(py), }) } /// Tuple associating variables and terms that are the result of a SPARQL ``SELECT`` query. /// /// It is the equivalent of a row in SQL. /// /// It could be indexes by variable name (:py:class:`Variable` or :py:class:`str`) or position in the tuple (:py:class:`int`). /// Unpacking also works. /// /// >>> store = Store() /// >>> store.add(Quad(NamedNode('http://example.com'), NamedNode('http://example.com/p'), Literal('1'))) /// >>> solution = next(store.query('SELECT ?s ?p ?o WHERE { ?s ?p ?o }')) /// >>> solution[Variable('s')] /// /// >>> solution['s'] /// /// >>> solution[0] /// /// >>> s, p, o = solution /// >>> s /// #[pyclass(unsendable, name = "QuerySolution", module = "oxigraph")] pub struct PyQuerySolution { inner: QuerySolution, } #[pyproto] impl PyObjectProtocol for PyQuerySolution { fn __repr__(&self) -> String { let mut buffer = String::new(); buffer.push_str("'); buffer } } #[pyproto] impl PyMappingProtocol for PyQuerySolution { fn __len__(&self) -> usize { self.inner.len() } fn __getitem__(&self, input: &PyAny) -> PyResult> { if let Ok(key) = usize::extract(input) { Ok(self.inner.get(key).map(|term| PyTerm::from(term.clone()))) } else if let Ok(key) = <&str>::extract(input) { Ok(self.inner.get(key).map(|term| PyTerm::from(term.clone()))) } else if let Ok(key) = input.downcast::>() { let key = &*key.borrow(); Ok(self .inner .get(<&Variable>::from(key)) .map(|term| PyTerm::from(term.clone()))) } else { Err(PyTypeError::new_err(format!( "{} is not an integer of a string", input.get_type().name()?, ))) } } } #[pyproto] impl PyIterProtocol for PyQuerySolution { fn __iter__(slf: PyRef) -> SolutionValueIter { SolutionValueIter { inner: slf .inner .values() .map(|v| v.cloned()) .collect::>() .into_iter(), } } } #[pyclass(module = "oxigraph")] pub struct SolutionValueIter { inner: IntoIter>, } #[pyproto] impl PyIterProtocol for SolutionValueIter { fn __iter__(slf: PyRefMut) -> Py { slf.into() } fn __next__(mut slf: PyRefMut) -> Option> { slf.inner.next().map(|v| v.map(PyTerm::from)) } } /// An iterator of :py:class:`QuerySolution` returned by a SPARQL ``SELECT`` query /// /// >>> store = Store() /// >>> store.add(Quad(NamedNode('http://example.com'), NamedNode('http://example.com/p'), Literal('1'))) /// >>> list(store.query('SELECT ?s WHERE { ?s ?p ?o }')) /// [>] #[pyclass(unsendable, name = "QuerySolutions", module = "oxigraph")] pub struct PyQuerySolutions { inner: QuerySolutionIter, } #[pymethods] impl PyQuerySolutions { /// :return: the ordered list of all variables that could appear in the query results /// :rtype: list(Variable) /// /// >>> store = Store() /// >>> store.query('SELECT ?s WHERE { ?s ?p ?o }').variables /// [] #[getter] fn variables(&self) -> Vec { self.inner .variables() .iter() .map(|v| v.clone().into()) .collect() } } #[pyproto] impl PyIterProtocol for PyQuerySolutions { fn __iter__(slf: PyRefMut) -> Py { slf.into() } fn __next__(mut slf: PyRefMut) -> PyResult> { Ok(slf .inner .next() .transpose() .map_err(map_evaluation_error)? .map(move |inner| PyQuerySolution { inner })) } } /// An iterator of :py:class:`Triple` returned by a SPARQL ``CONSTRUCT`` or ``DESCRIBE`` query /// /// >>> store = Store() /// >>> store.add(Quad(NamedNode('http://example.com'), NamedNode('http://example.com/p'), Literal('1'))) /// >>> list(store.query('CONSTRUCT WHERE { ?s ?p ?o }')) /// [ predicate= object=>>] #[pyclass(unsendable, name = "QueryTriples", module = "oxigraph")] pub struct PyQueryTriples { inner: QueryTripleIter, } #[pyproto] impl PyIterProtocol for PyQueryTriples { fn __iter__(slf: PyRefMut) -> Py { slf.into() } fn __next__(mut slf: PyRefMut) -> PyResult> { Ok(slf .inner .next() .transpose() .map_err(map_evaluation_error)? .map(|t| t.into())) } } pub fn map_evaluation_error(error: EvaluationError) -> PyErr { match error { EvaluationError::Parsing(error) => PySyntaxError::new_err(error.to_string()), EvaluationError::Io(error) => map_io_err(error), EvaluationError::Query(error) => PyValueError::new_err(error.to_string()), _ => PyRuntimeError::new_err(error.to_string()), } }