Skip to content

Commit 7cae4b8

Browse files
committed
refactor(async)!: remove async-std dependency, allow custom runtime
1 parent 1f4acad commit 7cae4b8

File tree

3 files changed

+60
-12
lines changed

3 files changed

+60
-12
lines changed

Cargo.toml

+7-3
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,9 @@ hex = { version = "0.2", package = "hex-conservative" }
2323
log = "^0.4"
2424
minreq = { version = "2.11.0", features = ["json-using-serde"], optional = true }
2525
reqwest = { version = "0.11", features = ["json"], default-features = false, optional = true }
26-
async-std = { version = "1.13.0", optional = true }
26+
27+
# default async runtime
28+
tokio = { version = "1.38", features = ["time"], optional = true }
2729

2830
[dev-dependencies]
2931
serde_json = "1.0"
@@ -32,13 +34,15 @@ electrsd = { version = "0.28.0", features = ["legacy", "esplora_a33e97e1", "bitc
3234
lazy_static = "1.4.0"
3335

3436
[features]
35-
default = ["blocking", "async", "async-https"]
37+
default = ["blocking", "async", "async-https", "tokio"]
3638
blocking = ["minreq", "minreq/proxy"]
3739
blocking-https = ["blocking", "minreq/https"]
3840
blocking-https-rustls = ["blocking", "minreq/https-rustls"]
3941
blocking-https-native = ["blocking", "minreq/https-native"]
4042
blocking-https-bundled = ["blocking", "minreq/https-bundled"]
41-
async = ["async-std", "reqwest", "reqwest/socks"]
43+
44+
tokio = ["dep:tokio"]
45+
async = ["reqwest", "reqwest/socks", "tokio?/time"]
4246
async-https = ["async", "reqwest/default-tls"]
4347
async-https-native = ["async", "reqwest/native-tls"]
4448
async-https-rustls = ["async", "reqwest/rustls-tls"]

src/async.rs

+26-5
Original file line numberDiff line numberDiff line change
@@ -11,8 +11,8 @@
1111

1212
//! Esplora by way of `reqwest` HTTP client.
1313
14-
use async_std::task;
1514
use std::collections::HashMap;
15+
use std::marker::PhantomData;
1616
use std::str::FromStr;
1717

1818
use bitcoin::consensus::{deserialize, serialize, Decodable, Encodable};
@@ -33,16 +33,19 @@ use crate::{
3333
};
3434

3535
#[derive(Debug, Clone)]
36-
pub struct AsyncClient {
36+
pub struct AsyncClient<S = DefaultSleeper> {
3737
/// The URL of the Esplora Server.
3838
url: String,
3939
/// The inner [`reqwest::Client`] to make HTTP requests.
4040
client: Client,
4141
/// Number of times to retry a request
4242
max_retries: usize,
43+
44+
/// Marker for the type of sleeper used
45+
marker: PhantomData<S>,
4346
}
4447

45-
impl AsyncClient {
48+
impl<S: Sleeper> AsyncClient<S> {
4649
/// Build an async client from a builder
4750
pub fn from_builder(builder: Builder) -> Result<Self, Error> {
4851
let mut client_builder = Client::builder();
@@ -73,15 +76,16 @@ impl AsyncClient {
7376
url: builder.base_url,
7477
client: client_builder.build()?,
7578
max_retries: builder.max_retries,
79+
marker: PhantomData,
7680
})
7781
}
7882

79-
/// Build an async client from the base url and [`Client`]
8083
pub fn from_client(url: String, client: Client) -> Self {
8184
AsyncClient {
8285
url,
8386
client,
8487
max_retries: crate::DEFAULT_MAX_RETRIES,
88+
marker: PhantomData,
8589
}
8690
}
8791

@@ -434,7 +438,7 @@ impl AsyncClient {
434438
loop {
435439
match self.client.get(url).send().await? {
436440
resp if attempts < self.max_retries && is_status_retryable(resp.status()) => {
437-
task::sleep(delay).await;
441+
S::sleep(delay).await;
438442
attempts += 1;
439443
delay *= 2;
440444
}
@@ -447,3 +451,20 @@ impl AsyncClient {
447451
fn is_status_retryable(status: reqwest::StatusCode) -> bool {
448452
RETRYABLE_ERROR_CODES.contains(&status.as_u16())
449453
}
454+
455+
pub trait Sleeper: 'static {
456+
type Sleep: std::future::Future<Output = ()>;
457+
fn sleep(dur: std::time::Duration) -> Self::Sleep;
458+
}
459+
460+
#[derive(Debug, Clone, Copy)]
461+
pub struct DefaultSleeper;
462+
463+
#[cfg(any(test, feature = "tokio"))]
464+
impl Sleeper for DefaultSleeper {
465+
type Sleep = tokio::time::Sleep;
466+
467+
fn sleep(dur: std::time::Duration) -> Self::Sleep {
468+
tokio::time::sleep(dur)
469+
}
470+
}

src/lib.rs

+27-4
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@
2626
//! Here is an example of how to create an asynchronous client.
2727
//!
2828
//! ```no_run
29-
//! # #[cfg(feature = "async")]
29+
//! # #[cfg(all(feature = "async", feature = "tokio"))]
3030
//! # {
3131
//! use esplora_client::Builder;
3232
//! let builder = Builder::new("https://blockstream.info/testnet/api");
@@ -71,8 +71,10 @@ use std::fmt;
7171
use std::num::TryFromIntError;
7272
use std::time::Duration;
7373

74-
pub mod api;
74+
#[cfg(feature = "async")]
75+
use r#async::Sleeper;
7576

77+
pub mod api;
7678
#[cfg(feature = "async")]
7779
pub mod r#async;
7880
#[cfg(feature = "blocking")]
@@ -178,11 +180,18 @@ impl Builder {
178180
BlockingClient::from_builder(self)
179181
}
180182

181-
// Build an asynchronous client from builder
182-
#[cfg(feature = "async")]
183+
/// Build an asynchronous client from builder
184+
#[cfg(all(feature = "async", feature = "tokio"))]
183185
pub fn build_async(self) -> Result<AsyncClient, Error> {
184186
AsyncClient::from_builder(self)
185187
}
188+
189+
/// Build an asynchronous client from builder where the returned client uses a
190+
/// user-defined [`Sleeper`].
191+
#[cfg(feature = "async")]
192+
pub fn build_async_with_sleeper<S: Sleeper>(self) -> Result<AsyncClient<S>, Error> {
193+
AsyncClient::from_builder(self)
194+
}
186195
}
187196

188197
/// Errors that can happen during a request to `Esplora` servers.
@@ -320,8 +329,15 @@ mod test {
320329
let blocking_client = builder.build_blocking();
321330

322331
let builder_async = Builder::new(&format!("http://{}", esplora_url));
332+
333+
#[cfg(feature = "tokio")]
323334
let async_client = builder_async.build_async().unwrap();
324335

336+
#[cfg(not(feature = "tokio"))]
337+
let async_client = builder_async
338+
.build_async_with_sleeper::<r#async::DefaultSleeper>()
339+
.unwrap();
340+
325341
(blocking_client, async_client)
326342
}
327343

@@ -992,4 +1008,11 @@ mod test {
9921008
let tx_async = async_client.get_tx(&txid).await.unwrap();
9931009
assert_eq!(tx, tx_async);
9941010
}
1011+
1012+
#[cfg(all(feature = "async", feature = "tokio"))]
1013+
#[test]
1014+
fn use_builder_with_tokio_as_normal() {
1015+
let builder = Builder::new("https://blockstream.info/testnet/api");
1016+
let _client = builder.build_async().unwrap();
1017+
}
9951018
}

0 commit comments

Comments
 (0)