xml: Some work on errors

This commit is contained in:
Lennart
2025-02-06 14:06:17 +01:00
parent 9b6ad4eb39
commit 6caa04a516
9 changed files with 37 additions and 33 deletions

View File

@@ -1,4 +1,5 @@
use actix_web::{http::StatusCode, HttpResponse}; use actix_web::{http::StatusCode, HttpResponse};
use rustical_xml::XmlError;
use thiserror::Error; use thiserror::Error;
use tracing::error; use tracing::error;
@@ -33,7 +34,15 @@ impl actix_web::error::ResponseError for Error {
Self::NotFound => StatusCode::NOT_FOUND, Self::NotFound => StatusCode::NOT_FOUND,
Self::BadRequest(_) => StatusCode::BAD_REQUEST, Self::BadRequest(_) => StatusCode::BAD_REQUEST,
Self::Unauthorized => StatusCode::UNAUTHORIZED, Self::Unauthorized => StatusCode::UNAUTHORIZED,
Self::XmlError(_) => StatusCode::BAD_REQUEST, Self::XmlError(error) => match &error {
XmlError::InvalidTag(..)
| XmlError::MissingField(_)
| XmlError::UnsupportedEvent(_)
| XmlError::InvalidVariant(_)
| XmlError::InvalidFieldName(_, _)
| XmlError::InvalidValue(_) => StatusCode::UNPROCESSABLE_ENTITY,
_ => StatusCode::BAD_REQUEST,
},
Error::PropReadOnly => StatusCode::CONFLICT, Error::PropReadOnly => StatusCode::CONFLICT,
Self::IOError(_) => StatusCode::INTERNAL_SERVER_ERROR, Self::IOError(_) => StatusCode::INTERNAL_SERVER_ERROR,
} }

View File

@@ -14,8 +14,8 @@ impl ValueDeserialize for SyncLevel {
"1" => Self::One, "1" => Self::One,
"Infinity" => Self::Infinity, "Infinity" => Self::Infinity,
_ => { _ => {
return Err(rustical_xml::XmlError::Other( return Err(rustical_xml::XmlError::InvalidValue(
"Invalid sync-level".to_owned(), rustical_xml::ParseValueError::Other("Invalid sync-level".to_owned()),
)) ))
} }
}) })

View File

@@ -30,10 +30,12 @@ impl rustical_xml::ValueDeserialize for CalendarObjectType {
"VEVENT" => Ok(Self::Event), "VEVENT" => Ok(Self::Event),
"VTODO" => Ok(Self::Todo), "VTODO" => Ok(Self::Todo),
"VJOURNAL" => Ok(Self::Journal), "VJOURNAL" => Ok(Self::Journal),
_ => Err(rustical_xml::XmlError::Other(format!( _ => Err(rustical_xml::XmlError::InvalidValue(
"Invalid value '{}', must be VEVENT, VTODO, or VJOURNAL", rustical_xml::ParseValueError::Other(format!(
val "Invalid value '{}', must be VEVENT, VTODO, or VJOURNAL",
))), val
)),
)),
} }
} }
} }

View File

