use std::net::SocketAddr; use axum::{response::IntoResponse, routing::get, Router}; use deadpool_postgres::Pool; use serde::{Deserialize, Serialize}; use time::Duration; use tokio::{signal, task::AbortHandle}; use tower_sessions::{session_store::ExpiredDeletion, Expiry, Session, SessionManagerLayer}; use tower_sessions_postgres_store::PostgresStore; const COUNTER_KEY: &str = "counter"; #[derive(Serialize, Deserialize, Default)] struct Counter(usize); async fn handler(session: Session) -> impl IntoResponse { let counter: Counter = session.get(COUNTER_KEY).await.unwrap().unwrap_or_default(); session.insert(COUNTER_KEY, counter.0 + 1).await.unwrap(); format!("Current count: {}", counter.0) } pub fn create_pool(database_url: &str) -> Pool { let config = database_url.parse().unwrap(); let manager = deadpool_postgres::Manager::new(config, tokio_postgres::NoTls); deadpool_postgres::Pool::builder(manager).build().unwrap() } #[tokio::main] async fn main() -> Result<(), Box> { let database_url = std::option_env!("DATABASE_URL").expect("Missing DATABASE_URL."); let pool = create_pool(database_url); let session_store = PostgresStore::new(pool); session_store.migrate().await?; let deletion_task = tokio::task::spawn( session_store .clone() .continuously_delete_expired(tokio::time::Duration::from_secs(60)), ); let session_layer = SessionManagerLayer::new(session_store) .with_secure(false) .with_expiry(Expiry::OnInactivity(Duration::seconds(10))); let app = Router::new().route("/", get(handler)).layer(session_layer); let addr = SocketAddr::from(([127, 0, 0, 1], 3000)); let listener = tokio::net::TcpListener::bind(&addr).await?; // Ensure we use a shutdown signal to abort the deletion task. axum::serve(listener, app.into_make_service()) .with_graceful_shutdown(shutdown_signal(deletion_task.abort_handle())) .await?; deletion_task.await??; Ok(()) } async fn shutdown_signal(deletion_task_abort_handle: AbortHandle) { let ctrl_c = async { signal::ctrl_c() .await .expect("failed to install Ctrl+C handler"); }; #[cfg(unix)] let terminate = async { signal::unix::signal(signal::unix::SignalKind::terminate()) .expect("failed to install signal handler") .recv() .await; }; #[cfg(not(unix))] let terminate = std::future::pending::<()>(); tokio::select! { _ = ctrl_c => { deletion_task_abort_handle.abort() }, _ = terminate => { deletion_task_abort_handle.abort() }, } }