7357: Dont show runnables from nested mods r=matklad a=matklad

bors r+
🤖

Co-authored-by: Aleksey Kladov <aleksey.kladov@gmail.com>
This commit is contained in:
bors[bot] 2021-01-20 11:49:59 +00:00 committed by GitHub
commit 6cf3d669e2
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 81 additions and 23 deletions

View file

@ -9,6 +9,7 @@ use syntax::{
ast::{self, AstNode, AttrsOwner}, ast::{self, AstNode, AttrsOwner},
match_ast, SyntaxNode, match_ast, SyntaxNode,
}; };
use test_utils::mark;
use crate::{ use crate::{
display::{ToNav, TryToNav}, display::{ToNav, TryToNav},
@ -96,28 +97,26 @@ impl Runnable {
pub(crate) fn runnables(db: &RootDatabase, file_id: FileId) -> Vec<Runnable> { pub(crate) fn runnables(db: &RootDatabase, file_id: FileId) -> Vec<Runnable> {
let sema = Semantics::new(db); let sema = Semantics::new(db);
let module = match sema.to_module_def(file_id) { let module = match sema.to_module_def(file_id) {
None => return vec![], None => return Vec::new(),
Some(it) => it, Some(it) => it,
}; };
runnables_mod(&sema, module) let mut res = Vec::new();
runnables_mod(&sema, &mut res, module);
res
} }
fn runnables_mod(sema: &Semantics<RootDatabase>, module: hir::Module) -> Vec<Runnable> { fn runnables_mod(sema: &Semantics<RootDatabase>, acc: &mut Vec<Runnable>, module: hir::Module) {
let mut res: Vec<Runnable> = module acc.extend(module.declarations(sema.db).into_iter().filter_map(|def| {
.declarations(sema.db) let runnable = match def {
.into_iter() hir::ModuleDef::Module(it) => runnable_mod(&sema, it),
.filter_map(|def| { hir::ModuleDef::Function(it) => runnable_fn(&sema, it),
let runnable = match def { _ => None,
hir::ModuleDef::Module(it) => runnable_mod(&sema, it), };
hir::ModuleDef::Function(it) => runnable_fn(&sema, it), runnable.or_else(|| module_def_doctest(&sema, def))
_ => None, }));
};
runnable.or_else(|| module_def_doctest(&sema, def))
})
.collect();
res.extend(module.impl_defs(sema.db).into_iter().flat_map(|it| it.items(sema.db)).filter_map( acc.extend(module.impl_defs(sema.db).into_iter().flat_map(|it| it.items(sema.db)).filter_map(
|def| match def { |def| match def {
hir::AssocItem::Function(it) => { hir::AssocItem::Function(it) => {
runnable_fn(&sema, it).or_else(|| module_def_doctest(&sema, it.into())) runnable_fn(&sema, it).or_else(|| module_def_doctest(&sema, it.into()))
@ -127,12 +126,14 @@ fn runnables_mod(sema: &Semantics<RootDatabase>, module: hir::Module) -> Vec<Run
}, },
)); ));
res.extend(module.declarations(sema.db).into_iter().flat_map(|def| match def { for def in module.declarations(sema.db) {
hir::ModuleDef::Module(it) => runnables_mod(sema, it), if let hir::ModuleDef::Module(submodule) = def {
_ => vec![], match submodule.definition_source(sema.db).value {
})); hir::ModuleSource::Module(_) => runnables_mod(sema, acc, submodule),
hir::ModuleSource::SourceFile(_) => mark::hit!(dont_recurse_in_outline_submodules),
res }
}
}
} }
pub(crate) fn runnable_fn(sema: &Semantics<RootDatabase>, def: hir::Function) -> Option<Runnable> { pub(crate) fn runnable_fn(sema: &Semantics<RootDatabase>, def: hir::Function) -> Option<Runnable> {
@ -326,6 +327,7 @@ fn has_test_function_or_multiple_test_submodules(
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use expect_test::{expect, Expect}; use expect_test::{expect, Expect};
use test_utils::mark;
use crate::fixture; use crate::fixture;
@ -1050,4 +1052,25 @@ mod tests {
"#]], "#]],
); );
} }
#[test]
fn dont_recurse_in_outline_submodules() {
mark::check!(dont_recurse_in_outline_submodules);
check(
r#"
//- /lib.rs
$0
mod m;
//- /m.rs
mod tests {
#[test]
fn t() {}
}
"#,
&[],
expect![[r#"
[]
"#]],
);
}
} }

View file

@ -280,6 +280,9 @@ Prefer `Default` even it has to be implemented manually.
**Rationale:** less typing in the common case, uniformity. **Rationale:** less typing in the common case, uniformity.
Use `Vec::new` rather than `vec![]`. **Rationale:** uniformity, strength
reduction.
## Functions Over Objects ## Functions Over Objects
Avoid creating "doer" objects. Avoid creating "doer" objects.
@ -418,12 +421,44 @@ fn frobnicate(s: &str) {
**Rationale:** reveals the costs. **Rationale:** reveals the costs.
It is also more efficient when the caller already owns the allocation. It is also more efficient when the caller already owns the allocation.
## Collection types ## Collection Types
Prefer `rustc_hash::FxHashMap` and `rustc_hash::FxHashSet` instead of the ones in `std::collections`. Prefer `rustc_hash::FxHashMap` and `rustc_hash::FxHashSet` instead of the ones in `std::collections`.
**Rationale:** they use a hasher that's significantly faster and using them consistently will reduce code size by some small amount. **Rationale:** they use a hasher that's significantly faster and using them consistently will reduce code size by some small amount.
## Avoid Intermediate Collections
When writing a recursive function to compute a sets of things, use an accumulator parameter instead of returning a fresh collection.
Accumulator goes first in the list of arguments.
```rust
// GOOD
pub fn reachable_nodes(node: Node) -> FxHashSet<Node> {
let mut res = FxHashSet::default();
go(&mut res, node);
res
}
fn go(acc: &mut FxHashSet<Node>, node: Node) {
acc.insert(node);
for n in node.neighbors() {
go(acc, n);
}
}
// BAD
pub fn reachable_nodes(node: Node) -> FxHashSet<Node> {
let mut res = FxHashSet::default();
res.insert(node);
for n in node.neighbors() {
res.extend(reachable_nodes(n));
}
res
}
```
**Rational:** re-use allocations, accumulator style is more concise for complex cases.
# Style # Style
## Order of Imports ## Order of Imports