axum-extra/form: Use rejection macros for FormRejection (#3111)

This commit is contained in:
Tobias Bieniek 2024-12-27 10:02:07 +01:00 committed by GitHub
parent 8e78a37660
commit ed4d560054
No known key found for this signature in database
GPG key ID: B5690EEEBB952194

View file

@ -1,11 +1,11 @@
use axum::extract::rejection::RawFormRejection;
use axum::{ use axum::{
extract::{rejection::RawFormRejection, FromRequest, RawForm, Request}, extract::{FromRequest, RawForm, Request},
response::{IntoResponse, Response}, RequestExt,
Error, RequestExt,
}; };
use http::StatusCode; use axum_core::__composite_rejection as composite_rejection;
use axum_core::__define_rejection as define_rejection;
use serde::de::DeserializeOwned; use serde::de::DeserializeOwned;
use std::fmt;
/// Extractor that deserializes `application/x-www-form-urlencoded` requests /// Extractor that deserializes `application/x-www-form-urlencoded` requests
/// into some type. /// into some type.
@ -51,76 +51,32 @@ where
type Rejection = FormRejection; type Rejection = FormRejection;
async fn from_request(req: Request, _state: &S) -> Result<Self, Self::Rejection> { async fn from_request(req: Request, _state: &S) -> Result<Self, Self::Rejection> {
let RawForm(bytes) = req let RawForm(bytes) = req.extract().await?;
.extract()
.await
.map_err(FormRejection::RawFormRejection)?;
let deserializer = serde_html_form::Deserializer::new(form_urlencoded::parse(&bytes)); let deserializer = serde_html_form::Deserializer::new(form_urlencoded::parse(&bytes));
serde_path_to_error::deserialize::<_, T>(deserializer) let value = serde_path_to_error::deserialize::<_, T>(deserializer)
.map(Self) .map_err(FailedToDeserializeForm::from_err)?;
.map_err(|err| FormRejection::FailedToDeserializeForm(Error::new(err)))
Ok(Self(value))
} }
} }
/// Rejection used for [`Form`]. define_rejection! {
/// #[status = BAD_REQUEST]
/// Contains one variant for each way the [`Form`] extractor can fail. #[body = "Failed to deserialize form"]
#[derive(Debug)] /// Rejection type used if the [`Form`](Form) extractor is unable to
#[non_exhaustive] /// deserialize the form into the target type.
#[cfg(feature = "form")] pub struct FailedToDeserializeForm(Error);
pub enum FormRejection {
#[allow(missing_docs)]
RawFormRejection(RawFormRejection),
#[allow(missing_docs)]
FailedToDeserializeForm(Error),
} }
impl FormRejection { composite_rejection! {
/// Get the status code used for this rejection. /// Rejection used for [`Form`].
pub fn status(&self) -> StatusCode { ///
// Make sure to keep this in sync with `IntoResponse` impl. /// Contains one variant for each way the [`Form`] extractor can fail.
match self { pub enum FormRejection {
Self::RawFormRejection(inner) => inner.status(), RawFormRejection,
Self::FailedToDeserializeForm(_) => StatusCode::BAD_REQUEST, FailedToDeserializeForm,
}
}
}
impl IntoResponse for FormRejection {
fn into_response(self) -> Response {
let status = self.status();
match self {
Self::RawFormRejection(inner) => inner.into_response(),
Self::FailedToDeserializeForm(inner) => {
let body = format!("Failed to deserialize form: {inner}");
axum_core::__log_rejection!(
rejection_type = Self,
body_text = body,
status = status,
);
(status, body).into_response()
}
}
}
}
impl fmt::Display for FormRejection {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::RawFormRejection(inner) => inner.fmt(f),
Self::FailedToDeserializeForm(inner) => inner.fmt(f),
}
}
}
impl std::error::Error for FormRejection {
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
match self {
Self::RawFormRejection(inner) => Some(inner),
Self::FailedToDeserializeForm(inner) => Some(inner),
}
} }
} }
@ -131,6 +87,7 @@ mod tests {
use axum::routing::{on, post, MethodFilter}; use axum::routing::{on, post, MethodFilter};
use axum::Router; use axum::Router;
use http::header::CONTENT_TYPE; use http::header::CONTENT_TYPE;
use http::StatusCode;
use mime::APPLICATION_WWW_FORM_URLENCODED; use mime::APPLICATION_WWW_FORM_URLENCODED;
use serde::Deserialize; use serde::Deserialize;