-
Notifications
You must be signed in to change notification settings - Fork 581
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
31c8e1d
commit 9bfe64c
Showing
5 changed files
with
348 additions
and
2 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,207 @@ | ||
use toydb::raft::NodeID; | ||
use toydb::Client; | ||
|
||
use rand::Rng; | ||
use std::collections::BTreeMap; | ||
use std::error::Error; | ||
use std::fmt::Write as _; | ||
use std::path::Path; | ||
use std::time::Duration; | ||
|
||
/// Timeout for command responses and node readiness. | ||
const TIMEOUT: Duration = Duration::from_secs(5); | ||
|
||
/// The base SQL port (+id). | ||
const SQL_BASE_PORT: u16 = 19600; | ||
|
||
/// The base Raft port (+id). | ||
const RAFT_BASE_PORT: u16 = 19700; | ||
|
||
/// Runs a toyDB cluster using the built binary in a temporary directory. The | ||
/// cluster will be killed and removed when dropped. | ||
/// | ||
/// This runs the cluster as child processes using the built binary instead of | ||
/// spawning in-memory threads for a couple of reasons: it avoids having to | ||
/// gracefully shut down the server (which is complicated by e.g. | ||
/// TcpListener::accept() not being interruptable), and it tests the entire | ||
/// server (and eventually the toySQL client) end-to-end. | ||
pub struct TestCluster { | ||
servers: BTreeMap<NodeID, TestServer>, | ||
#[allow(dead_code)] | ||
dir: tempfile::TempDir, // deleted when dropped | ||
} | ||
|
||
type NodePorts = BTreeMap<NodeID, (u16, u16)>; // raft,sql on localhost | ||
|
||
impl TestCluster { | ||
/// Runs and returns a test cluster. It keeps running until dropped. | ||
pub fn run(nodes: u8) -> Result<Self, Box<dyn Error>> { | ||
// Create temporary directory. | ||
let dir = tempfile::TempDir::with_prefix("toydb")?; | ||
|
||
// Allocate port numbers for nodes. | ||
let ports: NodePorts = (1..=nodes) | ||
.map(|id| (id, (RAFT_BASE_PORT + id as u16, SQL_BASE_PORT + id as u16))) | ||
.collect(); | ||
|
||
// Start nodes. | ||
let mut servers = BTreeMap::new(); | ||
for id in 1..=nodes { | ||
let dir = dir.path().join(format!("toydb{id}")); | ||
servers.insert(id, TestServer::run(id, &dir, &ports)?); | ||
} | ||
|
||
// Wait for the nodes to be ready, by fetching the server status. | ||
let started = std::time::Instant::now(); | ||
for server in servers.values_mut() { | ||
while let Err(error) = server.connect().and_then(|mut c| Ok(c.status()?)) { | ||
server.assert_alive(); | ||
if started.elapsed() >= TIMEOUT { | ||
return Err(error); | ||
} | ||
std::thread::sleep(Duration::from_millis(200)); | ||
} | ||
} | ||
|
||
Ok(Self { servers, dir }) | ||
} | ||
|
||
/// Connects to a random cluster node using the regular client. | ||
#[allow(dead_code)] | ||
pub fn connect(&self) -> Result<Client, Box<dyn Error>> { | ||
let id = rand::thread_rng().gen_range(1..=self.servers.len()) as NodeID; | ||
self.servers.get(&id).unwrap().connect() | ||
} | ||
|
||
/// Connects to a random cluster node using the toysql binary. | ||
pub fn connect_toysql(&self) -> Result<TestClient, Box<dyn Error>> { | ||
let id = rand::thread_rng().gen_range(1..=self.servers.len()) as NodeID; | ||
self.servers.get(&id).unwrap().connect_toysql() | ||
} | ||
} | ||
|
||
/// A toyDB server. | ||
pub struct TestServer { | ||
id: NodeID, | ||
child: std::process::Child, | ||
sql_port: u16, | ||
} | ||
|
||
impl TestServer { | ||
/// Runs a toyDB server. | ||
fn run(id: NodeID, dir: &Path, ports: &NodePorts) -> Result<Self, Box<dyn Error>> { | ||
// Build and write the configuration file. | ||
let configfile = dir.join("toydb.yaml"); | ||
std::fs::create_dir_all(dir)?; | ||
std::fs::write(&configfile, Self::build_config(id, dir, ports)?)?; | ||
|
||
// Build the binary. | ||
// TODO: this may contribute to slow start times, consider building once | ||
// and passing it in. | ||
let build = escargot::CargoBuild::new().bin("toydb").run()?; | ||
|
||
// Spawn process. Discard output. | ||
let child = build | ||
.command() | ||
.args(["-c", &configfile.to_string_lossy()]) | ||
.stdout(std::process::Stdio::null()) | ||
.stderr(std::process::Stdio::null()) | ||
.spawn()?; | ||
|
||
let (_, sql_port) = ports.get(&id).copied().expect("node not in ports"); | ||
Ok(Self { id, child, sql_port }) | ||
} | ||
|
||
/// Generates a config file for the given node. | ||
fn build_config(id: NodeID, dir: &Path, ports: &NodePorts) -> Result<String, Box<dyn Error>> { | ||
let (raft_port, sql_port) = ports.get(&id).expect("node not in ports"); | ||
let mut cfg = String::new(); | ||
writeln!(cfg, "id: {id}")?; | ||
writeln!(cfg, "data_dir: {}", dir.to_string_lossy())?; | ||
writeln!(cfg, "listen_raft: localhost:{raft_port}")?; | ||
writeln!(cfg, "listen_sql: localhost:{sql_port}")?; | ||
writeln!(cfg, "peers: {{")?; | ||
for (peer_id, (peer_raft_port, _)) in ports.iter().filter(|(peer, _)| **peer != id) { | ||
writeln!(cfg, " '{peer_id}': localhost:{peer_raft_port},")?; | ||
} | ||
writeln!(cfg, "}}")?; | ||
Ok(cfg) | ||
} | ||
|
||
/// Asserts that the server is still running. | ||
fn assert_alive(&mut self) { | ||
if let Some(status) = self.child.try_wait().expect("failed to check exit status") { | ||
panic!("node {id} exited with status {status}", id = self.id) | ||
} | ||
} | ||
|
||
/// Connects to the server using a regular client. | ||
fn connect(&self) -> Result<Client, Box<dyn Error>> { | ||
Ok(Client::connect(("localhost", self.sql_port))?) | ||
} | ||
|
||
/// Connects to the server using the toysql binary. | ||
pub fn connect_toysql(&self) -> Result<TestClient, Box<dyn Error>> { | ||
TestClient::connect(self.sql_port) | ||
} | ||
} | ||
|
||
impl Drop for TestServer { | ||
// Kills the child process when dropped. | ||
fn drop(&mut self) { | ||
self.child.kill().expect("failed to kill node"); | ||
self.child.wait().expect("failed to wait for node to terminate"); | ||
} | ||
} | ||
|
||
/// A toySQL client using the toysql binary. | ||
pub struct TestClient { | ||
session: rexpect::session::PtySession, | ||
} | ||
|
||
impl TestClient { | ||
/// Connects to a toyDB server at the given SQL port number, using | ||
/// the toysql binary. | ||
fn connect(port: u16) -> Result<Self, Box<dyn Error>> { | ||
// Build the binary. | ||
let build = escargot::CargoBuild::new().bin("toysql").run()?; | ||
|
||
// Run it, using rexpect to manage stdin/stdout. | ||
let mut command = build.command(); | ||
command.args(["-p", &port.to_string()]); | ||
let session = rexpect::spawn_with_options( | ||
command, | ||
rexpect::reader::Options { | ||
timeout_ms: Some(TIMEOUT.as_millis() as u64), | ||
strip_ansi_escape_codes: true, | ||
}, | ||
)?; | ||
|
||
// Wait for the initial prompt. | ||
let mut client = Self { session }; | ||
client.read_until_prompt()?; | ||
Ok(client) | ||
} | ||
|
||
/// Executes a command, returning it and the resulting toysql prompt. | ||
pub fn execute(&mut self, command: &str) -> Result<(String, String), Box<dyn Error>> { | ||
let mut command = command.to_string(); | ||
if !command.ends_with(';') && !command.starts_with('!') { | ||
command = format!("{command};"); | ||
} | ||
self.session.send_line(&command)?; | ||
self.session.exp_string(&command)?; // wait for echo | ||
self.read_until_prompt() | ||
} | ||
|
||
/// Reads output until the next prompt, returning both. | ||
fn read_until_prompt(&mut self) -> Result<(String, String), Box<dyn Error>> { | ||
static UNTIL: std::sync::OnceLock<rexpect::ReadUntil> = std::sync::OnceLock::new(); | ||
let until = UNTIL.get_or_init(|| { | ||
let re = regex::Regex::new(r"toydb(:\d+|@\d+)?>\s+").expect("invalid regex"); | ||
rexpect::ReadUntil::Regex(re) | ||
}); | ||
let (output, prompt) = self.session.reader.read_until(until)?; | ||
Ok((output.trim().to_string(), prompt.trim().to_string())) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
# Tests toysql status. | ||
|
||
cluster nodes=1 | ||
--- | ||
ok | ||
|
||
c1:> SELECT 1 + 2 | ||
--- | ||
c1: foo | ||
c1: | ||
|
||
c1:> !status | ||
--- | ||
c1: foo | ||
c1: |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,96 @@ | ||
#![warn(clippy::all)] | ||
|
||
mod e2e; | ||
mod cluster; | ||
|
||
use cluster::{TestClient, TestCluster}; | ||
|
||
use std::fmt::Write as _; | ||
use std::{collections::HashMap, error::Error}; | ||
use test_each_file::test_each_path; | ||
|
||
// Run goldenscript tests in src/raft/testscripts/node. | ||
test_each_path! { in "tests/scripts" => test_goldenscript } | ||
|
||
fn test_goldenscript(path: &std::path::Path) { | ||
goldenscript::run(&mut Runner::new(), path).expect("goldenscript failed") | ||
} | ||
|
||
/// Runs Raft goldenscript tests. See run() for available commands. | ||
struct Runner { | ||
cluster: Option<TestCluster>, | ||
clients: HashMap<String, TestClient>, | ||
} | ||
|
||
impl Runner { | ||
fn new() -> Self { | ||
Self { cluster: None, clients: HashMap::new() } | ||
} | ||
|
||
/// Ensures a client exists with the given name. Does not return it, since | ||
/// it would require taking a mutable borrow to the entire Runner instead of | ||
/// just the clients map for the lifetime of the client borrow. | ||
fn ensure_client(&mut self, name: &str) -> Result<(), Box<dyn Error>> { | ||
if self.clients.contains_key(name) { | ||
return Ok(()); | ||
} | ||
let Some(cluster) = self.cluster.as_mut() else { | ||
return Err("no cluster".into()); | ||
}; | ||
let client = cluster.connect_toysql()?; | ||
self.clients.insert(name.to_string(), client); | ||
Ok(()) | ||
} | ||
} | ||
|
||
impl goldenscript::Runner for Runner { | ||
/// Runs a goldenscript command. | ||
fn run(&mut self, command: &goldenscript::Command) -> Result<String, Box<dyn Error>> { | ||
let mut output = String::new(); | ||
let mut tags = command.tags.clone(); | ||
|
||
match command.name.as_str() { | ||
// cluster nodes=N | ||
"cluster" => { | ||
let mut args = command.consume_args(); | ||
let nodes = args.lookup_parse("nodes")?.unwrap_or(0); | ||
args.reject_rest()?; | ||
if self.cluster.is_some() { | ||
return Err("cluster already exists".into()); | ||
} | ||
self.cluster = Some(TestCluster::run(nodes)?); | ||
return Ok("foo\n".to_string()); | ||
} | ||
|
||
c if command.prefix.is_none() => return Err(format!("unknown command {c}").into()), | ||
_ => {} | ||
} | ||
|
||
// Take the entire command as a toysql command and run it, using the | ||
// prefix as a client identifier (if any). | ||
if !command.args.is_empty() { | ||
return Err("statements should be given as a command with no args".into()); | ||
} | ||
let prefix = command.prefix.as_deref().unwrap_or_default(); | ||
self.ensure_client(prefix)?; | ||
let client = self.clients.get_mut(prefix).expect("no client"); | ||
let input = &command.name; | ||
|
||
// Execute the command and display the output. | ||
let (stdout, prompt) = client.execute(input)?; | ||
write!(output, "{stdout}")?; | ||
|
||
// If requested, also display the resulting prompt. | ||
if tags.remove("prompt") { | ||
// TODO: goldenscript emits a spurious prefix line if the output | ||
// ends with \n. Fix it. | ||
writeln!(output)?; | ||
write!(output, "{prompt}")?; | ||
} | ||
|
||
if let Some(tag) = tags.iter().next() { | ||
return Err(format!("invalid tag {tag}").into()); | ||
} | ||
|
||
Ok(output) | ||
} | ||
} |