use std::sync::Arc;
use std::time::Duration;
use axum::{
extract::DefaultBodyLimit,
http::HeaderValue,
middleware::{self, Next},
response::Response,
routing::get,
Router,
};
use tower_http::{timeout::TimeoutLayer, trace::TraceLayer};
use crate::AppState;
pub mod health;
pub mod metrics_handler;
pub mod send;
async fn request_id_layer(mut req: axum::http::Request<axum::body::Body>, next: Next) -> Response {
let request_id = uuid::Uuid::new_v4().to_string();
if let Ok(val) = HeaderValue::from_str(&request_id) {
req.headers_mut().insert("x-internal-request-id", val.clone());
}
let mut resp = next.run(req).await;
if let Ok(val) = HeaderValue::from_str(&request_id) {
resp.headers_mut().insert("x-request-id", val);
}
resp
}
pub fn build_router(state: Arc<AppState>) -> Router {
let cfg = state.config();
let max_body = cfg.server.max_request_body_bytes;
let timeout_secs = cfg.server.request_timeout_seconds;
let concurrency = cfg.server.concurrency_limit;
let mut router = Router::new()
.route("/healthz", get(health::healthz))
.route("/readyz", get(health::readyz))
.route("/metrics", get(metrics_handler::metrics_handler))
.route("/v1/send", axum::routing::post(send::send_mail))
.layer(middleware::from_fn(request_id_layer))
.layer(TraceLayer::new_for_http())
.layer(TimeoutLayer::with_status_code(
axum::http::StatusCode::REQUEST_TIMEOUT,
Duration::from_secs(timeout_secs),
))
.layer(DefaultBodyLimit::max(max_body))
.with_state(state);
if concurrency > 0 {
router = router.layer(tower::limit::ConcurrencyLimitLayer::new(concurrency));
}
router
}