tokio-task-tracker

Crates.iotokio-task-tracker
lib.rstokio-task-tracker
version1.3.3
sourcesrc
created_at2023-10-06 20:14:56.763777
updated_at2024-08-07 17:07:15.363155
descriptionA simple graceful shutdown solution for tokio.
homepage
repositoryhttps://github.com/jwalton/rust-tokio-task-tracker/
max_upload_size
id995349
size20,458
Jason Walton (jwalton)

documentation

README

tokio-task-tracker

tokio-task-tracker is a simple graceful shutdown solution for tokio.

The basic idea is to use a TaskSpawner to create TaskTracker objects, and hold on to them in spawned tasks. Inside the task, you can check tracker.cancelled().await to wait for the task to be cancelled.

The TaskWaiter can be used to wait for an interrupt and then wait for all trackers to be dropped.

Example

use std::time::Duration;

#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    let (spawner, waiter) = tokio_task_tracker::new();

    // Start a task
    spawner.spawn(|tracker| async move {
        tokio::select! {
            _ = tracker.cancelled() => {
                // The token was cancelled, task should shut down.
            }
            _ = tokio::time::sleep(Duration::from_secs(9999)) => {
                // Long work has completed
            }
        }
    });

    // Wait for all tasks to complete, or for someone to hit ctrl-c.
    // If tasks down't complete within 5 seconds, we'll quit anyways.
    waiter.wait_for_shutdown(Duration::from_secs(5)).await?;

    Ok(())
}

If you do not wish to allow a task to be aborted, you still need to make sure the task captures the tracker:

    // Start a task
    spawner.spawn(|tracker| async move {
        // Move the tracker into the task.
        let _tracker = tracker;

        // Do some work that we don't want to abort.
        tokio::time::sleep(Duration::from_secs(9999)).await;
    });

You can also create a tracker via the task method:

    // Start a task
    let tracker = spawner.task();
    tokio::task::spawn(async move {
        // Move the tracker into the task.
        let _tracker = tracker;

        // ...
    });

Trackers can be used to spawn subtasks via tracker.subtask() or tracker.spawn().

Commit count: 14

cargo fmt