🗑 async-dropper

async-dropper is probably the least-worst ad-hoc AsyncDrop implementation you've seen, and it works in two ways:

The code in this crate was most directly inspired by this StackOverflow thread on Async Drop and many other conversations:

Install

console cargo add async-dropper # by default, tokio is enabled cargo add async-dropper --features async-std # use async-std

If you're editing Cargo.toml by hand:

```toml [dependencies] async-dropper = "0.1"

async-dropper = { version = "0.1", features = [ "async-std" ] }

```

Warning async-dropper does not allow using both async-std and tokio features at the same time (see the FAQ below).

Quickstart

async_dropper::simple

To use the "simple" version which uses a wrapper struct (AsyncDropper<T>), see examples/async_drop_simple.rs:

```rust use std::{ result::Result, time::Duration, };

use asyncdroppersimple::{AsyncDrop, AsyncDropper}; use asynctrait::asynctrait;

// NOTE: this example is rooted in crates/async-dropper

/// This object will be async-dropped (which must be wrapped in AsyncDropper)

[derive(Default)]

struct AsyncThing(String);

[async_trait]

impl AsyncDrop for AsyncThing { async fn asyncdrop(&mut self) { eprintln!("async dropping [{}]!", self.0); tokio::time::sleep(Duration::fromsecs(2)).await; eprintln!("dropped [{}]!", self.0); } }

[tokio::main]

async fn main() -> Result<(), Box> { { let exampleobj = AsyncDropper::new(AsyncThing(String::from("test"))); eprintln!("here comes the (async) drop"); // drop will be triggered here, and it will take however long it takes // you could also call drop(_example_obj) }

Ok(())

}

```

You can run the example and see the output:

console cargo run --example async-drop-simple --features=tokio

async_dropper::derive

The derive macro is an novel (and possibly foolhardy) attempt to implement AsyncDrop without actually wrapping the existing struct.

async_dropper::derive uses Default and PartialEq to check if the struct in question is equivalent to it's default.

For this approach to work well your T should have cheap-to-create Defaults, and comparing a default value to an existing value should meaningfully differ (and identify an object that is no longer in use). Please think thoroughly about whether this model works for your use case.

For an example, see examples/async_drop.rs:

```rust use std::{ result::Result, time::Duration, };

use asyncdropper::derive::AsyncDrop; use asynctrait::async_trait;

/// This object will be async-dropped /// /// Objects that are dropped must implement [Default] and [PartialEq] /// (so make members optional, hide them behind Rc/Arc as necessary)

[derive(Debug, Default, PartialEq, Eq, AsyncDrop)]

struct AsyncThing(String);

/// Implementation of [AsyncDrop] that specifies the actual behavior

[async_trait]

impl AsyncDrop for AsyncThing { async fn asyncdrop(&mut self) -> Result<(), AsyncDropError> { // Wait 2 seconds then "succeed" eprintln!("async dropping [{}]!", self.0); tokio::time::sleep(Duration::fromsecs(2)).await; eprintln!("dropped [{}]!", self.0); Ok(()) }

fn drop_timeout(&self) -> Duration {
    Duration::from_secs(5) // extended from default 3 seconds
}

// NOTE: below was not implemented since we want the default of DropFailAction::Contineue
// fn drop_fail_action(&self) -> DropFailAction;

}

[tokio::main]

async fn main() -> Result<(), Box> { { let exampleobj = AsyncThing(String::from("test")); eprintln!("here comes the (async) drop"); // drop will be triggered here // you could also call drop(_example_obj) }

Ok(())

} ```

You can run the example and see the output:

console cargo run --example async-drop --features=tokio

Supported environments

async-dropper works with the following async environments:

| Name | Supported? | |-----------------------------------|------------| | Async w/ tokio | ✅ | | Async w/ async-std | ✅ |

FAQ

Why does async-dropper assume that I'm using either async-std or tokio

Because you probably are. If this is a problem for you, it can be changed, please file an issue.

Development

To get started working on developing situwatiion, run the following just targets:

console just setup build

To check that your changes are fine, you'll probably want to run:

console just test

If you want to see the full list of targets available that you can run just without any arguments.

console just

There are a few useful targets like just build-watch which will continuously build the project thanks to cargo watch.

Releasing

From the top level of this repository:

console PUBLISH_CRATE=yes PKG=<crate name> just release <version>

For example, to create the next semver patch release for async-dropper-simple:

console PUBLISH_CRATE=yes PKG=async-dropper-simple just release patch

Contributing

Contributions are welcome! If you find a bug or an impovement that should be included in async-dropper, create an issue or open a pull request.