2024-07-25 13:34:43 +01:00
|
|
|
use std::{
|
|
|
|
env::{self, consts::EXE_SUFFIX},
|
|
|
|
process::{Command, Stdio},
|
|
|
|
str::from_utf8,
|
|
|
|
};
|
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
struct Cmd<'a> {
|
|
|
|
current_dir: Option<&'a str>,
|
|
|
|
args: &'a [&'a str],
|
|
|
|
stdout: Option<&'a str>,
|
|
|
|
full_stdout: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Cmd<'a> {
|
|
|
|
#[inline]
|
|
|
|
fn current_dir(&mut self, current_dir: &'a str) -> &mut Self {
|
|
|
|
self.current_dir = Some(current_dir);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn args(&mut self, args: &'a [&'a str]) -> &mut Self {
|
|
|
|
self.args = args;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn stdout(&mut self, stdout: &'a str) -> &mut Self {
|
|
|
|
self.stdout = Some(stdout);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn full_stdout(&mut self) -> &mut Self {
|
|
|
|
self.full_stdout = true;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
fn assert(&self, success: bool) {
|
|
|
|
let rustlings_bin = {
|
|
|
|
let mut path = env::current_exe().unwrap();
|
|
|
|
// Pop test binary name
|
|
|
|
path.pop();
|
|
|
|
// Pop `/deps`
|
|
|
|
path.pop();
|
|
|
|
|
|
|
|
path.push("rustlings");
|
|
|
|
let mut path = path.into_os_string();
|
|
|
|
path.push(EXE_SUFFIX);
|
|
|
|
path
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut cmd = Command::new(rustlings_bin);
|
|
|
|
|
|
|
|
if let Some(current_dir) = self.current_dir {
|
|
|
|
cmd.current_dir(current_dir);
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd.args(self.args)
|
|
|
|
.stdin(Stdio::null())
|
|
|
|
.stderr(Stdio::null());
|
|
|
|
|
|
|
|
let status = if let Some(expected_stdout) = self.stdout {
|
|
|
|
let output = cmd.output().unwrap();
|
|
|
|
let stdout = from_utf8(&output.stdout).unwrap();
|
|
|
|
|
|
|
|
if self.full_stdout {
|
|
|
|
assert_eq!(stdout, expected_stdout);
|
|
|
|
} else {
|
|
|
|
assert!(stdout.contains(expected_stdout));
|
|
|
|
}
|
|
|
|
|
|
|
|
output.status
|
|
|
|
} else {
|
|
|
|
cmd.stdout(Stdio::null()).status().unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
assert_eq!(status.success(), success);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn success(&self) {
|
|
|
|
self.assert(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn fail(&self) {
|
|
|
|
self.assert(false);
|
|
|
|
}
|
|
|
|
}
|
2019-03-20 20:05:45 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn fails_when_in_wrong_dir() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default().current_dir("tests").fail();
|
2019-03-20 20:05:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn run_single_compile_success() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default()
|
|
|
|
.current_dir("tests/fixture/success")
|
|
|
|
.args(&["run", "compSuccess"])
|
2019-03-20 20:05:45 +00:00
|
|
|
.success();
|
|
|
|
}
|
|
|
|
|
2019-04-07 17:49:34 +01:00
|
|
|
#[test]
|
|
|
|
fn run_single_compile_failure() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default()
|
|
|
|
.current_dir("tests/fixture/failure")
|
|
|
|
.args(&["run", "compFailure"])
|
|
|
|
.fail();
|
2019-04-07 17:49:34 +01:00
|
|
|
}
|
|
|
|
|
2019-03-20 20:05:45 +00:00
|
|
|
#[test]
|
|
|
|
fn run_single_test_success() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default()
|
|
|
|
.current_dir("tests/fixture/success")
|
|
|
|
.args(&["run", "testSuccess"])
|
2019-03-20 20:05:45 +00:00
|
|
|
.success();
|
|
|
|
}
|
|
|
|
|
2019-04-07 17:49:34 +01:00
|
|
|
#[test]
|
|
|
|
fn run_single_test_failure() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default()
|
|
|
|
.current_dir("tests/fixture/failure")
|
|
|
|
.args(&["run", "testFailure"])
|
|
|
|
.fail();
|
2019-04-07 17:49:34 +01:00
|
|
|
}
|
|
|
|
|
2019-05-09 18:16:06 +01:00
|
|
|
#[test]
|
|
|
|
fn run_single_test_not_passed() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default()
|
|
|
|
.current_dir("tests/fixture/failure")
|
|
|
|
.args(&["run", "testNotPassed.rs"])
|
|
|
|
.fail();
|
2019-05-09 18:16:06 +01:00
|
|
|
}
|
|
|
|
|
2019-03-20 20:05:45 +00:00
|
|
|
#[test]
|
|
|
|
fn run_single_test_no_exercise() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default()
|
2019-04-07 17:49:34 +01:00
|
|
|
.current_dir("tests/fixture/failure")
|
2024-07-25 13:34:43 +01:00
|
|
|
.args(&["run", "compNoExercise.rs"])
|
|
|
|
.fail();
|
2019-03-20 20:05:45 +00:00
|
|
|
}
|
2019-11-11 16:19:50 +00:00
|
|
|
|
2022-08-17 15:43:48 +01:00
|
|
|
#[test]
|
|
|
|
fn reset_single_exercise() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default().args(&["reset", "intro1"]).success();
|
2022-08-17 15:43:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn reset_no_exercise() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default().args(&["reset"]).fail();
|
2022-08-17 15:43:48 +01:00
|
|
|
}
|
|
|
|
|
2019-11-11 16:19:50 +00:00
|
|
|
#[test]
|
|
|
|
fn get_hint_for_single_test() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default()
|
2019-11-11 16:19:50 +00:00
|
|
|
.current_dir("tests/fixture/failure")
|
2024-07-25 13:34:43 +01:00
|
|
|
.args(&["hint", "testFailure"])
|
|
|
|
.stdout("Hello!\n")
|
|
|
|
.full_stdout()
|
|
|
|
.success();
|
2019-11-11 16:28:19 +00:00
|
|
|
}
|
2019-11-11 16:21:06 +00:00
|
|
|
|
2019-11-12 10:35:40 +00:00
|
|
|
#[test]
|
|
|
|
fn run_compile_exercise_does_not_prompt() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default()
|
2019-11-12 10:35:40 +00:00
|
|
|
.current_dir("tests/fixture/state")
|
2024-07-25 13:34:43 +01:00
|
|
|
.args(&["run", "pending_exercise"])
|
|
|
|
.success();
|
2019-11-12 10:35:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn run_test_exercise_does_not_prompt() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default()
|
2019-11-12 10:35:40 +00:00
|
|
|
.current_dir("tests/fixture/state")
|
2024-07-25 13:34:43 +01:00
|
|
|
.args(&["run", "pending_test_exercise"])
|
|
|
|
.success();
|
2019-11-12 10:35:40 +00:00
|
|
|
}
|
2020-06-04 15:31:17 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn run_single_test_success_with_output() {
|
2024-07-25 13:34:43 +01:00
|
|
|
Cmd::default()
|
|
|
|
.current_dir("tests/fixture/success")
|
|
|
|
.args(&["run", "testSuccess"])
|
|
|
|
.stdout("\nTHIS TEST TOO SHALL PASS\n")
|
|
|
|
.success();
|
2020-08-10 15:42:54 +01:00
|
|
|
}
|