use std::sync::Arc;
use futures::{SinkExt, StreamExt};
use rustls::pki_types::ServerName;
use smolmix::Tunnel;
use tokio_tungstenite::tungstenite::Message;
use tracing::info;
type BoxError = Box<dyn std::error::Error + Send + Sync>;
const WS_HOST: &str = "ws.postman-echo.com";
const WS_PATH: &str = "/raw";
const ECHO_MSG: &str = "Hello from the Nym mixnet!";
fn tls_connector() -> tokio_rustls::TlsConnector {
let mut root_store = rustls::RootCertStore::empty();
root_store.extend(webpki_roots::TLS_SERVER_ROOTS.iter().cloned());
let config = rustls::ClientConfig::builder()
.with_root_certificates(root_store)
.with_no_client_auth();
tokio_rustls::TlsConnector::from(Arc::new(config))
}
#[tokio::main]
async fn main() -> Result<(), BoxError> {
nym_bin_common::logging::setup_tracing_logger();
rustls::crypto::ring::default_provider()
.install_default()
.expect("Failed to install rustls crypto provider");
let addr = tokio::net::lookup_host(format!("{WS_HOST}:443"))
.await?
.next()
.ok_or("DNS resolution failed")?;
info!("Resolved {WS_HOST} -> {addr}");
let connector = tls_connector();
let domain = ServerName::try_from(WS_HOST)?.to_owned();
info!("Connecting via clearnet...");
let clearnet_start = tokio::time::Instant::now();
let clearnet_tcp = tokio::net::TcpStream::connect(addr).await?;
let clearnet_tls = connector.connect(domain.clone(), clearnet_tcp).await?;
let (mut clearnet_ws, _) =
tokio_tungstenite::client_async(format!("wss://{WS_HOST}{WS_PATH}"), clearnet_tls).await?;
clearnet_ws.send(Message::Text(ECHO_MSG.into())).await?;
let clearnet_reply = clearnet_ws.next().await.ok_or("no clearnet reply")??;
let clearnet_duration = clearnet_start.elapsed();
let clearnet_text = clearnet_reply.into_text()?;
clearnet_ws.close(None).await?;
info!("Clearnet: \"{clearnet_text}\" in {clearnet_duration:?}");
let args: Vec<String> = std::env::args().collect();
let ipr_addr = args
.iter()
.position(|a| a == "--ipr")
.and_then(|i| args.get(i + 1));
let mut builder = Tunnel::builder();
if let Some(addr) = ipr_addr {
builder = builder.ipr_address(addr.parse().expect("invalid IPR address"));
}
let tunnel = builder.build().await?;
info!("Allocated IP: {}", tunnel.allocated_ips().ipv4);
let setup_start = tokio::time::Instant::now();
info!("TCP connecting via mixnet...");
let mixnet_tcp = tunnel.tcp_connect(addr).await?;
info!("TCP connected ({:?})", setup_start.elapsed());
info!("TLS handshake...");
let mixnet_tls = connector.connect(domain, mixnet_tcp).await?;
info!("TLS established ({:?})", setup_start.elapsed());
info!("WebSocket upgrade...");
let (mut mixnet_ws, _) =
tokio_tungstenite::client_async(format!("wss://{WS_HOST}{WS_PATH}"), mixnet_tls).await?;
let setup_duration = setup_start.elapsed();
info!("Setup complete ({:?})", setup_duration);
let request_start = tokio::time::Instant::now();
mixnet_ws.send(Message::Text(ECHO_MSG.into())).await?;
let mixnet_reply = mixnet_ws.next().await.ok_or("no mixnet reply")??;
let request_duration = request_start.elapsed();
let mixnet_text = mixnet_reply.into_text()?;
mixnet_ws.close(None).await?;
info!("Echo: \"{mixnet_text}\" ({:?})", request_duration);
info!("Clearnet: \"{clearnet_text}\" in {clearnet_duration:?}");
info!(
"Mixnet: \"{mixnet_text}\" (setup {:?} + echo {:?} = {:?})",
setup_duration,
request_duration,
setup_duration + request_duration
);
info!("Clearnet echo match: {}", clearnet_text == ECHO_MSG);
info!("Mixnet echo match: {}", mixnet_text == ECHO_MSG);
let total = setup_duration + request_duration;
let slowdown = total.as_millis() as f64 / clearnet_duration.as_millis().max(1) as f64;
info!(
"Slowdown: {slowdown:.1}x (setup: {:.1}x, echo: {:.1}x)",
setup_duration.as_millis() as f64 / clearnet_duration.as_millis().max(1) as f64,
request_duration.as_millis() as f64 / clearnet_duration.as_millis().max(1) as f64
);
tunnel.shutdown().await;
Ok(())
}