impl get-request tests

This commit is contained in:
M Alghazwi 2025-10-28 09:24:30 +03:00
parent 0f6b61f106
commit 8c5346563b
No known key found for this signature in database
GPG Key ID: 646E567CAD7DB607
5 changed files with 148 additions and 14 deletions

View File

@ -1,6 +1,14 @@
[package] [package]
name = "tor-experiments" name = "tor-experiments"
authors = ["M.A. Alghazwi."]
version = "0.1.0" version = "0.1.0"
edition = "2024" edition = "2024"
license = "MIT OR Apache-2.0"
keywords = ["tor", "privacy", "anonymity"]
[dependencies] [dependencies]
anyhow = "1.0.75"
arti-client = { version = "0.35", features = ["full"] }
arti-ureq = "0.35"
tor-circmgr = "0.35"
tokio = { version = "1", features = ["rt-multi-thread", "macros", "io-util"] }

46
src/get_request_tests.rs Normal file
View File

@ -0,0 +1,46 @@
#[cfg(test)]
mod tests {
use anyhow::Result;
use crate::ureq_builder::*;
use arti_ureq::ureq::Agent;
use crate::tor_client_builder::{choose_tls_provider, make_tor_client};
const TEST_URL: &str = "https://check.torproject.org/api/ip";
const TEST_ONION: &str = "http://2gzyxa5ihm7nsggfxnu52rck2vv4rvmdlkiu3zzui5du4xyclen53wid.onion";
/// Builds all components
fn builds_ureq_agent() -> Result<Agent> {
let tls = choose_tls_provider();
let tor = make_tor_client()?;
let connector = make_connector(tor, tls)?;
let agent = make_ureq_agent(connector, tls)?;
Ok(agent)
}
/// Test: send get request to url over Tor.
/// This test may take a little while on the first run
#[test]
fn fetches_url_over_tor() -> Result<()> {
let agent = builds_ureq_agent()?;
let body = fetch_with_agent(&agent, TEST_URL)?;
//sanity check:
assert!(!body.trim().is_empty(), "empty body from {}", TEST_URL);
Ok(())
}
// /// Test: send get request to .onion over Tor.
// /// This test may take a little while on the first run
// #[test]
// fn fetches_onion_over_tor() -> Result<()> {
// let tls = choose_tls_provider();
// let tor = make_tor_client()?;
// let connector = make_connector(tor, tls)?;
// let agent = make_ureq_agent(connector, tls)?;
//
// let body = fetch_with_agent(&agent, TEST_ONION)?;
// //sanity check:
// assert!(!body.trim().is_empty(), "empty body from {}", TEST_ONION);
// eprintln!("Response: {}", body);
// Ok(())
// }
}

View File

@ -1,14 +1,3 @@
pub fn add(left: u64, right: u64) -> u64 { pub mod ureq_builder;
left + right pub mod get_request_tests;
} pub mod tor_client_builder;
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn it_works() {
let result = add(2, 2);
assert_eq!(result, 4);
}
}

31
src/tor_client_builder.rs Normal file
View File

@ -0,0 +1,31 @@
use anyhow::Context;
use arti_ureq::ureq;
/// Create a Tor client
pub fn make_tor_client() -> anyhow::Result<arti_client::TorClient<arti_ureq::tor_rtcompat::PreferredRuntime>> {
let rt = arti_ureq::tor_rtcompat::PreferredRuntime::create()
.context("Failed to create runtime.")?;
let tor_client = arti_client::TorClient::with_runtime(rt)
.config(make_tor_client_config())
.create_unbootstrapped()
.context("Error creating Tor Client.")?;
// let tor_client = arti_client::TorClient::builder()
// .bootstrap_behavior(BootstrapBehavior::OnDemand)
// .create_unbootstrapped()?;
Ok(tor_client)
}
/// Build a default Tor client config.
/// For our purposes, we can use the default now.
pub fn make_tor_client_config() -> arti_client::config::TorClientConfig {
let config = arti_client::config::TorClientConfig::default();
config
}
/// returns TLS provider to use, default for now
pub fn choose_tls_provider() -> ureq::tls::TlsProvider {
arti_ureq::get_default_tls_provider()
}

60
src/ureq_builder.rs Normal file
View File

@ -0,0 +1,60 @@
use anyhow::{Context, Result};
use arti_ureq::ureq;
use arti_ureq::ureq::tls::RootCerts;
/// Build a Connector from the Tor client and chosen TLS provider.
/// This is a ureq connector, and here we are giving it out already created tor client + tls
pub fn make_connector(
tor_client: arti_client::TorClient<arti_ureq::tor_rtcompat::PreferredRuntime>,
tls_provider: ureq::tls::TlsProvider,
) -> Result<arti_ureq::Connector<arti_ureq::tor_rtcompat::PreferredRuntime>> {
let builder = arti_ureq::Connector::<arti_ureq::tor_rtcompat::PreferredRuntime>::builder()
.context("Failed to create ConnectorBuilder")?
.tor_client(tor_client)
.tls_provider(tls_provider);
Ok(builder.build().context("Failed to build Connector")?)
}
/// Build a `ureq::Agent`
pub fn make_ureq_agent(
connector: arti_ureq::Connector<arti_ureq::tor_rtcompat::PreferredRuntime>,
tls_provider: ureq::tls::TlsProvider,
) -> Result<ureq::Agent> {
// Build ureq TLS config
let ureq_tls_config = ureq::tls::TlsConfig::builder()
.root_certs(RootCerts::PlatformVerifier)
.provider(tls_provider)
.build();
// Build ureq config.
let ureq_config = ureq::config::Config::builder()
.user_agent("arti-ureq-custom-user-agent")
.tls_config(ureq_tls_config)
.build();
connector
.agent_with_ureq_config(ureq_config)
.context("Failed to create ureq agent.")
}
/// Fetch a URL via the provided ureq agent, returning the body as String.
pub fn fetch_with_agent(agent: &ureq::Agent, url: &str) -> Result<String> {
println!("[+] Making request to: {}", url);
let mut request = agent
.get(url)
.call()
.context("Failed to make request.")?;
println!("[+] Response status: {}", request.status());
let response = request
.body_mut()
.read_to_string()
.context("Failed to read body.")?;
println!("Response: {response}");
Ok(response)
}