parent
							
								
									73af297b4c
								
							
						
					
					
						commit
						465b951b7d
					
				| @ -0,0 +1,33 @@ | |||||||
|  | [package] | ||||||
|  | name = "oxrdfio" | ||||||
|  | version = "0.1.0-alpha.1-dev" | ||||||
|  | authors = ["Tpt <thomas@pellissier-tanon.fr>"] | ||||||
|  | license = "MIT OR Apache-2.0" | ||||||
|  | readme = "README.md" | ||||||
|  | keywords = ["RDF"] | ||||||
|  | repository = "https://github.com/oxigraph/oxigraph/tree/master/lib/oxrdfxml" | ||||||
|  | homepage = "https://oxigraph.org/" | ||||||
|  | documentation = "https://docs.rs/oxrdfio" | ||||||
|  | description = """ | ||||||
|  | Parser for various RDF serializations | ||||||
|  | """ | ||||||
|  | edition = "2021" | ||||||
|  | rust-version = "1.65" | ||||||
|  | 
 | ||||||
|  | [features] | ||||||
|  | default = [] | ||||||
|  | async-tokio = ["dep:tokio", "oxrdfxml/async-tokio", "oxttl/async-tokio"] | ||||||
|  | rdf-star = ["oxrdf/rdf-star", "oxttl/rdf-star"] | ||||||
|  | 
 | ||||||
|  | [dependencies] | ||||||
|  | oxrdf = { version = "0.2.0-alpha.1-dev", path = "../oxrdf" } | ||||||
|  | oxrdfxml = { version = "0.1.0-alpha.1-dev", path = "../oxrdfxml" } | ||||||
|  | oxttl = { version = "0.1.0-alpha.1-dev" , path = "../oxttl" } | ||||||
|  | tokio = { version = "1", optional = true, features = ["io-util"] } | ||||||
|  | 
 | ||||||
|  | [dev-dependencies] | ||||||
|  | tokio = { version = "1", features = ["rt", "macros"] } | ||||||
|  | 
 | ||||||
|  | [package.metadata.docs.rs] | ||||||
|  | all-features = true | ||||||
|  | rustdoc-args = ["--cfg", "docsrs"] | ||||||
| @ -0,0 +1,61 @@ | |||||||
|  | OxRDF I/O | ||||||
|  | ========= | ||||||
|  | 
 | ||||||
