mirror of
https://github.com/denoland/deno.git
synced 2025-01-26 17:02:20 -05:00
84fb2ad71b
Step 1 of the Rustification of sanitizers, which unblocks the faster timers. This replaces the resource sanitizer with a Rust one, using the new APIs in deno_core.
230 lines
6.4 KiB
Rust
230 lines
6.4 KiB
Rust
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
|
|
use super::fmt::format_test_error;
|
|
use super::fmt::to_relative_path_or_remote_url;
|
|
use super::*;
|
|
|
|
pub(super) fn format_test_step_ancestry(
|
|
desc: &TestStepDescription,
|
|
tests: &IndexMap<usize, TestDescription>,
|
|
test_steps: &IndexMap<usize, TestStepDescription>,
|
|
) -> String {
|
|
let root;
|
|
let mut ancestor_names = vec![];
|
|
let mut current_desc = desc;
|
|
loop {
|
|
if let Some(step_desc) = test_steps.get(¤t_desc.parent_id) {
|
|
ancestor_names.push(&step_desc.name);
|
|
current_desc = step_desc;
|
|
} else {
|
|
root = tests.get(¤t_desc.parent_id).unwrap();
|
|
break;
|
|
}
|
|
}
|
|
ancestor_names.reverse();
|
|
let mut result = String::new();
|
|
result.push_str(&root.name);
|
|
result.push_str(" ... ");
|
|
for name in ancestor_names {
|
|
result.push_str(name);
|
|
result.push_str(" ... ");
|
|
}
|
|
result.push_str(&desc.name);
|
|
result
|
|
}
|
|
|
|
pub fn format_test_for_summary(
|
|
cwd: &Url,
|
|
desc: &TestFailureDescription,
|
|
) -> String {
|
|
format!(
|
|
"{} {}",
|
|
&desc.name,
|
|
colors::gray(format!(
|
|
"=> {}:{}:{}",
|
|
to_relative_path_or_remote_url(cwd, &desc.location.file_name),
|
|
desc.location.line_number,
|
|
desc.location.column_number
|
|
))
|
|
)
|
|
}
|
|
|
|
pub fn format_test_step_for_summary(
|
|
cwd: &Url,
|
|
desc: &TestStepDescription,
|
|
tests: &IndexMap<usize, TestDescription>,
|
|
test_steps: &IndexMap<usize, TestStepDescription>,
|
|
) -> String {
|
|
let long_name = format_test_step_ancestry(desc, tests, test_steps);
|
|
format!(
|
|
"{} {}",
|
|
long_name,
|
|
colors::gray(format!(
|
|
"=> {}:{}:{}",
|
|
to_relative_path_or_remote_url(cwd, &desc.location.file_name),
|
|
desc.location.line_number,
|
|
desc.location.column_number
|
|
))
|
|
)
|
|
}
|
|
|
|
pub(super) fn report_sigint(
|
|
writer: &mut dyn std::io::Write,
|
|
cwd: &Url,
|
|
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(format_test_for_summary(cwd, &desc.into()));
|
|
}
|
|
if let Some(desc) = test_steps.get(id) {
|
|
formatted_pending
|
|
.insert(format_test_step_for_summary(cwd, desc, tests, test_steps));
|
|
}
|
|
}
|
|
writeln!(
|
|
writer,
|
|
"\n{} The following tests were pending:\n",
|
|
colors::intense_blue("SIGINT")
|
|
)
|
|
.unwrap();
|
|
for entry in formatted_pending {
|
|
writeln!(writer, "{}", entry).unwrap();
|
|
}
|
|
writeln!(writer).unwrap();
|
|
}
|
|
|
|
pub(super) fn report_summary(
|
|
writer: &mut dyn std::io::Write,
|
|
cwd: &Url,
|
|
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<(&TestFailureDescription, &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
|
|
writeln!(writer, "\n{}\n", colors::white_bold_on_red(" ERRORS ")).unwrap();
|
|
for (origin, (failures, uncaught_error)) in failures_by_origin {
|
|
for (description, failure) in failures {
|
|
if !failure.hide_in_summary() {
|
|
let failure_title = format_test_for_summary(cwd, description);
|
|
writeln!(writer, "{}", &failure_title).unwrap();
|
|
writeln!(
|
|
writer,
|
|
"{}: {}",
|
|
colors::red_bold("error"),
|
|
failure.to_string()
|
|
)
|
|
.unwrap();
|
|
writeln!(writer).unwrap();
|
|
failure_titles.push(failure_title);
|
|
}
|
|
}
|
|
if let Some(js_error) = uncaught_error {
|
|
let failure_title = format!(
|
|
"{} (uncaught error)",
|
|
to_relative_path_or_remote_url(cwd, &origin)
|
|
);
|
|
writeln!(writer, "{}", &failure_title).unwrap();
|
|
writeln!(
|
|
writer,
|
|
"{}: {}",
|
|
colors::red_bold("error"),
|
|
format_test_error(js_error)
|
|
)
|
|
.unwrap();
|
|
writeln!(writer, "This error was not caught from a test and caused the test runner to fail on the referenced module.").unwrap();
|
|
writeln!(writer, "It most likely originated from a dangling promise, event/timeout handler or top-level code.").unwrap();
|
|
writeln!(writer).unwrap();
|
|
failure_titles.push(failure_title);
|
|
}
|
|
}
|
|
// note: the trailing whitespace is intentional to get a red background
|
|
writeln!(writer, "{}\n", colors::white_bold_on_red(" FAILURES ")).unwrap();
|
|
for failure_title in failure_titles {
|
|
writeln!(writer, "{failure_title}").unwrap();
|
|
}
|
|
}
|
|
|
|
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()
|
|
};
|
|
|
|
writeln!(
|
|
writer,
|
|
"\n{} | {} {}",
|
|
status,
|
|
summary_result,
|
|
colors::gray(format!("({})", display::human_elapsed(elapsed.as_millis()))),
|
|
)
|
|
.unwrap();
|
|
}
|