SSR: Disable matching within use declarations

It currently does the wrong thing when the use declaration contains
braces.
This commit is contained in:
David Lattimore 2020-07-23 21:28:31 +10:00
parent 63f500b0ee
commit 8d09ab86ed
2 changed files with 51 additions and 1 deletions

View file

@ -10,7 +10,8 @@ use ra_ide_db::{
defs::Definition, defs::Definition,
search::{Reference, SearchScope}, search::{Reference, SearchScope},
}; };
use ra_syntax::{ast, AstNode, SyntaxNode}; use ra_syntax::{ast, AstNode, SyntaxKind, SyntaxNode};
use test_utils::mark;
/// A cache for the results of find_usages. This is for when we have multiple patterns that have the /// A cache for the results of find_usages. This is for when we have multiple patterns that have the
/// same path. e.g. if the pattern was `foo::Bar` that can parse as a path, an expression, a type /// same path. e.g. if the pattern was `foo::Bar` that can parse as a path, an expression, a type
@ -59,6 +60,10 @@ impl<'db> MatchFinder<'db> {
.skip(first_path.depth as usize) .skip(first_path.depth as usize)
.next() .next()
{ {
if !is_search_permitted_ancestors(&node_to_match) {
mark::hit!(use_declaration_with_braces);
continue;
}
if let Ok(m) = if let Ok(m) =
matching::get_match(false, rule, &node_to_match, &None, &self.sema) matching::get_match(false, rule, &node_to_match, &None, &self.sema)
{ {
@ -123,6 +128,9 @@ impl<'db> MatchFinder<'db> {
restrict_range: &Option<FileRange>, restrict_range: &Option<FileRange>,
matches_out: &mut Vec<Match>, matches_out: &mut Vec<Match>,
) { ) {
if !is_search_permitted(code) {
return;
}
if let Ok(m) = matching::get_match(false, rule, &code, restrict_range, &self.sema) { if let Ok(m) = matching::get_match(false, rule, &code, restrict_range, &self.sema) {
matches_out.push(m); matches_out.push(m);
} }
@ -149,6 +157,25 @@ impl<'db> MatchFinder<'db> {
} }
} }
/// Returns whether we support matching within `node` and all of its ancestors.
fn is_search_permitted_ancestors(node: &SyntaxNode) -> bool {
if let Some(parent) = node.parent() {
if !is_search_permitted_ancestors(&parent) {
return false;
}
}
is_search_permitted(node)
}
/// Returns whether we support matching within this kind of node.
fn is_search_permitted(node: &SyntaxNode) -> bool {
// FIXME: Properly handle use declarations. At the moment, if our search pattern is `foo::bar`
// and the code is `use foo::{baz, bar}`, we'll match `bar`, since it resolves to `foo::bar`.
// However we'll then replace just the part we matched `bar`. We probably need to instead remove
// `bar` and insert a new use declaration.
node.kind() != SyntaxKind::USE_ITEM
}
impl UsageCache { impl UsageCache {
fn find(&mut self, definition: &Definition) -> Option<&[Reference]> { fn find(&mut self, definition: &Definition) -> Option<&[Reference]> {
// We expect a very small number of cache entries (generally 1), so a linear scan should be // We expect a very small number of cache entries (generally 1), so a linear scan should be

View file

@ -804,3 +804,26 @@ fn overlapping_possible_matches() {
&["foo(foo(42))", "foo(foo(foo(foo(42))))"], &["foo(foo(42))", "foo(foo(foo(foo(42))))"],
); );
} }
#[test]
fn use_declaration_with_braces() {
// It would be OK for a path rule to match and alter a use declaration. We shouldn't mess it up
// though. In particular, we must not change `use foo::{baz, bar}` to `use foo::{baz,
// foo2::bar2}`.
mark::check!(use_declaration_with_braces);
assert_ssr_transform(
"foo::bar ==>> foo2::bar2",
r#"
mod foo { pub fn bar() {} pub fn baz() {} }
mod foo2 { pub fn bar2() {} }
use foo::{baz, bar};
fn main() { bar() }
"#,
expect![["
mod foo { pub fn bar() {} pub fn baz() {} }
mod foo2 { pub fn bar2() {} }
use foo::{baz, bar};
fn main() { foo2::bar2() }
"]],
)
}