#mocking #async #tokio #run-time #compatible #clock

async-time-mock-tokio

Mockable time for use in async runtimes (tokio compatibility)

4 releases

new 0.1.2 Jan 13, 2025
0.1.1 Jan 22, 2024
0.1.0 Feb 19, 2023
0.0.1 Jan 11, 2023

#156 in Testing

Download history 11/week @ 2024-09-25 3/week @ 2024-10-02 12/week @ 2024-10-09 5/week @ 2024-10-16 28/week @ 2024-10-30 251/week @ 2024-11-06 130/week @ 2024-11-13 240/week @ 2024-11-20 522/week @ 2024-11-27 241/week @ 2024-12-04 265/week @ 2024-12-11 234/week @ 2024-12-18 158/week @ 2024-12-25 63/week @ 2025-01-08

494 downloads per month

MIT license

20KB
486 lines

async-time-mock-tokio

Asynchronous time mocking with a tokio compatible API based on async-time-mock-core, inspired by the approach described in Mocking Time In Async Rust.

NOTE: This library is still in it's infancy, the API is still likely to change (read: improve). Please leave your feedback and suggestions on GitHub.

Cargo features

  • mock: Enable the mock clock. If you only enable this in tests, this library turns into a thin wrapper around tokio's time functions.

Example

use async_time_mock_tokio::MockableClock;
use std::{
	sync::atomic::{AtomicBool, Ordering},
	time::Duration,
};

static HAS_SLEPT: AtomicBool = AtomicBool::new(false);

async fn sleep(clock: MockableClock) {
	// Sleep is either mocked or a real tokio::sleep, depending on which variant of `MockableClock` you pass in.
	let _guard = clock.sleep(Duration::from_secs(3600)).await;
	// Dropping this guard signifies that all the effects of the timer have finished.
	// This allows test code to wait until the condition to assert for has happened.

	println!("Slept for an hour");
	HAS_SLEPT.store(true, Ordering::SeqCst);
}

#[tokio::main]
async fn main() {
	let (clock, controller) = MockableClock::mock(); // In production, you can use MockableClock::Real instead

	tokio::spawn(sleep(clock));

	controller.advance_time(Duration::from_secs(600)).await;
	assert!(!HAS_SLEPT.load(Ordering::SeqCst), "Timer won't trigger after just 10 minutes.");

	// advance_time will first trigger the sleep in the task above and then wait until the `_guard` was dropped.
	// This ensures that the task had enough time to actually set `HAS_SLEPT` to `true`.
	controller.advance_time(Duration::from_secs(3000)).await;
	assert!(HAS_SLEPT.load(Ordering::SeqCst), "Timer has triggered after 1 hour.")
}

Dependencies

~2.3–8.5MB
~65K SLoC