1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-10 16:11:13 -05:00

refactor: use '--reporter' and '--junit-path' flags for 'deno test' (#20031)

This commit adds "--reporter" and "--junit-path" flags to "deno test"
subcommand instead of using "--dot" and "--junit" flags.
This commit is contained in:
Bartek Iwańczuk 2023-08-03 04:05:34 +02:00 committed by GitHub
parent 480894e5c8
commit db287e216d
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 103 additions and 118 deletions

View file

@ -223,8 +223,7 @@ pub enum TestReporterConfig {
#[default] #[default]
Pretty, Pretty,
Dot, Dot,
// Contains path to write to or "-" to print to stdout. Junit,
Junit(String),
} }
#[derive(Clone, Debug, Default, Eq, PartialEq)] #[derive(Clone, Debug, Default, Eq, PartialEq)]
@ -241,6 +240,7 @@ pub struct TestFlags {
pub trace_ops: bool, pub trace_ops: bool,
pub watch: Option<WatchFlags>, pub watch: Option<WatchFlags>,
pub reporter: TestReporterConfig, pub reporter: TestReporterConfig,
pub junit_path: Option<String>,
} }
#[derive(Clone, Debug, Eq, PartialEq)] #[derive(Clone, Debug, Eq, PartialEq)]
@ -1877,21 +1877,17 @@ Directory arguments are expanded to all contained files matching the glob
.arg(no_clear_screen_arg()) .arg(no_clear_screen_arg())
.arg(script_arg().last(true)) .arg(script_arg().last(true))
.arg( .arg(
Arg::new("junit") Arg::new("junit-path")
.long("junit") .long("junit-path")
.value_name("PATH") .value_name("PATH")
.value_hint(ValueHint::FilePath) .value_hint(ValueHint::FilePath)
.help("Write a JUnit XML test report to PATH. Use '-' to write to stdout which is the default when PATH is not provided.") .help("Write a JUnit XML test report to PATH. Use '-' to write to stdout which is the default when PATH is not provided.")
.num_args(0..=1)
.require_equals(true)
.default_missing_value("-")
) )
.arg( .arg(
Arg::new("dot-reporter") Arg::new("reporter")
.long("dot") .long("reporter")
.conflicts_with("junit") .help("Select reporter to use. Default to 'pretty'.")
.help("Use 'dot' test reporter with a concise output format") .value_parser(["pretty", "dot", "junit"])
.action(ArgAction::SetTrue),
) )
) )
} }
@ -3093,20 +3089,24 @@ fn test_parse(flags: &mut Flags, matches: &mut ArgMatches) {
Vec::new() Vec::new()
}; };
let junit_path = matches.remove_one::<String>("junit"); let junit_path = matches.remove_one::<String>("junit-path");
let dot_reporter = matches.get_flag("dot-reporter");
if dot_reporter {
flags.log_level = Some(Level::Error);
}
let reporter = if dot_reporter { let reporter =
TestReporterConfig::Dot if let Some(reporter) = matches.remove_one::<String>("reporter") {
} else if let Some(path) = junit_path { match reporter.as_str() {
TestReporterConfig::Junit(path) "pretty" => TestReporterConfig::Pretty,
"junit" => TestReporterConfig::Junit,
"dot" => TestReporterConfig::Dot,
_ => unreachable!(),
}
} else { } else {
TestReporterConfig::Pretty TestReporterConfig::Pretty
}; };
if matches!(reporter, TestReporterConfig::Dot) {
flags.log_level = Some(Level::Error);
}
flags.subcommand = DenoSubcommand::Test(TestFlags { flags.subcommand = DenoSubcommand::Test(TestFlags {
no_run, no_run,
doc, doc,
@ -3120,6 +3120,7 @@ fn test_parse(flags: &mut Flags, matches: &mut ArgMatches) {
trace_ops, trace_ops,
watch: watch_arg_parse(matches), watch: watch_arg_parse(matches),
reporter, reporter,
junit_path,
}); });
} }
@ -6024,6 +6025,7 @@ mod tests {
coverage_dir: Some("cov".to_string()), coverage_dir: Some("cov".to_string()),
watch: Default::default(), watch: Default::default(),
reporter: Default::default(), reporter: Default::default(),
junit_path: None,
}), }),
unstable: true, unstable: true,
no_prompt: true, no_prompt: true,
@ -6103,6 +6105,7 @@ mod tests {
trace_ops: false, trace_ops: false,
coverage_dir: None, coverage_dir: None,
watch: Default::default(), watch: Default::default(),
junit_path: None,
}), }),
type_check_mode: TypeCheckMode::Local, type_check_mode: TypeCheckMode::Local,
no_prompt: true, no_prompt: true,
@ -6136,6 +6139,7 @@ mod tests {
coverage_dir: None, coverage_dir: None,
watch: Default::default(), watch: Default::default(),
reporter: Default::default(), reporter: Default::default(),
junit_path: None,
}), }),
type_check_mode: TypeCheckMode::Local, type_check_mode: TypeCheckMode::Local,
no_prompt: true, no_prompt: true,
@ -6173,6 +6177,7 @@ mod tests {
coverage_dir: None, coverage_dir: None,
watch: Default::default(), watch: Default::default(),
reporter: Default::default(), reporter: Default::default(),
junit_path: None,
}), }),
no_prompt: true, no_prompt: true,
type_check_mode: TypeCheckMode::Local, type_check_mode: TypeCheckMode::Local,
@ -6183,27 +6188,28 @@ mod tests {
} }
#[test] #[test]
fn test_dot() { fn test_reporter() {
let r = flags_from_vec(svec!["deno", "test", "--dot"]); let r = flags_from_vec(svec!["deno", "test", "--reporter=pretty"]);
assert_eq!(
r.unwrap(),
Flags {
subcommand: DenoSubcommand::Test(TestFlags {
reporter: TestReporterConfig::Pretty,
..Default::default()
}),
no_prompt: true,
type_check_mode: TypeCheckMode::Local,
..Flags::default()
}
);
let r = flags_from_vec(svec!["deno", "test", "--reporter=dot"]);
assert_eq!( assert_eq!(
r.unwrap(), r.unwrap(),
Flags { Flags {
subcommand: DenoSubcommand::Test(TestFlags { subcommand: DenoSubcommand::Test(TestFlags {
no_run: false,
doc: false,
fail_fast: None,
filter: None,
reporter: TestReporterConfig::Dot, reporter: TestReporterConfig::Dot,
allow_none: false, ..Default::default()
shuffle: None,
files: FileFlags {
include: vec![],
ignore: vec![],
},
concurrent_jobs: None,
trace_ops: false,
coverage_dir: None,
watch: Default::default(),
}), }),
no_prompt: true, no_prompt: true,
type_check_mode: TypeCheckMode::Local, type_check_mode: TypeCheckMode::Local,
@ -6212,7 +6218,42 @@ mod tests {
} }
); );
let r = flags_from_vec(svec!["deno", "test", "--dot", "--junit"]); let r = flags_from_vec(svec!["deno", "test", "--reporter=junit"]);
assert_eq!(
r.unwrap(),
Flags {
subcommand: DenoSubcommand::Test(TestFlags {
reporter: TestReporterConfig::Junit,
..Default::default()
}),
no_prompt: true,
type_check_mode: TypeCheckMode::Local,
..Flags::default()
}
);
let r = flags_from_vec(svec![
"deno",
"test",
"--reporter=dot",
"--junit-path=report.xml"
]);
assert_eq!(
r.unwrap(),
Flags {
subcommand: DenoSubcommand::Test(TestFlags {
reporter: TestReporterConfig::Dot,
junit_path: Some("report.xml".to_string()),
..Default::default()
}),
no_prompt: true,
type_check_mode: TypeCheckMode::Local,
log_level: Some(Level::Error),
..Flags::default()
}
);
let r = flags_from_vec(svec!["deno", "test", "--junit-path"]);
assert!(r.is_err()); assert!(r.is_err());
} }
@ -6238,6 +6279,7 @@ mod tests {
coverage_dir: None, coverage_dir: None,
watch: Default::default(), watch: Default::default(),
reporter: Default::default(), reporter: Default::default(),
junit_path: None,
}), }),
no_prompt: true, no_prompt: true,
type_check_mode: TypeCheckMode::Local, type_check_mode: TypeCheckMode::Local,
@ -6270,6 +6312,7 @@ mod tests {
no_clear_screen: false, no_clear_screen: false,
}), }),
reporter: Default::default(), reporter: Default::default(),
junit_path: None,
}), }),
no_prompt: true, no_prompt: true,
type_check_mode: TypeCheckMode::Local, type_check_mode: TypeCheckMode::Local,
@ -6301,6 +6344,7 @@ mod tests {
no_clear_screen: false, no_clear_screen: false,
}), }),
reporter: Default::default(), reporter: Default::default(),
junit_path: None,
}), }),
no_prompt: true, no_prompt: true,
type_check_mode: TypeCheckMode::Local, type_check_mode: TypeCheckMode::Local,
@ -6334,66 +6378,7 @@ mod tests {
no_clear_screen: true, no_clear_screen: true,
}), }),
reporter: Default::default(), reporter: Default::default(),
}), junit_path: None,
type_check_mode: TypeCheckMode::Local,
no_prompt: true,
..Flags::default()
}
);
}
#[test]
fn test_junit_default() {
let r = flags_from_vec(svec!["deno", "test", "--junit"]);
assert_eq!(
r.unwrap(),
Flags {
subcommand: DenoSubcommand::Test(TestFlags {
no_run: false,
doc: false,
fail_fast: None,
filter: None,
allow_none: false,
shuffle: None,
files: FileFlags {
include: vec![],
ignore: vec![],
},
concurrent_jobs: None,
trace_ops: false,
coverage_dir: None,
watch: Default::default(),
reporter: TestReporterConfig::Junit("-".to_string()),
}),
type_check_mode: TypeCheckMode::Local,
no_prompt: true,
..Flags::default()
}
);
}
#[test]
fn test_junit_with_path() {
let r = flags_from_vec(svec!["deno", "test", "--junit=junit.xml"]);
assert_eq!(
r.unwrap(),
Flags {
subcommand: DenoSubcommand::Test(TestFlags {
no_run: false,
doc: false,
fail_fast: None,
filter: None,
allow_none: false,
shuffle: None,
files: FileFlags {
include: vec![],
ignore: vec![],
},
concurrent_jobs: None,
trace_ops: false,
coverage_dir: None,
watch: Default::default(),
reporter: TestReporterConfig::Junit("junit.xml".to_string()),
}), }),
type_check_mode: TypeCheckMode::Local, type_check_mode: TypeCheckMode::Local,
no_prompt: true, no_prompt: true,

View file

@ -228,6 +228,7 @@ pub struct TestOptions {
pub concurrent_jobs: NonZeroUsize, pub concurrent_jobs: NonZeroUsize,
pub trace_ops: bool, pub trace_ops: bool,
pub reporter: TestReporterConfig, pub reporter: TestReporterConfig,
pub junit_path: Option<String>,
} }
impl TestOptions { impl TestOptions {
@ -253,6 +254,7 @@ impl TestOptions {
shuffle: test_flags.shuffle, shuffle: test_flags.shuffle,
trace_ops: test_flags.trace_ops, trace_ops: test_flags.trace_ops,
reporter: test_flags.reporter, reporter: test_flags.reporter,
junit_path: test_flags.junit_path,
}) })
} }
} }

