Skip to content

Commit

Permalink
refactor(async)!: remove async-std dependency, allow custom runtime
Browse files Browse the repository at this point in the history
  • Loading branch information
praveenperera committed Nov 10, 2024
1 parent 1f4acad commit 7cae4b8
Show file tree
Hide file tree
Showing 3 changed files with 60 additions and 12 deletions.
10 changes: 7 additions & 3 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,9 @@ hex = { version = "0.2", package = "hex-conservative" }
log = "^0.4"
minreq = { version = "2.11.0", features = ["json-using-serde"], optional = true }
reqwest = { version = "0.11", features = ["json"], default-features = false, optional = true }
async-std = { version = "1.13.0", optional = true }

# default async runtime
tokio = { version = "1.38", features = ["time"], optional = true }

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

[features]
default = ["blocking", "async", "async-https"]
default = ["blocking", "async", "async-https", "tokio"]
blocking = ["minreq", "minreq/proxy"]
blocking-https = ["blocking", "minreq/https"]
blocking-https-rustls = ["blocking", "minreq/https-rustls"]
blocking-https-native = ["blocking", "minreq/https-native"]
blocking-https-bundled = ["blocking", "minreq/https-bundled"]
async = ["async-std", "reqwest", "reqwest/socks"]

tokio = ["dep:tokio"]
async = ["reqwest", "reqwest/socks", "tokio?/time"]
async-https = ["async", "reqwest/default-tls"]
async-https-native = ["async", "reqwest/native-tls"]
async-https-rustls = ["async", "reqwest/rustls-tls"]
Expand Down
31 changes: 26 additions & 5 deletions src/async.rs
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,8 @@

//! Esplora by way of `reqwest` HTTP client.
use async_std::task;
use std::collections::HashMap;
use std::marker::PhantomData;
use std::str::FromStr;

use bitcoin::consensus::{deserialize, serialize, Decodable, Encodable};
Expand All @@ -33,16 +33,19 @@ use crate::{
};

#[derive(Debug, Clone)]
pub struct AsyncClient {
pub struct AsyncClient<S = DefaultSleeper> {
/// The URL of the Esplora Server.
url: String,
/// The inner [`reqwest::Client`] to make HTTP requests.
client: Client,
/// Number of times to retry a request
max_retries: usize,

/// Marker for the type of sleeper used
marker: PhantomData<S>,
}

impl AsyncClient {
impl<S: Sleeper> AsyncClient<S> {
/// Build an async client from a builder
pub fn from_builder(builder: Builder) -> Result<Self, Error> {
let mut client_builder = Client::builder();
Expand Down Expand Up @@ -73,15 +76,16 @@ impl AsyncClient {
url: builder.base_url,
client: client_builder.build()?,
max_retries: builder.max_retries,
marker: PhantomData,
})
}

/// Build an async client from the base url and [`Client`]
pub fn from_client(url: String, client: Client) -> Self {
AsyncClient {
url,
client,
max_retries: crate::DEFAULT_MAX_RETRIES,
marker: PhantomData,
}
}

Expand Down Expand Up @@ -434,7 +438,7 @@ impl AsyncClient {
loop {
match self.client.get(url).send().await? {
resp if attempts < self.max_retries && is_status_retryable(resp.status()) => {
task::sleep(delay).await;
S::sleep(delay).await;
attempts += 1;
delay *= 2;
}
Expand All @@ -447,3 +451,20 @@ impl AsyncClient {
fn is_status_retryable(status: reqwest::StatusCode) -> bool {
RETRYABLE_ERROR_CODES.contains(&status.as_u16())
}

pub trait Sleeper: 'static {
type Sleep: std::future::Future<Output = ()>;
fn sleep(dur: std::time::Duration) -> Self::Sleep;
}

#[derive(Debug, Clone, Copy)]
pub struct DefaultSleeper;

#[cfg(any(test, feature = "tokio"))]
impl Sleeper for DefaultSleeper {
type Sleep = tokio::time::Sleep;

fn sleep(dur: std::time::Duration) -> Self::Sleep {
tokio::time::sleep(dur)
}
}
31 changes: 27 additions & 4 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@
//! Here is an example of how to create an asynchronous client.
//!
//! ```no_run
//! # #[cfg(feature = "async")]
//! # #[cfg(all(feature = "async", feature = "tokio"))]
//! # {
//! use esplora_client::Builder;
//! let builder = Builder::new("https://blockstream.info/testnet/api");
Expand Down Expand Up @@ -71,8 +71,10 @@ use std::fmt;
use std::num::TryFromIntError;
use std::time::Duration;

pub mod api;
#[cfg(feature = "async")]
use r#async::Sleeper;

pub mod api;
#[cfg(feature = "async")]
pub mod r#async;
#[cfg(feature = "blocking")]
Expand Down Expand Up @@ -178,11 +180,18 @@ impl Builder {
BlockingClient::from_builder(self)
}

// Build an asynchronous client from builder
#[cfg(feature = "async")]
/// Build an asynchronous client from builder
#[cfg(all(feature = "async", feature = "tokio"))]
pub fn build_async(self) -> Result<AsyncClient, Error> {
AsyncClient::from_builder(self)
}

/// Build an asynchronous client from builder where the returned client uses a
/// user-defined [`Sleeper`].
#[cfg(feature = "async")]
pub fn build_async_with_sleeper<S: Sleeper>(self) -> Result<AsyncClient<S>, Error> {
AsyncClient::from_builder(self)
}
}

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

let builder_async = Builder::new(&format!("http://{}", esplora_url));

#[cfg(feature = "tokio")]
let async_client = builder_async.build_async().unwrap();

#[cfg(not(feature = "tokio"))]
let async_client = builder_async
.build_async_with_sleeper::<r#async::DefaultSleeper>()
.unwrap();

(blocking_client, async_client)
}

Expand Down Expand Up @@ -992,4 +1008,11 @@ mod test {
let tx_async = async_client.get_tx(&txid).await.unwrap();
assert_eq!(tx, tx_async);
}

#[cfg(all(feature = "async", feature = "tokio"))]
#[test]
fn use_builder_with_tokio_as_normal() {
let builder = Builder::new("https://blockstream.info/testnet/api");
let _client = builder.build_async().unwrap();
}
}

0 comments on commit 7cae4b8

Please sign in to comment.