| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306 |
- use anyhow::{Ok, Result};
- use common::{run_rathole_client, PING, PONG};
- use rand::Rng;
- use std::time::Duration;
- use tokio::{
- io::{AsyncReadExt, AsyncWriteExt},
- net::{TcpStream, UdpSocket},
- sync::broadcast,
- time,
- };
- use tracing::{debug, info, instrument};
- use tracing_subscriber::EnvFilter;
- use crate::common::run_rathole_server;
- mod common;
- const ECHO_SERVER_ADDR: &str = "127.0.0.1:8080";
- const PINGPONG_SERVER_ADDR: &str = "127.0.0.1:8081";
- const ECHO_SERVER_ADDR_EXPOSED: &str = "127.0.0.1:2334";
- const PINGPONG_SERVER_ADDR_EXPOSED: &str = "127.0.0.1:2335";
- const HITTER_NUM: usize = 4;
- #[derive(Clone, Copy, Debug)]
- enum Type {
- Tcp,
- Udp,
- }
- fn init() {
- let level = "info";
- let _ = tracing_subscriber::fmt()
- .with_env_filter(
- EnvFilter::try_from_default_env().unwrap_or_else(|_| EnvFilter::from(level)),
- )
- .try_init();
- }
- #[tokio::test]
- async fn tcp() -> Result<()> {
- init();
- // Spawn a echo server
- tokio::spawn(async move {
- if let Err(e) = common::tcp::echo_server(ECHO_SERVER_ADDR).await {
- panic!("Failed to run the echo server for testing: {:?}", e);
- }
- });
- // Spawn a pingpong server
- tokio::spawn(async move {
- if let Err(e) = common::tcp::pingpong_server(PINGPONG_SERVER_ADDR).await {
- panic!("Failed to run the pingpong server for testing: {:?}", e);
- }
- });
- test("tests/for_tcp/tcp_transport.toml", Type::Tcp).await?;
- #[cfg(any(
- // FIXME: Self-signed certificate on macOS nativetls requires manual interference.
- all(target_os = "macos", feature = "rustls"),
- // On other OS accept run with either
- all(not(target_os = "macos"), any(feature = "native-tls", feature = "rustls")),
- ))]
- test("tests/for_tcp/tls_transport.toml", Type::Tcp).await?;
- #[cfg(feature = "noise")]
- test("tests/for_tcp/noise_transport.toml", Type::Tcp).await?;
- #[cfg(any(feature = "websocket-native-tls", feature = "websocket-rustls"))]
- test("tests/for_tcp/websocket_transport.toml", Type::Tcp).await?;
- #[cfg(not(target_os = "macos"))]
- #[cfg(any(feature = "websocket-native-tls", feature = "websocket-rustls"))]
- test("tests/for_tcp/websocket_tls_transport.toml", Type::Tcp).await?;
- Ok(())
- }
- #[tokio::test]
- async fn udp() -> Result<()> {
- init();
- // Spawn a echo server
- tokio::spawn(async move {
- if let Err(e) = common::udp::echo_server(ECHO_SERVER_ADDR).await {
- panic!("Failed to run the echo server for testing: {:?}", e);
- }
- });
- // Spawn a pingpong server
- tokio::spawn(async move {
- if let Err(e) = common::udp::pingpong_server(PINGPONG_SERVER_ADDR).await {
- panic!("Failed to run the pingpong server for testing: {:?}", e);
- }
- });
- test("tests/for_udp/tcp_transport.toml", Type::Udp).await?;
- #[cfg(any(
- // FIXME: Self-signed certificate on macOS nativetls requires manual interference.
- all(target_os = "macos", feature = "rustls"),
- // On other OS accept run with either
- all(not(target_os = "macos"), any(feature = "native-tls", feature = "rustls")),
- ))]
- test("tests/for_udp/tls_transport.toml", Type::Udp).await?;
- #[cfg(feature = "noise")]
- test("tests/for_udp/noise_transport.toml", Type::Udp).await?;
- #[cfg(any(feature = "websocket-native-tls", feature = "websocket-rustls"))]
- test("tests/for_udp/websocket_transport.toml", Type::Udp).await?;
- #[cfg(not(target_os = "macos"))]
- #[cfg(any(feature = "websocket-native-tls", feature = "websocket-rustls"))]
- test("tests/for_udp/websocket_tls_transport.toml", Type::Udp).await?;
- Ok(())
- }
- #[instrument]
- async fn test(config_path: &'static str, t: Type) -> Result<()> {
- if cfg!(not(all(feature = "client", feature = "server"))) {
- // Skip the test if the client or the server is not enabled
- return Ok(());
- }
- let (client_shutdown_tx, client_shutdown_rx) = broadcast::channel(1);
- let (server_shutdown_tx, server_shutdown_rx) = broadcast::channel(1);
- // Start the client
- info!("start the client");
- let client = tokio::spawn(async move {
- run_rathole_client(config_path, client_shutdown_rx)
- .await
- .unwrap();
- });
- // Sleep for 1 second. Expect the client keep retrying to reach the server
- time::sleep(Duration::from_secs(1)).await;
- // Start the server
- info!("start the server");
- let server = tokio::spawn(async move {
- run_rathole_server(config_path, server_shutdown_rx)
- .await
- .unwrap();
- });
- time::sleep(Duration::from_millis(2500)).await; // Wait for the client to retry
- info!("echo");
- echo_hitter(ECHO_SERVER_ADDR_EXPOSED, t).await.unwrap();
- info!("pingpong");
- pingpong_hitter(PINGPONG_SERVER_ADDR_EXPOSED, t)
- .await
- .unwrap();
- // Simulate the client crash and restart
- info!("shutdown the client");
- client_shutdown_tx.send(true)?;
- let _ = tokio::join!(client);
- info!("restart the client");
- let client_shutdown_rx = client_shutdown_tx.subscribe();
- let client = tokio::spawn(async move {
- run_rathole_client(config_path, client_shutdown_rx)
- .await
- .unwrap();
- });
- time::sleep(Duration::from_secs(1)).await; // Wait for the client to start
- info!("echo");
- echo_hitter(ECHO_SERVER_ADDR_EXPOSED, t).await.unwrap();
- info!("pingpong");
- pingpong_hitter(PINGPONG_SERVER_ADDR_EXPOSED, t)
- .await
- .unwrap();
- // Simulate the server crash and restart
- info!("shutdown the server");
- server_shutdown_tx.send(true)?;
- let _ = tokio::join!(server);
- info!("restart the server");
- let server_shutdown_rx = server_shutdown_tx.subscribe();
- let server = tokio::spawn(async move {
- run_rathole_server(config_path, server_shutdown_rx)
- .await
- .unwrap();
- });
- time::sleep(Duration::from_millis(2500)).await; // Wait for the client to retry
- // Simulate heavy load
- info!("lots of echo and pingpong");
- let mut v = Vec::new();
- for _ in 0..HITTER_NUM / 2 {
- v.push(tokio::spawn(async move {
- echo_hitter(ECHO_SERVER_ADDR_EXPOSED, t).await.unwrap();
- }));
- v.push(tokio::spawn(async move {
- pingpong_hitter(PINGPONG_SERVER_ADDR_EXPOSED, t)
- .await
- .unwrap();
- }));
- }
- for h in v {
- assert!(tokio::join!(h).0.is_ok());
- }
- // Shutdown
- info!("shutdown the server and the client");
- server_shutdown_tx.send(true)?;
- client_shutdown_tx.send(true)?;
- let _ = tokio::join!(server, client);
- Ok(())
- }
- async fn echo_hitter(addr: &'static str, t: Type) -> Result<()> {
- match t {
- Type::Tcp => tcp_echo_hitter(addr).await,
- Type::Udp => udp_echo_hitter(addr).await,
- }
- }
- async fn pingpong_hitter(addr: &'static str, t: Type) -> Result<()> {
- match t {
- Type::Tcp => tcp_pingpong_hitter(addr).await,
- Type::Udp => udp_pingpong_hitter(addr).await,
- }
- }
- async fn tcp_echo_hitter(addr: &'static str) -> Result<()> {
- let mut conn = TcpStream::connect(addr).await?;
- let mut wr = [0u8; 1024];
- let mut rd = [0u8; 1024];
- for _ in 0..100 {
- rand::thread_rng().fill(&mut wr);
- conn.write_all(&wr).await?;
- conn.read_exact(&mut rd).await?;
- assert_eq!(wr, rd);
- }
- Ok(())
- }
- async fn udp_echo_hitter(addr: &'static str) -> Result<()> {
- let conn = UdpSocket::bind("127.0.0.1:0").await?;
- conn.connect(addr).await?;
- let mut wr = [0u8; 128];
- let mut rd = [0u8; 128];
- for _ in 0..3 {
- rand::thread_rng().fill(&mut wr);
- conn.send(&wr).await?;
- debug!("send");
- conn.recv(&mut rd).await?;
- debug!("recv");
- assert_eq!(wr, rd);
- }
- Ok(())
- }
- async fn tcp_pingpong_hitter(addr: &'static str) -> Result<()> {
- let mut conn = TcpStream::connect(addr).await?;
- let wr = PING.as_bytes();
- let mut rd = [0u8; PONG.len()];
- for _ in 0..100 {
- conn.write_all(wr).await?;
- conn.read_exact(&mut rd).await?;
- assert_eq!(rd, PONG.as_bytes());
- }
- Ok(())
- }
- async fn udp_pingpong_hitter(addr: &'static str) -> Result<()> {
- let conn = UdpSocket::bind("127.0.0.1:0").await?;
- conn.connect(&addr).await?;
- let wr = PING.as_bytes();
- let mut rd = [0u8; PONG.len()];
- for _ in 0..3 {
- conn.send(wr).await?;
- debug!("ping");
- conn.recv(&mut rd).await?;
- debug!("pong");
- assert_eq!(rd, PONG.as_bytes());
- }
- Ok(())
- }
|