mod common; use common::TestUtils; use serde_json::Value; use serial_test::serial; use std::time::Duration; use tokio::io::{AsyncReadExt, AsyncWriteExt}; use tokio::net::TcpStream; use tokio::time::sleep; /// Setup function to ensure donut-proxy binary exists and cleanup stale proxies async fn setup_test() -> Result> { let cargo_manifest_dir = std::env::var("CARGO_MANIFEST_DIR")?; let project_root = std::path::PathBuf::from(cargo_manifest_dir) .parent() .unwrap() .to_path_buf(); // Build donut-proxy binary if it doesn't exist let proxy_binary = project_root .join("src-tauri") .join("target") .join("debug") .join("donut-proxy"); if !proxy_binary.exists() { println!("Building donut-proxy binary for integration tests..."); let build_status = std::process::Command::new("cargo") .args(["build", "--bin", "donut-proxy"]) .current_dir(project_root.join("src-tauri")) .status()?; if !build_status.success() { return Err("Failed to build donut-proxy binary".into()); } } if !proxy_binary.exists() { return Err("donut-proxy binary was not created successfully".into()); } // Clean up any stale proxies from previous test runs let _ = TestUtils::execute_command(&proxy_binary, &["proxy", "stop"]).await; Ok(proxy_binary) } /// Helper to track and cleanup proxy processes struct ProxyTestTracker { proxy_ids: Vec, binary_path: std::path::PathBuf, } impl ProxyTestTracker { fn new(binary_path: std::path::PathBuf) -> Self { Self { proxy_ids: Vec::new(), binary_path, } } fn track_proxy(&mut self, proxy_id: String) { self.proxy_ids.push(proxy_id); } async fn cleanup_all(&self) { for proxy_id in &self.proxy_ids { let _ = TestUtils::execute_command(&self.binary_path, &["proxy", "stop", "--id", proxy_id]).await; } } } impl Drop for ProxyTestTracker { fn drop(&mut self) { let proxy_ids = self.proxy_ids.clone(); let binary_path = self.binary_path.clone(); tokio::spawn(async move { for proxy_id in &proxy_ids { let _ = TestUtils::execute_command(&binary_path, &["proxy", "stop", "--id", proxy_id]).await; } }); } } /// Test starting a local proxy without upstream proxy (DIRECT) #[tokio::test] #[serial] async fn test_local_proxy_direct() -> Result<(), Box> { let binary_path = setup_test().await?; let mut tracker = ProxyTestTracker::new(binary_path.clone()); println!("Starting local proxy without upstream (DIRECT)..."); let output = TestUtils::execute_command(&binary_path, &["proxy", "start"]).await?; if !output.status.success() { let stderr = String::from_utf8_lossy(&output.stderr); let stdout = String::from_utf8_lossy(&output.stdout); return Err(format!("Proxy start failed - stdout: {stdout}, stderr: {stderr}").into()); } let stdout = String::from_utf8(output.stdout)?; let config: Value = serde_json::from_str(&stdout)?; let proxy_id = config["id"].as_str().unwrap().to_string(); let local_port = config["localPort"].as_u64().unwrap() as u16; let local_url = config["localUrl"].as_str().unwrap(); let upstream_url = config["upstreamUrl"].as_str().unwrap(); tracker.track_proxy(proxy_id.clone()); println!( "Proxy started: id={}, port={}, url={}, upstream={}", proxy_id, local_port, local_url, upstream_url ); // Verify proxy is listening sleep(Duration::from_millis(500)).await; match TcpStream::connect(("127.0.0.1", local_port)).await { Ok(_) => { println!("Proxy is listening on port {local_port}"); } Err(e) => { return Err(format!("Proxy port {local_port} is not listening: {e}").into()); } } // Test making an HTTP request through the proxy let mut stream = TcpStream::connect(("127.0.0.1", local_port)).await?; let request = b"GET http://httpbin.org/ip HTTP/1.1\r\nHost: httpbin.org\r\nConnection: close\r\n\r\n"; stream.write_all(request).await?; let mut response = Vec::new(); stream.read_to_end(&mut response).await?; let response_str = String::from_utf8_lossy(&response); if response_str.contains("200 OK") || response_str.contains("origin") { println!("Proxy successfully forwarded HTTP request"); } else { println!( "Warning: Proxy response may be unexpected: {}", &response_str[..response_str.len().min(200)] ); } // Cleanup tracker.cleanup_all().await; Ok(()) } /// Test chaining local proxies (local proxy -> local proxy -> internet) #[tokio::test] #[serial] async fn test_chained_local_proxies() -> Result<(), Box> { let binary_path = setup_test().await?; let mut tracker = ProxyTestTracker::new(binary_path.clone()); println!("Testing chained local proxies..."); // Start first proxy (DIRECT - connects to internet) let output1 = TestUtils::execute_command(&binary_path, &["proxy", "start"]).await?; if !output1.status.success() { let stderr = String::from_utf8_lossy(&output1.stderr); let stdout = String::from_utf8_lossy(&output1.stdout); return Err(format!("Failed to start first proxy - stdout: {stdout}, stderr: {stderr}").into()); } let config1: Value = serde_json::from_str(&String::from_utf8(output1.stdout)?)?; let proxy1_id = config1["id"].as_str().unwrap().to_string(); let proxy1_port = config1["localPort"].as_u64().unwrap() as u16; tracker.track_proxy(proxy1_id.clone()); println!("First proxy started on port {}", proxy1_port); // Wait for first proxy to be ready sleep(Duration::from_millis(500)).await; match TcpStream::connect(("127.0.0.1", proxy1_port)).await { Ok(_) => println!("First proxy is ready"), Err(e) => return Err(format!("First proxy not ready: {e}").into()), } // Start second proxy chained to first proxy let output2 = TestUtils::execute_command( &binary_path, &[ "proxy", "start", "--host", "127.0.0.1", "--proxy-port", &proxy1_port.to_string(), "--type", "http", ], ) .await?; if !output2.status.success() { let stderr = String::from_utf8_lossy(&output2.stderr); let stdout = String::from_utf8_lossy(&output2.stdout); return Err( format!("Failed to start second proxy - stdout: {stdout}, stderr: {stderr}").into(), ); } let config2: Value = serde_json::from_str(&String::from_utf8(output2.stdout)?)?; let proxy2_id = config2["id"].as_str().unwrap().to_string(); let proxy2_port = config2["localPort"].as_u64().unwrap() as u16; tracker.track_proxy(proxy2_id.clone()); println!( "Second proxy started on port {} (chained to proxy on port {})", proxy2_port, proxy1_port ); // Wait for second proxy to be ready sleep(Duration::from_millis(500)).await; match TcpStream::connect(("127.0.0.1", proxy2_port)).await { Ok(_) => println!("Second proxy is ready"), Err(e) => return Err(format!("Second proxy not ready: {e}").into()), } // Test making an HTTP request through the chained proxy let mut stream = TcpStream::connect(("127.0.0.1", proxy2_port)).await?; let request = b"GET http://httpbin.org/ip HTTP/1.1\r\nHost: httpbin.org\r\nConnection: close\r\n\r\n"; stream.write_all(request).await?; let mut response = Vec::new(); stream.read_to_end(&mut response).await?; let response_str = String::from_utf8_lossy(&response); if response_str.contains("200 OK") || response_str.contains("origin") { println!("Chained proxy successfully forwarded HTTP request"); } else { println!( "Warning: Chained proxy response may be unexpected: {}", &response_str[..response_str.len().min(200)] ); } // Cleanup tracker.cleanup_all().await; Ok(()) } /// Test starting a local proxy with HTTP upstream proxy #[tokio::test] #[serial] async fn test_local_proxy_with_http_upstream( ) -> Result<(), Box> { let binary_path = setup_test().await?; let mut tracker = ProxyTestTracker::new(binary_path.clone()); // Start a mock HTTP upstream proxy server let upstream_listener = tokio::net::TcpListener::bind("127.0.0.1:0").await?; let upstream_addr = upstream_listener.local_addr()?; let upstream_port = upstream_addr.port(); let upstream_handle = tokio::spawn(async move { use http_body_util::Full; use hyper::body::Bytes; use hyper::server::conn::http1; use hyper::service::service_fn; use hyper::{Response, StatusCode}; use hyper_util::rt::TokioIo; while let Ok((stream, _)) = upstream_listener.accept().await { let io = TokioIo::new(stream); tokio::task::spawn(async move { let service = service_fn(|_req| async { Ok::<_, hyper::Error>( Response::builder() .status(StatusCode::OK) .body(Full::new(Bytes::from("Upstream Proxy Response"))) .unwrap(), ) }); let _ = http1::Builder::new().serve_connection(io, service).await; }); } }); sleep(Duration::from_millis(200)).await; println!("Starting local proxy with HTTP upstream proxy..."); let output = TestUtils::execute_command( &binary_path, &[ "proxy", "start", "--host", "127.0.0.1", "--proxy-port", &upstream_port.to_string(), "--type", "http", ], ) .await?; if !output.status.success() { let stderr = String::from_utf8_lossy(&output.stderr); let stdout = String::from_utf8_lossy(&output.stdout); upstream_handle.abort(); return Err(format!("Proxy start failed - stdout: {stdout}, stderr: {stderr}").into()); } let stdout = String::from_utf8(output.stdout)?; let config: Value = serde_json::from_str(&stdout)?; let proxy_id = config["id"].as_str().unwrap().to_string(); let local_port = config["localPort"].as_u64().unwrap() as u16; tracker.track_proxy(proxy_id.clone()); println!("Proxy started: id={}, port={}", proxy_id, local_port); // Verify proxy is listening sleep(Duration::from_millis(500)).await; match TcpStream::connect(("127.0.0.1", local_port)).await { Ok(_) => { println!("Proxy is listening on port {local_port}"); } Err(e) => { upstream_handle.abort(); return Err(format!("Proxy port {local_port} is not listening: {e}").into()); } } // Cleanup tracker.cleanup_all().await; upstream_handle.abort(); Ok(()) } /// Test multiple proxies running simultaneously #[tokio::test] #[serial] async fn test_multiple_proxies_simultaneously( ) -> Result<(), Box> { let binary_path = setup_test().await?; let mut tracker = ProxyTestTracker::new(binary_path.clone()); println!("Starting multiple proxies simultaneously..."); let mut proxy_ports = Vec::new(); // Start 3 proxies, waiting for each to be ready before starting the next // This avoids race conditions on macOS where processes need time to initialize for i in 0..3 { let output = TestUtils::execute_command(&binary_path, &["proxy", "start"]).await?; if !output.status.success() { let stderr = String::from_utf8_lossy(&output.stderr); let stdout = String::from_utf8_lossy(&output.stdout); return Err( format!( "Failed to start proxy {} - stdout: {}, stderr: {}", i + 1, stdout, stderr ) .into(), ); } let config: Value = serde_json::from_str(&String::from_utf8(output.stdout)?)?; let proxy_id = config["id"].as_str().unwrap().to_string(); let local_port = config["localPort"].as_u64().unwrap() as u16; tracker.track_proxy(proxy_id); proxy_ports.push(local_port); println!("Proxy {} started on port {}", i + 1, local_port); // Wait for this proxy to be ready before starting the next one // This prevents race conditions on macOS where processes need time to initialize let mut attempts = 0; let max_attempts = 50; // 5 seconds max (50 * 100ms) loop { sleep(Duration::from_millis(100)).await; match TcpStream::connect(("127.0.0.1", local_port)).await { Ok(_) => { println!("Proxy {} is ready on port {}", i + 1, local_port); break; } Err(_) => { attempts += 1; if attempts >= max_attempts { return Err( format!( "Proxy {} on port {} failed to become ready after {} attempts", i + 1, local_port, max_attempts ) .into(), ); } } } } } // Verify all proxies are still listening for (i, port) in proxy_ports.iter().enumerate() { match TcpStream::connect(("127.0.0.1", *port)).await { Ok(_) => { println!("Proxy {} is listening on port {}", i + 1, port); } Err(e) => { return Err(format!("Proxy {} on port {} is not listening: {e}", i + 1, port).into()); } } } // Cleanup tracker.cleanup_all().await; Ok(()) } /// Test proxy listing #[tokio::test] #[serial] async fn test_proxy_list() -> Result<(), Box> { let binary_path = setup_test().await?; let mut tracker = ProxyTestTracker::new(binary_path.clone()); // Start a proxy let output = TestUtils::execute_command(&binary_path, &["proxy", "start"]).await?; if !output.status.success() { let stderr = String::from_utf8_lossy(&output.stderr); let stdout = String::from_utf8_lossy(&output.stdout); return Err(format!("Failed to start proxy - stdout: {stdout}, stderr: {stderr}").into()); } let config: Value = serde_json::from_str(&String::from_utf8(output.stdout)?)?; let proxy_id = config["id"].as_str().unwrap().to_string(); tracker.track_proxy(proxy_id.clone()); // List proxies let list_output = TestUtils::execute_command(&binary_path, &["proxy", "list"]).await?; if !list_output.status.success() { return Err("Failed to list proxies".into()); } let list_stdout = String::from_utf8(list_output.stdout)?; let proxies: Vec = serde_json::from_str(&list_stdout)?; // Verify our proxy is in the list let found = proxies.iter().any(|p| p["id"].as_str() == Some(&proxy_id)); assert!(found, "Proxy should be in the list"); // Cleanup tracker.cleanup_all().await; Ok(()) } /// Test traffic tracking through proxy #[tokio::test] #[serial] async fn test_traffic_tracking() -> Result<(), Box> { let binary_path = setup_test().await?; let mut tracker = ProxyTestTracker::new(binary_path.clone()); println!("Testing traffic tracking through proxy..."); // Start a proxy let output = TestUtils::execute_command(&binary_path, &["proxy", "start"]).await?; if !output.status.success() { let stderr = String::from_utf8_lossy(&output.stderr); let stdout = String::from_utf8_lossy(&output.stdout); return Err(format!("Failed to start proxy - stdout: {stdout}, stderr: {stderr}").into()); } let config: Value = serde_json::from_str(&String::from_utf8(output.stdout)?)?; let proxy_id = config["id"].as_str().unwrap().to_string(); let local_port = config["localPort"].as_u64().unwrap() as u16; tracker.track_proxy(proxy_id.clone()); println!("Proxy started on port {}", local_port); // Wait for proxy to be ready sleep(Duration::from_millis(500)).await; // Make an HTTP request through the proxy let mut stream = TcpStream::connect(("127.0.0.1", local_port)).await?; let request = b"GET http://httpbin.org/ip HTTP/1.1\r\nHost: httpbin.org\r\nConnection: close\r\n\r\n"; // Track bytes sent let bytes_sent = request.len(); stream.write_all(request).await?; // Read response let mut response = Vec::new(); stream.read_to_end(&mut response).await?; let bytes_received = response.len(); println!( "HTTP request completed: sent {} bytes, received {} bytes", bytes_sent, bytes_received ); // Wait for traffic stats to be flushed (happens every second) sleep(Duration::from_secs(2)).await; // Verify traffic was tracked by checking traffic stats file exists // Note: Traffic stats are stored in the cache directory let cache_dir = directories::BaseDirs::new() .expect("Failed to get base directories") .cache_dir() .to_path_buf(); let traffic_stats_dir = cache_dir.join("DonutBrowserDev").join("traffic_stats"); let stats_file = traffic_stats_dir.join(format!("{}.json", proxy_id)); if stats_file.exists() { let content = std::fs::read_to_string(&stats_file)?; let stats: Value = serde_json::from_str(&content)?; let total_sent = stats["total_bytes_sent"].as_u64().unwrap_or(0); let total_received = stats["total_bytes_received"].as_u64().unwrap_or(0); let total_requests = stats["total_requests"].as_u64().unwrap_or(0); println!( "Traffic stats recorded: sent {} bytes, received {} bytes, {} requests", total_sent, total_received, total_requests ); // Check if domains are being tracked let mut domain_traffic = false; if let Some(domains) = stats.get("domains") { if let Some(domain_map) = domains.as_object() { println!("Domains tracked: {}", domain_map.len()); for (domain, domain_stats) in domain_map { println!(" - {}", domain); // Check if any domain has traffic if let Some(domain_obj) = domain_stats.as_object() { let domain_sent = domain_obj .get("bytes_sent") .and_then(|v| v.as_u64()) .unwrap_or(0); let domain_recv = domain_obj .get("bytes_received") .and_then(|v| v.as_u64()) .unwrap_or(0); let domain_reqs = domain_obj .get("request_count") .and_then(|v| v.as_u64()) .unwrap_or(0); println!( " sent: {}, received: {}, requests: {}", domain_sent, domain_recv, domain_reqs ); if domain_sent > 0 || domain_recv > 0 || domain_reqs > 0 { domain_traffic = true; } } } } } // Verify that some traffic was recorded - check either total bytes or domain traffic assert!( total_sent > 0 || total_received > 0 || total_requests > 0 || domain_traffic, "Traffic stats should record some activity (sent: {}, received: {}, requests: {})", total_sent, total_received, total_requests ); println!("Traffic tracking test passed!"); } else { println!("Warning: Traffic stats file not found at {:?}", stats_file); // This is not necessarily a failure - the file may not have been created yet // The important thing is that the proxy is working } // Cleanup tracker.cleanup_all().await; // Clean up the traffic stats file if stats_file.exists() { let _ = std::fs::remove_file(&stats_file); } Ok(()) } /// Test proxy stop #[tokio::test] #[serial] async fn test_proxy_stop() -> Result<(), Box> { let binary_path = setup_test().await?; let _tracker = ProxyTestTracker::new(binary_path.clone()); // Start a proxy let output = TestUtils::execute_command(&binary_path, &["proxy", "start"]).await?; if !output.status.success() { let stderr = String::from_utf8_lossy(&output.stderr); let stdout = String::from_utf8_lossy(&output.stdout); return Err(format!("Failed to start proxy - stdout: {stdout}, stderr: {stderr}").into()); } let config: Value = serde_json::from_str(&String::from_utf8(output.stdout)?)?; let proxy_id = config["id"].as_str().unwrap().to_string(); let local_port = config["localPort"].as_u64().unwrap() as u16; // Verify proxy is running sleep(Duration::from_millis(500)).await; match TcpStream::connect(("127.0.0.1", local_port)).await { Ok(_) => println!("Proxy is running"), Err(_) => return Err("Proxy is not running".into()), } // Stop the proxy let stop_output = TestUtils::execute_command(&binary_path, &["proxy", "stop", "--id", &proxy_id]).await?; if !stop_output.status.success() { return Err("Failed to stop proxy".into()); } // Wait a bit for the process to exit sleep(Duration::from_millis(500)).await; // Verify proxy is stopped (connection should fail) match TcpStream::connect(("127.0.0.1", local_port)).await { Ok(_) => return Err("Proxy should be stopped but is still listening".into()), Err(_) => println!("Proxy successfully stopped"), } Ok(()) }