mirror of
https://github.com/denoland/deno.git
synced 2025-01-11 00:21:05 -05:00
refactor(cli): Creating a TestReporter trait (#19786)
This PR breaks the addition of the `TestReporter` trait and refactoring of `PrettyTestReporter` out of #19747. The goal is to enable the addition of test reporters, including machine readable output.
This commit is contained in:
parent
8dd9d5f523
commit
96efe3c176
1 changed files with 405 additions and 345 deletions
|
@ -377,6 +377,49 @@ impl TestSummary {
|
|||
}
|
||||
}
|
||||
|
||||
trait TestReporter {
|
||||
fn report_register(&mut self, description: &TestDescription);
|
||||
fn report_plan(&mut self, plan: &TestPlan);
|
||||
fn report_wait(&mut self, description: &TestDescription);
|
||||
fn report_output(&mut self, output: &[u8]);
|
||||
fn report_result(
|
||||
&mut self,
|
||||
description: &TestDescription,
|
||||
result: &TestResult,
|
||||
elapsed: u64,
|
||||
);
|
||||
fn report_uncaught_error(&mut self, origin: &str, error: Box<JsError>);
|
||||
fn report_step_register(&mut self, description: &TestStepDescription);
|
||||
fn report_step_wait(&mut self, description: &TestStepDescription);
|
||||
fn report_step_result(
|
||||
&mut self,
|
||||
desc: &TestStepDescription,
|
||||
result: &TestStepResult,
|
||||
elapsed: u64,
|
||||
tests: &IndexMap<usize, TestDescription>,
|
||||
test_steps: &IndexMap<usize, TestStepDescription>,
|
||||
);
|
||||
fn report_summary(
|
||||
&mut self,
|
||||
elapsed: &Duration,
|
||||
tests: &IndexMap<usize, TestDescription>,
|
||||
test_steps: &IndexMap<usize, TestStepDescription>,
|
||||
);
|
||||
fn report_sigint(
|
||||
&mut self,
|
||||
tests_pending: &HashSet<usize>,
|
||||
tests: &IndexMap<usize, TestDescription>,
|
||||
test_steps: &IndexMap<usize, TestStepDescription>,
|
||||
);
|
||||
}
|
||||
|
||||
fn get_test_reporter(options: &TestSpecifiersOptions) -> Box<dyn TestReporter> {
|
||||
Box::new(PrettyTestReporter::new(
|
||||
options.concurrent_jobs.get() > 1,
|
||||
options.log_level != Some(Level::Error),
|
||||
))
|
||||
}
|
||||
|
||||
struct PrettyTestReporter {
|
||||
parallel: bool,
|
||||
echo_output: bool,
|
||||
|
@ -387,6 +430,7 @@ struct PrettyTestReporter {
|
|||
started_tests: bool,
|
||||
child_results_buffer:
|
||||
HashMap<usize, IndexMap<usize, (TestStepDescription, TestStepResult, u64)>>,
|
||||
summary: TestSummary,
|
||||
}
|
||||
|
||||
impl PrettyTestReporter {
|
||||
|
@ -400,6 +444,7 @@ impl PrettyTestReporter {
|
|||
did_have_user_output: false,
|
||||
started_tests: false,
|
||||
child_results_buffer: Default::default(),
|
||||
summary: TestSummary::new(),
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -511,291 +556,6 @@ impl PrettyTestReporter {
|
|||
}
|
||||
}
|
||||
|
||||
fn report_register(&mut self, _description: &TestDescription) {}
|
||||
|
||||
fn report_plan(&mut self, plan: &TestPlan) {
|
||||
if self.parallel {
|
||||
return;
|
||||
}
|
||||
let inflection = if plan.total == 1 { "test" } else { "tests" };
|
||||
println!(
|
||||
"{}",
|
||||
colors::gray(format!(
|
||||
"running {} {} from {}",
|
||||
plan.total,
|
||||
inflection,
|
||||
self.to_relative_path_or_remote_url(&plan.origin)
|
||||
))
|
||||
);
|
||||
self.in_new_line = true;
|
||||
}
|
||||
|
||||
fn report_wait(&mut self, description: &TestDescription) {
|
||||
if !self.parallel {
|
||||
self.force_report_wait(description);
|
||||
}
|
||||
self.started_tests = true;
|
||||
}
|
||||
|
||||
fn report_output(&mut self, output: &[u8]) {
|
||||
if !self.echo_output {
|
||||
return;
|
||||
}
|
||||
|
||||
if !self.did_have_user_output && self.started_tests {
|
||||
self.did_have_user_output = true;
|
||||
if !self.in_new_line {
|
||||
println!();
|
||||
}
|
||||
println!("{}", colors::gray("------- output -------"));
|
||||
self.in_new_line = true;
|
||||
}
|
||||
|
||||
// output everything to stdout in order to prevent
|
||||
// stdout and stderr racing
|
||||
std::io::stdout().write_all(output).unwrap();
|
||||
}
|
||||
|
||||
fn report_result(
|
||||
&mut self,
|
||||
description: &TestDescription,
|
||||
result: &TestResult,
|
||||
elapsed: u64,
|
||||
) {
|
||||
if self.parallel {
|
||||
self.force_report_wait(description);
|
||||
}
|
||||
|
||||
self.write_output_end();
|
||||
if self.in_new_line || self.scope_test_id != Some(description.id) {
|
||||
self.force_report_wait(description);
|
||||
}
|
||||
|
||||
let status = match result {
|
||||
TestResult::Ok => colors::green("ok").to_string(),
|
||||
TestResult::Ignored => colors::yellow("ignored").to_string(),
|
||||
TestResult::Failed(failure) => failure.format_label(),
|
||||
TestResult::Cancelled => colors::gray("cancelled").to_string(),
|
||||
};
|
||||
print!(" {}", status);
|
||||
if let TestResult::Failed(failure) = result {
|
||||
if let Some(inline_summary) = failure.format_inline_summary() {
|
||||
print!(" ({})", inline_summary)
|
||||
}
|
||||
}
|
||||
println!(
|
||||
" {}",
|
||||
colors::gray(format!("({})", display::human_elapsed(elapsed.into())))
|
||||
);
|
||||
self.in_new_line = true;
|
||||
self.scope_test_id = None;
|
||||
}
|
||||
|
||||
fn report_uncaught_error(&mut self, origin: &str, _error: &JsError) {
|
||||
if !self.in_new_line {
|
||||
println!();
|
||||
}
|
||||
println!(
|
||||
"Uncaught error from {} {}",
|
||||
self.to_relative_path_or_remote_url(origin),
|
||||
colors::red("FAILED")
|
||||
);
|
||||
self.in_new_line = true;
|
||||
self.did_have_user_output = false;
|
||||
}
|
||||
|
||||
fn report_step_register(&mut self, _description: &TestStepDescription) {}
|
||||
|
||||
fn report_step_wait(&mut self, description: &TestStepDescription) {
|
||||
if !self.parallel && self.scope_test_id == Some(description.parent_id) {
|
||||
self.force_report_step_wait(description);
|
||||
}
|
||||
}
|
||||
|
||||
fn report_step_result(
|
||||
&mut self,
|
||||
desc: &TestStepDescription,
|
||||
result: &TestStepResult,
|
||||
elapsed: u64,
|
||||
tests: &IndexMap<usize, TestDescription>,
|
||||
test_steps: &IndexMap<usize, TestStepDescription>,
|
||||
) {
|
||||
if self.parallel {
|
||||
self.write_output_end();
|
||||
print!(
|
||||
"{} {} ...",
|
||||
colors::gray(format!(
|
||||
"{} =>",
|
||||
self.to_relative_path_or_remote_url(&desc.origin)
|
||||
)),
|
||||
self.format_test_step_ancestry(desc, tests, test_steps)
|
||||
);
|
||||
self.in_new_line = false;
|
||||
self.scope_test_id = Some(desc.id);
|
||||
self.force_report_step_result(desc, result, elapsed);
|
||||
} else {
|
||||
let sibling_results =
|
||||
self.child_results_buffer.entry(desc.parent_id).or_default();
|
||||
if self.scope_test_id == Some(desc.id)
|
||||
|| self.scope_test_id == Some(desc.parent_id)
|
||||
{
|
||||
let sibling_results = std::mem::take(sibling_results);
|
||||
self.force_report_step_result(desc, result, elapsed);
|
||||
// Flush buffered sibling results.
|
||||
for (desc, result, elapsed) in sibling_results.values() {
|
||||
self.force_report_step_result(desc, result, *elapsed);
|
||||
}
|
||||
} else {
|
||||
sibling_results
|
||||
.insert(desc.id, (desc.clone(), result.clone(), elapsed));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn report_summary(&mut self, summary: &TestSummary, elapsed: &Duration) {
|
||||
if !summary.failures.is_empty() || !summary.uncaught_errors.is_empty() {
|
||||
#[allow(clippy::type_complexity)] // Type alias doesn't look better here
|
||||
let mut failures_by_origin: BTreeMap<
|
||||
String,
|
||||
(Vec<(&TestDescription, &TestFailure)>, Option<&JsError>),
|
||||
> = BTreeMap::default();
|
||||
let mut failure_titles = vec![];
|
||||
for (description, failure) in &summary.failures {
|
||||
let (failures, _) = failures_by_origin
|
||||
.entry(description.origin.clone())
|
||||
.or_default();
|
||||
failures.push((description, failure));
|
||||
}
|
||||
for (origin, js_error) in &summary.uncaught_errors {
|
||||
let (_, uncaught_error) =
|
||||
failures_by_origin.entry(origin.clone()).or_default();
|
||||
let _ = uncaught_error.insert(js_error.as_ref());
|
||||
}
|
||||
// note: the trailing whitespace is intentional to get a red background
|
||||
println!("\n{}\n", colors::white_bold_on_red(" ERRORS "));
|
||||
for (origin, (failures, uncaught_error)) in failures_by_origin {
|
||||
for (description, failure) in failures {
|
||||
if !failure.hide_in_summary() {
|
||||
let failure_title = self.format_test_for_summary(description);
|
||||
println!("{}", &failure_title);
|
||||
println!("{}: {}", colors::red_bold("error"), failure.to_string());
|
||||
println!();
|
||||
failure_titles.push(failure_title);
|
||||
}
|
||||
}
|
||||
if let Some(js_error) = uncaught_error {
|
||||
let failure_title = format!(
|
||||
"{} (uncaught error)",
|
||||
self.to_relative_path_or_remote_url(&origin)
|
||||
);
|
||||
println!("{}", &failure_title);
|
||||
println!(
|
||||
"{}: {}",
|
||||
colors::red_bold("error"),
|
||||
format_test_error(js_error)
|
||||
);
|
||||
println!("This error was not caught from a test and caused the test runner to fail on the referenced module.");
|
||||
println!("It most likely originated from a dangling promise, event/timeout handler or top-level code.");
|
||||
println!();
|
||||
failure_titles.push(failure_title);
|
||||
}
|
||||
}
|
||||
// note: the trailing whitespace is intentional to get a red background
|
||||
println!("{}\n", colors::white_bold_on_red(" FAILURES "));
|
||||
for failure_title in failure_titles {
|
||||
println!("{failure_title}");
|
||||
}
|
||||
}
|
||||
|
||||
let status = if summary.has_failed() {
|
||||
colors::red("FAILED").to_string()
|
||||
} else {
|
||||
colors::green("ok").to_string()
|
||||
};
|
||||
|
||||
let get_steps_text = |count: usize| -> String {
|
||||
if count == 0 {
|
||||
String::new()
|
||||
} else if count == 1 {
|
||||
" (1 step)".to_string()
|
||||
} else {
|
||||
format!(" ({count} steps)")
|
||||
}
|
||||
};
|
||||
|
||||
let mut summary_result = String::new();
|
||||
|
||||
write!(
|
||||
summary_result,
|
||||
"{} passed{} | {} failed{}",
|
||||
summary.passed,
|
||||
get_steps_text(summary.passed_steps),
|
||||
summary.failed,
|
||||
get_steps_text(summary.failed_steps),
|
||||
)
|
||||
.unwrap();
|
||||
|
||||
let ignored_steps = get_steps_text(summary.ignored_steps);
|
||||
if summary.ignored > 0 || !ignored_steps.is_empty() {
|
||||
write!(
|
||||
summary_result,
|
||||
" | {} ignored{}",
|
||||
summary.ignored, ignored_steps
|
||||
)
|
||||
.unwrap()
|
||||
}
|
||||
|
||||
if summary.measured > 0 {
|
||||
write!(summary_result, " | {} measured", summary.measured,).unwrap();
|
||||
}
|
||||
|
||||
if summary.filtered_out > 0 {
|
||||
write!(summary_result, " | {} filtered out", summary.filtered_out)
|
||||
.unwrap()
|
||||
};
|
||||
|
||||
println!(
|
||||
"\n{} | {} {}\n",
|
||||
status,
|
||||
summary_result,
|
||||
colors::gray(format!(
|
||||
"({})",
|
||||
display::human_elapsed(elapsed.as_millis())
|
||||
)),
|
||||
);
|
||||
self.in_new_line = true;
|
||||
}
|
||||
|
||||
fn report_sigint(
|
||||
&mut self,
|
||||
tests_pending: &HashSet<usize>,
|
||||
tests: &IndexMap<usize, TestDescription>,
|
||||
test_steps: &IndexMap<usize, TestStepDescription>,
|
||||
) {
|
||||
if tests_pending.is_empty() {
|
||||
return;
|
||||
}
|
||||
let mut formatted_pending = BTreeSet::new();
|
||||
for id in tests_pending {
|
||||
if let Some(desc) = tests.get(id) {
|
||||
formatted_pending.insert(self.format_test_for_summary(desc));
|
||||
}
|
||||
if let Some(desc) = test_steps.get(id) {
|
||||
formatted_pending
|
||||
.insert(self.format_test_step_for_summary(desc, tests, test_steps));
|
||||
}
|
||||
}
|
||||
println!(
|
||||
"\n{} The following tests were pending:\n",
|
||||
colors::intense_blue("SIGINT")
|
||||
);
|
||||
for entry in formatted_pending {
|
||||
println!("{}", entry);
|
||||
}
|
||||
println!();
|
||||
self.in_new_line = true;
|
||||
}
|
||||
|
||||
fn format_test_step_ancestry(
|
||||
&self,
|
||||
desc: &TestStepDescription,
|
||||
|
@ -859,6 +619,354 @@ impl PrettyTestReporter {
|
|||
}
|
||||
}
|
||||
|
||||
impl TestReporter for PrettyTestReporter {
|
||||
fn report_register(&mut self, _description: &TestDescription) {}
|
||||
fn report_plan(&mut self, plan: &TestPlan) {
|
||||
self.summary.total += plan.total;
|
||||
self.summary.filtered_out += plan.filtered_out;
|
||||
if self.parallel {
|
||||
return;
|
||||
}
|
||||
let inflection = if plan.total == 1 { "test" } else { "tests" };
|
||||
println!(
|
||||
"{}",
|
||||
colors::gray(format!(
|
||||
"running {} {} from {}",
|
||||
plan.total,
|
||||
inflection,
|
||||
self.to_relative_path_or_remote_url(&plan.origin)
|
||||
))
|
||||
);
|
||||
self.in_new_line = true;
|
||||
}
|
||||
|
||||
fn report_wait(&mut self, description: &TestDescription) {
|
||||
if !self.parallel {
|
||||
self.force_report_wait(description);
|
||||
}
|
||||
self.started_tests = true;
|
||||
}
|
||||
|
||||
fn report_output(&mut self, output: &[u8]) {
|
||||
if !self.echo_output {
|
||||
return;
|
||||
}
|
||||
|
||||
if !self.did_have_user_output && self.started_tests {
|
||||
self.did_have_user_output = true;
|
||||
if !self.in_new_line {
|
||||
println!();
|
||||
}
|
||||
println!("{}", colors::gray("------- output -------"));
|
||||
self.in_new_line = true;
|
||||
}
|
||||
|
||||
// output everything to stdout in order to prevent
|
||||
// stdout and stderr racing
|
||||
std::io::stdout().write_all(output).unwrap();
|
||||
}
|
||||
|
||||
fn report_result(
|
||||
&mut self,
|
||||
description: &TestDescription,
|
||||
result: &TestResult,
|
||||
elapsed: u64,
|
||||
) {
|
||||
match &result {
|
||||
TestResult::Ok => {
|
||||
self.summary.passed += 1;
|
||||
}
|
||||
TestResult::Ignored => {
|
||||
self.summary.ignored += 1;
|
||||
}
|
||||
TestResult::Failed(failure) => {
|
||||
self.summary.failed += 1;
|
||||
self
|
||||
.summary
|
||||
.failures
|
||||
.push((description.clone(), failure.clone()));
|
||||
}
|
||||
TestResult::Cancelled => {
|
||||
self.summary.failed += 1;
|
||||
}
|
||||
}
|
||||
|
||||
if self.parallel {
|
||||
self.force_report_wait(description);
|
||||
}
|
||||
|
||||
self.write_output_end();
|
||||
if self.in_new_line || self.scope_test_id != Some(description.id) {
|
||||
self.force_report_wait(description);
|
||||
}
|
||||
|
||||
let status = match result {
|
||||
TestResult::Ok => colors::green("ok").to_string(),
|
||||
TestResult::Ignored => colors::yellow("ignored").to_string(),
|
||||
TestResult::Failed(failure) => failure.format_label(),
|
||||
TestResult::Cancelled => colors::gray("cancelled").to_string(),
|
||||
};
|
||||
print!(" {}", status);
|
||||
if let TestResult::Failed(failure) = result {
|
||||
if let Some(inline_summary) = failure.format_inline_summary() {
|
||||
print!(" ({})", inline_summary)
|
||||
}
|
||||
}
|
||||
println!(
|
||||
" {}",
|
||||
colors::gray(format!("({})", display::human_elapsed(elapsed.into())))
|
||||
);
|
||||
self.in_new_line = true;
|
||||
self.scope_test_id = None;
|
||||
}
|
||||
|
||||
fn report_uncaught_error(&mut self, origin: &str, error: Box<JsError>) {
|
||||
self.summary.failed += 1;
|
||||
self
|
||||
.summary
|
||||
.uncaught_errors
|
||||
.push((origin.to_string(), error));
|
||||
|
||||
if !self.in_new_line {
|
||||
println!();
|
||||
}
|
||||
println!(
|
||||
"Uncaught error from {} {}",
|
||||
self.to_relative_path_or_remote_url(origin),
|
||||
colors::red("FAILED")
|
||||
);
|
||||
self.in_new_line = true;
|
||||
self.did_have_user_output = false;
|
||||
}
|
||||
|
||||
fn report_step_register(&mut self, _description: &TestStepDescription) {}
|
||||
|
||||
fn report_step_wait(&mut self, description: &TestStepDescription) {
|
||||
if !self.parallel && self.scope_test_id == Some(description.parent_id) {
|
||||
self.force_report_step_wait(description);
|
||||
}
|
||||
}
|
||||
|
||||
fn report_step_result(
|
||||
&mut self,
|
||||
desc: &TestStepDescription,
|
||||
result: &TestStepResult,
|
||||
elapsed: u64,
|
||||
tests: &IndexMap<usize, TestDescription>,
|
||||
test_steps: &IndexMap<usize, TestStepDescription>,
|
||||
) {
|
||||
match &result {
|
||||
TestStepResult::Ok => {
|
||||
self.summary.passed_steps += 1;
|
||||
}
|
||||
TestStepResult::Ignored => {
|
||||
self.summary.ignored_steps += 1;
|
||||
}
|
||||
TestStepResult::Failed(failure) => {
|
||||
self.summary.failed_steps += 1;
|
||||
self.summary.failures.push((
|
||||
TestDescription {
|
||||
id: desc.id,
|
||||
name: self.format_test_step_ancestry(desc, tests, test_steps),
|
||||
ignore: false,
|
||||
only: false,
|
||||
origin: desc.origin.clone(),
|
||||
location: desc.location.clone(),
|
||||
},
|
||||
failure.clone(),
|
||||
))
|
||||
}
|
||||
}
|
||||
|
||||
if self.parallel {
|
||||
self.write_output_end();
|
||||
print!(
|
||||
"{} {} ...",
|
||||
colors::gray(format!(
|
||||
"{} =>",
|
||||
self.to_relative_path_or_remote_url(&desc.origin)
|
||||
)),
|
||||
self.format_test_step_ancestry(desc, tests, test_steps)
|
||||
);
|
||||
self.in_new_line = false;
|
||||
self.scope_test_id = Some(desc.id);
|
||||
self.force_report_step_result(desc, result, elapsed);
|
||||
} else {
|
||||
let sibling_results =
|
||||
self.child_results_buffer.entry(desc.parent_id).or_default();
|
||||
if self.scope_test_id == Some(desc.id)
|
||||
|| self.scope_test_id == Some(desc.parent_id)
|
||||
{
|
||||
let sibling_results = std::mem::take(sibling_results);
|
||||
self.force_report_step_result(desc, result, elapsed);
|
||||
// Flush buffered sibling results.
|
||||
for (desc, result, elapsed) in sibling_results.values() {
|
||||
self.force_report_step_result(desc, result, *elapsed);
|
||||
}
|
||||
} else {
|
||||
sibling_results
|
||||
.insert(desc.id, (desc.clone(), result.clone(), elapsed));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn report_summary(
|
||||
&mut self,
|
||||
elapsed: &Duration,
|
||||
_tests: &IndexMap<usize, TestDescription>,
|
||||
_test_steps: &IndexMap<usize, TestStepDescription>,
|
||||
) {
|
||||
if !self.summary.failures.is_empty()
|
||||
|| !self.summary.uncaught_errors.is_empty()
|
||||
{
|
||||
#[allow(clippy::type_complexity)] // Type alias doesn't look better here
|
||||
let mut failures_by_origin: BTreeMap<
|
||||
String,
|
||||
(Vec<(&TestDescription, &TestFailure)>, Option<&JsError>),
|
||||
> = BTreeMap::default();
|
||||
let mut failure_titles = vec![];
|
||||
for (description, failure) in &self.summary.failures {
|
||||
let (failures, _) = failures_by_origin
|
||||
.entry(description.origin.clone())
|
||||
.or_default();
|
||||
failures.push((description, failure));
|
||||
}
|
||||
|
||||
for (origin, js_error) in &self.summary.uncaught_errors {
|
||||
let (_, uncaught_error) =
|
||||
failures_by_origin.entry(origin.clone()).or_default();
|
||||
let _ = uncaught_error.insert(js_error.as_ref());
|
||||
}
|
||||
// note: the trailing whitespace is intentional to get a red background
|
||||
println!("\n{}\n", colors::white_bold_on_red(" ERRORS "));
|
||||
for (origin, (failures, uncaught_error)) in failures_by_origin {
|
||||
for (description, failure) in failures {
|
||||
if !failure.hide_in_summary() {
|
||||
let failure_title = self.format_test_for_summary(description);
|
||||
println!("{}", &failure_title);
|
||||
println!("{}: {}", colors::red_bold("error"), failure.to_string());
|
||||
println!();
|
||||
failure_titles.push(failure_title);
|
||||
}
|
||||
}
|
||||
if let Some(js_error) = uncaught_error {
|
||||
let failure_title = format!(
|
||||
"{} (uncaught error)",
|
||||
self.to_relative_path_or_remote_url(&origin)
|
||||
);
|
||||
println!("{}", &failure_title);
|
||||
println!(
|
||||
"{}: {}",
|
||||
colors::red_bold("error"),
|
||||
format_test_error(js_error)
|
||||
);
|
||||
println!("This error was not caught from a test and caused the test runner to fail on the referenced module.");
|
||||
println!("It most likely originated from a dangling promise, event/timeout handler or top-level code.");
|
||||
println!();
|
||||
failure_titles.push(failure_title);
|
||||
}
|
||||
}
|
||||
// note: the trailing whitespace is intentional to get a red background
|
||||
println!("{}\n", colors::white_bold_on_red(" FAILURES "));
|
||||
for failure_title in failure_titles {
|
||||
println!("{failure_title}");
|
||||
}
|
||||
}
|
||||
|
||||
let status = if self.summary.has_failed() {
|
||||
colors::red("FAILED").to_string()
|
||||
} else {
|
||||
colors::green("ok").to_string()
|
||||
};
|
||||
|
||||
let get_steps_text = |count: usize| -> String {
|
||||
if count == 0 {
|
||||
String::new()
|
||||
} else if count == 1 {
|
||||
" (1 step)".to_string()
|
||||
} else {
|
||||
format!(" ({count} steps)")
|
||||
}
|
||||
};
|
||||
|
||||
let mut summary_result = String::new();
|
||||
|
||||
write!(
|
||||
summary_result,
|
||||
"{} passed{} | {} failed{}",
|
||||
self.summary.passed,
|
||||
get_steps_text(self.summary.passed_steps),
|
||||
self.summary.failed,
|
||||
get_steps_text(self.summary.failed_steps),
|
||||
)
|
||||
.unwrap();
|
||||
|
||||
let ignored_steps = get_steps_text(self.summary.ignored_steps);
|
||||
if self.summary.ignored > 0 || !ignored_steps.is_empty() {
|
||||
write!(
|
||||
summary_result,
|
||||
" | {} ignored{}",
|
||||
self.summary.ignored, ignored_steps
|
||||
)
|
||||
.unwrap()
|
||||
}
|
||||
|
||||
if self.summary.measured > 0 {
|
||||
write!(summary_result, " | {} measured", self.summary.measured,).unwrap();
|
||||
}
|
||||
|
||||
if self.summary.filtered_out > 0 {
|
||||
write!(
|
||||
summary_result,
|
||||
" | {} filtered out",
|
||||
self.summary.filtered_out
|
||||
)
|
||||
.unwrap()
|
||||
};
|
||||
|
||||
println!(
|
||||
"\n{} | {} {}\n",
|
||||
status,
|
||||
summary_result,
|
||||
colors::gray(format!(
|
||||
"({})",
|
||||
display::human_elapsed(elapsed.as_millis())
|
||||
)),
|
||||
);
|
||||
self.in_new_line = true;
|
||||
}
|
||||
|
||||
fn report_sigint(
|
||||
&mut self,
|
||||
tests_pending: &HashSet<usize>,
|
||||
tests: &IndexMap<usize, TestDescription>,
|
||||
test_steps: &IndexMap<usize, TestStepDescription>,
|
||||
) {
|
||||
if tests_pending.is_empty() {
|
||||
return;
|
||||
}
|
||||
let mut formatted_pending = BTreeSet::new();
|
||||
for id in tests_pending {
|
||||
if let Some(desc) = tests.get(id) {
|
||||
formatted_pending.insert(self.format_test_for_summary(desc));
|
||||
}
|
||||
if let Some(desc) = test_steps.get(id) {
|
||||
formatted_pending
|
||||
.insert(self.format_test_step_for_summary(desc, tests, test_steps));
|
||||
}
|
||||
}
|
||||
println!(
|
||||
"\n{} The following tests were pending:\n",
|
||||
colors::intense_blue("SIGINT")
|
||||
);
|
||||
for entry in formatted_pending {
|
||||
println!("{}", entry);
|
||||
}
|
||||
println!();
|
||||
self.in_new_line = true;
|
||||
}
|
||||
}
|
||||
|
||||
fn abbreviate_test_error(js_error: &JsError) -> JsError {
|
||||
let mut js_error = js_error.clone();
|
||||
let frames = std::mem::take(&mut js_error.frames);
|
||||
|
@ -1339,6 +1447,7 @@ async fn test_specifiers(
|
|||
sender_.upgrade().map(|s| s.send(TestEvent::Sigint).ok());
|
||||
});
|
||||
HAS_TEST_RUN_SIGINT_HANDLER.store(true, Ordering::Relaxed);
|
||||
let mut reporter = get_test_reporter(&options);
|
||||
|
||||
let join_handles = specifiers.into_iter().map(move |specifier| {
|
||||
let worker_factory = worker_factory.clone();
|
||||
|
@ -1362,11 +1471,6 @@ async fn test_specifiers(
|
|||
.buffer_unordered(concurrent_jobs.get())
|
||||
.collect::<Vec<Result<Result<(), AnyError>, tokio::task::JoinError>>>();
|
||||
|
||||
let mut reporter = Box::new(PrettyTestReporter::new(
|
||||
concurrent_jobs.get() > 1,
|
||||
options.log_level != Some(Level::Error),
|
||||
));
|
||||
|
||||
let handler = {
|
||||
spawn(async move {
|
||||
let earlier = Instant::now();
|
||||
|
@ -1374,8 +1478,8 @@ async fn test_specifiers(
|
|||
let mut test_steps = IndexMap::new();
|
||||
let mut tests_started = HashSet::new();
|
||||
let mut tests_with_result = HashSet::new();
|
||||
let mut summary = TestSummary::new();
|
||||
let mut used_only = false;
|
||||
let mut failed = false;
|
||||
|
||||
while let Some(event) = receiver.recv().await {
|
||||
match event {
|
||||
|
@ -1385,9 +1489,6 @@ async fn test_specifiers(
|
|||
}
|
||||
|
||||
TestEvent::Plan(plan) => {
|
||||
summary.total += plan.total;
|
||||
summary.filtered_out += plan.filtered_out;
|
||||
|
||||
if plan.used_only {
|
||||
used_only = true;
|
||||
}
|
||||
|
@ -1407,32 +1508,19 @@ async fn test_specifiers(
|
|||
|
||||
TestEvent::Result(id, result, elapsed) => {
|
||||
if tests_with_result.insert(id) {
|
||||
let description = tests.get(&id).unwrap();
|
||||
match &result {
|
||||
TestResult::Ok => {
|
||||
summary.passed += 1;
|
||||
}
|
||||
TestResult::Ignored => {
|
||||
summary.ignored += 1;
|
||||
}
|
||||
TestResult::Failed(failure) => {
|
||||
summary.failed += 1;
|
||||
summary
|
||||
.failures
|
||||
.push((description.clone(), failure.clone()));
|
||||
}
|
||||
TestResult::Cancelled => {
|
||||
summary.failed += 1;
|
||||
match result {
|
||||
TestResult::Failed(_) | TestResult::Cancelled => {
|
||||
failed = true;
|
||||
}
|
||||
_ => (),
|
||||
}
|
||||
reporter.report_result(description, &result, elapsed);
|
||||
reporter.report_result(tests.get(&id).unwrap(), &result, elapsed);
|
||||
}
|
||||
}
|
||||
|
||||
TestEvent::UncaughtError(origin, error) => {
|
||||
reporter.report_uncaught_error(&origin, &error);
|
||||
summary.failed += 1;
|
||||
summary.uncaught_errors.push((origin.clone(), error));
|
||||
failed = true;
|
||||
reporter.report_uncaught_error(&origin, error);
|
||||
}
|
||||
|
||||
TestEvent::StepRegister(description) => {
|
||||
|
@ -1448,36 +1536,8 @@ async fn test_specifiers(
|
|||
|
||||
TestEvent::StepResult(id, result, duration) => {
|
||||
if tests_with_result.insert(id) {
|
||||
let description = test_steps.get(&id).unwrap();
|
||||
match &result {
|
||||
TestStepResult::Ok => {
|
||||
summary.passed_steps += 1;
|
||||
}
|
||||
TestStepResult::Ignored => {
|
||||
summary.ignored_steps += 1;
|
||||
}
|
||||
TestStepResult::Failed(failure) => {
|
||||
summary.failed_steps += 1;
|
||||
summary.failures.push((
|
||||
TestDescription {
|
||||
id: description.id,
|
||||
name: reporter.format_test_step_ancestry(
|
||||
description,
|
||||
&tests,
|
||||
&test_steps,
|
||||
),
|
||||
ignore: false,
|
||||
only: false,
|
||||
origin: description.origin.clone(),
|
||||
location: description.location.clone(),
|
||||
},
|
||||
failure.clone(),
|
||||
))
|
||||
}
|
||||
}
|
||||
|
||||
reporter.report_step_result(
|
||||
description,
|
||||
test_steps.get(&id).unwrap(),
|
||||
&result,
|
||||
duration,
|
||||
&tests,
|
||||
|
@ -1504,7 +1564,7 @@ async fn test_specifiers(
|
|||
HAS_TEST_RUN_SIGINT_HANDLER.store(false, Ordering::Relaxed);
|
||||
|
||||
let elapsed = Instant::now().duration_since(earlier);
|
||||
reporter.report_summary(&summary, &elapsed);
|
||||
reporter.report_summary(&elapsed, &tests, &test_steps);
|
||||
|
||||
if used_only {
|
||||
return Err(generic_error(
|
||||
|
@ -1512,7 +1572,7 @@ async fn test_specifiers(
|
|||
));
|
||||
}
|
||||
|
||||
if summary.failed > 0 {
|
||||
if failed {
|
||||
return Err(generic_error("Test failed"));
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in a new issue