use crate::app::context::AppContext;
use crate::service::http::middleware::Middleware;
use axum::Router;
use axum::http::HeaderName;
use axum_core::extract::FromRef;
use serde_derive::{Deserialize, Serialize};
use std::str::FromStr;
use tower_http::request_id::{MakeRequestUuid, PropagateRequestIdLayer, SetRequestIdLayer};
use validator::Validate;
pub const REQUEST_ID_HEADER_NAME: &str = "request-id";
#[derive(Debug, Clone, Validate, Serialize, Deserialize)]
#[serde(rename_all = "kebab-case", default)]
#[non_exhaustive]
pub struct CommonRequestIdConfig {
pub header_name: String,
}
impl Default for CommonRequestIdConfig {
fn default() -> Self {
Self {
header_name: REQUEST_ID_HEADER_NAME.to_string(),
}
}
}
#[derive(Debug, Clone, Default, Validate, Serialize, Deserialize)]
#[serde(rename_all = "kebab-case", default)]
#[non_exhaustive]
pub struct SetRequestIdConfig {
#[serde(flatten)]
#[validate(nested)]
pub common: CommonRequestIdConfig,
}
#[derive(Debug, Clone, Default, Validate, Serialize, Deserialize)]
#[serde(rename_all = "kebab-case", default)]
#[non_exhaustive]
pub struct PropagateRequestIdConfig {
#[serde(flatten)]
#[validate(nested)]
pub common: CommonRequestIdConfig,
}
pub struct SetRequestIdMiddleware;
impl<S> Middleware<S> for SetRequestIdMiddleware
where
S: 'static + Send + Sync + Clone,
AppContext: FromRef<S>,
{
type Error = crate::error::Error;
fn name(&self) -> String {
"set-request-id".to_string()
}
fn enabled(&self, state: &S) -> bool {
AppContext::from_ref(state)
.config()
.service
.http
.custom
.middleware
.set_request_id
.common
.enabled(state)
}
fn priority(&self, state: &S) -> i32 {
AppContext::from_ref(state)
.config()
.service
.http
.custom
.middleware
.set_request_id
.common
.priority
}
fn install(&self, state: &S, router: Router) -> Result<Router, Self::Error> {
let context = AppContext::from_ref(state);
let header_name = &context
.config()
.service
.http
.custom
.middleware
.set_request_id
.custom
.common
.header_name;
let router = router.layer(SetRequestIdLayer::new(
HeaderName::from_str(header_name)?,
MakeRequestUuid,
));
Ok(router)
}
}
pub struct PropagateRequestIdMiddleware;
impl<S> Middleware<S> for PropagateRequestIdMiddleware
where
S: 'static + Send + Sync + Clone,
AppContext: FromRef<S>,
{
type Error = crate::error::Error;
fn name(&self) -> String {
"propagate-request-id".to_string()
}
fn enabled(&self, state: &S) -> bool {
AppContext::from_ref(state)
.config()
.service
.http
.custom
.middleware
.propagate_request_id
.common
.enabled(state)
}
fn priority(&self, state: &S) -> i32 {
AppContext::from_ref(state)
.config()
.service
.http
.custom
.middleware
.propagate_request_id
.common
.priority
}
fn install(&self, state: &S, router: Router) -> Result<Router, Self::Error> {
let context = AppContext::from_ref(state);
let header_name = &context
.config()
.service
.http
.custom
.middleware
.propagate_request_id
.custom
.common
.header_name;
let router = router.layer(PropagateRequestIdLayer::new(HeaderName::from_str(
header_name,
)?));
Ok(router)
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::config::AppConfig;
use rstest::rstest;
#[rstest]
#[case(false, Some(true), true)]
#[case(false, Some(false), false)]
#[cfg_attr(coverage_nightly, coverage(off))]
fn set_request_id_enabled(
#[case] default_enable: bool,
#[case] enable: Option<bool>,
#[case] expected_enabled: bool,
) {
let mut config = AppConfig::test(None).unwrap();
config.service.http.custom.middleware.default_enable = default_enable;
config
.service
.http
.custom
.middleware
.set_request_id
.common
.enable = enable;
let context = AppContext::test(Some(config), None, None).unwrap();
let middleware = SetRequestIdMiddleware;
assert_eq!(middleware.enabled(&context), expected_enabled);
}
#[rstest]
#[case(None, -9990)]
#[case(Some(1234), 1234)]
#[cfg_attr(coverage_nightly, coverage(off))]
fn set_request_id_priority(
#[case] override_priority: Option<i32>,
#[case] expected_priority: i32,
) {
let mut config = AppConfig::test(None).unwrap();
if let Some(priority) = override_priority {
config
.service
.http
.custom
.middleware
.set_request_id
.common
.priority = priority;
}
let context = AppContext::test(Some(config), None, None).unwrap();
let middleware = SetRequestIdMiddleware;
assert_eq!(middleware.priority(&context), expected_priority);
}
#[rstest]
#[case(false, Some(true), true)]
#[case(false, Some(false), false)]
#[cfg_attr(coverage_nightly, coverage(off))]
fn propagate_request_id_enabled(
#[case] default_enable: bool,
#[case] enable: Option<bool>,
#[case] expected_enabled: bool,
) {
let mut config = AppConfig::test(None).unwrap();
config.service.http.custom.middleware.default_enable = default_enable;
config
.service
.http
.custom
.middleware
.propagate_request_id
.common
.enable = enable;
let context = AppContext::test(Some(config), None, None).unwrap();
let middleware = PropagateRequestIdMiddleware;
assert_eq!(middleware.enabled(&context), expected_enabled);
}
#[rstest]
#[case(None, 9990)]
#[case(Some(1234), 1234)]
#[cfg_attr(coverage_nightly, coverage(off))]
fn propagate_request_id_priority(
#[case] override_priority: Option<i32>,
#[case] expected_priority: i32,
) {
let mut config = AppConfig::test(None).unwrap();
if let Some(priority) = override_priority {
config
.service
.http
.custom
.middleware
.propagate_request_id
.common
.priority = priority;
}
let context = AppContext::test(Some(config), None, None).unwrap();
let middleware = PropagateRequestIdMiddleware;
assert_eq!(middleware.priority(&context), expected_priority);
}
}