axum-extra/form: Add FailedToDeserializeFormBody rejection variant (#3119)

This commit is contained in:
Tobias Bieniek 2024-12-27 12:21:38 +01:00 committed by GitHub
parent 53370b24e7
commit 09841ff895
No known key found for this signature in database
GPG key ID: B5690EEEBB952194

View file

@ -51,14 +51,22 @@ 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 is_get_or_head =
req.method() == http::Method::GET || req.method() == http::Method::HEAD;
let RawForm(bytes) = req.extract().await?; let RawForm(bytes) = req.extract().await?;
let deserializer = serde_html_form::Deserializer::new(form_urlencoded::parse(&bytes)); let deserializer = serde_html_form::Deserializer::new(form_urlencoded::parse(&bytes));
let value = serde_path_to_error::deserialize::<_, T>(deserializer) serde_path_to_error::deserialize::<_, T>(deserializer)
.map_err(FailedToDeserializeForm::from_err)?; .map(Self)
.map_err(|err| {
Ok(Self(value)) if is_get_or_head {
FailedToDeserializeForm::from_err(err).into()
} else {
FailedToDeserializeFormBody::from_err(err).into()
}
})
} }
} }
@ -70,6 +78,14 @@ define_rejection! {
pub struct FailedToDeserializeForm(Error); pub struct FailedToDeserializeForm(Error);
} }
define_rejection! {
#[status = UNPROCESSABLE_ENTITY]
#[body = "Failed to deserialize form body"]
/// Rejection type used if the [`Form`](Form) extractor is unable to
/// deserialize the form body into the target type.
pub struct FailedToDeserializeFormBody(Error);
}
composite_rejection! { composite_rejection! {
/// Rejection used for [`Form`]. /// Rejection used for [`Form`].
/// ///
@ -77,6 +93,7 @@ composite_rejection! {
pub enum FormRejection { pub enum FormRejection {
RawFormRejection, RawFormRejection,
FailedToDeserializeForm, FailedToDeserializeForm,
FailedToDeserializeFormBody,
} }
} }
@ -146,10 +163,10 @@ mod tests {
.header(CONTENT_TYPE, APPLICATION_WWW_FORM_URLENCODED.as_ref()) .header(CONTENT_TYPE, APPLICATION_WWW_FORM_URLENCODED.as_ref())
.body("a=false") .body("a=false")
.await; .await;
assert_eq!(res.status(), StatusCode::BAD_REQUEST); assert_eq!(res.status(), StatusCode::UNPROCESSABLE_ENTITY);
assert_eq!( assert_eq!(
res.text().await, res.text().await,
"Failed to deserialize form: a: invalid digit found in string" "Failed to deserialize form body: a: invalid digit found in string"
); );
} }
} }