use crate::io::*; use crate::model::*; use crate::store::map_storage_error; use oxigraph::io::RdfSerializer; use oxigraph::model::Term; use oxigraph::sparql::results::{ ParseError, QueryResultsParser, QueryResultsReader, QueryResultsSerializer, SolutionsReader, }; use oxigraph::sparql::{ EvaluationError, Query, QueryResults, QuerySolution, QuerySolutionIter, QueryTripleIter, Variable, }; use pyo3::basic::CompareOp; use pyo3::exceptions::{ PyNotImplementedError, PyRuntimeError, PySyntaxError, PyTypeError, PyValueError, }; use pyo3::prelude::*; use pyo3::types::PyBytes; use std::io::BufReader; use std::path::PathBuf; use std::vec::IntoIter; pub fn parse_query( query: &str, base_iri: Option<&str>, use_default_graph_as_union: bool, default_graph: Option<&PyAny>, named_graphs: Option<&PyAny>, py: Python<'_>, ) -> PyResult { let mut query = allow_threads_unsafe(py, || Query::parse(query, base_iri)) .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) -> PyObject { match results { QueryResults::Solutions(inner) => PyQuerySolutions { inner: PyQuerySolutionsVariant::Query(inner), } .into_py(py), QueryResults::Graph(inner) => PyQueryTriples { inner }.into_py(py), QueryResults::Boolean(inner) => PyQueryBoolean { inner }.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(frozen, unsendable, name = "QuerySolution", module = "pyoxigraph")] pub struct PyQuerySolution { inner: QuerySolution, } #[pymethods] impl PyQuerySolution { fn __repr__(&self) -> String { let mut buffer = String::new(); buffer.push_str("'); buffer } fn __richcmp__(&self, other: &Self, op: CompareOp) -> PyResult { match op { CompareOp::Eq => Ok(self.inner == other.inner), CompareOp::Ne => Ok(self.inner != other.inner), _ => Err(PyNotImplementedError::new_err( "Ordering is not implemented", )), } } 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.extract::>() { 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()?, ))) } } #[allow(clippy::unnecessary_to_owned)] fn __iter__(&self) -> SolutionValueIter { SolutionValueIter { inner: self.inner.values().to_vec().into_iter(), } } } #[pyclass(module = "pyoxigraph")] pub struct SolutionValueIter { inner: IntoIter>, } #[pymethods] impl SolutionValueIter { fn __iter__(slf: PyRef<'_, Self>) -> PyRef { slf } fn __next__(&mut self) -> Option> { self.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 = "pyoxigraph")] pub struct PyQuerySolutions { inner: PyQuerySolutionsVariant, } enum PyQuerySolutionsVariant { Query(QuerySolutionIter), Reader(SolutionsReader>), } #[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 { match &self.inner { PyQuerySolutionsVariant::Query(inner) => { inner.variables().iter().map(|v| v.clone().into()).collect() } PyQuerySolutionsVariant::Reader(inner) => { inner.variables().iter().map(|v| v.clone().into()).collect() } } } /// Writes the query results into a file. /// /// It currently supports the following formats: /// /// * `XML `_ (``application/sparql-results+xml`` or ``srx``) /// * `JSON `_ (``application/sparql-results+json`` or ``srj``) /// * `CSV `_ (``text/csv`` or ``csv``) /// * `TSV `_ (``text/tab-separated-values`` or ``tsv``) /// /// It supports also some media type and extension aliases. /// For example, ``application/json`` could also be used for `JSON `_. /// /// :param output: The binary I/O object or file path to write to. For example, it could be a file path as a string or a file writer opened in binary mode with ``open('my_file.ttl', 'wb')``. If :py:const:`None`, a :py:func:`bytes` buffer is returned with the serialized content. /// :type output: io(bytes) or str or pathlib.Path or None, optional /// :param format: the format of the query results serialization using a media type like ``text/csv`` or an extension like `csv`. If :py:const:`None`, the format is guessed from the file name extension. /// :type format: str or None, optional /// :rtype: bytes or None /// :raises ValueError: if the format is not supported. /// :raises OSError: if an error happens during a file writing. /// /// >>> store = Store() /// >>> store.add(Quad(NamedNode('http://example.com'), NamedNode('http://example.com/p'), Literal('1'))) /// >>> results = store.query("SELECT ?s ?p ?o WHERE { ?s ?p ?o }") /// >>> results.serialize(format="json") /// b'{"head":{"vars":["s","p","o"]},"results":{"bindings":[{"s":{"type":"uri","value":"http://example.com"},"p":{"type":"uri","value":"http://example.com/p"},"o":{"type":"literal","value":"1"}}]}}' #[pyo3(signature = (output = None, /, format = None))] fn serialize<'a>( &mut self, output: Option<&PyAny>, format: Option<&str>, py: Python<'a>, ) -> PyResult> { PyWritable::do_write( |output, format| { let mut writer = QueryResultsSerializer::from_format(format) .solutions_writer( output, match &self.inner { PyQuerySolutionsVariant::Query(inner) => inner.variables().to_vec(), PyQuerySolutionsVariant::Reader(inner) => inner.variables().to_vec(), }, ) .map_err(map_io_err)?; match &mut self.inner { PyQuerySolutionsVariant::Query(inner) => { for solution in inner { writer .write(&solution.map_err(map_evaluation_error)?) .map_err(map_io_err)?; } } PyQuerySolutionsVariant::Reader(inner) => { for solution in inner { writer .write(&solution.map_err(map_query_results_parse_error)?) .map_err(map_io_err)?; } } } writer.finish().map_err(map_io_err) }, output, format, py, ) } fn __iter__(slf: PyRef<'_, Self>) -> PyRef { slf } fn __next__(&mut self, py: Python<'_>) -> PyResult> { Ok(match &mut self.inner { PyQuerySolutionsVariant::Query(inner) => allow_threads_unsafe(py, || { inner.next().transpose().map_err(map_evaluation_error) }), PyQuerySolutionsVariant::Reader(inner) => inner .next() .transpose() .map_err(map_query_results_parse_error), }? .map(move |inner| PyQuerySolution { inner })) } } /// A boolean returned by a SPARQL ``ASK`` query. /// /// It can be easily casted to a regular boolean using the :py:func:`bool` function. /// /// >>> store = Store() /// >>> store.add(Quad(NamedNode('http://example.com'), NamedNode('http://example.com/p'), Literal('1'))) /// >>> bool(store.query('ASK { ?s ?p ?o }')) /// True #[pyclass(unsendable, name = "QueryBoolean", module = "pyoxigraph")] pub struct PyQueryBoolean { inner: bool, } #[pymethods] impl PyQueryBoolean { /// Writes the query results into a file. /// /// It currently supports the following formats: /// /// * `XML `_ (``application/sparql-results+xml`` or ``srx``) /// * `JSON `_ (``application/sparql-results+json`` or ``srj``) /// * `CSV `_ (``text/csv`` or ``csv``) /// * `TSV `_ (``text/tab-separated-values`` or ``tsv``) /// /// It supports also some media type and extension aliases. /// For example, ``application/json`` could also be used for `JSON `_. /// /// :param output: The binary I/O object or file path to write to. For example, it could be a file path as a string or a file writer opened in binary mode with ``open('my_file.ttl', 'wb')``. If :py:const:`None`, a :py:func:`bytes` buffer is returned with the serialized content. /// :type output: io(bytes) or str or pathlib.Path or None, optional /// :param format: the format of the query results serialization using a media type like ``text/csv`` or an extension like `csv`. If :py:const:`None`, the format is guessed from the file name extension. /// :type format: str or None, optional /// :rtype: bytes or None /// :raises ValueError: if the format is not supported. /// :raises OSError: if an error happens during a file writing. /// /// >>> store = Store() /// >>> store.add(Quad(NamedNode('http://example.com'), NamedNode('http://example.com/p'), Literal('1'))) /// >>> results = store.query("ASK { ?s ?p ?o }") /// >>> results.serialize(format="json") /// b'{"head":{},"boolean":true}' #[pyo3(signature = (output = None, /, format = None))] fn serialize<'a>( &mut self, output: Option<&PyAny>, format: Option<&str>, py: Python<'a>, ) -> PyResult> { PyWritable::do_write( |output, format| { py.allow_threads(|| { QueryResultsSerializer::from_format(format) .write_boolean_result(output, self.inner) .map_err(map_io_err) }) }, output, format, py, ) } fn __bool__(&self) -> bool { self.inner } fn __richcmp__(&self, other: &Self, op: CompareOp) -> bool { op.matches(self.inner.cmp(&other.inner)) } fn __hash__(&self) -> u64 { self.inner.into() } fn __repr__(&self) -> String { format!("", self.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 = "pyoxigraph")] pub struct PyQueryTriples { inner: QueryTripleIter, } #[pymethods] impl PyQueryTriples { /// Writes the query results into a file. /// /// It currently supports the following formats: /// /// * `N-Triples `_ (``application/n-triples`` or ``nt``) /// * `N-Quads `_ (``application/n-quads`` or ``nq``) /// * `Turtle `_ (``text/turtle`` or ``ttl``) /// * `TriG `_ (``application/trig`` or ``trig``) /// * `N3 `_ (``text/n3`` or ``n3``) /// * `RDF/XML `_ (``application/rdf+xml`` or ``rdf``) /// /// It supports also some media type and extension aliases. /// For example, ``application/turtle`` could also be used for `Turtle `_ /// and ``application/xml`` or ``xml`` for `RDF/XML `_. /// /// :param output: The binary I/O object or file path to write to. For example, it could be a file path as a string or a file writer opened in binary mode with ``open('my_file.ttl', 'wb')``. If :py:const:`None`, a :py:func:`bytes` buffer is returned with the serialized content. /// :type output: io(bytes) or str or pathlib.Path or None, optional /// :param format: the format of the RDF serialization using a media type like ``text/turtle`` or an extension like `ttl`. If :py:const:`None`, the format is guessed from the file name extension. /// :type format: str or None, optional /// :rtype: bytes or None /// :raises ValueError: if the format is not supported. /// :raises OSError: if an error happens during a file writing. /// /// >>> store = Store() /// >>> store.add(Quad(NamedNode('http://example.com'), NamedNode('http://example.com/p'), Literal('1'))) /// >>> results = store.query("CONSTRUCT { ?s ?p ?o } WHERE { ?s ?p ?o }") /// >>> results.serialize(format="nt") /// b' "1" .\n' #[pyo3(signature = (output = None, /, format = None))] fn serialize<'a>( &mut self, output: Option<&PyAny>, format: Option<&str>, py: Python<'a>, ) -> PyResult> { PyWritable::do_write( |output, format| { let mut writer = RdfSerializer::from_format(format).serialize_to_write(output); for triple in &mut self.inner { writer .write_triple(&triple.map_err(map_evaluation_error)?) .map_err(map_io_err)?; } writer.finish().map_err(map_io_err) }, output, format, py, ) } fn __iter__(slf: PyRef<'_, Self>) -> PyRef { slf } fn __next__(&mut self, py: Python<'_>) -> PyResult> { Ok(allow_threads_unsafe(py, || self.inner.next()) .transpose() .map_err(map_evaluation_error)? .map(Into::into)) } } /// Parses SPARQL query results. /// /// It currently supports the following formats: /// /// * `XML `_ (``application/sparql-results+xml`` or ``srx``) /// * `JSON `_ (``application/sparql-results+json`` or ``srj``) /// * `CSV `_ (``text/csv`` or ``csv``) /// * `TSV `_ (``text/tab-separated-values`` or ``tsv``) /// /// It supports also some media type and extension aliases. /// For example, ``application/json`` could also be used for `JSON `_. /// /// :param input: The I/O object or file path to read from. For example, it could be a file path as a string or a file reader opened in binary mode with ``open('my_file.ttl', 'rb')``. /// :type input: io(bytes) or io(str) or str or pathlib.Path /// :param format: the format of the RDF serialization using a media type like ``text/turtle`` or an extension like `ttl`. If :py:const:`None`, the format is guessed from the file name extension. /// :type format: str or None, optional /// :return: an iterator of :py:class:`QuerySolution` or a :py:func:`bool`. /// :rtype: QuerySolutions or QueryBoolean /// :raises ValueError: if the format is not supported. /// :raises SyntaxError: if the provided data is invalid. /// /// >>> input = io.BytesIO(b'?s\t?p\t?o\n\t\t1\n') /// >>> list(parse_query_results(input, "text/tsv")) /// [ p= o=>>] /// /// >>> input = io.BytesIO(b'{"head":{},"boolean":true}') /// >>> parse_query_results(input, "application/sparql-results+json") /// #[pyfunction] #[pyo3(signature = (input, /, format = None))] pub fn parse_query_results( input: &PyAny, format: Option<&str>, py: Python<'_>, ) -> PyResult { let file_path = input.extract::().ok(); let format = parse_format(format, file_path.as_deref())?; let input = if let Some(file_path) = &file_path { PyReadable::from_file(file_path, py).map_err(map_io_err)? } else { PyReadable::from_data(input) }; let results = QueryResultsParser::from_format(format) .read_results(BufReader::new(input)) .map_err(map_query_results_parse_error)?; Ok(match results { QueryResultsReader::Solutions(inner) => PyQuerySolutions { inner: PyQuerySolutionsVariant::Reader(inner), } .into_py(py), QueryResultsReader::Boolean(inner) => PyQueryBoolean { inner }.into_py(py), }) } pub fn map_evaluation_error(error: EvaluationError) -> PyErr { match error { EvaluationError::Parsing(error) => PySyntaxError::new_err(error.to_string()), EvaluationError::Storage(error) => map_storage_error(error), EvaluationError::GraphParsing(error) => map_parse_error(error, None), EvaluationError::ResultsParsing(error) => map_query_results_parse_error(error), EvaluationError::ResultsSerialization(error) => map_io_err(error), EvaluationError::Service(error) => match error.downcast() { Ok(error) => map_io_err(*error), Err(error) => PyRuntimeError::new_err(error.to_string()), }, _ => PyRuntimeError::new_err(error.to_string()), } } pub fn map_query_results_parse_error(error: ParseError) -> PyErr { match error { ParseError::Syntax(error) => PySyntaxError::new_err(error.to_string()), ParseError::Io(error) => map_io_err(error), } }