Fork of https://github.com/oxigraph/oxigraph.git for the purpose of NextGraph project
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
204 lines
6.6 KiB
204 lines
6.6 KiB
1 year ago
|
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())
|
||
|
}
|
||
|
}
|