3 releases (breaking)
0.2.0 | Apr 11, 2024 |
---|---|
0.1.0 | Jul 22, 2023 |
0.0.1 | Jul 19, 2023 |
#2 in #polling
Used in cache_browns
24KB
351 lines
interruptible-polling
General purpose polling tasks with RAII-driven fast clean exits.
See docs.rs for full details.
License
This project is licensed under either of
at your option.
The SPDX license identifier for this project is MIT OR Apache-2.0
.
lib.rs
:
This library provides PollingTask
and SelfUpdatingPollingTask
structs for scheduling a
closure to execute as a recurring task.
It is common for a service to have long-lived polling operations for the life of the process. The intended use case is to offer a RAII container for a polled operation that will interrupt pending sleeps to allow a low-latency clean exit.
If the poll operation is still running, the task drop will join the background thread which will exit after the closure finishes.
Examples
-
Use
PollingTask
to emit a heart beat every 30 seconds.use interruptible_polling::PollingTask; use std::time::Duration; let task = PollingTask::new(Duration::from_secs(30), || { println!("BeatBeat"); });
-
Some polled operations such as configuration updates contain the updated rate at which the service should continue to poll for future updates. The
SelfUpdatingPollingTask
passes a callback to the poll task that allows it to conveniently apply the new state to future polls.use interruptible_polling::SelfUpdatingPollingTask; use std::time::Duration; use serde_json::{Value, from_reader}; use std::fs::File; use std::io::BufReader; let task = SelfUpdatingPollingTask::new(Duration::from_secs(30), move |interval: &mut Duration| { let file = File::open("app.config").unwrap(); let reader = BufReader::new(file); let config: Value = from_reader(reader).expect("JSON was not well-formatted"); // Do other work with config *interval = Duration::from_secs(config["pollingInterval"].as_u64().expect("Polling interval isn't u64 convertable")); } );
-
If your poll operation is long-lived or internally iterative, there are opportunities to assert if the task is still active to allow the blocked clean exit to occur faster. If you create the task with
PollingTask::new_with_checker
orSelfUpdatingPollingTask::new_with_checker
your closure will receive a lookup function to peek if the managed task is still active.
use interruptible_polling::PollingTask;
use std::time::Duration;
let task = PollingTask::new_with_checker(
Duration::from_secs(30),
|checker: &dyn Fn() -> bool|
{
let keys = vec![1 ,2, 3];
for key in keys {
// Early exit if signaled. The task will not poll again either way, but you have
// returned control to the parent task earlier.
if !checker() {
break;
}
// Some long or potentially long operation such as a synchronous web request.
}
});
Fire and Forget
For convenience, if you also need to run polling threads that don't require clean exits, fire and forget can be
enabled. This is gated behind feature fire-forget
to encourage use of the primary abstractions. It's not hard to make a
polling thread, so typical crate users are here for the clean exit constructs. However, some projects need both.
If you need both, enable the feature to make both available. By default, it's disabled.