diff --git a/crates/ide/src/hover.rs b/crates/ide/src/hover.rs index 8cb4a51d8e..939efa43ff 100644 --- a/crates/ide/src/hover.rs +++ b/crates/ide/src/hover.rs @@ -17,7 +17,7 @@ use crate::{ doc_links::{remove_links, rewrite_links}, markdown_remove::remove_markdown, markup::Markup, - runnables::{runnable, runnable_fn}, + runnables::{runnable_fn, runnable_mod}, FileId, FilePosition, NavigationTarget, RangeInfo, Runnable, }; @@ -192,7 +192,7 @@ fn runnable_action( Definition::ModuleDef(it) => match it { ModuleDef::Module(it) => match it.definition_source(sema.db).value { ModuleSource::Module(it) => { - runnable(&sema, it.syntax().clone()).map(|it| HoverAction::Runnable(it)) + runnable_mod(&sema, it).map(|it| HoverAction::Runnable(it)) } _ => None, }, diff --git a/crates/ide/src/runnables.rs b/crates/ide/src/runnables.rs index 557563d7e1..3a1e204db5 100644 --- a/crates/ide/src/runnables.rs +++ b/crates/ide/src/runnables.rs @@ -96,21 +96,23 @@ impl Runnable { pub(crate) fn runnables(db: &RootDatabase, file_id: FileId) -> Vec { let sema = Semantics::new(db); let source_file = sema.parse(file_id); - source_file.syntax().descendants().filter_map(|i| runnable(&sema, i)).collect() -} - -pub(crate) fn runnable(sema: &Semantics, item: SyntaxNode) -> Option { - let runnable_item = match_ast! { - match (item.clone()) { - ast::Fn(func) => { - let def = sema.to_def(&func)?; - runnable_fn(sema, def) - }, - ast::Module(it) => runnable_mod(sema, it), - _ => None, - } - }; - runnable_item.or_else(|| runnable_doctest(sema, item)) + source_file + .syntax() + .descendants() + .filter_map(|item| { + let runnable = match_ast! { + match item { + ast::Fn(func) => { + let def = sema.to_def(&func)?; + runnable_fn(&sema, def) + }, + ast::Module(it) => runnable_mod(&sema, it), + _ => None, + } + }; + runnable.or_else(|| runnable_doctest(&sema, item)) + }) + .collect() } pub(crate) fn runnable_fn(sema: &Semantics, def: hir::Function) -> Option { @@ -145,6 +147,29 @@ pub(crate) fn runnable_fn(sema: &Semantics, def: hir::Function) -> Some(Runnable { nav, kind, cfg }) } +pub(crate) fn runnable_mod( + sema: &Semantics, + module: ast::Module, +) -> Option { + if !has_test_function_or_multiple_test_submodules(&module) { + return None; + } + let module_def = sema.to_def(&module)?; + + let path = module_def + .path_to_root(sema.db) + .into_iter() + .rev() + .filter_map(|it| it.name(sema.db)) + .join("::"); + + let def = sema.to_def(&module)?; + let attrs = def.attrs(sema.db); + let cfg = attrs.cfg(); + let nav = module_def.to_nav(sema.db); + Some(Runnable { nav, kind: RunnableKind::TestMod { path }, cfg }) +} + fn runnable_doctest(sema: &Semantics, item: SyntaxNode) -> Option { match_ast! { match item { @@ -253,26 +278,6 @@ fn has_runnable_doc_test(attrs: &hir::Attrs) -> bool { }) } -fn runnable_mod(sema: &Semantics, module: ast::Module) -> Option { - if !has_test_function_or_multiple_test_submodules(&module) { - return None; - } - let module_def = sema.to_def(&module)?; - - let path = module_def - .path_to_root(sema.db) - .into_iter() - .rev() - .filter_map(|it| it.name(sema.db)) - .join("::"); - - let def = sema.to_def(&module)?; - let attrs = def.attrs(sema.db); - let cfg = attrs.cfg(); - let nav = module_def.to_nav(sema.db); - Some(Runnable { nav, kind: RunnableKind::TestMod { path }, cfg }) -} - // We could create runnables for modules with number_of_test_submodules > 0, // but that bloats the runnables for no real benefit, since all tests can be run by the submodule already fn has_test_function_or_multiple_test_submodules(module: &ast::Module) -> bool {