// Copyright 2018-2025 the Deno authors. MIT license. use std::collections::HashMap; use std::collections::HashSet; use deno_core::error::AnyError; use deno_core::ModuleSpecifier; use lsp::Range; use tower_lsp::lsp_types as lsp; use super::lsp_custom; use super::lsp_custom::TestData; use crate::lsp::client::TestingNotification; use crate::lsp::logging::lsp_warn; use crate::lsp::urls::url_to_uri; use crate::tools::test::TestDescription; use crate::tools::test::TestStepDescription; use crate::util::checksum; #[derive(Debug, Clone, PartialEq)] pub struct TestDefinition { pub id: String, pub name: String, pub range: Option, pub is_dynamic: bool, pub parent_id: Option, pub step_ids: HashSet, } #[derive(Debug, Clone, PartialEq)] pub struct TestModule { pub specifier: ModuleSpecifier, pub defs: HashMap, } impl TestModule { pub fn new(specifier: ModuleSpecifier) -> Self { Self { specifier, defs: Default::default(), } } /// Returns `(id, is_newly_registered)`. pub fn register( &mut self, name: String, range: Option, is_dynamic: bool, parent_id: Option, ) -> (String, bool) { let mut id_components = Vec::with_capacity(7); id_components.push(name.as_bytes()); let mut current_parent_id = &parent_id; while let Some(parent_id) = current_parent_id { let parent = match self.defs.get(parent_id) { Some(d) => d, None => { lsp_warn!("Internal Error: parent_id \"{}\" of test \"{}\" was not registered.", parent_id, &name); id_components.push("".as_bytes()); break; } }; id_components.push(parent.name.as_bytes()); current_parent_id = &parent.parent_id; } id_components.push(self.specifier.as_str().as_bytes()); id_components.reverse(); let id = checksum::gen(&id_components); if self.defs.contains_key(&id) { return (id, false); } if let Some(parent_id) = &parent_id { let parent = self.defs.get_mut(parent_id).unwrap(); parent.step_ids.insert(id.clone()); } self.defs.insert( id.clone(), TestDefinition { id: id.clone(), name, range, is_dynamic, parent_id, step_ids: Default::default(), }, ); (id, true) } /// Returns `(id, was_newly_registered)`. pub fn register_dynamic(&mut self, desc: &TestDescription) -> (String, bool) { self.register(desc.name.clone(), None, true, None) } /// Returns `(id, was_newly_registered)`. pub fn register_step_dynamic( &mut self, desc: &TestStepDescription, parent_static_id: &str, ) -> (String, bool) { self.register( desc.name.clone(), None, true, Some(parent_static_id.to_string()), ) } pub fn get(&self, id: &str) -> Option<&TestDefinition> { self.defs.get(id) } pub fn get_test_data(&self, id: &str) -> TestData { fn get_test_data_inner(tm: &TestModule, id: &str) -> TestData { let def = tm.defs.get(id).unwrap(); TestData { id: def.id.clone(), label: def.name.clone(), steps: def .step_ids .iter() .map(|id| get_test_data_inner(tm, id)) .collect(), range: def.range, } } let def = self.defs.get(id).unwrap(); let mut current_data = get_test_data_inner(self, &def.id); let mut current_parent_id = &def.parent_id; while let Some(parent_id) = current_parent_id { let parent = self.defs.get(parent_id).unwrap(); current_data = TestData { id: parent.id.clone(), label: parent.name.clone(), steps: vec![current_data], range: None, }; current_parent_id = &parent.parent_id; } current_data } /// Return the test definitions as a testing module notification. pub fn as_replace_notification( &self, maybe_root_uri: Option<&ModuleSpecifier>, ) -> Result { let label = self.label(maybe_root_uri); Ok(TestingNotification::Module( lsp_custom::TestModuleNotificationParams { text_document: lsp::TextDocumentIdentifier { uri: url_to_uri(&self.specifier)?, }, kind: lsp_custom::TestModuleNotificationKind::Replace, label, tests: self .defs .iter() .filter(|(_, def)| def.parent_id.is_none()) .map(|(id, _)| self.get_test_data(id)) .collect(), }, )) } pub fn label(&self, maybe_root_uri: Option<&ModuleSpecifier>) -> String { if let Some(root) = maybe_root_uri { self.specifier.as_str().replace(root.as_str(), "") } else { self .specifier .path_segments() .and_then(|s| s.last().map(|s| s.to_string())) .unwrap_or_else(|| "".to_string()) } } pub fn is_empty(&self) -> bool { self.defs.is_empty() } }