@@ -30,7 +30,9 @@ impl ValueDeserialize for UtcDateTime {
Ok(Self( Ok(Self(
NaiveDateTime::parse_from_str(&input, UTC_DATE_TIME) NaiveDateTime::parse_from_str(&input, UTC_DATE_TIME)
.map_err(|_| { .map_err(|_| {
rustical_xml::XmlError::Other("Could not parse as UTC timestamp".to_owned()) rustical_xml::XmlError::InvalidValue(rustical_xml::ParseValueError::Other(
"Could not parse as UTC timestamp".to_owned(),
))
})? })?
.and_utc(), .and_utc(),
)) ))

View File

@@ -165,7 +165,6 @@ impl Enum {
Event::Start(start) | Event::Empty(start) => { Event::Start(start) | Event::Empty(start) => {
return <Self as ::rustical_xml::XmlDeserialize>::deserialize(&mut reader, &start, empty); return <Self as ::rustical_xml::XmlDeserialize>::deserialize(&mut reader, &start, empty);
} }
Event::Eof => return Err(::rustical_xml::XmlError::Eof), Event::Eof => return Err(::rustical_xml::XmlError::Eof),
Event::Text(bytes_text) => { Event::Text(bytes_text) => {
return Err(::rustical_xml::XmlError::UnsupportedEvent("Text")); return Err(::rustical_xml::XmlError::UnsupportedEvent("Text"));
@@ -173,19 +172,14 @@ impl Enum {
Event::CData(cdata) => { Event::CData(cdata) => {
return Err(::rustical_xml::XmlError::UnsupportedEvent("CDATA")); return Err(::rustical_xml::XmlError::UnsupportedEvent("CDATA"));
} }
Event::Decl(_) => { /* <?xml ... ?> ignore this */ }
Event::Comment(_) => { /* ignore */ } Event::Comment(_) => { /* ignore */ }
Event::Decl(_) => { Event::DocType(_) => { /* ignore */ }
/* ignore */
// return Err(::rustical_xml::XmlError::UnsupportedEvent("Declaration"));
}
Event::PI(_) => { Event::PI(_) => {
return Err(::rustical_xml::XmlError::UnsupportedEvent("Processing instruction")); return Err(::rustical_xml::XmlError::UnsupportedEvent("Processing instruction"));
} }
Event::DocType(doctype) => {
return Err(::rustical_xml::XmlError::UnsupportedEvent("Doctype in the middle of the document"));
}
Event::End(end) => { Event::End(end) => {
return Err(::rustical_xml::XmlError::UnsupportedEvent("Premature end")); unreachable!("Premature end of xml document, should be handled by quick_xml");
} }
}; };
} }

View File

@@ -179,22 +179,15 @@ impl NamedStruct {
let text = String::from_utf8(cdata.to_vec())?; let text = String::from_utf8(cdata.to_vec())?;
#(#text_field_branches)* #(#text_field_branches)*
} }
Event::Decl(_) => { /* <?xml ... ?> ignore this */ }
Event::Comment(_) => { /* ignore */ } Event::Comment(_) => { /* ignore */ }
Event::Decl(_) => { Event::DocType(_) => { /* ignore */ }
// Error: not supported
return Err(XmlError::UnsupportedEvent("Declaration"));
}
Event::PI(_) => { Event::PI(_) => {
// Error: not supported // Error: not supported
return Err(XmlError::UnsupportedEvent("Processing instruction")); return Err(XmlError::UnsupportedEvent("Processing instruction"));
} }
Event::DocType(doctype) => {
// Error: start of new document
return Err(XmlError::UnsupportedEvent("Doctype in the middle of the document"));
}
Event::End(end) => { Event::End(end) => {
// This should actually be unreachable unreachable!("Unexpected closing tag for wrong element, should be handled by quick_xml");
return Err(XmlError::Other("Unexpected closing tag for wrong element".to_owned()));
} }
} }
} }

View File

@@ -67,6 +67,7 @@ impl<T: XmlRootTag + XmlDeserialize> XmlDocument for T {
return Self::deserialize(&mut reader, &start, empty); return Self::deserialize(&mut reader, &start, empty);
} }
Event::Eof => return Err(XmlError::Eof),
_ => return Err(XmlError::UnsupportedEvent("unknown, todo")), _ => return Err(XmlError::UnsupportedEvent("unknown, todo")),
}; };
} }

View File

@@ -4,22 +4,23 @@ use thiserror::Error;
#[derive(Debug, Error)] #[derive(Debug, Error)]
pub enum XmlError { pub enum XmlError {
// Syntactix errors
#[error(transparent)] #[error(transparent)]
QuickXmlError(#[from] quick_xml::Error), QuickXmlError(#[from] quick_xml::Error),
#[error(transparent)] #[error(transparent)]
QuickXmlAttrError(#[from] quick_xml::events::attributes::AttrError), QuickXmlAttrError(#[from] quick_xml::events::attributes::AttrError),
#[error(transparent)] #[error(transparent)]
FromUtf8Error(#[from] FromUtf8Error), FromUtf8Error(#[from] FromUtf8Error),
#[error("Invalid tag [{0}]{1}. Expected [{2}]{3}")]
InvalidTag(String, String, String, String),
#[error("Missing field {0}")]
MissingField(&'static str),
#[error("End of file, expected closing tags")] #[error("End of file, expected closing tags")]
Eof, Eof,
#[error("Unsupported xml event: {0}")] #[error("Unsupported xml event: {0}")]
UnsupportedEvent(&'static str), UnsupportedEvent(&'static str),
#[error("{0}")]
Other(String), // Semantic errors
#[error("Invalid tag [{0}]{1}. Expected [{2}]{3}")]
InvalidTag(String, String, String, String),
#[error("Missing field {0}")]
MissingField(&'static str),
#[error("Invalid variant: {0}")] #[error("Invalid variant: {0}")]
InvalidVariant(String), InvalidVariant(String),
#[error("Invalid field name in {0}: {1}")] #[error("Invalid field name in {0}: {1}")]

View File

@@ -22,6 +22,8 @@ pub enum ParseValueError {
ParseIntError(#[from] ParseIntError), ParseIntError(#[from] ParseIntError),
#[error(transparent)] #[error(transparent)]
ParseFloatError(#[from] ParseFloatError), ParseFloatError(#[from] ParseFloatError),
#[error("{0}")]
Other(String),
} }
macro_rules! impl_value_parse { macro_rules! impl_value_parse {