use oxilangtag::LanguageTagParseError; use oxiri::IriParseError; use std::error::Error; use std::sync::Arc; use std::{fmt, io}; /// Error returned during RDF/XML 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 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 for ParseError { #[inline] fn from(error: io::Error) -> Self { Self::Io(error) } } impl From for ParseError { #[inline] fn from(error: SyntaxError) -> Self { Self::Syntax(error) } } impl From for io::Error { #[inline] fn from(error: ParseError) -> Self { match error { ParseError::Io(error) => error, ParseError::Syntax(error) => error.into(), } } } impl From for ParseError { #[inline] fn from(error: quick_xml::Error) -> Self { match error { quick_xml::Error::Io(error) => Self::Io(match Arc::try_unwrap(error) { Ok(error) => error, Err(error) => io::Error::new(error.kind(), error), }), error => Self::Syntax(SyntaxError { inner: SyntaxErrorKind::Xml(error), }), } } } impl From for ParseError { #[inline] fn from(error: quick_xml::events::attributes::AttrError) -> Self { Self::Syntax(SyntaxError { inner: SyntaxErrorKind::XmlAttribute(error), }) } } /// An error in the syntax of the parsed file. #[derive(Debug)] pub struct SyntaxError { pub(crate) inner: SyntaxErrorKind, } #[derive(Debug)] pub enum SyntaxErrorKind { Xml(quick_xml::Error), XmlAttribute(quick_xml::events::attributes::AttrError), InvalidIri { iri: String, error: IriParseError, }, InvalidLanguageTag { tag: String, error: LanguageTagParseError, }, Msg { msg: String, }, } impl SyntaxError { /// Builds an error from a printable error message. #[inline] pub(crate) fn msg(msg: impl Into) -> Self { Self { inner: SyntaxErrorKind::Msg { msg: msg.into() }, } } } impl fmt::Display for SyntaxError { #[inline] fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { match &self.inner { SyntaxErrorKind::Xml(error) => error.fmt(f), SyntaxErrorKind::XmlAttribute(error) => error.fmt(f), SyntaxErrorKind::InvalidIri { iri, error } => { write!(f, "error while parsing IRI '{}': {}", iri, error) } SyntaxErrorKind::InvalidLanguageTag { tag, error } => { write!(f, "error while parsing language tag '{}': {}", tag, error) } SyntaxErrorKind::Msg { msg } => f.write_str(msg), } } } impl Error for SyntaxError { #[inline] fn source(&self) -> Option<&(dyn Error + 'static)> { match &self.inner { SyntaxErrorKind::Xml(error) => Some(error), SyntaxErrorKind::XmlAttribute(error) => Some(error), SyntaxErrorKind::InvalidIri { error, .. } => Some(error), SyntaxErrorKind::InvalidLanguageTag { error, .. } => Some(error), SyntaxErrorKind::Msg { .. } => None, } } } impl From for io::Error { #[inline] fn from(error: SyntaxError) -> Self { match error.inner { SyntaxErrorKind::Xml(error) => match error { quick_xml::Error::Io(error) => match Arc::try_unwrap(error) { Ok(error) => error, Err(error) => Self::new(error.kind(), error), }, quick_xml::Error::UnexpectedEof(error) => { Self::new(io::ErrorKind::UnexpectedEof, error) } error => Self::new(io::ErrorKind::InvalidData, error), }, SyntaxErrorKind::Msg { msg } => Self::new(io::ErrorKind::InvalidData, msg), _ => Self::new(io::ErrorKind::InvalidData, error), } } }