From 907d9bb4d720a7b01bffb098c72c789665f2415b Mon Sep 17 00:00:00 2001 From: Nayeem Rahman Date: Fri, 25 Aug 2023 22:32:22 +0100 Subject: [PATCH] fix(lsp): test explorer panic on step result (#20289) Fixes https://github.com/denoland/vscode_deno/issues/843. Prevents step results from being reported twice. Refactors `LspTestReporter` to use a complete `(test_id, descriptor)` map instead of a brittle `LspTestReporter::stack`. --- cli/lsp/testing/execution.rs | 213 ++++++++++++++--------------- cli/tests/integration/lsp_tests.rs | 69 +++++++++- 2 files changed, 172 insertions(+), 110 deletions(-) diff --git a/cli/lsp/testing/execution.rs b/cli/lsp/testing/execution.rs index 28c6a5de9d..4a1e703222 100644 --- a/cli/lsp/testing/execution.rs +++ b/cli/lsp/testing/execution.rs @@ -334,6 +334,7 @@ impl TestRun { spawn(async move { let earlier = Instant::now(); let mut summary = test::TestSummary::new(); + let mut tests_with_result = HashSet::new(); let mut used_only = false; while let Some(event) = receiver.recv().await { @@ -359,20 +360,21 @@ impl TestRun { reporter.report_output(&output); } test::TestEvent::Result(id, result, elapsed) => { - let description = tests.read().get(&id).unwrap().clone(); - match &result { - test::TestResult::Ok => summary.passed += 1, - test::TestResult::Ignored => summary.ignored += 1, - test::TestResult::Failed(error) => { - summary.failed += 1; - summary.failures.push((description.clone(), error.clone())); - } - test::TestResult::Cancelled => { - summary.failed += 1; + if tests_with_result.insert(id) { + let description = tests.read().get(&id).unwrap().clone(); + match &result { + test::TestResult::Ok => summary.passed += 1, + test::TestResult::Ignored => summary.ignored += 1, + test::TestResult::Failed(error) => { + summary.failed += 1; + summary.failures.push((description.clone(), error.clone())); + } + test::TestResult::Cancelled => { + summary.failed += 1; + } } + reporter.report_result(&description, &result, elapsed); } - - reporter.report_result(&description, &result, elapsed); } test::TestEvent::UncaughtError(origin, error) => { reporter.report_uncaught_error(&origin, &error); @@ -387,22 +389,24 @@ impl TestRun { reporter.report_step_wait(test_steps.get(&id).unwrap()); } test::TestEvent::StepResult(id, result, duration) => { - match &result { - test::TestStepResult::Ok => { - summary.passed_steps += 1; - } - test::TestStepResult::Ignored => { - summary.ignored_steps += 1; - } - test::TestStepResult::Failed(_) => { - summary.failed_steps += 1; + if tests_with_result.insert(id) { + match &result { + test::TestStepResult::Ok => { + summary.passed_steps += 1; + } + test::TestStepResult::Ignored => { + summary.ignored_steps += 1; + } + test::TestStepResult::Failed(_) => { + summary.failed_steps += 1; + } } + reporter.report_step_result( + test_steps.get(&id).unwrap(), + &result, + duration, + ); } - reporter.report_step_result( - test_steps.get(&id).unwrap(), - &result, - duration, - ); } test::TestEvent::Sigint => {} } @@ -479,6 +483,15 @@ enum TestOrTestStepDescription { TestStepDescription(test::TestStepDescription), } +impl TestOrTestStepDescription { + fn origin(&self) -> &str { + match self { + TestOrTestStepDescription::TestDescription(d) => d.origin.as_str(), + TestOrTestStepDescription::TestStepDescription(d) => d.origin.as_str(), + } + } +} + impl From<&test::TestDescription> for TestOrTestStepDescription { fn from(desc: &test::TestDescription) -> Self { Self::TestDescription(desc.clone()) @@ -560,11 +573,11 @@ impl From<&test::TestStepDescription> for lsp_custom::TestIdentifier { struct LspTestReporter { client: Client, - current_origin: Option, - maybe_root_uri: Option, id: u32, - stack: HashMap>, - tests: Arc>>, + maybe_root_uri: Option, + files: Arc>>, + tests: IndexMap, + current_test: Option, } impl LspTestReporter { @@ -572,15 +585,15 @@ impl LspTestReporter { run: &TestRun, client: Client, maybe_root_uri: Option<&ModuleSpecifier>, - tests: Arc>>, + files: Arc>>, ) -> Self { Self { client, - current_origin: None, - maybe_root_uri: maybe_root_uri.cloned(), id: run.id, - stack: HashMap::new(), - tests, + maybe_root_uri: maybe_root_uri.cloned(), + files, + tests: Default::default(), + current_test: Default::default(), } } @@ -598,8 +611,9 @@ impl LspTestReporter { fn report_plan(&mut self, _plan: &test::TestPlan) {} fn report_register(&mut self, desc: &test::TestDescription) { - let mut tests = self.tests.lock(); - let tds = tests + self.tests.insert(desc.id, desc.into()); + let mut files = self.files.lock(); + let tds = files .entry(ModuleSpecifier::parse(&desc.location.file_name).unwrap()) .or_default(); if tds.inject(desc.into()) { @@ -626,23 +640,17 @@ impl LspTestReporter { } fn report_wait(&mut self, desc: &test::TestDescription) { - self.current_origin = Some(desc.origin.clone()); - let test: lsp_custom::TestIdentifier = desc.into(); - let stack = self.stack.entry(desc.origin.clone()).or_default(); - assert!(stack.is_empty()); - stack.push(desc.into()); + self.current_test = Some(desc.id); + let test = desc.into(); self.progress(lsp_custom::TestRunProgressMessage::Started { test }); } fn report_output(&mut self, output: &[u8]) { - let test = self.current_origin.as_ref().and_then(|origin| { - self - .stack - .get(origin) - .and_then(|v| v.last().map(|td| td.into())) - }); + let test = self + .current_test + .as_ref() + .map(|id| self.tests.get(id).unwrap().into()); let value = String::from_utf8_lossy(output).replace('\n', "\r\n"); - self.progress(lsp_custom::TestRunProgressMessage::Output { value, test, @@ -657,10 +665,7 @@ impl LspTestReporter { result: &test::TestResult, elapsed: u64, ) { - let stack = self.stack.entry(desc.origin.clone()).or_default(); - assert_eq!(stack.len(), 1); - assert_eq!(stack.pop(), Some(desc.into())); - self.current_origin = None; + self.current_test = None; match result { test::TestResult::Ok => { self.progress(lsp_custom::TestRunProgressMessage::Passed { @@ -691,78 +696,69 @@ impl LspTestReporter { } fn report_uncaught_error(&mut self, origin: &str, js_error: &JsError) { - if self.current_origin == Some(origin.to_string()) { - self.current_origin = None; - } - let stack = self.stack.remove(origin).unwrap_or_default(); + self.current_test = None; let err_string = format!( "Uncaught error from {}: {}\nThis error was not caught from a test and caused the test runner to fail on the referenced module.\nIt most likely originated from a dangling promise, event/timeout handler or top-level code.", origin, test::fmt::format_test_error(js_error) ); let messages = as_test_messages(err_string, false); - for t in stack.iter().rev() { - match t { - TestOrTestStepDescription::TestDescription(desc) => { - self.progress(lsp_custom::TestRunProgressMessage::Failed { - test: desc.into(), - messages: messages.clone(), - duration: None, - }); - } - TestOrTestStepDescription::TestStepDescription(desc) => { - self.progress(lsp_custom::TestRunProgressMessage::Failed { - test: desc.into(), - messages: messages.clone(), - duration: None, - }); - } - } + for desc in self.tests.values().filter(|d| d.origin() == origin) { + self.progress(lsp_custom::TestRunProgressMessage::Failed { + test: desc.into(), + messages: messages.clone(), + duration: None, + }); } } fn report_step_register(&mut self, desc: &test::TestStepDescription) { - let mut tests = self.tests.lock(); - let tds = tests + self.tests.insert(desc.id, desc.into()); + let mut files = self.files.lock(); + let tds = files .entry(ModuleSpecifier::parse(&desc.location.file_name).unwrap()) .or_default(); - if tds.inject(desc.into()) { - let specifier = ModuleSpecifier::parse(&desc.origin).unwrap(); - let mut prev: lsp_custom::TestData = desc.into(); - if let Some(stack) = self.stack.get(&desc.origin) { - for item in stack.iter().rev() { - let mut data: lsp_custom::TestData = item.into(); - data.steps = vec![prev]; - prev = data; + let data: lsp_custom::TestData = desc.into(); + if tds.inject(data.clone()) { + let mut step_desc = Some(desc); + let mut data = data; + while let Some(desc) = step_desc { + let parent_desc = self.tests.get(&desc.parent_id).unwrap(); + let mut parent_data: lsp_custom::TestData = parent_desc.into(); + parent_data.steps = vec![data]; + data = parent_data; + step_desc = match parent_desc { + TestOrTestStepDescription::TestDescription(_) => None, + TestOrTestStepDescription::TestStepDescription(d) => Some(d), } - let label = if let Some(root) = &self.maybe_root_uri { - specifier.as_str().replace(root.as_str(), "") - } else { - specifier - .path_segments() - .and_then(|s| s.last().map(|s| s.to_string())) - .unwrap_or_else(|| "".to_string()) - }; - self - .client - .send_test_notification(TestingNotification::Module( - lsp_custom::TestModuleNotificationParams { - text_document: lsp::TextDocumentIdentifier { uri: specifier }, - kind: lsp_custom::TestModuleNotificationKind::Insert, - label, - tests: vec![prev], - }, - )); } + let specifier = ModuleSpecifier::parse(&desc.origin).unwrap(); + let label = if let Some(root) = &self.maybe_root_uri { + specifier.as_str().replace(root.as_str(), "") + } else { + specifier + .path_segments() + .and_then(|s| s.last().map(|s| s.to_string())) + .unwrap_or_else(|| "".to_string()) + }; + self + .client + .send_test_notification(TestingNotification::Module( + lsp_custom::TestModuleNotificationParams { + text_document: lsp::TextDocumentIdentifier { uri: specifier }, + kind: lsp_custom::TestModuleNotificationKind::Insert, + label, + tests: vec![data], + }, + )); } } fn report_step_wait(&mut self, desc: &test::TestStepDescription) { + if self.current_test == Some(desc.parent_id) { + self.current_test = Some(desc.id); + } let test: lsp_custom::TestIdentifier = desc.into(); - let stack = self.stack.entry(desc.origin.clone()).or_default(); - self.current_origin = Some(desc.origin.clone()); - assert!(!stack.is_empty()); - stack.push(desc.into()); self.progress(lsp_custom::TestRunProgressMessage::Started { test }); } @@ -772,8 +768,9 @@ impl LspTestReporter { result: &test::TestStepResult, elapsed: u64, ) { - let stack = self.stack.entry(desc.origin.clone()).or_default(); - assert_eq!(stack.pop(), Some(desc.into())); + if self.current_test == Some(desc.id) { + self.current_test = Some(desc.parent_id); + } match result { test::TestStepResult::Ok => { self.progress(lsp_custom::TestRunProgressMessage::Passed { diff --git a/cli/tests/integration/lsp_tests.rs b/cli/tests/integration/lsp_tests.rs index d2cb368063..e3fdcc2811 100644 --- a/cli/tests/integration/lsp_tests.rs +++ b/cli/tests/integration/lsp_tests.rs @@ -8256,8 +8256,9 @@ fn lsp_testing_api() { let contents = r#" Deno.test({ name: "test a", - fn() { + async fn(t) { console.log("test a"); + await t.step("step of test a", () => {}); } }); "#; @@ -8287,7 +8288,6 @@ Deno.test({ assert_eq!(params.tests.len(), 1); let test = ¶ms.tests[0]; assert_eq!(test.label, "test a"); - assert!(test.steps.is_none()); assert_eq!( test.range, Some(lsp::Range { @@ -8301,6 +8301,23 @@ Deno.test({ } }) ); + let steps = test.steps.as_ref().unwrap(); + assert_eq!(steps.len(), 1); + let step = &steps[0]; + assert_eq!(step.label, "step of test a"); + assert_eq!( + step.range, + Some(lsp::Range { + start: lsp::Position { + line: 5, + character: 12, + }, + end: lsp::Position { + line: 5, + character: 16, + } + }) + ); let res = client.write_request_with_res_as::( "deno/testRun", @@ -8368,6 +8385,54 @@ Deno.test({ })) ); + let (method, notification) = client.read_notification::(); + assert_eq!(method, "deno/testRunProgress"); + assert_eq!( + notification, + Some(json!({ + "id": 1, + "message": { + "type": "started", + "test": { + "textDocument": { + "uri": specifier, + }, + "id": id, + "stepId": step.id, + }, + } + })) + ); + + let (method, notification) = client.read_notification::(); + assert_eq!(method, "deno/testRunProgress"); + let mut notification = notification.unwrap(); + let duration = notification + .as_object_mut() + .unwrap() + .get_mut("message") + .unwrap() + .as_object_mut() + .unwrap() + .remove("duration"); + assert!(duration.is_some()); + assert_eq!( + notification, + json!({ + "id": 1, + "message": { + "type": "passed", + "test": { + "textDocument": { + "uri": specifier, + }, + "id": id, + "stepId": step.id, + }, + } + }) + ); + let (method, notification) = client.read_notification::(); assert_eq!(method, "deno/testRunProgress"); let notification = notification.unwrap();