|  | [](https://crates.io/crates/oxrdfio) | ||||||
|  | [](https://docs.rs/oxrdfio) | ||||||
|  | [](https://crates.io/crates/oxrdfio) | ||||||
|  | [](https://github.com/oxigraph/oxigraph/actions) | ||||||
|  | [](https://gitter.im/oxigraph/community?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge) | ||||||
|  | 
 | ||||||
|  | OxRDF I/O is a set of parsers and serializers for RDF. | ||||||
|  | 
 | ||||||
|  | It supports: | ||||||
|  | * [N3](https://w3c.github.io/N3/spec/) using [`oxttl`](https://crates.io/crates/oxttl) | ||||||
|  | * [N-Quads](https://www.w3.org/TR/n-quads/) using [`oxttl`](https://crates.io/crates/oxttl) | ||||||
|  | * [N-Triples](https://www.w3.org/TR/n-triples/) using [`oxttl`](https://crates.io/crates/oxttl) | ||||||
|  | * [RDF/XML](https://www.w3.org/TR/rdf-syntax-grammar/) using [`oxrdfxml`](https://crates.io/crates/oxrdfxml) | ||||||
|  | * [TriG](https://www.w3.org/TR/trig/) using [`oxttl`](https://crates.io/crates/oxttl) | ||||||
|  | * [Turtle](https://www.w3.org/TR/turtle/) using [`oxttl`](https://crates.io/crates/oxttl) | ||||||
|  | 
 | ||||||
|  | Support for [SPARQL-star](https://w3c.github.io/rdf-star/cg-spec/2021-12-17.html) is also available behind the `rdf-star`feature for [Turtle-star](https://w3c.github.io/rdf-star/cg-spec/2021-12-17.html#turtle-star), [TriG-star](https://w3c.github.io/rdf-star/cg-spec/2021-12-17.html#trig-star), [N-Triples-star](https://w3c.github.io/rdf-star/cg-spec/2021-12-17.html#n-triples-star) and [N-Quads-star](https://w3c.github.io/rdf-star/cg-spec/2021-12-17.html#n-quads-star). | ||||||
|  | 
 | ||||||
|  | It is designed as a low level parser compatible with both synchronous and asynchronous I/O (behind the `async-tokio` feature). | ||||||
|  | 
 | ||||||
|  | Usage example counting the number of people in a Turtle file: | ||||||
|  | ```rust | ||||||
|  | use oxrdf::{NamedNodeRef, vocab::rdf}; | ||||||
|  | use oxrdfio::{RdfFormat, RdfParser}; | ||||||
|  | 
 | ||||||
|  | let file = b"@base <http://example.com/> . | ||||||
|  | @prefix schema: <http://schema.org/> . | ||||||
|  | <foo> a schema:Person ; | ||||||
|  |     schema:name \"Foo\" . | ||||||
|  | <bar> a schema:Person ; | ||||||
|  |     schema:name \"Bar\" ."; | ||||||
|  | 
 | ||||||
|  | let schema_person = NamedNodeRef::new("http://schema.org/Person").unwrap(); | ||||||
|  | let mut count = 0; | ||||||
|  | for quad in RdfParser::from_format(RdfFormat::Turtle).parse_read(file.as_ref()) { | ||||||
|  |     let quad = quad.unwrap(); | ||||||
|  |     if quad.predicate == rdf::TYPE && quad.object == schema_person.into() { | ||||||
|  |         count += 1; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | assert_eq!(2, count); | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ## License | ||||||
|  | 
 | ||||||
|  | This project is licensed under either of | ||||||
|  | 
 | ||||||
|  | * Apache License, Version 2.0, ([LICENSE-APACHE](../LICENSE-APACHE) or | ||||||
|  |   `<http://www.apache.org/licenses/LICENSE-2.0>`) | ||||||
|  | * MIT license ([LICENSE-MIT](../LICENSE-MIT) or | ||||||
|  |   `<http://opensource.org/licenses/MIT>`) | ||||||
|  | 
 | ||||||
|  | at your option. | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ### Contribution | ||||||
|  | 
 | ||||||
|  | Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in Oxigraph by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions. | ||||||
| @ -0,0 +1,148 @@ | |||||||
|  | use std::error::Error; | ||||||
|  | use std::{fmt, io}; | ||||||
|  | 
 | ||||||
|  | /// Error returned during RDF format parsing.
 | ||||||
|  | #[derive(Debug)] | ||||||
|  | pub enum ParseError { | ||||||
|  |     /// I/O error during parsing (file not found...).
 | ||||||
|  |     Io(io::Error), | ||||||
|  |     /// An error in the file syntax.
 | ||||||
|  |     Syntax(SyntaxError), | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl ParseError { | ||||||
|  |     pub(crate) fn msg(msg: &'static str) -> Self { | ||||||
|  |         Self::Syntax(SyntaxError { | ||||||
|  |             inner: SyntaxErrorKind::Msg { msg }, | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::Display for ParseError { | ||||||
|  |     #[inline] | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||||
|  |         match self { | ||||||
|  |             Self::Io(e) => e.fmt(f), | ||||||
|  |             Self::Syntax(e) => e.fmt(f), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Error for ParseError { | ||||||
|  |     #[inline] | ||||||
|  |     fn source(&self) -> Option<&(dyn Error + 'static)> { | ||||||
|  |         match self { | ||||||
|  |             Self::Io(e) => Some(e), | ||||||
|  |             Self::Syntax(e) => Some(e), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<oxttl::SyntaxError> for SyntaxError { | ||||||
|  |     #[inline] | ||||||
|  |     fn from(error: oxttl::SyntaxError) -> Self { | ||||||
|  |         SyntaxError { | ||||||
|  |             inner: SyntaxErrorKind::Turtle(error), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<oxttl::ParseError> for ParseError { | ||||||
|  |     #[inline] | ||||||
|  |     fn from(error: oxttl::ParseError) -> Self { | ||||||
|  |         match error { | ||||||
|  |             oxttl::ParseError::Syntax(e) => Self::Syntax(e.into()), | ||||||
|  |             oxttl::ParseError::Io(e) => Self::Io(e), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<oxrdfxml::SyntaxError> for SyntaxError { | ||||||
|  |     #[inline] | ||||||
|  |     fn from(error: oxrdfxml::SyntaxError) -> Self { | ||||||
|  |         SyntaxError { | ||||||
|  |             inner: SyntaxErrorKind::RdfXml(error), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<oxrdfxml::ParseError> for ParseError { | ||||||
|  |     #[inline] | ||||||
|  |     fn from(error: oxrdfxml::ParseError) -> Self { | ||||||
|  |         match error { | ||||||
|  |             oxrdfxml::ParseError::Syntax(e) => Self::Syntax(e.into()), | ||||||
|  |             oxrdfxml::ParseError::Io(e) => Self::Io(e), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<io::Error> for ParseError { | ||||||
|  |     #[inline] | ||||||
|  |     fn from(error: io::Error) -> Self { | ||||||
|  |         Self::Io(error) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<SyntaxError> for ParseError { | ||||||
|  |     #[inline] | ||||||
|  |     fn from(error: SyntaxError) -> Self { | ||||||
|  |         Self::Syntax(error) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<ParseError> for io::Error { | ||||||
|  |     #[inline] | ||||||
|  |     fn from(error: ParseError) -> Self { | ||||||
|  |         match error { | ||||||
|  |             ParseError::Io(error) => error, | ||||||
|  |             ParseError::Syntax(error) => error.into(), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// An error in the syntax of the parsed file.
 | ||||||
|  | #[derive(Debug)] | ||||||
|  | pub struct SyntaxError { | ||||||
|  |     inner: SyntaxErrorKind, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[derive(Debug)] | ||||||
|  | enum SyntaxErrorKind { | ||||||
|  |     Turtle(oxttl::SyntaxError), | ||||||
|  |     RdfXml(oxrdfxml::SyntaxError), | ||||||
|  | 
 | ||||||
|  |     Msg { msg: &'static str }, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::Display for SyntaxError { | ||||||
|  |     #[inline] | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||||
|  |         match &self.inner { | ||||||
|  |             SyntaxErrorKind::Turtle(e) => e.fmt(f), | ||||||
|  |             SyntaxErrorKind::RdfXml(e) => e.fmt(f), | ||||||
|  |             SyntaxErrorKind::Msg { msg } => write!(f, "{msg}"), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Error for SyntaxError { | ||||||
|  |     #[inline] | ||||||
|  |     fn source(&self) -> Option<&(dyn Error + 'static)> { | ||||||
|  |         match &self.inner { | ||||||
|  |             SyntaxErrorKind::Turtle(e) => Some(e), | ||||||
|  |             SyntaxErrorKind::RdfXml(e) => Some(e), | ||||||
|  |             SyntaxErrorKind::Msg { .. } => None, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<SyntaxError> for io::Error { | ||||||
|  |     #[inline] | ||||||
|  |     fn from(error: SyntaxError) -> Self { | ||||||
|  |         match error.inner { | ||||||
|  |             SyntaxErrorKind::Turtle(error) => error.into(), | ||||||
|  |             SyntaxErrorKind::RdfXml(error) => error.into(), | ||||||
|  |             SyntaxErrorKind::Msg { msg } => io::Error::new(io::ErrorKind::InvalidData, msg), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,203 @@ | |||||||
|  | use std::fmt; | ||||||
|  | 
 | ||||||
|  | /// RDF serialization formats.
 | ||||||
|  | ///
 | ||||||
|  | /// This enumeration is non exhaustive. New formats like JSON-LD might be added in the future.
 | ||||||
|  | #[derive(Eq, PartialEq, Debug, Clone, Copy, Hash)] | ||||||
|  | #[non_exhaustive] | ||||||
|  | pub enum RdfFormat { | ||||||
|  |     /// [N3](https://w3c.github.io/N3/spec/)
 | ||||||
|  |     N3, | ||||||
|  |     /// [N-Quads](https://www.w3.org/TR/n-quads/)
 | ||||||
|  |     NQuads, | ||||||
|  |     /// [N-Triples](https://www.w3.org/TR/n-triples/)
 | ||||||
|  |     NTriples, | ||||||
|  |     /// [RDF/XML](https://www.w3.org/TR/rdf-syntax-grammar/)
 | ||||||
|  |     RdfXml, | ||||||
|  |     /// [TriG](https://www.w3.org/TR/trig/)
 | ||||||
|  |     TriG, | ||||||
|  |     /// [Turtle](https://www.w3.org/TR/turtle/)
 | ||||||
|  |     Turtle, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl RdfFormat { | ||||||
|  |     /// The format canonical IRI according to the [Unique URIs for file formats registry](https://www.w3.org/ns/formats/).
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::RdfFormat;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(RdfFormat::NTriples.iri(), "http://www.w3.org/ns/formats/N-Triples")
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     pub const fn iri(self) -> &'static str { | ||||||
|  |         match self { | ||||||
|  |             Self::N3 => "http://www.w3.org/ns/formats/N3", | ||||||
|  |             Self::NQuads => "http://www.w3.org/ns/formats/N-Quads", | ||||||
|  |             Self::NTriples => "http://www.w3.org/ns/formats/N-Triples", | ||||||
|  |             Self::RdfXml => "http://www.w3.org/ns/formats/RDF_XML", | ||||||
|  |             Self::TriG => "http://www.w3.org/ns/formats/TriG", | ||||||
|  |             Self::Turtle => "http://www.w3.org/ns/formats/Turtle", | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// The format [IANA media type](https://tools.ietf.org/html/rfc2046).
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::RdfFormat;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(RdfFormat::NTriples.media_type(), "application/n-triples")
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     pub const fn media_type(self) -> &'static str { | ||||||
|  |         match self { | ||||||
|  |             Self::N3 => "text/n3", | ||||||
|  |             Self::NQuads => "application/n-quads", | ||||||
|  |             Self::NTriples => "application/n-triples", | ||||||
|  |             Self::RdfXml => "application/rdf+xml", | ||||||
|  |             Self::TriG => "application/trig", | ||||||
|  |             Self::Turtle => "text/turtle", | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// The format [IANA-registered](https://tools.ietf.org/html/rfc2046) file extension.
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::RdfFormat;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(RdfFormat::NTriples.file_extension(), "nt")
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     pub const fn file_extension(self) -> &'static str { | ||||||
|  |         match self { | ||||||
|  |             Self::N3 => "n3", | ||||||
|  |             Self::NQuads => "nq", | ||||||
|  |             Self::NTriples => "nt", | ||||||
|  |             Self::RdfXml => "rdf", | ||||||
|  |             Self::TriG => "trig", | ||||||
|  |             Self::Turtle => "ttl", | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// The format name.
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::RdfFormat;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(RdfFormat::NTriples.name(), "N-Triples")
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     pub const fn name(self) -> &'static str { | ||||||
|  |         match self { | ||||||
|  |             Self::N3 => "N3", | ||||||
|  |             Self::NQuads => "N-Quads", | ||||||
|  |             Self::NTriples => "N-Triples", | ||||||
|  |             Self::RdfXml => "RDF/XML", | ||||||
|  |             Self::TriG => "TriG", | ||||||
|  |             Self::Turtle => "Turtle", | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Checks if the formats supports [RDF datasets](https://www.w3.org/TR/rdf11-concepts/#dfn-rdf-dataset) and not only [RDF graphs](https://www.w3.org/TR/rdf11-concepts/#dfn-rdf-graph).
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::RdfFormat;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(RdfFormat::NTriples.supports_datasets(), false);
 | ||||||
|  |     /// assert_eq!(RdfFormat::NQuads.supports_datasets(), true);
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     pub const fn supports_datasets(self) -> bool { | ||||||
|  |         matches!(self, Self::NQuads | Self::TriG) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Checks if the formats supports [RDF-star quoted triples](https://w3c.github.io/rdf-star/cg-spec/2021-12-17.html#dfn-quoted).
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::RdfFormat;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(RdfFormat::NTriples.supports_rdf_star(), true);
 | ||||||
|  |     /// assert_eq!(RdfFormat::RdfXml.supports_rdf_star(), false);
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     #[cfg(feature = "rdf-star")] | ||||||
|  |     pub const fn supports_rdf_star(self) -> bool { | ||||||
|  |         matches!( | ||||||
|  |             self, | ||||||
|  |             Self::NTriples | Self::NQuads | Self::Turtle | Self::TriG | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Looks for a known format from a media type.
 | ||||||
|  |     ///
 | ||||||
|  |     /// It supports some media type aliases.
 | ||||||
|  |     /// For example, "application/xml" is going to return `RdfFormat::RdfXml` even if it is not its canonical media type.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Example:
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::RdfFormat;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(RdfFormat::from_media_type("text/turtle; charset=utf-8"), Some(RdfFormat::Turtle))
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn from_media_type(media_type: &str) -> Option<Self> { | ||||||
|  |         const MEDIA_TYPES: [(&str, RdfFormat); 14] = [ | ||||||
|  |             ("application/n-quads", RdfFormat::NQuads), | ||||||
|  |             ("application/n-triples", RdfFormat::NTriples), | ||||||
|  |             ("application/rdf+xml", RdfFormat::RdfXml), | ||||||
|  |             ("application/trig", RdfFormat::TriG), | ||||||
|  |             ("application/turtle", RdfFormat::Turtle), | ||||||
|  |             ("application/xml", RdfFormat::RdfXml), | ||||||
|  |             ("application/x-trig", RdfFormat::TriG), | ||||||
|  |             ("application/x-turtle", RdfFormat::Turtle), | ||||||
|  |             ("text/n3", RdfFormat::N3), | ||||||
|  |             ("text/nquads", RdfFormat::NQuads), | ||||||
|  |             ("text/plain", RdfFormat::NTriples), | ||||||
|  |             ("text/turtle", RdfFormat::Turtle), | ||||||
|  |             ("text/xml", RdfFormat::RdfXml), | ||||||
|  |             ("text/x-nquads", RdfFormat::NQuads), | ||||||
|  |         ]; | ||||||
|  |         let media_type = media_type.split(';').next()?.trim(); | ||||||
|  |         for (candidate_media_type, candidate_id) in MEDIA_TYPES { | ||||||
|  |             if candidate_media_type.eq_ignore_ascii_case(media_type) { | ||||||
|  |                 return Some(candidate_id); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         None | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Looks for a known format from an extension.
 | ||||||
|  |     ///
 | ||||||
|  |     /// It supports some aliases.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Example:
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::RdfFormat;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(RdfFormat::from_extension("nt"), Some(RdfFormat::NTriples))
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn from_extension(extension: &str) -> Option<Self> { | ||||||
|  |         const MEDIA_TYPES: [(&str, RdfFormat); 8] = [ | ||||||
|  |             ("n3", RdfFormat::N3), | ||||||
|  |             ("nq", RdfFormat::NQuads), | ||||||
|  |             ("nt", RdfFormat::NTriples), | ||||||
|  |             ("rdf", RdfFormat::RdfXml), | ||||||
|  |             ("trig", RdfFormat::TriG), | ||||||
|  |             ("ttl", RdfFormat::Turtle), | ||||||
|  |             ("txt", RdfFormat::NTriples), | ||||||
|  |             ("xml", RdfFormat::RdfXml), | ||||||
|  |         ]; | ||||||
|  |         for (candidate_extension, candidate_id) in MEDIA_TYPES { | ||||||
|  |             if candidate_extension.eq_ignore_ascii_case(extension) { | ||||||
|  |                 return Some(candidate_id); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         None | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::Display for RdfFormat { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||||
|  |         f.write_str(self.name()) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,19 @@ | |||||||
|  | #![doc = include_str!("../README.md")] | ||||||
|  | #![doc(test(attr(deny(warnings))))] | ||||||
|  | #![cfg_attr(docsrs, feature(doc_auto_cfg))] | ||||||
|  | #![doc(html_favicon_url = "https://raw.githubusercontent.com/oxigraph/oxigraph/main/logo.svg")] | ||||||
|  | #![doc(html_logo_url = "https://raw.githubusercontent.com/oxigraph/oxigraph/main/logo.svg")] | ||||||
|  | 
 | ||||||
|  | mod error; | ||||||
|  | mod format; | ||||||
|  | mod parser; | ||||||
|  | mod serializer; | ||||||
|  | 
 | ||||||
|  | pub use error::{ParseError, SyntaxError}; | ||||||
|  | pub use format::RdfFormat; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | pub use parser::FromTokioAsyncReadQuadReader; | ||||||
|  | pub use parser::{FromReadQuadReader, RdfParser}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | pub use serializer::ToTokioAsyncWriteQuadWriter; | ||||||
|  | pub use serializer::{RdfSerializer, ToWriteQuadWriter}; | ||||||
| @ -0,0 +1,577 @@ | |||||||
|  | //! Utilities to read RDF graphs and datasets.
 | ||||||
|  | 
 | ||||||
|  | pub use crate::error::{ParseError, SyntaxError}; | ||||||
|  | use crate::format::RdfFormat; | ||||||
|  | use oxrdf::{BlankNode, GraphName, IriParseError, Quad, Subject, Term, Triple}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxrdfxml::FromTokioAsyncReadRdfXmlReader; | ||||||
|  | use oxrdfxml::{FromReadRdfXmlReader, RdfXmlParser}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxttl::n3::FromTokioAsyncReadN3Reader; | ||||||
|  | use oxttl::n3::{FromReadN3Reader, N3Parser, N3Quad, N3Term}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxttl::nquads::FromTokioAsyncReadNQuadsReader; | ||||||
|  | use oxttl::nquads::{FromReadNQuadsReader, NQuadsParser}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxttl::ntriples::FromTokioAsyncReadNTriplesReader; | ||||||
|  | use oxttl::ntriples::{FromReadNTriplesReader, NTriplesParser}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxttl::trig::FromTokioAsyncReadTriGReader; | ||||||
|  | use oxttl::trig::{FromReadTriGReader, TriGParser}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxttl::turtle::FromTokioAsyncReadTurtleReader; | ||||||
|  | use oxttl::turtle::{FromReadTurtleReader, TurtleParser}; | ||||||
|  | use std::collections::HashMap; | ||||||
|  | use std::io::Read; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use tokio::io::AsyncRead; | ||||||
|  | 
 | ||||||
|  | /// Parsers for RDF serialization formats.
 | ||||||
|  | ///
 | ||||||
|  | /// It currently supports the following formats:
 | ||||||
|  | /// * [N3](https://w3c.github.io/N3/spec/) ([`RdfFormat::N3`])
 | ||||||
|  | /// * [N-Quads](https://www.w3.org/TR/n-quads/) ([`RdfFormat::NQuads`])
 | ||||||
|  | /// * [N-Triples](https://www.w3.org/TR/n-triples/) ([`RdfFormat::NTriples`])
 | ||||||
|  | /// * [RDF/XML](https://www.w3.org/TR/rdf-syntax-grammar/) ([`RdfFormat::RdfXml`])
 | ||||||
|  | /// * [TriG](https://www.w3.org/TR/trig/) ([`RdfFormat::TriG`])
 | ||||||
|  | /// * [Turtle](https://www.w3.org/TR/turtle/) ([`RdfFormat::Turtle`])
 | ||||||
|  | ///
 | ||||||
|  | /// Note the useful options:
 | ||||||
|  | /// - [`with_base_iri`](RdfParser::with_base_iri) to resolve the relative IRIs.
 | ||||||
|  | /// - [`rename_blank_nodes`](RdfParser::rename_blank_nodes) to rename the blank nodes to auto-generated numbers to avoid conflicts when merging RDF graphs together.
 | ||||||
|  | /// - [`without_named_graphs`](RdfParser::without_named_graphs) to parse a single graph.
 | ||||||
|  | ///
 | ||||||
|  | /// ```
 | ||||||
|  | /// use oxrdfio::{RdfFormat, RdfParser};
 | ||||||
|  | ///
 | ||||||
|  | /// let file = "<http://example.com/s> <http://example.com/p> <http://example.com/o> .";
 | ||||||
|  | ///
 | ||||||
|  | /// let parser = RdfParser::from_format(RdfFormat::NTriples);
 | ||||||
|  | /// let quads = parser.parse_read(file.as_bytes()).collect::<Result<Vec<_>,_>>()?;
 | ||||||
|  | ///
 | ||||||
|  | /// assert_eq!(quads.len(), 1);
 | ||||||
|  | /// assert_eq!(quads[0].subject.to_string(), "<http://example.com/s>");
 | ||||||
|  | /// # std::io::Result::Ok(())
 | ||||||
|  | /// ```
 | ||||||
|  | pub struct RdfParser { | ||||||
|  |     inner: RdfParserKind, | ||||||
|  |     default_graph: GraphName, | ||||||
|  |     without_named_graphs: bool, | ||||||
|  |     rename_blank_nodes: bool, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | enum RdfParserKind { | ||||||
|  |     N3(N3Parser), | ||||||
|  |     NQuads(NQuadsParser), | ||||||
|  |     NTriples(NTriplesParser), | ||||||
|  |     RdfXml(RdfXmlParser), | ||||||
|  |     TriG(TriGParser), | ||||||
|  |     Turtle(TurtleParser), | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl RdfParser { | ||||||
|  |     /// Builds a parser for the given format.
 | ||||||
|  |     #[inline] | ||||||
|  |     #[must_use] | ||||||
|  |     pub fn from_format(format: RdfFormat) -> Self { | ||||||
|  |         Self { | ||||||
|  |             inner: match format { | ||||||
|  |                 RdfFormat::N3 => RdfParserKind::N3(N3Parser::new()), | ||||||
|  |                 RdfFormat::NQuads => RdfParserKind::NQuads({ | ||||||
|  |                     #[cfg(feature = "rdf-star")] | ||||||
|  |                     { | ||||||
|  |                         NQuadsParser::new().with_quoted_triples() | ||||||
|  |                     } | ||||||
|  |                     #[cfg(not(feature = "rdf-star"))] | ||||||
|  |                     { | ||||||
|  |                         NQuadsParser::new() | ||||||
|  |                     } | ||||||
|  |                 }), | ||||||
|  |                 RdfFormat::NTriples => RdfParserKind::NTriples({ | ||||||
|  |                     #[cfg(feature = "rdf-star")] | ||||||
|  |                     { | ||||||
|  |                         NTriplesParser::new().with_quoted_triples() | ||||||
|  |                     } | ||||||
|  |                     #[cfg(not(feature = "rdf-star"))] | ||||||
|  |                     { | ||||||
|  |                         NTriplesParser::new() | ||||||
|  |                     } | ||||||
|  |                 }), | ||||||
|  |                 RdfFormat::RdfXml => RdfParserKind::RdfXml(RdfXmlParser::new()), | ||||||
|  |                 RdfFormat::TriG => RdfParserKind::TriG({ | ||||||
|  |                     #[cfg(feature = "rdf-star")] | ||||||
|  |                     { | ||||||
|  |                         TriGParser::new().with_quoted_triples() | ||||||
|  |                     } | ||||||
|  |                     #[cfg(not(feature = "rdf-star"))] | ||||||
|  |                     { | ||||||
|  |                         TriGParser::new() | ||||||
|  |                     } | ||||||
|  |                 }), | ||||||
|  |                 RdfFormat::Turtle => RdfParserKind::Turtle({ | ||||||
|  |                     #[cfg(feature = "rdf-star")] | ||||||
|  |                     { | ||||||
|  |                         TurtleParser::new().with_quoted_triples() | ||||||
|  |                     } | ||||||
|  |                     #[cfg(not(feature = "rdf-star"))] | ||||||
|  |                     { | ||||||
|  |                         TurtleParser::new() | ||||||
|  |                     } | ||||||
|  |                 }), | ||||||
|  |             }, | ||||||
|  |             default_graph: GraphName::DefaultGraph, | ||||||
|  |             without_named_graphs: false, | ||||||
|  |             rename_blank_nodes: false, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Provides an IRI that could be used to resolve the file relative IRIs.
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::{RdfFormat, RdfParser};
 | ||||||
|  |     ///
 | ||||||
|  |     /// let file = "</s> </p> </o> .";
 | ||||||
|  |     ///
 | ||||||
|  |     /// let parser = RdfParser::from_format(RdfFormat::Turtle).with_base_iri("http://example.com")?;
 | ||||||
|  |     /// let quads = parser.parse_read(file.as_bytes()).collect::<Result<Vec<_>,_>>()?;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(quads.len(), 1);
 | ||||||
|  |     /// assert_eq!(quads[0].subject.to_string(), "<http://example.com/s>");
 | ||||||
|  |     /// # Result::<_,Box<dyn std::error::Error>>::Ok(())
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn with_base_iri(self, base_iri: impl Into<String>) -> Result<Self, IriParseError> { | ||||||
|  |         Ok(Self { | ||||||
|  |             inner: match self.inner { | ||||||
|  |                 RdfParserKind::N3(p) => RdfParserKind::N3(p), | ||||||
|  |                 RdfParserKind::NTriples(p) => RdfParserKind::NTriples(p), | ||||||
|  |                 RdfParserKind::NQuads(p) => RdfParserKind::NQuads(p), | ||||||
|  |                 RdfParserKind::RdfXml(p) => RdfParserKind::RdfXml(p.with_base_iri(base_iri)?), | ||||||
|  |                 RdfParserKind::TriG(p) => RdfParserKind::TriG(p.with_base_iri(base_iri)?), | ||||||
|  |                 RdfParserKind::Turtle(p) => RdfParserKind::Turtle(p.with_base_iri(base_iri)?), | ||||||
|  |             }, | ||||||
|  |             default_graph: self.default_graph, | ||||||
|  |             without_named_graphs: self.without_named_graphs, | ||||||
|  |             rename_blank_nodes: self.rename_blank_nodes, | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Provides the name graph name that should replace the default graph in the returned quads.
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdf::NamedNode;
 | ||||||
|  |     /// use oxrdfio::{RdfFormat, RdfParser};
 | ||||||
|  |     ///
 | ||||||
|  |     /// let file = "<http://example.com/s> <http://example.com/p> <http://example.com/o> .";
 | ||||||
|  |     ///
 | ||||||
|  |     /// let parser = RdfParser::from_format(RdfFormat::Turtle).with_default_graph(NamedNode::new("http://example.com/g")?);
 | ||||||
|  |     /// let quads = parser.parse_read(file.as_bytes()).collect::<Result<Vec<_>,_>>()?;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(quads.len(), 1);
 | ||||||
|  |     /// assert_eq!(quads[0].graph_name.to_string(), "<http://example.com/g>");
 | ||||||
|  |     /// # Result::<_,Box<dyn std::error::Error>>::Ok(())
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     #[must_use] | ||||||
|  |     pub fn with_default_graph(self, default_graph: impl Into<GraphName>) -> Self { | ||||||
|  |         Self { | ||||||
|  |             inner: self.inner, | ||||||
|  |             default_graph: default_graph.into(), | ||||||
|  |             without_named_graphs: self.without_named_graphs, | ||||||
|  |             rename_blank_nodes: self.rename_blank_nodes, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Sets that the parser must fail if parsing a named graph.
 | ||||||
|  |     ///
 | ||||||
|  |     /// This function restricts the parser to only parse a single [RDF graph](https://www.w3.org/TR/rdf11-concepts/#dfn-rdf-graph) and not an [RDF dataset](https://www.w3.org/TR/rdf11-concepts/#dfn-rdf-dataset).
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::{RdfFormat, RdfParser};
 | ||||||
|  |     ///
 | ||||||
|  |     /// let file = "<http://example.com/s> <http://example.com/p> <http://example.com/o> <http://example.com/g> .";
 | ||||||
|  |     ///
 | ||||||
|  |     /// let parser = RdfParser::from_format(RdfFormat::NQuads).without_named_graphs();
 | ||||||
|  |     /// assert!(parser.parse_read(file.as_bytes()).next().unwrap().is_err());
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     #[must_use] | ||||||
|  |     pub fn without_named_graphs(self) -> Self { | ||||||
|  |         Self { | ||||||
|  |             inner: self.inner, | ||||||
|  |             default_graph: self.default_graph, | ||||||
|  |             without_named_graphs: true, | ||||||
|  |             rename_blank_nodes: self.rename_blank_nodes, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Renames the blank nodes ids from the ones set in the serialization to random ids.
 | ||||||
|  |     ///
 | ||||||
|  |     /// This allows to avoid id conflicts when merging graphs together.
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::{RdfFormat, RdfParser};
 | ||||||
|  |     ///
 | ||||||
|  |     /// let file = "_:a <http://example.com/p> <http://example.com/o> .";
 | ||||||
|  |     ///
 | ||||||
|  |     /// let parser = RdfParser::from_format(RdfFormat::NQuads).rename_blank_nodes();
 | ||||||
|  |     /// let result1 = parser.parse_read(file.as_bytes()).collect::<Result<Vec<_>,_>>()?;
 | ||||||
|  |     /// let result2 = parser.parse_read(file.as_bytes()).collect::<Result<Vec<_>,_>>()?;
 | ||||||
|  |     /// assert_ne!(result1, result2);
 | ||||||
|  |     /// # Result::<_,Box<dyn std::error::Error>>::Ok(())
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[inline] | ||||||
|  |     #[must_use] | ||||||
|  |     pub fn rename_blank_nodes(self) -> Self { | ||||||
|  |         Self { | ||||||
|  |             inner: self.inner, | ||||||
|  |             default_graph: self.default_graph, | ||||||
|  |             without_named_graphs: self.without_named_graphs, | ||||||
|  |             rename_blank_nodes: true, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Parses from a [`Read`] implementation and returns an iterator of quads.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Reads are buffered.
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::{RdfFormat, RdfParser};
 | ||||||
|  |     ///
 | ||||||
|  |     /// let file = "<http://example.com/s> <http://example.com/p> <http://example.com/o> .";
 | ||||||
|  |     ///
 | ||||||
|  |     /// let parser = RdfParser::from_format(RdfFormat::NTriples);
 | ||||||
|  |     /// let quads = parser.parse_read(file.as_bytes()).collect::<Result<Vec<_>,_>>()?;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(quads.len(), 1);
 | ||||||
|  |     /// assert_eq!(quads[0].subject.to_string(), "<http://example.com/s>");
 | ||||||
|  |     /// # std::io::Result::Ok(())
 | ||||||
|  |     /// ```
 | ||||||
|  |     pub fn parse_read<R: Read>(&self, reader: R) -> FromReadQuadReader<R> { | ||||||
|  |         FromReadQuadReader { | ||||||
|  |             parser: match &self.inner { | ||||||
|  |                 RdfParserKind::N3(p) => FromReadQuadReaderKind::N3(p.parse_read(reader)), | ||||||
|  |                 RdfParserKind::NQuads(p) => FromReadQuadReaderKind::NQuads(p.parse_read(reader)), | ||||||
|  |                 RdfParserKind::NTriples(p) => { | ||||||
|  |                     FromReadQuadReaderKind::NTriples(p.parse_read(reader)) | ||||||
|  |                 } | ||||||
|  |                 RdfParserKind::RdfXml(p) => FromReadQuadReaderKind::RdfXml(p.parse_read(reader)), | ||||||
|  |                 RdfParserKind::TriG(p) => FromReadQuadReaderKind::TriG(p.parse_read(reader)), | ||||||
|  |                 RdfParserKind::Turtle(p) => FromReadQuadReaderKind::Turtle(p.parse_read(reader)), | ||||||
|  |             }, | ||||||
|  |             mapper: QuadMapper { | ||||||
|  |                 default_graph: self.default_graph.clone(), | ||||||
|  |                 without_named_graphs: self.without_named_graphs, | ||||||
|  |                 blank_node_map: self.rename_blank_nodes.then(HashMap::new), | ||||||
|  |             }, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Parses from a Tokio [`AsyncRead`] implementation and returns an async iterator of quads.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Reads are buffered.
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::{RdfFormat, RdfParser, ParseError};
 | ||||||
|  |     ///
 | ||||||
|  |     /// #[tokio::main(flavor = "current_thread")]
 | ||||||
|  |     /// async fn main() -> Result<(), ParseError> {
 | ||||||
|  |     ///     let file = "<http://example.com/s> <http://example.com/p> <http://example.com/o> .";
 | ||||||
|  |     ///
 | ||||||
|  |     ///     let parser = RdfParser::from_format(RdfFormat::NTriples);
 | ||||||
|  |     ///     let mut reader = parser.parse_tokio_async_read(file.as_bytes());
 | ||||||
|  |     ///     if let Some(quad) = reader.next().await {
 | ||||||
|  |     ///         assert_eq!(quad?.subject.to_string(), "<http://example.com/s>");
 | ||||||
|  |     ///     }
 | ||||||
|  |     ///     Ok(())
 | ||||||
|  |     /// }
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[cfg(feature = "async-tokio")] | ||||||
|  |     pub fn parse_tokio_async_read<R: AsyncRead + Unpin>( | ||||||
|  |         &self, | ||||||
|  |         reader: R, | ||||||
|  |     ) -> FromTokioAsyncReadQuadReader<R> { | ||||||
|  |         FromTokioAsyncReadQuadReader { | ||||||
|  |             parser: match &self.inner { | ||||||
|  |                 RdfParserKind::N3(p) => { | ||||||
|  |                     FromTokioAsyncReadQuadReaderKind::N3(p.parse_tokio_async_read(reader)) | ||||||
|  |                 } | ||||||
|  |                 RdfParserKind::NQuads(p) => { | ||||||
|  |                     FromTokioAsyncReadQuadReaderKind::NQuads(p.parse_tokio_async_read(reader)) | ||||||
|  |                 } | ||||||
|  |                 RdfParserKind::NTriples(p) => { | ||||||
|  |                     FromTokioAsyncReadQuadReaderKind::NTriples(p.parse_tokio_async_read(reader)) | ||||||
|  |                 } | ||||||
|  |                 RdfParserKind::RdfXml(p) => { | ||||||
|  |                     FromTokioAsyncReadQuadReaderKind::RdfXml(p.parse_tokio_async_read(reader)) | ||||||
|  |                 } | ||||||
|  |                 RdfParserKind::TriG(p) => { | ||||||
|  |                     FromTokioAsyncReadQuadReaderKind::TriG(p.parse_tokio_async_read(reader)) | ||||||
|  |                 } | ||||||
|  |                 RdfParserKind::Turtle(p) => { | ||||||
|  |                     FromTokioAsyncReadQuadReaderKind::Turtle(p.parse_tokio_async_read(reader)) | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             mapper: QuadMapper { | ||||||
|  |                 default_graph: self.default_graph.clone(), | ||||||
|  |                 without_named_graphs: self.without_named_graphs, | ||||||
|  |                 blank_node_map: self.rename_blank_nodes.then(HashMap::new), | ||||||
|  |             }, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Parses a RDF file from a [`Read`] implementation. Can be built using [`RdfParser::parse_read`].
 | ||||||
|  | ///
 | ||||||
|  | /// Reads are buffered.
 | ||||||
|  | ///
 | ||||||
|  | /// ```
 | ||||||
|  | /// use oxrdfio::{RdfFormat, RdfParser};
 | ||||||
|  | ///
 | ||||||
|  | /// let file = "<http://example.com/s> <http://example.com/p> <http://example.com/o> .";
 | ||||||
|  | ///
 | ||||||
|  | /// let parser = RdfParser::from_format(RdfFormat::NTriples);
 | ||||||
|  | /// let quads = parser.parse_read(file.as_bytes()).collect::<Result<Vec<_>,_>>()?;
 | ||||||
|  | ///
 | ||||||
|  | /// assert_eq!(quads.len(), 1);
 | ||||||
|  | /// assert_eq!(quads[0].subject.to_string(), "<http://example.com/s>");
 | ||||||
|  | /// # std::io::Result::Ok(())
 | ||||||
|  | /// ```
 | ||||||
|  | #[must_use] | ||||||
|  | pub struct FromReadQuadReader<R: Read> { | ||||||
|  |     parser: FromReadQuadReaderKind<R>, | ||||||
|  |     mapper: QuadMapper, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | enum FromReadQuadReaderKind<R: Read> { | ||||||
|  |     N3(FromReadN3Reader<R>), | ||||||
|  |     NQuads(FromReadNQuadsReader<R>), | ||||||
|  |     NTriples(FromReadNTriplesReader<R>), | ||||||
|  |     RdfXml(FromReadRdfXmlReader<R>), | ||||||
|  |     TriG(FromReadTriGReader<R>), | ||||||
|  |     Turtle(FromReadTurtleReader<R>), | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<R: Read> Iterator for FromReadQuadReader<R> { | ||||||
|  |     type Item = Result<Quad, ParseError>; | ||||||
|  | 
 | ||||||
|  |     fn next(&mut self) -> Option<Result<Quad, ParseError>> { | ||||||
|  |         Some(match &mut self.parser { | ||||||
|  |             FromReadQuadReaderKind::N3(parser) => match parser.next()? { | ||||||
|  |                 Ok(quad) => self.mapper.map_n3_quad(quad), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |             FromReadQuadReaderKind::NQuads(parser) => match parser.next()? { | ||||||
|  |                 Ok(quad) => self.mapper.map_quad(quad), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |             FromReadQuadReaderKind::NTriples(parser) => match parser.next()? { | ||||||
|  |                 Ok(triple) => Ok(self.mapper.map_triple_to_quad(triple)), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |             FromReadQuadReaderKind::RdfXml(parser) => match parser.next()? { | ||||||
|  |                 Ok(triple) => Ok(self.mapper.map_triple_to_quad(triple)), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |             FromReadQuadReaderKind::TriG(parser) => match parser.next()? { | ||||||
|  |                 Ok(quad) => self.mapper.map_quad(quad), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |             FromReadQuadReaderKind::Turtle(parser) => match parser.next()? { | ||||||
|  |                 Ok(triple) => Ok(self.mapper.map_triple_to_quad(triple)), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Parses a RDF file from a Tokio [`AsyncRead`] implementation. Can be built using [`RdfParser::parse_tokio_async_read`].
 | ||||||
|  | ///
 | ||||||
|  | /// Reads are buffered.
 | ||||||
|  | ///
 | ||||||
|  | /// ```
 | ||||||
|  | /// use oxrdfio::{RdfFormat, RdfParser, ParseError};
 | ||||||
|  | ///
 | ||||||
|  | /// #[tokio::main(flavor = "current_thread")]
 | ||||||
|  | /// async fn main() -> Result<(), ParseError> {
 | ||||||
|  | ///     let file = "<http://example.com/s> <http://example.com/p> <http://example.com/o> .";
 | ||||||
|  | ///
 | ||||||
|  | ///     let parser = RdfParser::from_format(RdfFormat::NTriples);
 | ||||||
|  | ///     let mut reader = parser.parse_tokio_async_read(file.as_bytes());
 | ||||||
|  | ///     if let Some(quad) = reader.next().await {
 | ||||||
|  | ///         assert_eq!(quad?.subject.to_string(), "<http://example.com/s>");
 | ||||||
|  | ///     }
 | ||||||
|  | ///     Ok(())
 | ||||||
|  | /// }
 | ||||||
|  | /// ```
 | ||||||
|  | #[must_use] | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | pub struct FromTokioAsyncReadQuadReader<R: AsyncRead + Unpin> { | ||||||
|  |     parser: FromTokioAsyncReadQuadReaderKind<R>, | ||||||
|  |     mapper: QuadMapper, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | enum FromTokioAsyncReadQuadReaderKind<R: AsyncRead + Unpin> { | ||||||
|  |     N3(FromTokioAsyncReadN3Reader<R>), | ||||||
|  |     NQuads(FromTokioAsyncReadNQuadsReader<R>), | ||||||
|  |     NTriples(FromTokioAsyncReadNTriplesReader<R>), | ||||||
|  |     RdfXml(FromTokioAsyncReadRdfXmlReader<R>), | ||||||
|  |     TriG(FromTokioAsyncReadTriGReader<R>), | ||||||
|  |     Turtle(FromTokioAsyncReadTurtleReader<R>), | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | impl<R: AsyncRead + Unpin> FromTokioAsyncReadQuadReader<R> { | ||||||
|  |     pub async fn next(&mut self) -> Option<Result<Quad, ParseError>> { | ||||||
|  |         Some(match &mut self.parser { | ||||||
|  |             FromTokioAsyncReadQuadReaderKind::N3(parser) => match parser.next().await? { | ||||||
|  |                 Ok(quad) => self.mapper.map_n3_quad(quad), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |             FromTokioAsyncReadQuadReaderKind::NQuads(parser) => match parser.next().await? { | ||||||
|  |                 Ok(quad) => self.mapper.map_quad(quad), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |             FromTokioAsyncReadQuadReaderKind::NTriples(parser) => match parser.next().await? { | ||||||
|  |                 Ok(triple) => Ok(self.mapper.map_triple_to_quad(triple)), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |             FromTokioAsyncReadQuadReaderKind::RdfXml(parser) => match parser.next().await? { | ||||||
|  |                 Ok(triple) => Ok(self.mapper.map_triple_to_quad(triple)), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |             FromTokioAsyncReadQuadReaderKind::TriG(parser) => match parser.next().await? { | ||||||
|  |                 Ok(quad) => self.mapper.map_quad(quad), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |             FromTokioAsyncReadQuadReaderKind::Turtle(parser) => match parser.next().await? { | ||||||
|  |                 Ok(triple) => Ok(self.mapper.map_triple_to_quad(triple)), | ||||||
|  |                 Err(e) => Err(e.into()), | ||||||
|  |             }, | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct QuadMapper { | ||||||
|  |     default_graph: GraphName, | ||||||
|  |     without_named_graphs: bool, | ||||||
|  |     blank_node_map: Option<HashMap<BlankNode, BlankNode>>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl QuadMapper { | ||||||
|  |     fn map_blank_node(&mut self, node: BlankNode) -> BlankNode { | ||||||
|  |         if let Some(blank_node_map) = &mut self.blank_node_map { | ||||||
|  |             blank_node_map | ||||||
|  |                 .entry(node) | ||||||
|  |                 .or_insert_with(BlankNode::default) | ||||||
|  |                 .clone() | ||||||
|  |         } else { | ||||||
|  |             node | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn map_subject(&mut self, node: Subject) -> Subject { | ||||||
|  |         match node { | ||||||
|  |             Subject::NamedNode(node) => node.into(), | ||||||
|  |             Subject::BlankNode(node) => self.map_blank_node(node).into(), | ||||||
|  |             #[cfg(feature = "rdf-star")] | ||||||
|  |             Subject::Triple(triple) => self.map_triple(*triple).into(), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn map_term(&mut self, node: Term) -> Term { | ||||||
|  |         match node { | ||||||
|  |             Term::NamedNode(node) => node.into(), | ||||||
|  |             Term::BlankNode(node) => self.map_blank_node(node).into(), | ||||||
|  |             Term::Literal(literal) => literal.into(), | ||||||
|  |             #[cfg(feature = "rdf-star")] | ||||||
|  |             Term::Triple(triple) => self.map_triple(*triple).into(), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn map_triple(&mut self, triple: Triple) -> Triple { | ||||||
|  |         Triple { | ||||||
|  |             subject: self.map_subject(triple.subject), | ||||||
|  |             predicate: triple.predicate, | ||||||
|  |             object: self.map_term(triple.object), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn map_graph_name(&mut self, graph_name: GraphName) -> Result<GraphName, ParseError> { | ||||||
|  |         match graph_name { | ||||||
|  |             GraphName::NamedNode(node) => { | ||||||
|  |                 if self.without_named_graphs { | ||||||
|  |                     Err(ParseError::msg("Named graphs are not allowed")) | ||||||
|  |                 } else { | ||||||
|  |                     Ok(node.into()) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             GraphName::BlankNode(node) => { | ||||||
|  |                 if self.without_named_graphs { | ||||||
|  |                     Err(ParseError::msg("Named graphs are not allowed")) | ||||||
|  |                 } else { | ||||||
|  |                     Ok(self.map_blank_node(node).into()) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             GraphName::DefaultGraph => Ok(self.default_graph.clone()), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn map_quad(&mut self, quad: Quad) -> Result<Quad, ParseError> { | ||||||
|  |         Ok(Quad { | ||||||
|  |             subject: self.map_subject(quad.subject), | ||||||
|  |             predicate: quad.predicate, | ||||||
|  |             object: self.map_term(quad.object), | ||||||
|  |             graph_name: self.map_graph_name(quad.graph_name)?, | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn map_triple_to_quad(&mut self, triple: Triple) -> Quad { | ||||||
|  |         self.map_triple(triple).in_graph(self.default_graph.clone()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn map_n3_quad(&mut self, quad: N3Quad) -> Result<Quad, ParseError> { | ||||||
|  |         Ok(Quad { | ||||||
|  |             subject: match quad.subject { | ||||||
|  |                 N3Term::NamedNode(s) => Ok(s.into()), | ||||||
|  |                 N3Term::BlankNode(s) => Ok(self.map_blank_node(s).into()), | ||||||
|  |                 N3Term::Literal(_) => Err(ParseError::msg( | ||||||
|  |                     "literals are not allowed in regular RDF subjects", | ||||||
|  |                 )), | ||||||
|  |                 #[cfg(feature = "rdf-star")] | ||||||
|  |                 N3Term::Triple(s) => Ok(self.map_triple(*s).into()), | ||||||
|  |                 N3Term::Variable(_) => Err(ParseError::msg( | ||||||
|  |                     "variables are not allowed in regular RDF subjects", | ||||||
|  |                 )), | ||||||
|  |             }?, | ||||||
|  |             predicate: match quad.predicate { | ||||||
|  |                 N3Term::NamedNode(p) => Ok(p), | ||||||
|  |                 N3Term::BlankNode(_) => Err(ParseError::msg( | ||||||
|  |                     "blank nodes are not allowed in regular RDF predicates", | ||||||
|  |                 )), | ||||||
|  |                 N3Term::Literal(_) => Err(ParseError::msg( | ||||||
|  |                     "literals are not allowed in regular RDF predicates", | ||||||
|  |                 )), | ||||||
|  |                 #[cfg(feature = "rdf-star")] | ||||||
|  |                 N3Term::Triple(_) => Err(ParseError::msg( | ||||||
|  |                     "quoted triples are not allowed in regular RDF predicates", | ||||||
|  |                 )), | ||||||
|  |                 N3Term::Variable(_) => Err(ParseError::msg( | ||||||
|  |                     "variables are not allowed in regular RDF predicates", | ||||||
|  |                 )), | ||||||
|  |             }?, | ||||||
|  |             object: match quad.object { | ||||||
|  |                 N3Term::NamedNode(o) => Ok(o.into()), | ||||||
|  |                 N3Term::BlankNode(o) => Ok(self.map_blank_node(o).into()), | ||||||
|  |                 N3Term::Literal(o) => Ok(o.into()), | ||||||
|  |                 #[cfg(feature = "rdf-star")] | ||||||
|  |                 N3Term::Triple(o) => Ok(self.map_triple(*o).into()), | ||||||
|  |                 N3Term::Variable(_) => Err(ParseError::msg( | ||||||
|  |                     "variables are not allowed in regular RDF objects", | ||||||
|  |                 )), | ||||||
|  |             }?, | ||||||
|  |             graph_name: self.map_graph_name(quad.graph_name)?, | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,322 @@ | |||||||
|  | //! Utilities to write RDF graphs and datasets.
 | ||||||
|  | 
 | ||||||
|  | use crate::format::RdfFormat; | ||||||
|  | use oxrdf::{GraphNameRef, QuadRef, TripleRef}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxrdfxml::ToTokioAsyncWriteRdfXmlWriter; | ||||||
|  | use oxrdfxml::{RdfXmlSerializer, ToWriteRdfXmlWriter}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxttl::nquads::ToTokioAsyncWriteNQuadsWriter; | ||||||
|  | use oxttl::nquads::{NQuadsSerializer, ToWriteNQuadsWriter}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxttl::ntriples::ToTokioAsyncWriteNTriplesWriter; | ||||||
|  | use oxttl::ntriples::{NTriplesSerializer, ToWriteNTriplesWriter}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxttl::trig::ToTokioAsyncWriteTriGWriter; | ||||||
|  | use oxttl::trig::{ToWriteTriGWriter, TriGSerializer}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use oxttl::turtle::ToTokioAsyncWriteTurtleWriter; | ||||||
|  | use oxttl::turtle::{ToWriteTurtleWriter, TurtleSerializer}; | ||||||
|  | use std::io::{self, Write}; | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | use tokio::io::{AsyncWrite, AsyncWriteExt}; | ||||||
|  | 
 | ||||||
|  | /// A serializer for RDF serialization formats.
 | ||||||
|  | ///
 | ||||||
|  | /// It currently supports the following formats:
 | ||||||
|  | /// * [N3](https://w3c.github.io/N3/spec/) ([`RdfFormat::N3`])
 | ||||||
|  | /// * [N-Quads](https://www.w3.org/TR/n-quads/) ([`RdfFormat::NQuads`])
 | ||||||
|  | /// * [N-Triples](https://www.w3.org/TR/n-triples/) ([`RdfFormat::NTriples`])
 | ||||||
|  | /// * [RDF/XML](https://www.w3.org/TR/rdf-syntax-grammar/) ([`RdfFormat::RdfXml`])
 | ||||||
|  | /// * [TriG](https://www.w3.org/TR/trig/) ([`RdfFormat::TriG`])
 | ||||||
|  | /// * [Turtle](https://www.w3.org/TR/turtle/) ([`RdfFormat::Turtle`])
 | ||||||
|  | ///
 | ||||||
|  | /// ```
 | ||||||
|  | /// use oxrdfio::{RdfFormat, RdfSerializer};
 | ||||||
|  | /// use oxrdf::{Quad, NamedNode};
 | ||||||
|  | ///
 | ||||||
|  | /// let mut buffer = Vec::new();
 | ||||||
|  | /// let mut writer = RdfSerializer::from_format(RdfFormat::NQuads).serialize_to_write(&mut buffer);
 | ||||||
|  | /// writer.write_quad(&Quad {
 | ||||||
|  | ///    subject: NamedNode::new("http://example.com/s")?.into(),
 | ||||||
|  | ///    predicate: NamedNode::new("http://example.com/p")?,
 | ||||||
|  | ///    object: NamedNode::new("http://example.com/o")?.into(),
 | ||||||
|  | ///    graph_name: NamedNode::new("http://example.com/g")?.into()
 | ||||||
|  | /// })?;
 | ||||||
|  | /// writer.finish()?;
 | ||||||
|  | ///
 | ||||||
|  | /// assert_eq!(buffer.as_slice(), "<http://example.com/s> <http://example.com/p> <http://example.com/o> <http://example.com/g> .\n".as_bytes());
 | ||||||
|  | /// # Result::<_,Box<dyn std::error::Error>>::Ok(())
 | ||||||
|  | /// ```
 | ||||||
|  | pub struct RdfSerializer { | ||||||
|  |     format: RdfFormat, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl RdfSerializer { | ||||||
|  |     /// Builds a serializer for the given format
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn from_format(format: RdfFormat) -> Self { | ||||||
|  |         Self { format } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Writes to a [`Write`] implementation.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Warning: Do not forget to run the [`finish`](ToWriteQuadWriter::finish()) method to properly write the last bytes of the file.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Warning: This writer does unbuffered writes. You might want to use [`BufWriter`](io::BufWriter) to avoid that.
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::{RdfFormat, RdfSerializer};
 | ||||||
|  |     /// use oxrdf::{Quad, NamedNode};
 | ||||||
|  |     ///
 | ||||||
|  |     /// let mut buffer = Vec::new();
 | ||||||
|  |     /// let mut writer = RdfSerializer::from_format(RdfFormat::NQuads).serialize_to_write(&mut buffer);
 | ||||||
|  |     /// writer.write_quad(&Quad {
 | ||||||
|  |     ///    subject: NamedNode::new("http://example.com/s")?.into(),
 | ||||||
|  |     ///    predicate: NamedNode::new("http://example.com/p")?,
 | ||||||
|  |     ///    object: NamedNode::new("http://example.com/o")?.into(),
 | ||||||
|  |     ///    graph_name: NamedNode::new("http://example.com/g")?.into()
 | ||||||
|  |     /// })?;
 | ||||||
|  |     /// writer.finish()?;
 | ||||||
|  |     ///
 | ||||||
|  |     /// assert_eq!(buffer.as_slice(), "<http://example.com/s> <http://example.com/p> <http://example.com/o> <http://example.com/g> .\n".as_bytes());
 | ||||||
|  |     /// # Result::<_,Box<dyn std::error::Error>>::Ok(())
 | ||||||
|  |     /// ```
 | ||||||
|  |     pub fn serialize_to_write<W: Write>(&self, writer: W) -> ToWriteQuadWriter<W> { | ||||||
|  |         ToWriteQuadWriter { | ||||||
|  |             formatter: match self.format { | ||||||
|  |                 RdfFormat::NQuads => ToWriteQuadWriterKind::NQuads( | ||||||
|  |                     NQuadsSerializer::new().serialize_to_write(writer), | ||||||
|  |                 ), | ||||||
|  |                 RdfFormat::NTriples => ToWriteQuadWriterKind::NTriples( | ||||||
|  |                     NTriplesSerializer::new().serialize_to_write(writer), | ||||||
|  |                 ), | ||||||
|  |                 RdfFormat::RdfXml => ToWriteQuadWriterKind::RdfXml( | ||||||
|  |                     RdfXmlSerializer::new().serialize_to_write(writer), | ||||||
|  |                 ), | ||||||
|  |                 RdfFormat::TriG => { | ||||||
|  |                     ToWriteQuadWriterKind::TriG(TriGSerializer::new().serialize_to_write(writer)) | ||||||
|  |                 } | ||||||
|  |                 RdfFormat::Turtle | RdfFormat::N3 => ToWriteQuadWriterKind::Turtle( | ||||||
|  |                     TurtleSerializer::new().serialize_to_write(writer), | ||||||
|  |                 ), | ||||||
|  |             }, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Writes to a Tokio [`AsyncWrite`] implementation.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Warning: Do not forget to run the [`finish`](ToTokioAsyncWriteQuadWriter::finish()) method to properly write the last bytes of the file.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Warning: This writer does unbuffered writes. You might want to use [`BufWriter`](tokio::io::BufWriter) to avoid that.
 | ||||||
|  |     ///
 | ||||||
|  |     /// ```
 | ||||||
|  |     /// use oxrdfio::{RdfFormat, RdfSerializer};
 | ||||||
|  |     /// use oxrdf::{Quad, NamedNode};
 | ||||||
|  |     /// use std::io;
 | ||||||
|  |     ///
 | ||||||
|  |     /// #[tokio::main(flavor = "current_thread")]
 | ||||||
|  |     /// async fn main() -> io::Result<()> {
 | ||||||
|  |     ///     let mut buffer = Vec::new();
 | ||||||
|  |     ///     let mut writer = RdfSerializer::from_format(RdfFormat::NQuads).serialize_to_tokio_async_write(&mut buffer);
 | ||||||
|  |     ///     writer.write_quad(&Quad {
 | ||||||
|  |     ///         subject: NamedNode::new_unchecked("http://example.com/s").into(),
 | ||||||
|  |     ///         predicate: NamedNode::new_unchecked("http://example.com/p"),
 | ||||||
|  |     ///         object: NamedNode::new_unchecked("http://example.com/o").into(),
 | ||||||
|  |     ///         graph_name: NamedNode::new_unchecked("http://example.com/g").into()
 | ||||||
|  |     ///     }).await?;
 | ||||||
|  |     ///     writer.finish().await?;
 | ||||||
|  |     ///
 | ||||||
|  |     ///     assert_eq!(buffer.as_slice(), "<http://example.com/s> <http://example.com/p> <http://example.com/o> <http://example.com/g> .\n".as_bytes());
 | ||||||
|  |     ///     Ok(())
 | ||||||
|  |     /// }
 | ||||||
|  |     /// ```
 | ||||||
|  |     #[cfg(feature = "async-tokio")] | ||||||
|  |     pub fn serialize_to_tokio_async_write<W: AsyncWrite + Unpin>( | ||||||
|  |         &self, | ||||||
|  |         writer: W, | ||||||
|  |     ) -> ToTokioAsyncWriteQuadWriter<W> { | ||||||
|  |         ToTokioAsyncWriteQuadWriter { | ||||||
|  |             formatter: match self.format { | ||||||
|  |                 RdfFormat::NQuads => ToTokioAsyncWriteQuadWriterKind::NQuads( | ||||||
|  |                     NQuadsSerializer::new().serialize_to_tokio_async_write(writer), | ||||||
|  |                 ), | ||||||
|  |                 RdfFormat::NTriples => ToTokioAsyncWriteQuadWriterKind::NTriples( | ||||||
|  |                     NTriplesSerializer::new().serialize_to_tokio_async_write(writer), | ||||||
|  |                 ), | ||||||
|  |                 RdfFormat::RdfXml => ToTokioAsyncWriteQuadWriterKind::RdfXml( | ||||||
|  |                     RdfXmlSerializer::new().serialize_to_tokio_async_write(writer), | ||||||
|  |                 ), | ||||||
|  |                 RdfFormat::TriG => ToTokioAsyncWriteQuadWriterKind::TriG( | ||||||
|  |                     TriGSerializer::new().serialize_to_tokio_async_write(writer), | ||||||
|  |                 ), | ||||||
|  |                 RdfFormat::Turtle | RdfFormat::N3 => ToTokioAsyncWriteQuadWriterKind::Turtle( | ||||||
|  |                     TurtleSerializer::new().serialize_to_tokio_async_write(writer), | ||||||
|  |                 ), | ||||||
|  |             }, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Writes quads or triples to a [`Write`] implementation.
 | ||||||
|  | ///
 | ||||||
|  | /// Can be built using [`RdfSerializer::serialize_to_write`].
 | ||||||
|  | ///
 | ||||||
|  | /// Warning: Do not forget to run the [`finish`](ToWriteQuadWriter::finish()) method to properly write the last bytes of the file.
 | ||||||
|  | ///
 | ||||||
|  | /// Warning: This writer does unbuffered writes. You might want to use [`BufWriter`](io::BufWriter) to avoid that.
 | ||||||
|  | ///
 | ||||||
|  | /// ```
 | ||||||
|  | /// use oxrdfio::{RdfFormat, RdfSerializer};
 | ||||||
|  | /// use oxrdf::{Quad, NamedNode};
 | ||||||
|  | ///
 | ||||||
|  | /// let mut buffer = Vec::new();
 | ||||||
|  | /// let mut writer = RdfSerializer::from_format(RdfFormat::NQuads).serialize_to_write(&mut buffer);
 | ||||||
|  | /// writer.write_quad(&Quad {
 | ||||||
|  | ///    subject: NamedNode::new("http://example.com/s")?.into(),
 | ||||||
|  | ///    predicate: NamedNode::new("http://example.com/p")?,
 | ||||||
|  | ///    object: NamedNode::new("http://example.com/o")?.into(),
 | ||||||
|  | ///    graph_name: NamedNode::new("http://example.com/g")?.into(),
 | ||||||
|  | /// })?;
 | ||||||
|  | /// writer.finish()?;
 | ||||||
|  | ///
 | ||||||
|  | /// assert_eq!(buffer.as_slice(), "<http://example.com/s> <http://example.com/p> <http://example.com/o> <http://example.com/g> .\n".as_bytes());
 | ||||||
|  | /// # Result::<_,Box<dyn std::error::Error>>::Ok(())
 | ||||||
|  | /// ```
 | ||||||
|  | #[must_use] | ||||||
|  | pub struct ToWriteQuadWriter<W: Write> { | ||||||
|  |     formatter: ToWriteQuadWriterKind<W>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | enum ToWriteQuadWriterKind<W: Write> { | ||||||
|  |     NQuads(ToWriteNQuadsWriter<W>), | ||||||
|  |     NTriples(ToWriteNTriplesWriter<W>), | ||||||
|  |     RdfXml(ToWriteRdfXmlWriter<W>), | ||||||
|  |     TriG(ToWriteTriGWriter<W>), | ||||||
|  |     Turtle(ToWriteTurtleWriter<W>), | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<W: Write> ToWriteQuadWriter<W> { | ||||||
|  |     /// Writes a [`QuadRef`]
 | ||||||
|  |     pub fn write_quad<'a>(&mut self, quad: impl Into<QuadRef<'a>>) -> io::Result<()> { | ||||||
|  |         match &mut self.formatter { | ||||||
|  |             ToWriteQuadWriterKind::NQuads(writer) => writer.write_quad(quad), | ||||||
|  |             ToWriteQuadWriterKind::NTriples(writer) => writer.write_triple(to_triple(quad)?), | ||||||
|  |             ToWriteQuadWriterKind::RdfXml(writer) => writer.write_triple(to_triple(quad)?), | ||||||
|  |             ToWriteQuadWriterKind::TriG(writer) => writer.write_quad(quad), | ||||||
|  |             ToWriteQuadWriterKind::Turtle(writer) => writer.write_triple(to_triple(quad)?), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Writes a [`TripleRef`]
 | ||||||
|  |     pub fn write_triple<'a>(&mut self, triple: impl Into<TripleRef<'a>>) -> io::Result<()> { | ||||||
|  |         self.write_quad(triple.into().in_graph(GraphNameRef::DefaultGraph)) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Writes the last bytes of the file
 | ||||||
|  |     pub fn finish(self) -> io::Result<()> { | ||||||
|  |         match self.formatter { | ||||||
|  |             ToWriteQuadWriterKind::NQuads(writer) => writer.finish(), | ||||||
|  |             ToWriteQuadWriterKind::NTriples(writer) => writer.finish(), | ||||||
|  |             ToWriteQuadWriterKind::RdfXml(writer) => writer.finish()?, | ||||||
|  |             ToWriteQuadWriterKind::TriG(writer) => writer.finish()?, | ||||||
|  |             ToWriteQuadWriterKind::Turtle(writer) => writer.finish()?, | ||||||
|  |         } | ||||||
|  |         .flush() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Writes quads or triples to a [`Write`] implementation.
 | ||||||
|  | ///
 | ||||||
|  | /// Can be built using [`RdfSerializer::serialize_to_write`].
 | ||||||
|  | ///
 | ||||||
|  | /// Warning: Do not forget to run the [`finish`](ToWriteQuadWriter::finish()) method to properly write the last bytes of the file.
 | ||||||
|  | ///
 | ||||||
|  | /// Warning: This writer does unbuffered writes. You might want to use [`BufWriter`](io::BufWriter) to avoid that.
 | ||||||
|  | ///
 | ||||||
|  | /// ```
 | ||||||
|  | /// use oxrdfio::{RdfFormat, RdfSerializer};
 | ||||||
|  | /// use oxrdf::{Quad, NamedNode};
 | ||||||
|  | /// use std::io;
 | ||||||
|  | ///
 | ||||||
|  | /// #[tokio::main(flavor = "current_thread")]
 | ||||||
|  | /// async fn main() -> io::Result<()> {
 | ||||||
|  | ///     let mut buffer = Vec::new();
 | ||||||
|  | ///     let mut writer = RdfSerializer::from_format(RdfFormat::NQuads).serialize_to_tokio_async_write(&mut buffer);
 | ||||||
|  | ///     writer.write_quad(&Quad {
 | ||||||
|  | ///         subject: NamedNode::new_unchecked("http://example.com/s").into(),
 | ||||||
|  | ///         predicate: NamedNode::new_unchecked("http://example.com/p"),
 | ||||||
|  | ///         object: NamedNode::new_unchecked("http://example.com/o").into(),
 | ||||||
|  | ///         graph_name: NamedNode::new_unchecked("http://example.com/g").into()
 | ||||||
|  | ///     }).await?;
 | ||||||
|  | ///     writer.finish().await?;
 | ||||||
|  | ///
 | ||||||
|  | ///     assert_eq!(buffer.as_slice(), "<http://example.com/s> <http://example.com/p> <http://example.com/o> <http://example.com/g> .\n".as_bytes());
 | ||||||
|  | ///     Ok(())
 | ||||||
|  | /// }
 | ||||||
|  | /// ```
 | ||||||
|  | #[must_use] | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | pub struct ToTokioAsyncWriteQuadWriter<W: AsyncWrite + Unpin> { | ||||||
|  |     formatter: ToTokioAsyncWriteQuadWriterKind<W>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | enum ToTokioAsyncWriteQuadWriterKind<W: AsyncWrite + Unpin> { | ||||||
|  |     NQuads(ToTokioAsyncWriteNQuadsWriter<W>), | ||||||
|  |     NTriples(ToTokioAsyncWriteNTriplesWriter<W>), | ||||||
|  |     RdfXml(ToTokioAsyncWriteRdfXmlWriter<W>), | ||||||
|  |     TriG(ToTokioAsyncWriteTriGWriter<W>), | ||||||
|  |     Turtle(ToTokioAsyncWriteTurtleWriter<W>), | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(feature = "async-tokio")] | ||||||
|  | impl<W: AsyncWrite + Unpin> ToTokioAsyncWriteQuadWriter<W> { | ||||||
|  |     /// Writes a [`QuadRef`]
 | ||||||
|  |     pub async fn write_quad<'a>(&mut self, quad: impl Into<QuadRef<'a>>) -> io::Result<()> { | ||||||
|  |         match &mut self.formatter { | ||||||
|  |             ToTokioAsyncWriteQuadWriterKind::NQuads(writer) => writer.write_quad(quad).await, | ||||||
|  |             ToTokioAsyncWriteQuadWriterKind::NTriples(writer) => { | ||||||
|  |                 writer.write_triple(to_triple(quad)?).await | ||||||
|  |             } | ||||||
|  |             ToTokioAsyncWriteQuadWriterKind::RdfXml(writer) => { | ||||||
|  |                 writer.write_triple(to_triple(quad)?).await | ||||||
|  |             } | ||||||
|  |             ToTokioAsyncWriteQuadWriterKind::TriG(writer) => writer.write_quad(quad).await, | ||||||
|  |             ToTokioAsyncWriteQuadWriterKind::Turtle(writer) => { | ||||||
|  |                 writer.write_triple(to_triple(quad)?).await | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Writes a [`TripleRef`]
 | ||||||
|  |     pub async fn write_triple<'a>(&mut self, triple: impl Into<TripleRef<'a>>) -> io::Result<()> { | ||||||
|  |         self.write_quad(triple.into().in_graph(GraphNameRef::DefaultGraph)) | ||||||
|  |             .await | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Writes the last bytes of the file
 | ||||||
|  |     pub async fn finish(self) -> io::Result<()> { | ||||||
|  |         match self.formatter { | ||||||
|  |             ToTokioAsyncWriteQuadWriterKind::NQuads(writer) => writer.finish(), | ||||||
|  |             ToTokioAsyncWriteQuadWriterKind::NTriples(writer) => writer.finish(), | ||||||
|  |             ToTokioAsyncWriteQuadWriterKind::RdfXml(writer) => writer.finish().await?, | ||||||
|  |             ToTokioAsyncWriteQuadWriterKind::TriG(writer) => writer.finish().await?, | ||||||
|  |             ToTokioAsyncWriteQuadWriterKind::Turtle(writer) => writer.finish().await?, | ||||||
|  |         } | ||||||
|  |         .flush() | ||||||
|  |         .await | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn to_triple<'a>(quad: impl Into<QuadRef<'a>>) -> io::Result<TripleRef<'a>> { | ||||||
|  |     let quad = quad.into(); | ||||||
|  |     if quad.graph_name.is_default_graph() { | ||||||
|  |         Ok(quad.into()) | ||||||
|  |     } else { | ||||||
|  |         Err(io::Error::new( | ||||||
|  |             io::ErrorKind::InvalidInput, | ||||||
|  |             "Only quads in the default graph can be serialized to a RDF graph format", | ||||||
|  |         )) | ||||||
|  |     } | ||||||
|  | } | ||||||
					Loading…
					
					
				
		Reference in new issue
	
	 Tpt
						Tpt