View file

@ -334,19 +334,19 @@ itest!(steps_ignored_steps {
}); });
itest!(steps_dot_passing_steps { itest!(steps_dot_passing_steps {
args: "test --dot test/steps/passing_steps.ts", args: "test --reporter=dot test/steps/passing_steps.ts",
exit_code: 0, exit_code: 0,
output: "test/steps/passing_steps.dot.out", output: "test/steps/passing_steps.dot.out",
}); });
itest!(steps_dot_failing_steps { itest!(steps_dot_failing_steps {
args: "test --dot test/steps/failing_steps.ts", args: "test --reporter=dot test/steps/failing_steps.ts",
exit_code: 1, exit_code: 1,
output: "test/steps/failing_steps.dot.out", output: "test/steps/failing_steps.dot.out",
}); });
itest!(steps_dot_ignored_steps { itest!(steps_dot_ignored_steps {
args: "test --dot test/steps/ignored_steps.ts", args: "test --reporter=dot test/steps/ignored_steps.ts",
exit_code: 0, exit_code: 0,
output: "test/steps/ignored_steps.dot.out", output: "test/steps/ignored_steps.dot.out",
}); });

View file

@ -361,6 +361,7 @@ struct TestSpecifiersOptions {
log_level: Option<log::Level>, log_level: Option<log::Level>,
specifier: TestSpecifierOptions, specifier: TestSpecifierOptions,
reporter: TestReporterConfig, reporter: TestReporterConfig,
junit_path: Option<String>,
} }
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
@ -394,28 +395,23 @@ impl TestSummary {
fn get_test_reporter(options: &TestSpecifiersOptions) -> Box<dyn TestReporter> { fn get_test_reporter(options: &TestSpecifiersOptions) -> Box<dyn TestReporter> {
let parallel = options.concurrent_jobs.get() > 1; let parallel = options.concurrent_jobs.get() > 1;
match &options.reporter { let reporter: Box<dyn TestReporter> = match &options.reporter {
TestReporterConfig::Dot => Box::new(DotTestReporter::new()), TestReporterConfig::Dot => Box::new(DotTestReporter::new()),
TestReporterConfig::Pretty => Box::new(PrettyTestReporter::new( TestReporterConfig::Pretty => Box::new(PrettyTestReporter::new(
parallel, parallel,
options.log_level != Some(Level::Error), options.log_level != Some(Level::Error),
)), )),
TestReporterConfig::Junit(path) => { TestReporterConfig::Junit => {
let junit = Box::new(JunitTestReporter::new(path.clone())); Box::new(JunitTestReporter::new("-".to_string()))
// If junit is writing to stdout, only enable the junit reporter
if path == "-" {
junit
} else {
Box::new(CompoundTestReporter::new(vec![
Box::new(PrettyTestReporter::new(
parallel,
options.log_level != Some(Level::Error),
)),
junit,
]))
}
} }
};
if let Some(junit_path) = &options.junit_path {
let junit = Box::new(JunitTestReporter::new(junit_path.to_string()));
return Box::new(CompoundTestReporter::new(vec![reporter, junit]));
} }
reporter
} }
/// Test a single specifier as documentation containing test programs, an executable test module or /// Test a single specifier as documentation containing test programs, an executable test module or
@ -1162,6 +1158,7 @@ pub async fn run_tests(
fail_fast: test_options.fail_fast, fail_fast: test_options.fail_fast,
log_level, log_level,
reporter: test_options.reporter, reporter: test_options.reporter,
junit_path: test_options.junit_path,
specifier: TestSpecifierOptions { specifier: TestSpecifierOptions {
filter: TestFilter::from_flag(&test_options.filter), filter: TestFilter::from_flag(&test_options.filter),
shuffle: test_options.shuffle, shuffle: test_options.shuffle,
@ -1293,6 +1290,7 @@ pub async fn run_tests_with_watch(
fail_fast: test_options.fail_fast, fail_fast: test_options.fail_fast,
log_level, log_level,
reporter: test_options.reporter, reporter: test_options.reporter,
junit_path: test_options.junit_path,
specifier: TestSpecifierOptions { specifier: TestSpecifierOptions {
filter: TestFilter::from_flag(&test_options.filter), filter: TestFilter::from_flag(&test_options.filter),
shuffle: test_options.shuffle, shuffle: test_options.shuffle,