2022-02-06 14:06:27 +03:00
|
|
|
use std::{fs::File, io::Read, path::PathBuf};
|
|
|
|
|
|
|
|
use tempfile::{tempdir, TempDir};
|
|
|
|
|
|
|
|
mod common;
|
2022-02-23 00:00:12 +03:00
|
|
|
use common::hyperfine;
|
2022-02-06 14:06:27 +03:00
|
|
|
|
|
|
|
struct ExecutionOrderTest {
|
|
|
|
cmd: assert_cmd::Command,
|
|
|
|
expected_content: String,
|
|
|
|
logfile_path: PathBuf,
|
|
|
|
#[allow(dead_code)]
|
|
|
|
tempdir: TempDir,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ExecutionOrderTest {
|
|
|
|
fn new() -> Self {
|
|
|
|
let tempdir = tempdir().unwrap();
|
|
|
|
let logfile_path = tempdir.path().join("output.log");
|
|
|
|
|
|
|
|
ExecutionOrderTest {
|
2022-02-23 00:00:12 +03:00
|
|
|
cmd: hyperfine(),
|
2022-02-06 14:06:27 +03:00
|
|
|
expected_content: String::new(),
|
|
|
|
logfile_path,
|
|
|
|
tempdir,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn arg<S: AsRef<str>>(&mut self, arg: S) -> &mut Self {
|
|
|
|
self.cmd.arg(arg.as_ref());
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_command(&self, output: &str) -> String {
|
|
|
|
format!(
|
|
|
|
"echo {output} >> {path}",
|
|
|
|
output = output,
|
|
|
|
path = self.logfile_path.to_string_lossy()
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn command(&mut self, output: &str) -> &mut Self {
|
|
|
|
self.arg(self.get_command(output));
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
fn setup(&mut self, output: &str) -> &mut Self {
|
|
|
|
self.arg("--setup");
|
|
|
|
self.command(output)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn prepare(&mut self, output: &str) -> &mut Self {
|
|
|
|
self.arg("--prepare");
|
|
|
|
self.command(output)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn cleanup(&mut self, output: &str) -> &mut Self {
|
|
|
|
self.arg("--cleanup");
|
|
|
|
self.command(output)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn expect_output(&mut self, output: &str) -> &mut Self {
|
|
|
|
self.expected_content.push_str(output);
|
|
|
|
|
|
|
|
#[cfg(windows)]
|
|
|
|
{
|
|
|
|
self.expected_content.push_str(" \r");
|
|
|
|
}
|
|
|
|
|
|
|
|
self.expected_content.push('\n');
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
fn run(&mut self) {
|
|
|
|
self.cmd.assert().success();
|
|
|
|
|
|
|
|
let mut f = File::open(&self.logfile_path).unwrap();
|
|
|
|
let mut content = String::new();
|
|
|
|
f.read_to_string(&mut content).unwrap();
|
|
|
|
|
|
|
|
assert_eq!(content, self.expected_content);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for ExecutionOrderTest {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self::new()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn benchmarks_are_executed_sequentially_one() {
|
|
|
|
ExecutionOrderTest::new()
|
|
|
|
.arg("--runs=1")
|
|
|
|
.command("command 1")
|
|
|
|
.command("command 2")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn benchmarks_are_executed_sequentially() {
|
|
|
|
ExecutionOrderTest::new()
|
|
|
|
.arg("--runs=2")
|
|
|
|
.command("command 1")
|
|
|
|
.command("command 2")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn warmup_runs_are_executed_before_benchmarking_runs() {
|
|
|
|
ExecutionOrderTest::new()
|
|
|
|
.arg("--runs=2")
|
|
|
|
.arg("--warmup=3")
|
|
|
|
.command("command 1")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn setup_commands_are_executed_before_each_series_of_timing_runs() {
|
|
|
|
ExecutionOrderTest::new()
|
|
|
|
.arg("--runs=2")
|
|
|
|
.setup("setup")
|
|
|
|
.command("command 1")
|
|
|
|
.command("command 2")
|
|
|
|
.expect_output("setup")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("setup")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn prepare_commands_are_executed_before_each_timing_run() {
|
|
|
|
ExecutionOrderTest::new()
|
|
|
|
.arg("--runs=2")
|
|
|
|
.prepare("prepare")
|
|
|
|
.command("command 1")
|
|
|
|
.command("command 2")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
2022-02-07 00:49:41 +03:00
|
|
|
#[test]
|
|
|
|
fn prepare_commands_are_executed_before_each_warmup() {
|
|
|
|
ExecutionOrderTest::new()
|
|
|
|
.arg("--warmup=2")
|
|
|
|
.arg("--runs=1")
|
|
|
|
.prepare("prepare")
|
|
|
|
.command("command 1")
|
|
|
|
.command("command 2")
|
|
|
|
// warmup 1
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command 1")
|
|
|
|
// benchmark 1
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command 1")
|
|
|
|
// warmup 2
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command 2")
|
|
|
|
// benchmark 2
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
2022-02-06 14:06:27 +03:00
|
|
|
#[test]
|
|
|
|
fn cleanup_commands_are_executed_once_after_each_benchmark() {
|
|
|
|
ExecutionOrderTest::new()
|
|
|
|
.arg("--runs=2")
|
|
|
|
.cleanup("cleanup")
|
|
|
|
.command("command 1")
|
|
|
|
.command("command 2")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("cleanup")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.expect_output("cleanup")
|
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn setup_prepare_cleanup_combined() {
|
|
|
|
ExecutionOrderTest::new()
|
2022-02-07 00:49:41 +03:00
|
|
|
.arg("--warmup=1")
|
2022-02-06 14:06:27 +03:00
|
|
|
.arg("--runs=2")
|
2022-02-07 00:49:41 +03:00
|
|
|
.setup("setup")
|
|
|
|
.prepare("prepare")
|
|
|
|
.command("command1")
|
|
|
|
.command("command2")
|
|
|
|
.cleanup("cleanup")
|
|
|
|
// 1
|
|
|
|
.expect_output("setup")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command1")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command1")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command1")
|
|
|
|
.expect_output("cleanup")
|
|
|
|
// 2
|
|
|
|
.expect_output("setup")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command2")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command2")
|
|
|
|
.expect_output("prepare")
|
|
|
|
.expect_output("command2")
|
|
|
|
.expect_output("cleanup")
|
2022-02-06 14:06:27 +03:00
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn single_parameter_value() {
|
|
|
|
ExecutionOrderTest::new()
|
|
|
|
.arg("--runs=2")
|
|
|
|
.arg("--parameter-list")
|
|
|
|
.arg("number")
|
|
|
|
.arg("1,2,3")
|
|
|
|
.command("command {number}")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 1")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.expect_output("command 2")
|
|
|
|
.expect_output("command 3")
|
|
|
|
.expect_output("command 3")
|
|
|
|
.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn multiple_parameter_values() {
|
|
|
|
ExecutionOrderTest::new()
|
|
|
|
.arg("--runs=2")
|
|
|
|
.arg("--parameter-list")
|
|
|
|
.arg("number")
|
|
|
|
.arg("1,2,3")
|
|
|
|
.arg("--parameter-list")
|
|
|
|
.arg("letter")
|
|
|
|
.arg("a,b")
|
|
|
|
.command("command {number} {letter}")
|
|
|
|
.expect_output("command 1 a")
|
|
|
|
.expect_output("command 1 a")
|
|
|
|
.expect_output("command 2 a")
|
|
|
|
.expect_output("command 2 a")
|
|
|
|
.expect_output("command 3 a")
|
|
|
|
.expect_output("command 3 a")
|
|
|
|
.expect_output("command 1 b")
|
|
|
|
.expect_output("command 1 b")
|
|
|
|
.expect_output("command 2 b")
|
|
|
|
.expect_output("command 2 b")
|
|
|
|
.expect_output("command 3 b")
|
|
|
|
.expect_output("command 3 b")
|
|
|
|
.run();
|
|
|
|
}
|