objectstore_server/web/
app.rs1use std::net::SocketAddr;
2
3use anyhow::Result;
4use axum::ServiceExt;
5use axum::extract::Request;
6use sentry::integrations::tower::{NewSentryLayer, SentryHttpLayer};
7use tokio::net::TcpListener;
8use tower::ServiceBuilder;
9use tower_http::catch_panic::CatchPanicLayer;
10use tower_http::trace::{DefaultOnFailure, TraceLayer};
11use tracing::Level;
12
13use crate::endpoints;
14use crate::state::ServiceState;
15use crate::web::middleware as m;
16
17#[derive(Debug)]
19pub struct App {
20 router: axum::Router,
21 graceful_shutdown: bool,
22}
23
24impl App {
25 pub fn new(state: ServiceState) -> Self {
30 let middleware = ServiceBuilder::new()
35 .layer(axum::middleware::from_fn(m::emit_request_metrics))
36 .layer(axum::middleware::from_fn_with_state(
37 state.request_counter.clone(),
38 m::limit_web_concurrency,
39 ))
40 .layer(state.request_counter.layer())
41 .layer(CatchPanicLayer::custom(m::handle_panic))
42 .layer(m::set_server_header())
43 .layer(NewSentryLayer::new_from_top())
44 .layer(SentryHttpLayer::new().enable_transaction())
45 .layer(
46 TraceLayer::new_for_http()
47 .make_span_with(m::make_http_span)
48 .on_failure(DefaultOnFailure::new().level(Level::DEBUG)),
49 );
50
51 let router = endpoints::routes()
52 .layer(middleware)
53 .with_state(state.clone());
54
55 App {
56 router,
57 graceful_shutdown: false,
58 }
59 }
60
61 pub fn graceful_shutdown(mut self, enable: bool) -> Self {
65 self.graceful_shutdown = enable;
66 self
67 }
68
69 pub async fn serve(self, listener: TcpListener) -> Result<()> {
74 let Self {
75 router,
76 graceful_shutdown,
77 } = self;
78
79 let service =
80 ServiceExt::<Request>::into_make_service_with_connect_info::<SocketAddr>(router);
81
82 if graceful_shutdown {
83 let guard = elegant_departure::get_shutdown_guard();
84 axum::serve(listener, service)
85 .with_graceful_shutdown(guard.wait_owned())
86 .await?;
87 } else {
88 axum::serve(listener, service).await?;
89 }
90
91 Ok(())
92 }
93}