2021-08-20 09:26:31 +02:00
|
|
|
//! Run with
|
|
|
|
//!
|
|
|
|
//! ```not_rust
|
2023-03-10 12:02:11 +01:00
|
|
|
//! cargo run -p example-print-request-response
|
2021-08-20 09:26:31 +02:00
|
|
|
//! ```
|
|
|
|
|
|
|
|
use axum::{
|
2021-12-05 19:16:46 +01:00
|
|
|
body::{Body, Bytes},
|
2023-03-20 21:02:40 +01:00
|
|
|
extract::Request,
|
|
|
|
http::StatusCode,
|
2022-01-25 10:19:06 +01:00
|
|
|
middleware::{self, Next},
|
2021-12-27 14:01:26 +01:00
|
|
|
response::{IntoResponse, Response},
|
2021-10-24 22:05:16 +02:00
|
|
|
routing::post,
|
2021-08-20 09:26:31 +02:00
|
|
|
Router,
|
|
|
|
};
|
2023-11-23 12:03:03 +01:00
|
|
|
use http_body_util::BodyExt;
|
2022-03-06 12:37:00 +01:00
|
|
|
use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt};
|
2021-08-20 09:26:31 +02:00
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
async fn main() {
|
2022-03-06 12:37:00 +01:00
|
|
|
tracing_subscriber::registry()
|
2022-11-30 10:46:19 +01:00
|
|
|
.with(
|
2024-08-24 08:36:08 +02:00
|
|
|
tracing_subscriber::EnvFilter::try_from_default_env().unwrap_or_else(|_| {
|
|
|
|
format!("{}=debug,tower_http=debug", env!("CARGO_CRATE_NAME")).into()
|
|
|
|
}),
|
2022-11-30 10:46:19 +01:00
|
|
|
)
|
2022-03-06 12:37:00 +01:00
|
|
|
.with(tracing_subscriber::fmt::layer())
|
|
|
|
.init();
|
2021-08-20 09:26:31 +02:00
|
|
|
|
|
|
|
let app = Router::new()
|
|
|
|
.route("/", post(|| async move { "Hello from `POST /`" }))
|
2021-12-27 14:01:26 +01:00
|
|
|
.layer(middleware::from_fn(print_request_response));
|
2021-08-20 09:26:31 +02:00
|
|
|
|
2023-03-22 23:42:14 +01:00
|
|
|
let listener = tokio::net::TcpListener::bind("127.0.0.1:3000")
|
2021-08-20 09:26:31 +02:00
|
|
|
.await
|
|
|
|
.unwrap();
|
2023-03-22 23:42:14 +01:00
|
|
|
tracing::debug!("listening on {}", listener.local_addr().unwrap());
|
|
|
|
axum::serve(listener, app).await.unwrap();
|
2021-08-20 09:26:31 +02:00
|
|
|
}
|
|
|
|
|
2021-12-27 14:01:26 +01:00
|
|
|
async fn print_request_response(
|
2023-03-20 21:02:40 +01:00
|
|
|
req: Request,
|
|
|
|
next: Next,
|
2021-12-27 14:01:26 +01:00
|
|
|
) -> Result<impl IntoResponse, (StatusCode, String)> {
|
2021-08-20 09:26:31 +02:00
|
|
|
let (parts, body) = req.into_parts();
|
|
|
|
let bytes = buffer_and_print("request", body).await?;
|
|
|
|
let req = Request::from_parts(parts, Body::from(bytes));
|
|
|
|
|
2021-12-27 14:01:26 +01:00
|
|
|
let res = next.run(req).await;
|
|
|
|
|
2021-08-20 09:26:31 +02:00
|
|
|
let (parts, body) = res.into_parts();
|
|
|
|
let bytes = buffer_and_print("response", body).await?;
|
|
|
|
let res = Response::from_parts(parts, Body::from(bytes));
|
2021-12-27 14:01:26 +01:00
|
|
|
|
2021-08-20 09:26:31 +02:00
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
|
2021-12-27 14:01:26 +01:00
|
|
|
async fn buffer_and_print<B>(direction: &str, body: B) -> Result<Bytes, (StatusCode, String)>
|
2021-08-20 09:26:31 +02:00
|
|
|
where
|
|
|
|
B: axum::body::HttpBody<Data = Bytes>,
|
2021-12-27 14:01:26 +01:00
|
|
|
B::Error: std::fmt::Display,
|
2021-08-20 09:26:31 +02:00
|
|
|
{
|
2023-11-23 12:03:03 +01:00
|
|
|
let bytes = match body.collect().await {
|
|
|
|
Ok(collected) => collected.to_bytes(),
|
2021-12-27 14:01:26 +01:00
|
|
|
Err(err) => {
|
|
|
|
return Err((
|
|
|
|
StatusCode::BAD_REQUEST,
|
2023-09-19 08:51:57 +02:00
|
|
|
format!("failed to read {direction} body: {err}"),
|
2021-12-27 14:01:26 +01:00
|
|
|
));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-08-20 09:26:31 +02:00
|
|
|
if let Ok(body) = std::str::from_utf8(&bytes) {
|
2023-09-19 08:51:57 +02:00
|
|
|
tracing::debug!("{direction} body = {body:?}");
|
2021-08-20 09:26:31 +02:00
|
|
|
}
|
2021-12-27 14:01:26 +01:00
|
|
|
|
2021-08-20 09:26:31 +02:00
|
|
|
Ok(bytes)
|
|
|
|
}
|