use std::net::SocketAddr; use axum::{response::IntoResponse, routing::get, Router}; use serde::{Deserialize, Serialize}; use time::Duration; use tokio::{signal, task::AbortHandle}; use tower_sessions::{session_store::ExpiredDeletion, Expiry, Session, SessionManagerLayer}; use tower_sessions_rusqlite_store::{tokio_rusqlite::Connection, RusqliteStore}; 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) } #[tokio::main] async fn main() -> Result<(), Box> { // in memory: let conn = Connection::open_in_memory().await.unwrap(); // or on disk: // let conn = Connection::open(&"rusqlite-example.db").await.unwrap(); let session_store = RusqliteStore::new(conn); session_store.migrate().await?; let deletion_task = tokio::task::spawn( session_store .clone() .continuously_delete_expired(tokio::time::Duration::from_secs(5)), ); let session_layer = SessionManagerLayer::new(session_store) .with_secure(false) .with_expiry(Expiry::OnInactivity(Duration::seconds(3))); 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?; println!("Listening on {}", addr); // 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() }, } }