diff --git a/chitchat-test/tests/cli.rs b/chitchat-test/tests/cli.rs index a142a61..ad6d229 100644 --- a/chitchat-test/tests/cli.rs +++ b/chitchat-test/tests/cli.rs @@ -2,9 +2,9 @@ mod helpers; +use std::net::TcpListener; use std::process::Child; -use std::thread; -use std::time::Duration; +use std::{thread, time::Duration}; use chitchat_test::{ApiResponse, SetKeyValueResponse}; use helpers::spawn_command; @@ -17,33 +17,82 @@ impl Drop for KillOnDrop { } } +struct NodeProcess { + node_id: String, + port: u16, + seed_addr: Option, + _child: Option, // Never accessed, but included to keep the value alive. +} + +impl NodeProcess { + pub fn new(node_id: String, port: u16, seed_addr: Option<&str>) -> Self { + Self { + node_id, + port, + seed_addr: seed_addr.map(|x| x.to_string()), + _child: None, + } + } + + fn start(&mut self) -> anyhow::Result<()> { + // Construct the command arguments + let mut args = vec![ + "--listen_addr".to_string(), + format!("127.0.0.1:{}", self.port), + "--node_id".to_string(), + self.node_id.clone(), + "--interval_ms".to_string(), + "50".to_string(), + ]; + + if let Some(seed) = &self.seed_addr { + args.push("--seed".to_string()); + args.push(seed.clone()); + } + + // Start the node process + let child = spawn_command(&args.join(" "))?; + self._child = Some(KillOnDrop(child)); + return Ok(()); + } + + fn base_url(&self) -> String { + format!("http://127.0.0.1:{}", self.port) + } +} + +fn create_node_process(node_id: &str, seed_addr: Option<&str>) -> anyhow::Result { + // First, find a free port by binding to port 0 to let the OS choose an available port. + let listener = TcpListener::bind("127.0.0.1:0")?; + let port = listener.local_addr()?.port(); + drop(listener); + + // Second, use the port to construct a new node process. + let mut process = NodeProcess::new(node_id.to_string(), port, seed_addr); + process.start()?; + + Ok(process) +} + fn setup_nodes( - port_offset: usize, num_nodes: usize, wait_stabilization_secs: u64, seed_requires_dns: bool, -) -> Vec { - let seed_port = port_offset; - let command_args = - format!("--listen_addr 127.0.0.1:{seed_port} --node_id node_0 --interval_ms 120"); - let seed_node = spawn_command(&command_args).unwrap(); +) -> Vec { + let child_handle = create_node_process("node_0", None).unwrap(); + let seed_port = child_handle.port; + let mut child_process_handles = vec![child_handle]; - let mut child_process_handles = vec![KillOnDrop(seed_node)]; + let seed_host_name = if seed_requires_dns { + "localhost" + } else { + "127.0.0.1" + }; + let seed_addr = format!("{}:{}", seed_host_name, seed_port); for i in 1..num_nodes { let node_id = format!("node_{i}"); - let node_port = port_offset + i; - let seed_host_name = if seed_requires_dns { - "localhost" - } else { - "127.0.0.1" - }; - let command_args = format!( - "--listen_addr 127.0.0.1:{node_port} --seed {seed_host_name}:{seed_port} --node_id \ - {node_id} --interval_ms 50" - ); - let node = spawn_command(&command_args).unwrap(); - child_process_handles.push(KillOnDrop(node)); + child_process_handles.push(create_node_process(&node_id, Some(&seed_addr)).unwrap()); } thread::sleep(Duration::from_secs(wait_stabilization_secs)); child_process_handles @@ -61,16 +110,18 @@ fn set_kv(node_api_endpoint: &str, key: &str, value: &str) -> anyhow::Result