mirror of
https://github.com/tokio-rs/axum.git
synced 2025-10-02 15:24:54 +00:00

* Only allow last extractor to mutate the request * Change `FromRequest` and add `FromRequestParts` trait (#1275) * Add `Once`/`Mut` type parameter for `FromRequest` and `RequestParts` * 🪄 * split traits * `FromRequest` for tuples * Remove `BodyAlreadyExtracted` * don't need fully qualified path * don't export `Once` and `Mut` * remove temp tests * depend on axum again Co-authored-by: Jonas Platte <jplatte+git@posteo.de> * Port `Handler` and most extractors (#1277) * Port `Handler` and most extractors * Put `M` inside `Handler` impls, not trait itself * comment out tuples for now * fix lints * Reorder arguments to `Handler` (#1281) I think `Request<B>, Arc<S>` is better since its consistent with `FromRequest` and `FromRequestParts`. * Port most things in axum-extra (#1282) * Port `#[derive(TypedPath)]` and `#[debug_handler]` (#1283) * port #[derive(TypedPath)] * wip: #[debug_handler] * fix #[debug_handler] * don't need itertools * also require `Send` * update expected error * support fully qualified `self` * Implement FromRequest[Parts] for tuples (#1286) * Port docs for axum and axum-core (#1285) * Port axum-extra (#1287) * Port axum-extra * Update axum-core/Cargo.toml Co-authored-by: Jonas Platte <jplatte+git@posteo.de> * remove `impl FromRequest for Either*` Co-authored-by: Jonas Platte <jplatte+git@posteo.de> * New FromRequest[Parts] trait cleanup (#1288) * Make private module truly private again * Simplify tuple FromRequest implementation * Port `#[derive(FromRequest)]` (#1289) * fix tests * fix docs * revert examples * fix docs link * fix intra docs links * Port examples (#1291) * Document wrapping other extractors (#1292) * axum-extra doesn't need to depend on axum-core (#1294) Missed this in https://github.com/tokio-rs/axum/pull/1287 * Add `FromRequest` changes to changelogs (#1293) * Update changelog * Remove default type for `S` in `Handler` * Clarify which types have default types for `S` * Apply suggestions from code review Co-authored-by: Jonas Platte <jplatte+git@posteo.de> Co-authored-by: Jonas Platte <jplatte+git@posteo.de> * remove unused import * Rename `Mut` and `Once` (#1296) * fix trybuild expected output Co-authored-by: Jonas Platte <jplatte+git@posteo.de>
61 lines
2.2 KiB
Rust
61 lines
2.2 KiB
Rust
//! Uses `axum_macros::FromRequest` to wrap another extractor and customize the
|
|
//! rejection
|
|
//!
|
|
//! + Easy learning curve: Deriving `FromRequest` generates a `FromRequest`
|
|
//! implementation for your type using another extractor. You only need
|
|
//! to provide a `From` impl between the original rejection type and the
|
|
//! target rejection. Crates like [`thiserror`] can provide such conversion
|
|
//! using derive macros.
|
|
//! - Boilerplate: Requires deriving `FromRequest` for every custom rejection
|
|
//! - There are some known limitations: [FromRequest#known-limitations]
|
|
//!
|
|
//! [`thiserror`]: https://crates.io/crates/thiserror
|
|
//! [FromRequest#known-limitations]: https://docs.rs/axum-macros/*/axum_macros/derive.FromRequest.html#known-limitations
|
|
use axum::{extract::rejection::JsonRejection, http::StatusCode, response::IntoResponse};
|
|
use axum_macros::FromRequest;
|
|
use serde_json::{json, Value};
|
|
|
|
pub async fn handler(Json(value): Json<Value>) -> impl IntoResponse {
|
|
Json(dbg!(value));
|
|
}
|
|
|
|
// create an extractor that internally uses `axum::Json` but has a custom rejection
|
|
#[derive(FromRequest)]
|
|
#[from_request(via(axum::Json), rejection(ApiError))]
|
|
pub struct Json<T>(T);
|
|
|
|
// We create our own rejection type
|
|
#[derive(Debug)]
|
|
pub struct ApiError {
|
|
code: StatusCode,
|
|
message: String,
|
|
}
|
|
|
|
// We implement `From<JsonRejection> for ApiError`
|
|
impl From<JsonRejection> for ApiError {
|
|
fn from(rejection: JsonRejection) -> Self {
|
|
let code = match rejection {
|
|
JsonRejection::JsonDataError(_) => StatusCode::UNPROCESSABLE_ENTITY,
|
|
JsonRejection::JsonSyntaxError(_) => StatusCode::BAD_REQUEST,
|
|
JsonRejection::MissingJsonContentType(_) => StatusCode::UNSUPPORTED_MEDIA_TYPE,
|
|
_ => StatusCode::INTERNAL_SERVER_ERROR,
|
|
};
|
|
Self {
|
|
code,
|
|
message: rejection.to_string(),
|
|
}
|
|
}
|
|
}
|
|
|
|
// We implement `IntoResponse` so `ApiError` can be used as a response
|
|
impl IntoResponse for ApiError {
|
|
fn into_response(self) -> axum::response::Response {
|
|
let payload = json!({
|
|
"message": self.message,
|
|
"origin": "derive_from_request"
|
|
});
|
|
|
|
(self.code, axum::Json(payload)).into_response()
|
|
}
|
|
}
|