use std::sync::Arc;
use std::time::Duration;
#[cfg(feature = "async-std")]
use async_io::Timer;
use env_logger::init;
use futures::channel::oneshot::channel;
#[cfg(not(feature = "tokio"))]
use futures::executor::block_on;
#[cfg(feature = "tokio")]
use tokio::runtime::Runtime;
#[cfg(feature = "tokio")]
use tokio::time::sleep;
use typhoon::bytes::StaticByteBuffer;
use typhoon::certificate::ServerKeyPair;
use typhoon::defaults::{AsyncExecutor, DefaultClientConnectionHandler, DefaultExecutor, DefaultServerConnectionHandler};
use typhoon::flow::{FakeBodyMode, FakeHeaderConfig, FlowConfig};
use typhoon::settings::SettingsBuilder;
use typhoon::socket::{ClientSocketBuilder, ListenerBuilder, ServerFlowConfiguration};
const ROUND_TRIPS: usize = 200;
#[cfg(feature = "tokio")]
fn main() {
Runtime::new().expect("failed to create tokio runtime").block_on(run());
}
#[cfg(not(feature = "tokio"))]
fn main() {
block_on(run());
}
#[cfg(feature = "tokio")]
pub async fn wait(duration: Duration) {
sleep(duration).await;
}
#[cfg(feature = "async-std")]
pub async fn wait(duration: Duration) {
Timer::after(duration).await;
}
async fn run() {
init();
let settings = Arc::new(SettingsBuilder::<DefaultExecutor>::new().build().expect("default settings should be valid"));
let server_addr = "127.0.0.1:19993".parse().expect("valid address");
let key_pair = ServerKeyPair::generate();
let certificate = key_pair.to_client_certificate(vec![server_addr]);
let flow_config = FlowConfig::new(FakeBodyMode::Empty, FakeHeaderConfig::new(vec![]));
let listener: Arc<_> = Arc::new(ListenerBuilder::<StaticByteBuffer, DefaultExecutor, DefaultServerConnectionHandler>::new(key_pair, DefaultServerConnectionHandler).add_flow(ServerFlowConfiguration::with_address(flow_config, server_addr)).with_settings(settings.clone()).build().await.expect("listener should build"));
listener.start().await;
println!("Server: listening on {server_addr}");
let (done_tx, done_rx) = channel::<usize>();
let listener_handle = listener.clone();
settings.executor().spawn(async move {
let client = listener_handle.accept().await.expect("accept should succeed");
println!("Server: client connected, running echo loop for {ROUND_TRIPS} round trips");
let mut count = 0;
while count < ROUND_TRIPS {
let data = client.receive_bytes().await.expect("receive should succeed");
client.send_bytes(&data).await.expect("echo send should succeed");
count += 1;
}
let _ = done_tx.send(count);
});
let socket = ClientSocketBuilder::<StaticByteBuffer, DefaultExecutor, DefaultClientConnectionHandler>::new(certificate, DefaultClientConnectionHandler).with_settings(settings.clone()).build().await.expect("client socket should build");
println!("Client: connected, running {} round trips", ROUND_TRIPS);
for i in 0..ROUND_TRIPS {
let sent = format!("round-{i:04}");
socket.send_bytes(sent.as_bytes()).await.expect("send should succeed");
let received = socket.receive_bytes().await.expect("receive should succeed");
assert_eq!(received.as_slice(), sent.as_bytes(), "payload mismatch on round trip {}", i);
if (i + 1) % 50 == 0 {
println!("Client: completed {}/{} round trips", i + 1, ROUND_TRIPS);
}
wait(Duration::from_secs_f32(1.5)).await;
}
let server_count = done_rx.await.expect("server task should complete");
assert_eq!(server_count, ROUND_TRIPS);
println!("Success! All {ROUND_TRIPS} round trips completed with correct payloads.");
}