Skip to content

Commit ca6cfe8

Browse files
Merge pull request #4 from ValuedMammal/review/103
Review 103
2 parents 7c27c63 + 8dce43f commit ca6cfe8

File tree

2 files changed

+51
-77
lines changed

2 files changed

+51
-77
lines changed

src/async.rs

+21-12
Original file line numberDiff line numberDiff line change
@@ -11,9 +11,11 @@
1111

1212
//! Esplora by way of `reqwest` HTTP client.
1313
14+
use async_trait::async_trait;
1415
use std::collections::HashMap;
1516
use std::marker::PhantomData;
1617
use std::str::FromStr;
18+
use std::time::Duration;
1719

1820
use bitcoin::consensus::{deserialize, serialize, Decodable, Encodable};
1921
use bitcoin::hashes::{sha256, Hash};
@@ -28,8 +30,8 @@ use log::{debug, error, info, trace};
2830
use reqwest::{header, Client, Response};
2931

3032
use crate::{
31-
BlockStatus, BlockSummary, Builder, DefaultSleeper, Error, MerkleProof, OutputStatus, Tx,
32-
TxStatus, BASE_BACKOFF_MILLIS, RETRYABLE_ERROR_CODES,
33+
BlockStatus, BlockSummary, Builder, Error, MerkleProof, OutputStatus, Tx, TxStatus,
34+
BASE_BACKOFF_MILLIS, RETRYABLE_ERROR_CODES,
3335
};
3436

3537
#[derive(Debug, Clone)]
@@ -41,12 +43,12 @@ pub struct AsyncClient<S = DefaultSleeper> {
4143
/// Number of times to retry a request
4244
max_retries: usize,
4345

44-
runtime: PhantomData<S>,
46+
marker: PhantomData<S>,
4547
}
4648

4749
impl<S: Sleeper> AsyncClient<S> {
4850
/// Build an async client from a builder
49-
pub fn from_builder(builder: Builder<S>) -> Result<Self, Error> {
51+
pub fn from_builder(builder: Builder) -> Result<Self, Error> {
5052
let mut client_builder = Client::builder();
5153

5254
#[cfg(not(target_arch = "wasm32"))]
@@ -75,7 +77,7 @@ impl<S: Sleeper> AsyncClient<S> {
7577
url: builder.base_url,
7678
client: client_builder.build()?,
7779
max_retries: builder.max_retries,
78-
runtime: PhantomData,
80+
marker: PhantomData,
7981
})
8082
}
8183

@@ -84,7 +86,7 @@ impl<S: Sleeper> AsyncClient<S> {
8486
url,
8587
client,
8688
max_retries: crate::DEFAULT_MAX_RETRIES,
87-
runtime: PhantomData,
89+
marker: PhantomData,
8890
}
8991
}
9092

@@ -451,15 +453,22 @@ fn is_status_retryable(status: reqwest::StatusCode) -> bool {
451453
RETRYABLE_ERROR_CODES.contains(&status.as_u16())
452454
}
453455

454-
#[async_trait::async_trait]
456+
/// Trait that defines the ability to sleep within an async runtime
457+
#[async_trait]
455458
pub trait Sleeper {
456-
async fn sleep(duration: std::time::Duration);
459+
/// Wait until the specified `duration` has elapsed
460+
async fn sleep(duration: Duration);
457461
}
458462

459-
#[cfg(feature = "tokio")]
460-
#[async_trait::async_trait]
461-
impl Sleeper for crate::DefaultSleeper {
462-
async fn sleep(duration: std::time::Duration) {
463+
/// Default sleeper. Note this may only be used as a [`Sleeper`] implementation
464+
/// if the "tokio" feature is enabled.
465+
#[derive(Debug, Clone, Copy)]
466+
pub struct DefaultSleeper;
467+
468+
#[cfg(any(test, feature = "tokio"))]
469+
#[async_trait]
470+
impl Sleeper for DefaultSleeper {
471+
async fn sleep(duration: Duration) {
463472
tokio::time::sleep(duration).await;
464473
}
465474
}

src/lib.rs

+30-65
Original file line numberDiff line numberDiff line change
@@ -66,13 +66,15 @@
6666
6767
#![allow(clippy::result_large_err)]
6868

69+
use std::collections::HashMap;
6970
use std::fmt;
7071
use std::num::TryFromIntError;
7172
use std::time::Duration;
72-
use std::{collections::HashMap, marker::PhantomData};
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")]
@@ -111,7 +113,7 @@ pub fn convert_fee_rate(target: usize, estimates: HashMap<u16, f64>) -> Option<f
111113
}
112114

113115
#[derive(Debug, Clone)]
114-
pub struct Builder<S = DefaultSleeper> {
116+
pub struct Builder {
115117
/// The URL of the Esplora server.
116118
pub base_url: String,
117119
/// Optional URL of the proxy to use to make requests to the Esplora server
@@ -133,13 +135,8 @@ pub struct Builder<S = DefaultSleeper> {
133135
pub headers: HashMap<String, String>,
134136
/// Max retries
135137
pub max_retries: usize,
136-
/// Async runtime, trait must implement `sleep` function, default is `tokio`
137-
marker: PhantomData<S>,
138138
}
139139

140-
#[derive(Debug, Clone, Copy)]
141-
pub struct DefaultSleeper;
142-
143140
impl Builder {
144141
/// Instantiate a new builder
145142
pub fn new(base_url: &str) -> Self {
@@ -149,38 +146,9 @@ impl Builder {
149146
timeout: None,
150147
headers: HashMap::new(),
151148
max_retries: DEFAULT_MAX_RETRIES,
152-
marker: PhantomData,
153-
}
154-
}
155-
156-
/// Build a blocking client from builder
157-
#[cfg(feature = "blocking")]
158-
pub fn build_blocking(self) -> BlockingClient {
159-
BlockingClient::from_builder(self)
160-
}
161-
}
162-
163-
#[cfg(feature = "async")]
164-
impl<S: r#async::Sleeper> Builder<S> {
165-
/// Instantiate a new builder, with a custom runtime
166-
pub fn new_custom_runtime(base_url: &str) -> Self {
167-
Builder {
168-
base_url: base_url.to_string(),
169-
proxy: None,
170-
timeout: None,
171-
headers: HashMap::new(),
172-
max_retries: DEFAULT_MAX_RETRIES,
173-
marker: PhantomData,
174149
}
175150
}
176151

177-
// Build an asynchronous client from builder
178-
pub fn build_async(self) -> Result<AsyncClient<S>, Error> {
179-
AsyncClient::from_builder(self)
180-
}
181-
}
182-
183-
impl<S> Builder<S> {
184152
/// Set the proxy of the builder
185153
pub fn proxy(mut self, proxy: &str) -> Self {
186154
self.proxy = Some(proxy.to_string());
@@ -205,6 +173,25 @@ impl<S> Builder<S> {
205173
self.max_retries = count;
206174
self
207175
}
176+
177+
/// Build a blocking client from builder
178+
#[cfg(feature = "blocking")]
179+
pub fn build_blocking(self) -> BlockingClient {
180+
BlockingClient::from_builder(self)
181+
}
182+
183+
/// Build an asynchronous client from builder
184+
#[cfg(feature = "tokio")]
185+
pub fn build_async(self) -> Result<AsyncClient, Error> {
186+
AsyncClient::from_builder(self)
187+
}
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+
}
208195
}
209196

210197
/// Errors that can happen during a request to `Esplora` servers.
@@ -284,6 +271,7 @@ mod test {
284271
bitcoind::bitcoincore_rpc::json::AddressType, bitcoind::bitcoincore_rpc::RpcApi,
285272
electrum_client::ElectrumApi,
286273
},
274+
r#async::DefaultSleeper,
287275
std::time::Duration,
288276
tokio::sync::OnceCell,
289277
};
@@ -342,7 +330,9 @@ mod test {
342330
let blocking_client = builder.build_blocking();
343331

344332
let builder_async = Builder::new(&format!("http://{}", esplora_url));
345-
let async_client = builder_async.build_async().unwrap();
333+
let async_client = builder_async
334+
.build_async_with_sleeper::<DefaultSleeper>()
335+
.unwrap();
346336

347337
(blocking_client, async_client)
348338
}
@@ -1015,35 +1005,10 @@ mod test {
10151005
assert_eq!(tx, tx_async);
10161006
}
10171007

1018-
#[cfg(all(feature = "async", feature = "tokio"))]
1008+
#[cfg(feature = "tokio")]
10191009
#[test]
10201010
fn use_builder_with_tokio_as_normal() {
10211011
let builder = Builder::new("https://blockstream.info/testnet/api");
1022-
let client = builder.build_async();
1023-
assert!(client.is_ok());
1024-
}
1025-
1026-
#[cfg(all(feature = "async", not(feature = "tokio")))]
1027-
mod custom_async_runtime {
1028-
use super::*;
1029-
use crate::r#async::Sleeper;
1030-
1031-
struct TestRuntime;
1032-
1033-
#[async_trait::async_trait]
1034-
impl Sleeper for TestRuntime {
1035-
async fn sleep(duration: Duration) {
1036-
tokio::time::sleep(duration).await;
1037-
}
1038-
}
1039-
1040-
#[test]
1041-
fn use_with_custom_runtime() {
1042-
let builder =
1043-
Builder::<TestRuntime>::new_custom_runtime("https://blockstream.info/testnet/api");
1044-
1045-
let client = builder.build_async();
1046-
assert!(client.is_ok());
1047-
}
1012+
let _client = builder.build_async().unwrap();
10481013
}
10491014
}

0 commit comments

Comments
 (0)