Helpers for "fusing" asynchronous computations.
A fused operation has a well-defined behavior once the operation has completed. For [Fuse] it means that an operation that has completed will block forever by returning [Poll::Pending].
This is similar to the Fuse type provided in futures-rs, but provides more utility allowing it to interact with types which does not implement [FusedFuture] or [FusedStream] as is now the case with all Tokio types since 1.0.
We also use [Fuse] to represent optional values, just like [Option]. But [Fuse] provides implementations and functions which allow us to safely perform operations over the value when it's pinned. Like what's needed to poll a future.
So we can take code that looks like this:
```rust let mut maybe_future = Some(future);
tokio::select! {
// NB: The async block is necessary because the future is polled eagerly
// regardless of the condition, which could cause the unwrap
to panic.
value = async { maybefuture.asmut().unwrap().await }, if maybefuture.issome() => {
/* do something */
}
}
```
And rewrite it into:
```rust let mut maybe_future = Fuse::new(future);
tokio::select! { value = &mut maybefuture, if !maybefuture.is_empty() => { /* do something */ } } ```
If we don't need the [else branch] to evalute, we can skip the [branch precondition]. Allowing us to further reduce it to:
```rust let mut maybe_future = Fuse::new(future);
tokio::select! { value = &mut maybe_future => { /* do something */ } } ```
stream
- Makes the [Fuse] implement the [Stream] trait if it contains a
stream.For the first example we'll be fusing the value on the stack using [tokio::pin]. We'll also be updating the fuse as it completes with another sleep with a configurable delay. Mimicking the behavior of [Interval].
This is available as the
stack_ticker
example:sh cargo run --example stack_ticker
```rust use async_fuse::Fuse; use std::time::Duration; use tokio::time;
let mut duration = Duration::from_millis(500);
let sleep = Fuse::new(time::sleep(duration)); tokio::pin!(sleep);
let updateduration = Fuse::new(time::sleep(Duration::fromsecs(2))); tokio::pin!(update_duration);
for _ in 0..10usize { tokio::select! { _ = &mut sleep => { println!("Tick"); sleep.set(Fuse::new(time::sleep(duration))); } _ = &mut updateduration => { println!("Tick faster!"); duration = Duration::frommillis(250); } } } ```
For some types it might be easier to fuse the value on the heap. To make this easier, we provide the [Fuse::pin] constructor which provides a fused value which is pinned on the heap.
As a result, it looks pretty similar to the above example.
This is available as the
heap_ticker
example:sh cargo run --example heap_ticker
```rust use async_fuse::Fuse; use std::time::Duration; use tokio::time;
let mut duration = Duration::from_millis(500);
let mut sleep = Fuse::pin(time::sleep(duration)); let mut updateduration = Fuse::pin(time::sleep(Duration::fromsecs(2)));
for _ in 0..10usize { tokio::select! { _ = &mut sleep => { println!("Tick"); sleep.set(Box::pin(time::sleep(duration))); } _ = &mut updateduration => { println!("Tick faster!"); duration = Duration::frommillis(250); } } } ```
The following showcases how we can fuse a trait object. Trait objects are useful since they allow the fused value to change between distinct implementations. The price is that we perform dynamic dispatch which has a small cost.
Also note that because [CoerceUnsized] is not yet stable, we cannot use [Fuse::pin] for convenience and have to pass a pinned box through [Fuse::new].
This is available as the
trait_object_ticker
example:sh cargo run --example trait_object_ticker
```rust use async_fuse::Fuse; use std::future::Future; use std::pin::Pin; use std::time::Duration; use tokio::time;
let mut duration = Duration::from_millis(500);
let mut sleep: Fuse let mut updateduration: Fuse for _ in 0..10usize {
tokio::select! {
_ = &mut sleep => {
println!("Tick");
sleep.set(Box::pin(time::sleep(duration)));
}
_ = &mut updateduration => {
println!("Tick faster!");
duration = Duration::frommillis(250);
}
}
}
``` License: MIT/Apache-2.0