use std::fmt::{self, Debug, Display, Formatter};
type BoxError = Box<dyn std::error::Error + Send + Sync>;
#[non_exhaustive]
pub enum Error {
UnknownField { field_name: Option<String> },
IncompleteFieldData { field_name: Option<String> },
IncompleteHeaders,
ReadHeaderFailed(httparse::Error),
DecodeHeaderName { name: String, cause: BoxError },
DecodeHeaderValue { value: Vec<u8>, cause: BoxError },
IncompleteStream,
FieldSizeExceeded { limit: u64, field_name: Option<String> },
StreamSizeExceeded { limit: u64 },
StreamReadFailed(BoxError),
LockFailure,
NoMultipart,
DecodeContentType(mime::FromStrError),
NoBoundary,
#[cfg(feature = "json")]
#[cfg_attr(nightly, doc(cfg(feature = "json")))]
DecodeJson(serde_json::Error),
}
impl Debug for Error {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
Display::fmt(self, f)
}
}
impl Display for Error {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
match self {
Error::UnknownField { field_name } => {
let name = field_name.as_deref().unwrap_or("<unknown>");
write!(f, "unknown field received: {:?}", name)
}
Error::IncompleteFieldData { field_name } => {
let name = field_name.as_deref().unwrap_or("<unknown>");
write!(f, "field {:?} received with incomplete data", name)
}
Error::DecodeHeaderName { name, .. } => {
write!(f, "failed to decode field's raw header name: {:?}", name)
}
Error::DecodeHeaderValue { .. } => {
write!(f, "failed to decode field's raw header value")
}
Error::FieldSizeExceeded { limit, field_name } => {
let name = field_name.as_deref().unwrap_or("<unknown>");
write!(f, "field {:?} exceeded the size limit: {} bytes", name, limit)
}
Error::StreamSizeExceeded { limit } => {
write!(f, "stream size exceeded limit: {} bytes", limit)
}
Error::ReadHeaderFailed(_) => write!(f, "failed to read headers"),
Error::StreamReadFailed(_) => write!(f, "failed to read stream"),
Error::DecodeContentType(_) => write!(f, "failed to decode Content-Type"),
Error::IncompleteHeaders => write!(f, "failed to read field complete headers"),
Error::IncompleteStream => write!(f, "incomplete multipart stream"),
Error::LockFailure => write!(f, "failed to lock multipart state"),
Error::NoMultipart => write!(f, "Content-Type is not multipart/form-data"),
Error::NoBoundary => write!(f, "multipart boundary not found in Content-Type"),
#[cfg(feature = "json")]
Error::DecodeJson(_) => write!(f, "failed to decode field data as JSON"),
}
}
}
impl std::error::Error for Error {
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
match self {
Error::ReadHeaderFailed(e) => Some(e),
Error::DecodeHeaderName { cause, .. } => Some(cause.as_ref()),
Error::DecodeHeaderValue { cause, .. } => Some(cause.as_ref()),
Error::StreamReadFailed(e) => Some(e.as_ref()),
Error::DecodeContentType(e) => Some(e),
#[cfg(feature = "json")]
Error::DecodeJson(e) => Some(e),
Error::UnknownField { .. }
| Error::IncompleteFieldData { .. }
| Error::IncompleteHeaders
| Error::IncompleteStream
| Error::FieldSizeExceeded { .. }
| Error::StreamSizeExceeded { .. }
| Error::LockFailure
| Error::NoMultipart
| Error::NoBoundary => None,
}
}
}
impl PartialEq for Error {
fn eq(&self, other: &Self) -> bool {
self.to_string().eq(&other.to_string())
}
}
impl Eq for Error {}