mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-28 04:45:05 +00:00
SSR: Disable matching within use declarations
It currently does the wrong thing when the use declaration contains braces.
This commit is contained in:
parent
63f500b0ee
commit
8d09ab86ed
2 changed files with 51 additions and 1 deletions
|
@ -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
|
||||||
|
|
|
@ -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() }
|
||||||
|
"]],
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue