Skip to content

Commit f32c8d1

Browse files
committed
feat(tor): add new AsyncTorClient
- feat(tor): add new async client, `AsyncTorClient`, which uses `arti-client` to establish Tor connections, and `hyper` as HTTP client over custom Tor anonymized data stream. - feat(tor): implements the common methods: `get_response`, `get_response_json`, `get_response_hex` and their `opt` versions too.
1 parent ac64e04 commit f32c8d1

File tree

5 files changed

+1117
-6
lines changed

5 files changed

+1117
-6
lines changed

Cargo.toml

+40-2
Original file line numberDiff line numberDiff line change
@@ -24,21 +24,59 @@ 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 }
2626

27+
arti-client = { version = "0.21.0", default-features = false, optional = true }
28+
tor-rtcompat = { version = "0.21.0", default-features = false, optional = true }
29+
hyper = { version = "1.4.1", features = ["client", "http1"], default-features = false, optional = true }
30+
hyper-util = { version = "0.1.7", features = ["tokio"], default-features = false, optional = true }
31+
tokio = { version = "1.38.1", optional = true }
32+
http-body-util = { version = "0.1.2", optional = true}
33+
http = { version = "1.1.0", optional = true }
34+
serde_json = { version = "1.0.127", optional = true }
35+
tokio-rustls = { version = "0.26.0", default-features = false, features = [
36+
"logging",
37+
"tls12",
38+
"ring",
39+
], optional = true }
40+
webpki-roots = { version = "0.26.3", optional = true }
41+
rustls-pki-types = { version = "1.8.0", optional = true }
42+
2743
[dev-dependencies]
28-
serde_json = "1.0"
2944
tokio = { version = "1.20.1", features = ["full"] }
3045
electrsd = { version = "0.28.0", features = ["legacy", "esplora_a33e97e1", "bitcoind_25_0"] }
3146
lazy_static = "1.4.0"
3247

3348
[features]
34-
default = ["blocking", "async", "async-https"]
49+
default = ["blocking", "async", "async-https", "async-tor"]
3550
blocking = ["minreq", "minreq/proxy"]
3651
blocking-https = ["blocking", "minreq/https"]
3752
blocking-https-rustls = ["blocking", "minreq/https-rustls"]
3853
blocking-https-native = ["blocking", "minreq/https-native"]
3954
blocking-https-bundled = ["blocking", "minreq/https-bundled"]
55+
4056
async = ["reqwest", "reqwest/socks"]
4157
async-https = ["async", "reqwest/default-tls"]
4258
async-https-native = ["async", "reqwest/native-tls"]
4359
async-https-rustls = ["async", "reqwest/rustls-tls"]
4460
async-https-rustls-manual-roots = ["async", "reqwest/rustls-tls-manual-roots"]
61+
62+
async-tor = [
63+
"dep:arti-client",
64+
"arti-client/tokio",
65+
"arti-client/onion-service-client",
66+
"arti-client/native-tls",
67+
68+
"dep:tor-rtcompat",
69+
"tor-rtcompat/tokio",
70+
71+
"dep:hyper",
72+
"dep:hyper-util",
73+
"dep:tokio",
74+
"dep:http-body-util",
75+
"dep:http",
76+
"dep:serde_json",
77+
"dep:tokio-rustls",
78+
"dep:webpki-roots",
79+
"dep:rustls-pki-types"
80+
]
81+
async-tor-https-native = ["async-tor", "arti-client/native-tls"]
82+
async-tor-https-rustls = ["async-tor", "arti-client/rustls"]

examples/tor.rs

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
use bitcoin::{consensus::encode::deserialize_hex, Transaction};
2+
use esplora_client::{r#async_tor::AsyncTorClient, Builder};
3+
4+
extern crate esplora_client;
5+
6+
// const MEMPOOL_SPACE_API: &str = "https://mempool.space/api";
7+
const MEMPOOL_SPACE_API: &str = "https://blockstream.info/api";
8+
9+
#[tokio::main]
10+
async fn main() {
11+
let builder = Builder::new(MEMPOOL_SPACE_API);
12+
let esplora_client = AsyncTorClient::from_builder(builder).await.unwrap();
13+
14+
let raw_tx = "01000000010000000000000000000000000000000000000000000000000000000000000000ffffffff4d04ffff001d0104455468652054696d65732030332f4a616e2f32303039204368616e63656c6c6f72206f6e206272696e6b206f66207365636f6e64206261696c6f757420666f722062616e6b73ffffffff0100f2052a01000000434104678afdb0fe5548271967f1a67130b7105cd6a828e03909a67962e0ea1f61deb649f6bc3f4cef38c4f35504e51ec112de5c384df7ba0b8d578a4c702b6bf11d5fac00000000";
15+
let tx: Transaction = deserialize_hex(raw_tx).unwrap();
16+
esplora_client.broadcast(&tx).await.unwrap();
17+
18+
print!(
19+
"successfully broadcasted transaction, with txid: {:?}",
20+
tx.compute_txid()
21+
);
22+
23+
// let tx_id =
24+
// Txid::from_str("4a5e1e4baab89f3a32518a88c31bc87f618f76673e2cc77ab2127b7afdeda33b").unwrap();
25+
// let tx = esplora_client.get_tx(&tx_id).await.unwrap().unwrap();
26+
27+
// println!("successfully fetched the transaction {:?}", tx);
28+
}

src/async.rs

+5-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111

1212
//! Esplora by way of `reqwest` HTTP client.
1313
14+
use core::str;
1415
use std::collections::HashMap;
1516
use std::str::FromStr;
1617

@@ -28,6 +29,7 @@ use reqwest::{header, Client};
2829

2930
use crate::{BlockStatus, BlockSummary, Builder, Error, MerkleProof, OutputStatus, Tx, TxStatus};
3031

32+
#[cfg(feature = "async")]
3133
#[derive(Debug, Clone)]
3234
pub struct AsyncClient {
3335
/// The URL of the Esplora Server.
@@ -36,6 +38,7 @@ pub struct AsyncClient {
3638
client: Client,
3739
}
3840

41+
#[cfg(feature = "async")]
3942
impl AsyncClient {
4043
/// Build an async client from a builder
4144
pub fn from_builder(builder: Builder) -> Result<Self, Error> {
@@ -84,7 +87,8 @@ impl AsyncClient {
8487
/// [`bitcoin::consensus::Decodable`] deserialization.
8588
async fn get_response<T: Decodable>(&self, path: &str) -> Result<T, Error> {
8689
let url = format!("{}{}", self.url, path);
87-
let response = self.client.get(url).send().await?;
90+
let request = self.client.get(url);
91+
let response = request.send().await?;
8892

8993
match response.status().is_success() {
9094
true => Ok(deserialize::<T>(&response.bytes().await?)?),

0 commit comments

Comments
 (0)