tower-sessions-ext-mongodb-store

Crates.iotower-sessions-ext-mongodb-store
lib.rstower-sessions-ext-mongodb-store
version0.13.0
created_at2025-06-15 20:13:37.09928+00
updated_at2025-09-09 17:34:38.427137+00
descriptionMongoDB session store for `tower-sessions-ext`.
homepagehttps://github.com/sagoez/tower-sessions-ext-stores
repositoryhttps://github.com/sagoez/tower-sessions-ext-stores
max_upload_size
id1713602
size80,809
Samuel (samgj18)

documentation

https://docs.rs/tower-sessions-ext-mongodb-store

README

tower-sessions-ext-mongodb-store

MongoDB session store for `tower-sessions-ext`.

[!NOTE] This is a maintained fork of the original implementation. The repository has been migrated from maxcountryman to sagoez to ensure continued maintenance and support.

🤸 Usage

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_ext::{Expiry, Session, SessionManagerLayer};
use tower_sessions_ext_core::ExpiredDeletion;
use tower_sessions_ext_mongodb_store::{mongodb::Client, MongoDBStore};

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<dyn std::error::Error>> {
    let database_url = std::option_env!("DATABASE_URL").expect("Missing DATABASE_URL.");
    let client = Client::with_uri_str(database_url).await?;
    let session_store = MongoDBStore::new(client, "tower-sessions-ext".to_string());

    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() },
    }
}

## 🙏 Acknowledgments

Special thanks to [maxcountryman](https://github.com/maxcountryman) for the original implementation and groundwork for this project.
Commit count: 0

cargo fmt