use assert_cmd::prelude::*; use pretty_assertions::assert_eq; use std::io::Write; use std::process::Command; use tempfile::NamedTempFile; type TestResult = Result<(), Box>; #[cfg(test)] fn run_test(input: &str, expected: &str) -> TestResult { let mut file = NamedTempFile::new()?; let name = file.path(); let mut cmd = Command::cargo_bin("engine-q")?; cmd.arg(name); writeln!(file, "{}", input)?; let output = cmd.output()?; let stderr = String::from_utf8_lossy(&output.stderr).to_string(); let stdout = String::from_utf8_lossy(&output.stdout).to_string(); println!("stdout: {}", stdout); println!("stderr: {}", stderr); assert!(output.status.success()); assert_eq!(stdout.trim(), expected); Ok(()) } #[cfg(test)] fn fail_test(input: &str, expected: &str) -> TestResult { let mut file = NamedTempFile::new()?; let name = file.path(); let mut cmd = Command::cargo_bin("engine-q")?; cmd.arg(name); writeln!(file, "{}", input)?; let output = cmd.output()?; let stderr = String::from_utf8_lossy(&output.stderr).to_string(); let stdout = String::from_utf8_lossy(&output.stdout).to_string(); println!("stdout: {}", stdout); println!("stderr: {}", stderr); assert!(stderr.contains(expected)); Ok(()) } #[test] fn add_simple() -> TestResult { run_test("3 + 4", "7") } #[test] fn add_simple2() -> TestResult { run_test("3 + 4 + 9", "16") } #[test] fn broken_math() -> TestResult { fail_test("3 + ", "incomplete") } #[test] fn if_test1() -> TestResult { run_test("if $true { 10 } else { 20 } ", "10") } #[test] fn if_test2() -> TestResult { run_test("if $false { 10 } else { 20 } ", "20") } #[test] fn simple_if() -> TestResult { run_test("if $true { 10 } ", "10") } #[test] fn simple_if2() -> TestResult { run_test("if $false { 10 } ", "") } #[test] fn if_cond() -> TestResult { run_test("if 2 < 3 { 3 } ", "3") } #[test] fn if_cond2() -> TestResult { run_test("if 2 > 3 { 3 } ", "") } #[test] fn if_cond3() -> TestResult { run_test("if 2 < 3 { 5 } else { 4 } ", "5") } #[test] fn if_cond4() -> TestResult { run_test("if 2 > 3 { 5 } else { 4 } ", "4") } #[test] fn if_elseif1() -> TestResult { run_test("if 2 > 3 { 5 } else if 6 < 7 { 4 } ", "4") } #[test] fn if_elseif2() -> TestResult { run_test("if 2 < 3 { 5 } else if 6 < 7 { 4 } else { 8 } ", "5") } #[test] fn if_elseif3() -> TestResult { run_test("if 2 > 3 { 5 } else if 6 > 7 { 4 } else { 8 } ", "8") } #[test] fn if_elseif4() -> TestResult { run_test("if 2 > 3 { 5 } else if 6 < 7 { 4 } else { 8 } ", "4") } #[test] fn no_scope_leak1() -> TestResult { fail_test( "if $false { let $x = 10 } else { let $x = 20 }; $x", "variable not found", ) } #[test] fn no_scope_leak2() -> TestResult { fail_test( "def foo [] { $x }; def bar [] { let $x = 10; foo }; bar", "Variable not found", ) } #[test] fn no_scope_leak3() -> TestResult { run_test( "def foo [$x] { $x }; def bar [] { let $x = 10; foo 20}; bar", "20", ) } #[test] fn no_scope_leak4() -> TestResult { run_test( "def foo [$x] { $x }; def bar [] { let $x = 10; (foo 20) + $x}; bar", "30", ) } #[test] fn simple_var_closing() -> TestResult { run_test("let $x = 10; def foo [] { $x }; foo", "10") } #[test] fn predecl_check() -> TestResult { run_test("def bob [] { sam }; def sam [] { 3 }; bob", "3") } #[test] fn def_with_no_dollar() -> TestResult { run_test("def bob [x] { $x + 3 }; bob 4", "7") } #[test] fn env_shorthand() -> TestResult { run_test("FOO=BAR if $false { 3 } else { 4 }", "4") } #[test] fn floating_add() -> TestResult { run_test("10.1 + 0.8", "10.9") } #[test] fn subcommand() -> TestResult { run_test("def foo [] {}; def \"foo bar\" [] {3}; foo bar", "3") } #[test] fn alias_1() -> TestResult { run_test("def foo [$x] { $x + 10 }; alias f = foo; f 100", "110") } #[test] fn alias_2() -> TestResult { run_test( "def foo [$x $y] { $x + $y + 10 }; alias f = foo 33; f 100", "143", ) }