async-fuse

Documentation Crates Actions Status

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 */ } } ```

Features

Fusing on the stack

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); } } } ```

Fusing on the heap

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); } } } ```

Fusing trait objects

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: Fusesecs(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); } } } ```

License: MIT/Apache-2.0