mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-13 05:38:46 +00:00
Merge #5518
5518: Use resolved paths in SSR rules r=matklad a=davidlattimore The main user-visible changes are: * SSR now matches paths based on whether they resolve to the same thing instead of whether they're written the same. * So `foo()` won't match `foo()` if it's a different function `foo()`, but will match `bar::foo()` if it's the same `foo`. * Paths in the replacement will now be rendered with appropriate qualification for their context. * For example `foo::Bar` will render as just `Bar` inside the module `foo`, but might render as `baz::foo::Bar` from elsewhere. * This means that all paths in the search pattern and replacement template must be able to be resolved. * It now also matters where you invoke SSR from, since paths are resolved relative to wherever that is. * Search now uses find-uses on paths to locate places to try matching. This means that when a path is present in the pattern, search will generally be pretty fast. * Function calls can now match method calls again, but this time only if they resolve to the same function. Co-authored-by: David Lattimore <dml@google.com>
This commit is contained in:
commit
c3defe2532
20 changed files with 1437 additions and 399 deletions
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -1164,6 +1164,7 @@ dependencies = [
|
||||||
name = "ra_ssr"
|
name = "ra_ssr"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"expect",
|
||||||
"ra_db",
|
"ra_db",
|
||||||
"ra_hir",
|
"ra_hir",
|
||||||
"ra_ide_db",
|
"ra_ide_db",
|
||||||
|
|
|
@ -505,9 +505,10 @@ impl Analysis {
|
||||||
&self,
|
&self,
|
||||||
query: &str,
|
query: &str,
|
||||||
parse_only: bool,
|
parse_only: bool,
|
||||||
|
position: FilePosition,
|
||||||
) -> Cancelable<Result<SourceChange, SsrError>> {
|
) -> Cancelable<Result<SourceChange, SsrError>> {
|
||||||
self.with_db(|db| {
|
self.with_db(|db| {
|
||||||
let edits = ssr::parse_search_replace(query, parse_only, db)?;
|
let edits = ssr::parse_search_replace(query, parse_only, db, position)?;
|
||||||
Ok(SourceChange::from(edits))
|
Ok(SourceChange::from(edits))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
use ra_db::SourceDatabaseExt;
|
use ra_db::FilePosition;
|
||||||
use ra_ide_db::{symbol_index::SymbolsDatabase, RootDatabase};
|
use ra_ide_db::RootDatabase;
|
||||||
|
|
||||||
use crate::SourceFileEdit;
|
use crate::SourceFileEdit;
|
||||||
use ra_ssr::{MatchFinder, SsrError, SsrRule};
|
use ra_ssr::{MatchFinder, SsrError, SsrRule};
|
||||||
|
@ -11,6 +11,19 @@ use ra_ssr::{MatchFinder, SsrError, SsrRule};
|
||||||
// A `$<name>` placeholder in the search pattern will match any AST node and `$<name>` will reference it in the replacement.
|
// A `$<name>` placeholder in the search pattern will match any AST node and `$<name>` will reference it in the replacement.
|
||||||
// Within a macro call, a placeholder will match up until whatever token follows the placeholder.
|
// Within a macro call, a placeholder will match up until whatever token follows the placeholder.
|
||||||
//
|
//
|
||||||
|
// All paths in both the search pattern and the replacement template must resolve in the context
|
||||||
|
// in which this command is invoked. Paths in the search pattern will then match the code if they
|
||||||
|
// resolve to the same item, even if they're written differently. For example if we invoke the
|
||||||
|
// command in the module `foo` with a pattern of `Bar`, then code in the parent module that refers
|
||||||
|
// to `foo::Bar` will match.
|
||||||
|
//
|
||||||
|
// Paths in the replacement template will be rendered appropriately for the context in which the
|
||||||
|
// replacement occurs. For example if our replacement template is `foo::Bar` and we match some
|
||||||
|
// code in the `foo` module, we'll insert just `Bar`.
|
||||||
|
//
|
||||||
|
// Method calls should generally be written in UFCS form. e.g. `foo::Bar::baz($s, $a)` will match
|
||||||
|
// `$s.baz($a)`, provided the method call `baz` resolves to the method `foo::Bar::baz`.
|
||||||
|
//
|
||||||
// Placeholders may be given constraints by writing them as `${<name>:<constraint1>:<constraint2>...}`.
|
// Placeholders may be given constraints by writing them as `${<name>:<constraint1>:<constraint2>...}`.
|
||||||
//
|
//
|
||||||
// Supported constraints:
|
// Supported constraints:
|
||||||
|
@ -43,21 +56,13 @@ pub fn parse_search_replace(
|
||||||
rule: &str,
|
rule: &str,
|
||||||
parse_only: bool,
|
parse_only: bool,
|
||||||
db: &RootDatabase,
|
db: &RootDatabase,
|
||||||
|
position: FilePosition,
|
||||||
) -> Result<Vec<SourceFileEdit>, SsrError> {
|
) -> Result<Vec<SourceFileEdit>, SsrError> {
|
||||||
let mut edits = vec![];
|
|
||||||
let rule: SsrRule = rule.parse()?;
|
let rule: SsrRule = rule.parse()?;
|
||||||
|
let mut match_finder = MatchFinder::in_context(db, position);
|
||||||
|
match_finder.add_rule(rule)?;
|
||||||
if parse_only {
|
if parse_only {
|
||||||
return Ok(edits);
|
return Ok(Vec::new());
|
||||||
}
|
}
|
||||||
let mut match_finder = MatchFinder::new(db);
|
Ok(match_finder.edits())
|
||||||
match_finder.add_rule(rule);
|
|
||||||
for &root in db.local_roots().iter() {
|
|
||||||
let sr = db.source_root(root);
|
|
||||||
for file_id in sr.iter() {
|
|
||||||
if let Some(edit) = match_finder.edits_for_file(file_id) {
|
|
||||||
edits.push(SourceFileEdit { file_id, edit });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(edits)
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -290,20 +290,25 @@ pub fn classify_name_ref(
|
||||||
|
|
||||||
let path = name_ref.syntax().ancestors().find_map(ast::Path::cast)?;
|
let path = name_ref.syntax().ancestors().find_map(ast::Path::cast)?;
|
||||||
let resolved = sema.resolve_path(&path)?;
|
let resolved = sema.resolve_path(&path)?;
|
||||||
let res = match resolved {
|
Some(NameRefClass::Definition(resolved.into()))
|
||||||
PathResolution::Def(def) => Definition::ModuleDef(def),
|
}
|
||||||
PathResolution::AssocItem(item) => {
|
|
||||||
let def = match item {
|
impl From<PathResolution> for Definition {
|
||||||
hir::AssocItem::Function(it) => it.into(),
|
fn from(path_resolution: PathResolution) -> Self {
|
||||||
hir::AssocItem::Const(it) => it.into(),
|
match path_resolution {
|
||||||
hir::AssocItem::TypeAlias(it) => it.into(),
|
PathResolution::Def(def) => Definition::ModuleDef(def),
|
||||||
};
|
PathResolution::AssocItem(item) => {
|
||||||
Definition::ModuleDef(def)
|
let def = match item {
|
||||||
}
|
hir::AssocItem::Function(it) => it.into(),
|
||||||
PathResolution::Local(local) => Definition::Local(local),
|
hir::AssocItem::Const(it) => it.into(),
|
||||||
PathResolution::TypeParam(par) => Definition::TypeParam(par),
|
hir::AssocItem::TypeAlias(it) => it.into(),
|
||||||
PathResolution::Macro(def) => Definition::Macro(def),
|
};
|
||||||
PathResolution::SelfType(impl_def) => Definition::SelfType(impl_def),
|
Definition::ModuleDef(def)
|
||||||
};
|
}
|
||||||
Some(NameRefClass::Definition(res))
|
PathResolution::Local(local) => Definition::Local(local),
|
||||||
|
PathResolution::TypeParam(par) => Definition::TypeParam(par),
|
||||||
|
PathResolution::Macro(def) => Definition::Macro(def),
|
||||||
|
PathResolution::SelfType(impl_def) => Definition::SelfType(impl_def),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -60,6 +60,10 @@ impl SearchScope {
|
||||||
SearchScope::new(std::iter::once((file, None)).collect())
|
SearchScope::new(std::iter::once((file, None)).collect())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn files(files: &[FileId]) -> SearchScope {
|
||||||
|
SearchScope::new(files.iter().map(|f| (*f, None)).collect())
|
||||||
|
}
|
||||||
|
|
||||||
pub fn intersection(&self, other: &SearchScope) -> SearchScope {
|
pub fn intersection(&self, other: &SearchScope) -> SearchScope {
|
||||||
let (mut small, mut large) = (&self.entries, &other.entries);
|
let (mut small, mut large) = (&self.entries, &other.entries);
|
||||||
if small.len() > large.len() {
|
if small.len() > large.len() {
|
||||||
|
|
|
@ -18,3 +18,6 @@ ra_ide_db = { path = "../ra_ide_db" }
|
||||||
hir = { path = "../ra_hir", package = "ra_hir" }
|
hir = { path = "../ra_hir", package = "ra_hir" }
|
||||||
rustc-hash = "1.1.0"
|
rustc-hash = "1.1.0"
|
||||||
test_utils = { path = "../test_utils" }
|
test_utils = { path = "../test_utils" }
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
expect = { path = "../expect" }
|
||||||
|
|
|
@ -4,44 +4,41 @@
|
||||||
//! based on a template.
|
//! based on a template.
|
||||||
|
|
||||||
mod matching;
|
mod matching;
|
||||||
|
mod nester;
|
||||||
mod parsing;
|
mod parsing;
|
||||||
mod replacing;
|
mod replacing;
|
||||||
|
mod resolving;
|
||||||
|
mod search;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
mod errors;
|
mod errors;
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests;
|
mod tests;
|
||||||
|
|
||||||
|
use crate::errors::bail;
|
||||||
pub use crate::errors::SsrError;
|
pub use crate::errors::SsrError;
|
||||||
pub use crate::matching::Match;
|
pub use crate::matching::Match;
|
||||||
use crate::matching::{record_match_fails_reasons_scope, MatchFailureReason};
|
use crate::matching::MatchFailureReason;
|
||||||
use hir::Semantics;
|
use hir::Semantics;
|
||||||
use ra_db::{FileId, FileRange};
|
use ra_db::{FileId, FilePosition, FileRange};
|
||||||
use ra_syntax::{ast, AstNode, SmolStr, SyntaxKind, SyntaxNode, TextRange};
|
use ra_ide_db::source_change::SourceFileEdit;
|
||||||
use ra_text_edit::TextEdit;
|
use ra_syntax::{ast, AstNode, SyntaxNode, TextRange};
|
||||||
|
use resolving::ResolvedRule;
|
||||||
use rustc_hash::FxHashMap;
|
use rustc_hash::FxHashMap;
|
||||||
|
|
||||||
// A structured search replace rule. Create by calling `parse` on a str.
|
// A structured search replace rule. Create by calling `parse` on a str.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct SsrRule {
|
pub struct SsrRule {
|
||||||
/// A structured pattern that we're searching for.
|
/// A structured pattern that we're searching for.
|
||||||
pattern: SsrPattern,
|
pattern: parsing::RawPattern,
|
||||||
/// What we'll replace it with.
|
/// What we'll replace it with.
|
||||||
template: parsing::SsrTemplate,
|
template: parsing::RawPattern,
|
||||||
|
parsed_rules: Vec<parsing::ParsedRule>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct SsrPattern {
|
pub struct SsrPattern {
|
||||||
raw: parsing::RawSearchPattern,
|
raw: parsing::RawPattern,
|
||||||
/// Placeholders keyed by the stand-in ident that we use in Rust source code.
|
parsed_rules: Vec<parsing::ParsedRule>,
|
||||||
placeholders_by_stand_in: FxHashMap<SmolStr, parsing::Placeholder>,
|
|
||||||
// We store our search pattern, parsed as each different kind of thing we can look for. As we
|
|
||||||
// traverse the AST, we get the appropriate one of these for the type of node we're on. For many
|
|
||||||
// search patterns, only some of these will be present.
|
|
||||||
expr: Option<SyntaxNode>,
|
|
||||||
type_ref: Option<SyntaxNode>,
|
|
||||||
item: Option<SyntaxNode>,
|
|
||||||
path: Option<SyntaxNode>,
|
|
||||||
pattern: Option<SyntaxNode>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Default)]
|
#[derive(Debug, Default)]
|
||||||
|
@ -53,40 +50,112 @@ pub struct SsrMatches {
|
||||||
pub struct MatchFinder<'db> {
|
pub struct MatchFinder<'db> {
|
||||||
/// Our source of information about the user's code.
|
/// Our source of information about the user's code.
|
||||||
sema: Semantics<'db, ra_ide_db::RootDatabase>,
|
sema: Semantics<'db, ra_ide_db::RootDatabase>,
|
||||||
rules: Vec<SsrRule>,
|
rules: Vec<ResolvedRule>,
|
||||||
|
scope: hir::SemanticsScope<'db>,
|
||||||
|
hygiene: hir::Hygiene,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'db> MatchFinder<'db> {
|
impl<'db> MatchFinder<'db> {
|
||||||
pub fn new(db: &'db ra_ide_db::RootDatabase) -> MatchFinder<'db> {
|
/// Constructs a new instance where names will be looked up as if they appeared at
|
||||||
MatchFinder { sema: Semantics::new(db), rules: Vec::new() }
|
/// `lookup_context`.
|
||||||
|
pub fn in_context(
|
||||||
|
db: &'db ra_ide_db::RootDatabase,
|
||||||
|
lookup_context: FilePosition,
|
||||||
|
) -> MatchFinder<'db> {
|
||||||
|
let sema = Semantics::new(db);
|
||||||
|
let file = sema.parse(lookup_context.file_id);
|
||||||
|
// Find a node at the requested position, falling back to the whole file.
|
||||||
|
let node = file
|
||||||
|
.syntax()
|
||||||
|
.token_at_offset(lookup_context.offset)
|
||||||
|
.left_biased()
|
||||||
|
.map(|token| token.parent())
|
||||||
|
.unwrap_or_else(|| file.syntax().clone());
|
||||||
|
let scope = sema.scope(&node);
|
||||||
|
MatchFinder {
|
||||||
|
sema: Semantics::new(db),
|
||||||
|
rules: Vec::new(),
|
||||||
|
scope,
|
||||||
|
hygiene: hir::Hygiene::new(db, lookup_context.file_id.into()),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn add_rule(&mut self, rule: SsrRule) {
|
/// Constructs an instance using the start of the first file in `db` as the lookup context.
|
||||||
self.rules.push(rule);
|
pub fn at_first_file(db: &'db ra_ide_db::RootDatabase) -> Result<MatchFinder<'db>, SsrError> {
|
||||||
|
use ra_db::SourceDatabaseExt;
|
||||||
|
use ra_ide_db::symbol_index::SymbolsDatabase;
|
||||||
|
if let Some(first_file_id) = db
|
||||||
|
.local_roots()
|
||||||
|
.iter()
|
||||||
|
.next()
|
||||||
|
.and_then(|root| db.source_root(root.clone()).iter().next())
|
||||||
|
{
|
||||||
|
Ok(MatchFinder::in_context(
|
||||||
|
db,
|
||||||
|
FilePosition { file_id: first_file_id, offset: 0.into() },
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
bail!("No files to search");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Adds a rule to be applied. The order in which rules are added matters. Earlier rules take
|
||||||
|
/// precedence. If a node is matched by an earlier rule, then later rules won't be permitted to
|
||||||
|
/// match to it.
|
||||||
|
pub fn add_rule(&mut self, rule: SsrRule) -> Result<(), SsrError> {
|
||||||
|
for parsed_rule in rule.parsed_rules {
|
||||||
|
self.rules.push(ResolvedRule::new(
|
||||||
|
parsed_rule,
|
||||||
|
&self.scope,
|
||||||
|
&self.hygiene,
|
||||||
|
self.rules.len(),
|
||||||
|
)?);
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Finds matches for all added rules and returns edits for all found matches.
|
||||||
|
pub fn edits(&self) -> Vec<SourceFileEdit> {
|
||||||
|
use ra_db::SourceDatabaseExt;
|
||||||
|
let mut matches_by_file = FxHashMap::default();
|
||||||
|
for m in self.matches().matches {
|
||||||
|
matches_by_file
|
||||||
|
.entry(m.range.file_id)
|
||||||
|
.or_insert_with(|| SsrMatches::default())
|
||||||
|
.matches
|
||||||
|
.push(m);
|
||||||
|
}
|
||||||
|
let mut edits = vec![];
|
||||||
|
for (file_id, matches) in matches_by_file {
|
||||||
|
let edit =
|
||||||
|
replacing::matches_to_edit(&matches, &self.sema.db.file_text(file_id), &self.rules);
|
||||||
|
edits.push(SourceFileEdit { file_id, edit });
|
||||||
|
}
|
||||||
|
edits
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Adds a search pattern. For use if you intend to only call `find_matches_in_file`. If you
|
/// Adds a search pattern. For use if you intend to only call `find_matches_in_file`. If you
|
||||||
/// intend to do replacement, use `add_rule` instead.
|
/// intend to do replacement, use `add_rule` instead.
|
||||||
pub fn add_search_pattern(&mut self, pattern: SsrPattern) {
|
pub fn add_search_pattern(&mut self, pattern: SsrPattern) -> Result<(), SsrError> {
|
||||||
self.add_rule(SsrRule { pattern, template: "()".parse().unwrap() })
|
for parsed_rule in pattern.parsed_rules {
|
||||||
}
|
self.rules.push(ResolvedRule::new(
|
||||||
|
parsed_rule,
|
||||||
pub fn edits_for_file(&self, file_id: FileId) -> Option<TextEdit> {
|
&self.scope,
|
||||||
let matches = self.find_matches_in_file(file_id);
|
&self.hygiene,
|
||||||
if matches.matches.is_empty() {
|
self.rules.len(),
|
||||||
None
|
)?);
|
||||||
} else {
|
|
||||||
use ra_db::SourceDatabaseExt;
|
|
||||||
Some(replacing::matches_to_edit(&matches, &self.sema.db.file_text(file_id)))
|
|
||||||
}
|
}
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn find_matches_in_file(&self, file_id: FileId) -> SsrMatches {
|
/// Returns matches for all added rules.
|
||||||
let file = self.sema.parse(file_id);
|
pub fn matches(&self) -> SsrMatches {
|
||||||
let code = file.syntax();
|
let mut matches = Vec::new();
|
||||||
let mut matches = SsrMatches::default();
|
let mut usage_cache = search::UsageCache::default();
|
||||||
self.find_matches(code, &None, &mut matches);
|
for rule in &self.rules {
|
||||||
matches
|
self.find_matches_for_rule(rule, &mut usage_cache, &mut matches);
|
||||||
|
}
|
||||||
|
nester::nest_and_remove_collisions(matches, &self.sema)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Finds all nodes in `file_id` whose text is exactly equal to `snippet` and attempts to match
|
/// Finds all nodes in `file_id` whose text is exactly equal to `snippet` and attempts to match
|
||||||
|
@ -115,53 +184,6 @@ impl<'db> MatchFinder<'db> {
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
|
|
||||||
fn find_matches(
|
|
||||||
&self,
|
|
||||||
code: &SyntaxNode,
|
|
||||||
restrict_range: &Option<FileRange>,
|
|
||||||
matches_out: &mut SsrMatches,
|
|
||||||
) {
|
|
||||||
for rule in &self.rules {
|
|
||||||
if let Ok(mut m) = matching::get_match(false, rule, &code, restrict_range, &self.sema) {
|
|
||||||
// Continue searching in each of our placeholders.
|
|
||||||
for placeholder_value in m.placeholder_values.values_mut() {
|
|
||||||
if let Some(placeholder_node) = &placeholder_value.node {
|
|
||||||
// Don't search our placeholder if it's the entire matched node, otherwise we'd
|
|
||||||
// find the same match over and over until we got a stack overflow.
|
|
||||||
if placeholder_node != code {
|
|
||||||
self.find_matches(
|
|
||||||
placeholder_node,
|
|
||||||
restrict_range,
|
|
||||||
&mut placeholder_value.inner_matches,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
matches_out.matches.push(m);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// If we've got a macro call, we already tried matching it pre-expansion, which is the only
|
|
||||||
// way to match the whole macro, now try expanding it and matching the expansion.
|
|
||||||
if let Some(macro_call) = ast::MacroCall::cast(code.clone()) {
|
|
||||||
if let Some(expanded) = self.sema.expand(¯o_call) {
|
|
||||||
if let Some(tt) = macro_call.token_tree() {
|
|
||||||
// When matching within a macro expansion, we only want to allow matches of
|
|
||||||
// nodes that originated entirely from within the token tree of the macro call.
|
|
||||||
// i.e. we don't want to match something that came from the macro itself.
|
|
||||||
self.find_matches(
|
|
||||||
&expanded,
|
|
||||||
&Some(self.sema.original_range(tt.syntax())),
|
|
||||||
matches_out,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for child in code.children() {
|
|
||||||
self.find_matches(&child, restrict_range, matches_out);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn output_debug_for_nodes_at_range(
|
fn output_debug_for_nodes_at_range(
|
||||||
&self,
|
&self,
|
||||||
node: &SyntaxNode,
|
node: &SyntaxNode,
|
||||||
|
@ -177,8 +199,17 @@ impl<'db> MatchFinder<'db> {
|
||||||
}
|
}
|
||||||
if node_range.range == range.range {
|
if node_range.range == range.range {
|
||||||
for rule in &self.rules {
|
for rule in &self.rules {
|
||||||
let pattern =
|
// For now we ignore rules that have a different kind than our node, otherwise
|
||||||
rule.pattern.tree_for_kind_with_reason(node.kind()).map(|p| p.clone());
|
// we get lots of noise. If at some point we add support for restricting rules
|
||||||
|
// to a particular kind of thing (e.g. only match type references), then we can
|
||||||
|
// relax this. We special-case expressions, since function calls can match
|
||||||
|
// method calls.
|
||||||
|
if rule.pattern.node.kind() != node.kind()
|
||||||
|
&& !(ast::Expr::can_cast(rule.pattern.node.kind())
|
||||||
|
&& ast::Expr::can_cast(node.kind()))
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
out.push(MatchDebugInfo {
|
out.push(MatchDebugInfo {
|
||||||
matched: matching::get_match(true, rule, &node, restrict_range, &self.sema)
|
matched: matching::get_match(true, rule, &node, restrict_range, &self.sema)
|
||||||
.map_err(|e| MatchFailureReason {
|
.map_err(|e| MatchFailureReason {
|
||||||
|
@ -186,7 +217,7 @@ impl<'db> MatchFinder<'db> {
|
||||||
"Match failed, but no reason was given".to_owned()
|
"Match failed, but no reason was given".to_owned()
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
pattern,
|
pattern: rule.pattern.node.clone(),
|
||||||
node: node.clone(),
|
node: node.clone(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -209,9 +240,8 @@ impl<'db> MatchFinder<'db> {
|
||||||
|
|
||||||
pub struct MatchDebugInfo {
|
pub struct MatchDebugInfo {
|
||||||
node: SyntaxNode,
|
node: SyntaxNode,
|
||||||
/// Our search pattern parsed as the same kind of syntax node as `node`. e.g. expression, item,
|
/// Our search pattern parsed as an expression or item, etc
|
||||||
/// etc. Will be absent if the pattern can't be parsed as that kind.
|
pattern: SyntaxNode,
|
||||||
pattern: Result<SyntaxNode, MatchFailureReason>,
|
|
||||||
matched: Result<Match, MatchFailureReason>,
|
matched: Result<Match, MatchFailureReason>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -228,29 +258,12 @@ impl std::fmt::Debug for MatchDebugInfo {
|
||||||
self.node
|
self.node
|
||||||
)?;
|
)?;
|
||||||
writeln!(f, "========= PATTERN ==========")?;
|
writeln!(f, "========= PATTERN ==========")?;
|
||||||
match &self.pattern {
|
writeln!(f, "{:#?}", self.pattern)?;
|
||||||
Ok(pattern) => {
|
|
||||||
writeln!(f, "{:#?}", pattern)?;
|
|
||||||
}
|
|
||||||
Err(err) => {
|
|
||||||
writeln!(f, "{}", err.reason)?;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
writeln!(f, "============================")?;
|
writeln!(f, "============================")?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SsrPattern {
|
|
||||||
fn tree_for_kind_with_reason(
|
|
||||||
&self,
|
|
||||||
kind: SyntaxKind,
|
|
||||||
) -> Result<&SyntaxNode, MatchFailureReason> {
|
|
||||||
record_match_fails_reasons_scope(true, || self.tree_for_kind(kind))
|
|
||||||
.map_err(|e| MatchFailureReason { reason: e.reason.unwrap() })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl SsrMatches {
|
impl SsrMatches {
|
||||||
/// Returns `self` with any nested matches removed and made into top-level matches.
|
/// Returns `self` with any nested matches removed and made into top-level matches.
|
||||||
pub fn flattened(self) -> SsrMatches {
|
pub fn flattened(self) -> SsrMatches {
|
||||||
|
|
|
@ -2,8 +2,9 @@
|
||||||
//! process of matching, placeholder values are recorded.
|
//! process of matching, placeholder values are recorded.
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
parsing::{Constraint, NodeKind, Placeholder, SsrTemplate},
|
parsing::{Constraint, NodeKind, Placeholder},
|
||||||
SsrMatches, SsrPattern, SsrRule,
|
resolving::{ResolvedPattern, ResolvedRule},
|
||||||
|
SsrMatches,
|
||||||
};
|
};
|
||||||
use hir::Semantics;
|
use hir::Semantics;
|
||||||
use ra_db::FileRange;
|
use ra_db::FileRange;
|
||||||
|
@ -48,9 +49,11 @@ pub struct Match {
|
||||||
pub(crate) matched_node: SyntaxNode,
|
pub(crate) matched_node: SyntaxNode,
|
||||||
pub(crate) placeholder_values: FxHashMap<Var, PlaceholderMatch>,
|
pub(crate) placeholder_values: FxHashMap<Var, PlaceholderMatch>,
|
||||||
pub(crate) ignored_comments: Vec<ast::Comment>,
|
pub(crate) ignored_comments: Vec<ast::Comment>,
|
||||||
// A copy of the template for the rule that produced this match. We store this on the match for
|
pub(crate) rule_index: usize,
|
||||||
// if/when we do replacement.
|
/// The depth of matched_node.
|
||||||
pub(crate) template: SsrTemplate,
|
pub(crate) depth: usize,
|
||||||
|
// Each path in the template rendered for the module in which the match was found.
|
||||||
|
pub(crate) rendered_template_paths: FxHashMap<SyntaxNode, hir::ModPath>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Represents a `$var` in an SSR query.
|
/// Represents a `$var` in an SSR query.
|
||||||
|
@ -86,7 +89,7 @@ pub(crate) struct MatchFailed {
|
||||||
/// parent module, we don't populate nested matches.
|
/// parent module, we don't populate nested matches.
|
||||||
pub(crate) fn get_match(
|
pub(crate) fn get_match(
|
||||||
debug_active: bool,
|
debug_active: bool,
|
||||||
rule: &SsrRule,
|
rule: &ResolvedRule,
|
||||||
code: &SyntaxNode,
|
code: &SyntaxNode,
|
||||||
restrict_range: &Option<FileRange>,
|
restrict_range: &Option<FileRange>,
|
||||||
sema: &Semantics<ra_ide_db::RootDatabase>,
|
sema: &Semantics<ra_ide_db::RootDatabase>,
|
||||||
|
@ -102,7 +105,7 @@ struct Matcher<'db, 'sema> {
|
||||||
/// If any placeholders come from anywhere outside of this range, then the match will be
|
/// If any placeholders come from anywhere outside of this range, then the match will be
|
||||||
/// rejected.
|
/// rejected.
|
||||||
restrict_range: Option<FileRange>,
|
restrict_range: Option<FileRange>,
|
||||||
rule: &'sema SsrRule,
|
rule: &'sema ResolvedRule,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Which phase of matching we're currently performing. We do two phases because most attempted
|
/// Which phase of matching we're currently performing. We do two phases because most attempted
|
||||||
|
@ -117,26 +120,35 @@ enum Phase<'a> {
|
||||||
|
|
||||||
impl<'db, 'sema> Matcher<'db, 'sema> {
|
impl<'db, 'sema> Matcher<'db, 'sema> {
|
||||||
fn try_match(
|
fn try_match(
|
||||||
rule: &'sema SsrRule,
|
rule: &ResolvedRule,
|
||||||
code: &SyntaxNode,
|
code: &SyntaxNode,
|
||||||
restrict_range: &Option<FileRange>,
|
restrict_range: &Option<FileRange>,
|
||||||
sema: &'sema Semantics<'db, ra_ide_db::RootDatabase>,
|
sema: &'sema Semantics<'db, ra_ide_db::RootDatabase>,
|
||||||
) -> Result<Match, MatchFailed> {
|
) -> Result<Match, MatchFailed> {
|
||||||
let match_state = Matcher { sema, restrict_range: restrict_range.clone(), rule };
|
let match_state = Matcher { sema, restrict_range: restrict_range.clone(), rule };
|
||||||
let pattern_tree = rule.pattern.tree_for_kind(code.kind())?;
|
|
||||||
// First pass at matching, where we check that node types and idents match.
|
// First pass at matching, where we check that node types and idents match.
|
||||||
match_state.attempt_match_node(&mut Phase::First, &pattern_tree, code)?;
|
match_state.attempt_match_node(&mut Phase::First, &rule.pattern.node, code)?;
|
||||||
match_state.validate_range(&sema.original_range(code))?;
|
match_state.validate_range(&sema.original_range(code))?;
|
||||||
let mut the_match = Match {
|
let mut the_match = Match {
|
||||||
range: sema.original_range(code),
|
range: sema.original_range(code),
|
||||||
matched_node: code.clone(),
|
matched_node: code.clone(),
|
||||||
placeholder_values: FxHashMap::default(),
|
placeholder_values: FxHashMap::default(),
|
||||||
ignored_comments: Vec::new(),
|
ignored_comments: Vec::new(),
|
||||||
template: rule.template.clone(),
|
rule_index: rule.index,
|
||||||
|
depth: 0,
|
||||||
|
rendered_template_paths: FxHashMap::default(),
|
||||||
};
|
};
|
||||||
// Second matching pass, where we record placeholder matches, ignored comments and maybe do
|
// Second matching pass, where we record placeholder matches, ignored comments and maybe do
|
||||||
// any other more expensive checks that we didn't want to do on the first pass.
|
// any other more expensive checks that we didn't want to do on the first pass.
|
||||||
match_state.attempt_match_node(&mut Phase::Second(&mut the_match), &pattern_tree, code)?;
|
match_state.attempt_match_node(
|
||||||
|
&mut Phase::Second(&mut the_match),
|
||||||
|
&rule.pattern.node,
|
||||||
|
code,
|
||||||
|
)?;
|
||||||
|
the_match.depth = sema.ancestors_with_macros(the_match.matched_node.clone()).count();
|
||||||
|
if let Some(template) = &rule.template {
|
||||||
|
the_match.render_template_paths(template, sema)?;
|
||||||
|
}
|
||||||
Ok(the_match)
|
Ok(the_match)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -177,10 +189,17 @@ impl<'db, 'sema> Matcher<'db, 'sema> {
|
||||||
}
|
}
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
// Non-placeholders.
|
// We allow a UFCS call to match a method call, provided they resolve to the same function.
|
||||||
|
if let Some(pattern_function) = self.rule.pattern.ufcs_function_calls.get(pattern) {
|
||||||
|
if let (Some(pattern), Some(code)) =
|
||||||
|
(ast::CallExpr::cast(pattern.clone()), ast::MethodCallExpr::cast(code.clone()))
|
||||||
|
{
|
||||||
|
return self.attempt_match_ufcs(phase, &pattern, &code, *pattern_function);
|
||||||
|
}
|
||||||
|
}
|
||||||
if pattern.kind() != code.kind() {
|
if pattern.kind() != code.kind() {
|
||||||
fail_match!(
|
fail_match!(
|
||||||
"Pattern had a `{}` ({:?}), code had `{}` ({:?})",
|
"Pattern had `{}` ({:?}), code had `{}` ({:?})",
|
||||||
pattern.text(),
|
pattern.text(),
|
||||||
pattern.kind(),
|
pattern.kind(),
|
||||||
code.text(),
|
code.text(),
|
||||||
|
@ -194,6 +213,7 @@ impl<'db, 'sema> Matcher<'db, 'sema> {
|
||||||
self.attempt_match_record_field_list(phase, pattern, code)
|
self.attempt_match_record_field_list(phase, pattern, code)
|
||||||
}
|
}
|
||||||
SyntaxKind::TOKEN_TREE => self.attempt_match_token_tree(phase, pattern, code),
|
SyntaxKind::TOKEN_TREE => self.attempt_match_token_tree(phase, pattern, code),
|
||||||
|
SyntaxKind::PATH => self.attempt_match_path(phase, pattern, code),
|
||||||
_ => self.attempt_match_node_children(phase, pattern, code),
|
_ => self.attempt_match_node_children(phase, pattern, code),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -310,6 +330,64 @@ impl<'db, 'sema> Matcher<'db, 'sema> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Paths are matched based on whether they refer to the same thing, even if they're written
|
||||||
|
/// differently.
|
||||||
|
fn attempt_match_path(
|
||||||
|
&self,
|
||||||
|
phase: &mut Phase,
|
||||||
|
pattern: &SyntaxNode,
|
||||||
|
code: &SyntaxNode,
|
||||||
|
) -> Result<(), MatchFailed> {
|
||||||
|
if let Some(pattern_resolved) = self.rule.pattern.resolved_paths.get(pattern) {
|
||||||
|
let pattern_path = ast::Path::cast(pattern.clone()).unwrap();
|
||||||
|
let code_path = ast::Path::cast(code.clone()).unwrap();
|
||||||
|
if let (Some(pattern_segment), Some(code_segment)) =
|
||||||
|
(pattern_path.segment(), code_path.segment())
|
||||||
|
{
|
||||||
|
// Match everything within the segment except for the name-ref, which is handled
|
||||||
|
// separately via comparing what the path resolves to below.
|
||||||
|
self.attempt_match_opt(
|
||||||
|
phase,
|
||||||
|
pattern_segment.type_arg_list(),
|
||||||
|
code_segment.type_arg_list(),
|
||||||
|
)?;
|
||||||
|
self.attempt_match_opt(
|
||||||
|
phase,
|
||||||
|
pattern_segment.param_list(),
|
||||||
|
code_segment.param_list(),
|
||||||
|
)?;
|
||||||
|
}
|
||||||
|
if matches!(phase, Phase::Second(_)) {
|
||||||
|
let resolution = self
|
||||||
|
.sema
|
||||||
|
.resolve_path(&code_path)
|
||||||
|
.ok_or_else(|| match_error!("Failed to resolve path `{}`", code.text()))?;
|
||||||
|
if pattern_resolved.resolution != resolution {
|
||||||
|
fail_match!("Pattern had path `{}` code had `{}`", pattern.text(), code.text());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return self.attempt_match_node_children(phase, pattern, code);
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn attempt_match_opt<T: AstNode>(
|
||||||
|
&self,
|
||||||
|
phase: &mut Phase,
|
||||||
|
pattern: Option<T>,
|
||||||
|
code: Option<T>,
|
||||||
|
) -> Result<(), MatchFailed> {
|
||||||
|
match (pattern, code) {
|
||||||
|
(Some(p), Some(c)) => self.attempt_match_node(phase, &p.syntax(), &c.syntax()),
|
||||||
|
(None, None) => Ok(()),
|
||||||
|
(Some(p), None) => fail_match!("Pattern `{}` had nothing to match", p.syntax().text()),
|
||||||
|
(None, Some(c)) => {
|
||||||
|
fail_match!("Nothing in pattern to match code `{}`", c.syntax().text())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// We want to allow the records to match in any order, so we have special matching logic for
|
/// We want to allow the records to match in any order, so we have special matching logic for
|
||||||
/// them.
|
/// them.
|
||||||
fn attempt_match_record_field_list(
|
fn attempt_match_record_field_list(
|
||||||
|
@ -443,9 +521,61 @@ impl<'db, 'sema> Matcher<'db, 'sema> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn attempt_match_ufcs(
|
||||||
|
&self,
|
||||||
|
phase: &mut Phase,
|
||||||
|
pattern: &ast::CallExpr,
|
||||||
|
code: &ast::MethodCallExpr,
|
||||||
|
pattern_function: hir::Function,
|
||||||
|
) -> Result<(), MatchFailed> {
|
||||||
|
use ast::ArgListOwner;
|
||||||
|
let code_resolved_function = self
|
||||||
|
.sema
|
||||||
|
.resolve_method_call(code)
|
||||||
|
.ok_or_else(|| match_error!("Failed to resolve method call"))?;
|
||||||
|
if pattern_function != code_resolved_function {
|
||||||
|
fail_match!("Method call resolved to a different function");
|
||||||
|
}
|
||||||
|
// Check arguments.
|
||||||
|
let mut pattern_args = pattern
|
||||||
|
.arg_list()
|
||||||
|
.ok_or_else(|| match_error!("Pattern function call has no args"))?
|
||||||
|
.args();
|
||||||
|
self.attempt_match_opt(phase, pattern_args.next(), code.expr())?;
|
||||||
|
let mut code_args =
|
||||||
|
code.arg_list().ok_or_else(|| match_error!("Code method call has no args"))?.args();
|
||||||
|
loop {
|
||||||
|
match (pattern_args.next(), code_args.next()) {
|
||||||
|
(None, None) => return Ok(()),
|
||||||
|
(p, c) => self.attempt_match_opt(phase, p, c)?,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn get_placeholder(&self, element: &SyntaxElement) -> Option<&Placeholder> {
|
fn get_placeholder(&self, element: &SyntaxElement) -> Option<&Placeholder> {
|
||||||
only_ident(element.clone())
|
only_ident(element.clone()).and_then(|ident| self.rule.get_placeholder(&ident))
|
||||||
.and_then(|ident| self.rule.pattern.placeholders_by_stand_in.get(ident.text()))
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Match {
|
||||||
|
fn render_template_paths(
|
||||||
|
&mut self,
|
||||||
|
template: &ResolvedPattern,
|
||||||
|
sema: &Semantics<ra_ide_db::RootDatabase>,
|
||||||
|
) -> Result<(), MatchFailed> {
|
||||||
|
let module = sema
|
||||||
|
.scope(&self.matched_node)
|
||||||
|
.module()
|
||||||
|
.ok_or_else(|| match_error!("Matched node isn't in a module"))?;
|
||||||
|
for (path, resolved_path) in &template.resolved_paths {
|
||||||
|
if let hir::PathResolution::Def(module_def) = resolved_path.resolution {
|
||||||
|
let mod_path = module.find_use_path(sema.db, module_def).ok_or_else(|| {
|
||||||
|
match_error!("Failed to render template path `{}` at match location")
|
||||||
|
})?;
|
||||||
|
self.rendered_template_paths.insert(path.clone(), mod_path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -510,28 +640,6 @@ impl PlaceholderMatch {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SsrPattern {
|
|
||||||
pub(crate) fn tree_for_kind(&self, kind: SyntaxKind) -> Result<&SyntaxNode, MatchFailed> {
|
|
||||||
let (tree, kind_name) = if ast::Expr::can_cast(kind) {
|
|
||||||
(&self.expr, "expression")
|
|
||||||
} else if ast::TypeRef::can_cast(kind) {
|
|
||||||
(&self.type_ref, "type reference")
|
|
||||||
} else if ast::ModuleItem::can_cast(kind) {
|
|
||||||
(&self.item, "item")
|
|
||||||
} else if ast::Path::can_cast(kind) {
|
|
||||||
(&self.path, "path")
|
|
||||||
} else if ast::Pat::can_cast(kind) {
|
|
||||||
(&self.pattern, "pattern")
|
|
||||||
} else {
|
|
||||||
fail_match!("Matching nodes of kind {:?} is not supported", kind);
|
|
||||||
};
|
|
||||||
match tree {
|
|
||||||
Some(tree) => Ok(tree),
|
|
||||||
None => fail_match!("Pattern cannot be parsed as a {}", kind_name),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl NodeKind {
|
impl NodeKind {
|
||||||
fn matches(&self, node: &SyntaxNode) -> Result<(), MatchFailed> {
|
fn matches(&self, node: &SyntaxNode) -> Result<(), MatchFailed> {
|
||||||
let ok = match self {
|
let ok = match self {
|
||||||
|
@ -596,13 +704,12 @@ mod tests {
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_match_replace() {
|
fn parse_match_replace() {
|
||||||
let rule: SsrRule = "foo($x) ==>> bar($x)".parse().unwrap();
|
let rule: SsrRule = "foo($x) ==>> bar($x)".parse().unwrap();
|
||||||
let input = "fn foo() {} fn main() { foo(1+2); }";
|
let input = "fn foo() {} fn bar() {} fn main() { foo(1+2); }";
|
||||||
|
|
||||||
use ra_db::fixture::WithFixture;
|
let (db, position) = crate::tests::single_file(input);
|
||||||
let (db, file_id) = ra_ide_db::RootDatabase::with_single_file(input);
|
let mut match_finder = MatchFinder::in_context(&db, position);
|
||||||
let mut match_finder = MatchFinder::new(&db);
|
match_finder.add_rule(rule).unwrap();
|
||||||
match_finder.add_rule(rule);
|
let matches = match_finder.matches();
|
||||||
let matches = match_finder.find_matches_in_file(file_id);
|
|
||||||
assert_eq!(matches.matches.len(), 1);
|
assert_eq!(matches.matches.len(), 1);
|
||||||
assert_eq!(matches.matches[0].matched_node.text(), "foo(1+2)");
|
assert_eq!(matches.matches[0].matched_node.text(), "foo(1+2)");
|
||||||
assert_eq!(matches.matches[0].placeholder_values.len(), 1);
|
assert_eq!(matches.matches[0].placeholder_values.len(), 1);
|
||||||
|
@ -615,9 +722,11 @@ mod tests {
|
||||||
"1+2"
|
"1+2"
|
||||||
);
|
);
|
||||||
|
|
||||||
let edit = crate::replacing::matches_to_edit(&matches, input);
|
let edits = match_finder.edits();
|
||||||
|
assert_eq!(edits.len(), 1);
|
||||||
|
let edit = &edits[0];
|
||||||
let mut after = input.to_string();
|
let mut after = input.to_string();
|
||||||
edit.apply(&mut after);
|
edit.edit.apply(&mut after);
|
||||||
assert_eq!(after, "fn foo() {} fn main() { bar(1+2); }");
|
assert_eq!(after, "fn foo() {} fn bar() {} fn main() { bar(1+2); }");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
98
crates/ra_ssr/src/nester.rs
Normal file
98
crates/ra_ssr/src/nester.rs
Normal file
|
@ -0,0 +1,98 @@
|
||||||
|
//! Converts a flat collection of matches into a nested form suitable for replacement. When there
|
||||||
|
//! are multiple matches for a node, or that overlap, priority is given to the earlier rule. Nested
|
||||||
|
//! matches are only permitted if the inner match is contained entirely within a placeholder of an
|
||||||
|
//! outer match.
|
||||||
|
//!
|
||||||
|
//! For example, if our search pattern is `foo(foo($a))` and the code had `foo(foo(foo(foo(42))))`,
|
||||||
|
//! then we'll get 3 matches, however only the outermost and innermost matches can be accepted. The
|
||||||
|
//! middle match would take the second `foo` from the outer match.
|
||||||
|
|
||||||
|
use crate::{Match, SsrMatches};
|
||||||
|
use ra_syntax::SyntaxNode;
|
||||||
|
use rustc_hash::FxHashMap;
|
||||||
|
|
||||||
|
pub(crate) fn nest_and_remove_collisions(
|
||||||
|
mut matches: Vec<Match>,
|
||||||
|
sema: &hir::Semantics<ra_ide_db::RootDatabase>,
|
||||||
|
) -> SsrMatches {
|
||||||
|
// We sort the matches by depth then by rule index. Sorting by depth means that by the time we
|
||||||
|
// see a match, any parent matches or conflicting matches will have already been seen. Sorting
|
||||||
|
// by rule_index means that if there are two matches for the same node, the rule added first
|
||||||
|
// will take precedence.
|
||||||
|
matches.sort_by(|a, b| a.depth.cmp(&b.depth).then_with(|| a.rule_index.cmp(&b.rule_index)));
|
||||||
|
let mut collector = MatchCollector::default();
|
||||||
|
for m in matches {
|
||||||
|
collector.add_match(m, sema);
|
||||||
|
}
|
||||||
|
collector.into()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
struct MatchCollector {
|
||||||
|
matches_by_node: FxHashMap<SyntaxNode, Match>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MatchCollector {
|
||||||
|
/// Attempts to add `m` to matches. If it conflicts with an existing match, it is discarded. If
|
||||||
|
/// it is entirely within the a placeholder of an existing match, then it is added as a child
|
||||||
|
/// match of the existing match.
|
||||||
|
fn add_match(&mut self, m: Match, sema: &hir::Semantics<ra_ide_db::RootDatabase>) {
|
||||||
|
let matched_node = m.matched_node.clone();
|
||||||
|
if let Some(existing) = self.matches_by_node.get_mut(&matched_node) {
|
||||||
|
try_add_sub_match(m, existing, sema);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for ancestor in sema.ancestors_with_macros(m.matched_node.clone()) {
|
||||||
|
if let Some(existing) = self.matches_by_node.get_mut(&ancestor) {
|
||||||
|
try_add_sub_match(m, existing, sema);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self.matches_by_node.insert(matched_node, m);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Attempts to add `m` as a sub-match of `existing`.
|
||||||
|
fn try_add_sub_match(
|
||||||
|
m: Match,
|
||||||
|
existing: &mut Match,
|
||||||
|
sema: &hir::Semantics<ra_ide_db::RootDatabase>,
|
||||||
|
) {
|
||||||
|
for p in existing.placeholder_values.values_mut() {
|
||||||
|
// Note, no need to check if p.range.file is equal to m.range.file, since we
|
||||||
|
// already know we're within `existing`.
|
||||||
|
if p.range.range.contains_range(m.range.range) {
|
||||||
|
// Convert the inner matches in `p` into a temporary MatchCollector. When
|
||||||
|
// we're done, we then convert it back into an SsrMatches. If we expected
|
||||||
|
// lots of inner matches, it might be worthwhile keeping a MatchCollector
|
||||||
|
// around for each placeholder match. However we expect most placeholder
|
||||||
|
// will have 0 and a few will have 1. More than that should hopefully be
|
||||||
|
// exceptional.
|
||||||
|
let mut collector = MatchCollector::default();
|
||||||
|
for m in std::mem::replace(&mut p.inner_matches.matches, Vec::new()) {
|
||||||
|
collector.matches_by_node.insert(m.matched_node.clone(), m);
|
||||||
|
}
|
||||||
|
collector.add_match(m, sema);
|
||||||
|
p.inner_matches = collector.into();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<MatchCollector> for SsrMatches {
|
||||||
|
fn from(mut match_collector: MatchCollector) -> Self {
|
||||||
|
let mut matches = SsrMatches::default();
|
||||||
|
for (_, m) in match_collector.matches_by_node.drain() {
|
||||||
|
matches.matches.push(m);
|
||||||
|
}
|
||||||
|
matches.matches.sort_by(|a, b| {
|
||||||
|
// Order matches by file_id then by start range. This should be sufficient since ranges
|
||||||
|
// shouldn't be overlapping.
|
||||||
|
a.range
|
||||||
|
.file_id
|
||||||
|
.cmp(&b.range.file_id)
|
||||||
|
.then_with(|| a.range.range.start().cmp(&b.range.range.start()))
|
||||||
|
});
|
||||||
|
matches
|
||||||
|
}
|
||||||
|
}
|
|
@ -7,17 +7,19 @@
|
||||||
|
|
||||||
use crate::errors::bail;
|
use crate::errors::bail;
|
||||||
use crate::{SsrError, SsrPattern, SsrRule};
|
use crate::{SsrError, SsrPattern, SsrRule};
|
||||||
use ra_syntax::{ast, AstNode, SmolStr, SyntaxKind, T};
|
use ra_syntax::{ast, AstNode, SmolStr, SyntaxKind, SyntaxNode, T};
|
||||||
use rustc_hash::{FxHashMap, FxHashSet};
|
use rustc_hash::{FxHashMap, FxHashSet};
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Debug)]
|
||||||
pub(crate) struct SsrTemplate {
|
pub(crate) struct ParsedRule {
|
||||||
pub(crate) tokens: Vec<PatternElement>,
|
pub(crate) placeholders_by_stand_in: FxHashMap<SmolStr, Placeholder>,
|
||||||
|
pub(crate) pattern: SyntaxNode,
|
||||||
|
pub(crate) template: Option<SyntaxNode>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub(crate) struct RawSearchPattern {
|
pub(crate) struct RawPattern {
|
||||||
tokens: Vec<PatternElement>,
|
tokens: Vec<PatternElement>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -54,6 +56,60 @@ pub(crate) struct Token {
|
||||||
pub(crate) text: SmolStr,
|
pub(crate) text: SmolStr,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ParsedRule {
|
||||||
|
fn new(
|
||||||
|
pattern: &RawPattern,
|
||||||
|
template: Option<&RawPattern>,
|
||||||
|
) -> Result<Vec<ParsedRule>, SsrError> {
|
||||||
|
let raw_pattern = pattern.as_rust_code();
|
||||||
|
let raw_template = template.map(|t| t.as_rust_code());
|
||||||
|
let raw_template = raw_template.as_ref().map(|s| s.as_str());
|
||||||
|
let mut builder = RuleBuilder {
|
||||||
|
placeholders_by_stand_in: pattern.placeholders_by_stand_in(),
|
||||||
|
rules: Vec::new(),
|
||||||
|
};
|
||||||
|
builder.try_add(ast::Expr::parse(&raw_pattern), raw_template.map(ast::Expr::parse));
|
||||||
|
builder.try_add(ast::TypeRef::parse(&raw_pattern), raw_template.map(ast::TypeRef::parse));
|
||||||
|
builder.try_add(
|
||||||
|
ast::ModuleItem::parse(&raw_pattern),
|
||||||
|
raw_template.map(ast::ModuleItem::parse),
|
||||||
|
);
|
||||||
|
builder.try_add(ast::Path::parse(&raw_pattern), raw_template.map(ast::Path::parse));
|
||||||
|
builder.try_add(ast::Pat::parse(&raw_pattern), raw_template.map(ast::Pat::parse));
|
||||||
|
builder.build()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct RuleBuilder {
|
||||||
|
placeholders_by_stand_in: FxHashMap<SmolStr, Placeholder>,
|
||||||
|
rules: Vec<ParsedRule>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RuleBuilder {
|
||||||
|
fn try_add<T: AstNode>(&mut self, pattern: Result<T, ()>, template: Option<Result<T, ()>>) {
|
||||||
|
match (pattern, template) {
|
||||||
|
(Ok(pattern), Some(Ok(template))) => self.rules.push(ParsedRule {
|
||||||
|
placeholders_by_stand_in: self.placeholders_by_stand_in.clone(),
|
||||||
|
pattern: pattern.syntax().clone(),
|
||||||
|
template: Some(template.syntax().clone()),
|
||||||
|
}),
|
||||||
|
(Ok(pattern), None) => self.rules.push(ParsedRule {
|
||||||
|
placeholders_by_stand_in: self.placeholders_by_stand_in.clone(),
|
||||||
|
pattern: pattern.syntax().clone(),
|
||||||
|
template: None,
|
||||||
|
}),
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn build(self) -> Result<Vec<ParsedRule>, SsrError> {
|
||||||
|
if self.rules.is_empty() {
|
||||||
|
bail!("Not a valid Rust expression, type, item, path or pattern");
|
||||||
|
}
|
||||||
|
Ok(self.rules)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl FromStr for SsrRule {
|
impl FromStr for SsrRule {
|
||||||
type Err = SsrError;
|
type Err = SsrError;
|
||||||
|
|
||||||
|
@ -68,21 +124,24 @@ impl FromStr for SsrRule {
|
||||||
if it.next().is_some() {
|
if it.next().is_some() {
|
||||||
return Err(SsrError("More than one delimiter found".into()));
|
return Err(SsrError("More than one delimiter found".into()));
|
||||||
}
|
}
|
||||||
let rule = SsrRule { pattern: pattern.parse()?, template: template.parse()? };
|
let raw_pattern = pattern.parse()?;
|
||||||
|
let raw_template = template.parse()?;
|
||||||
|
let parsed_rules = ParsedRule::new(&raw_pattern, Some(&raw_template))?;
|
||||||
|
let rule = SsrRule { pattern: raw_pattern, template: raw_template, parsed_rules };
|
||||||
validate_rule(&rule)?;
|
validate_rule(&rule)?;
|
||||||
Ok(rule)
|
Ok(rule)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl FromStr for RawSearchPattern {
|
impl FromStr for RawPattern {
|
||||||
type Err = SsrError;
|
type Err = SsrError;
|
||||||
|
|
||||||
fn from_str(pattern_str: &str) -> Result<RawSearchPattern, SsrError> {
|
fn from_str(pattern_str: &str) -> Result<RawPattern, SsrError> {
|
||||||
Ok(RawSearchPattern { tokens: parse_pattern(pattern_str)? })
|
Ok(RawPattern { tokens: parse_pattern(pattern_str)? })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl RawSearchPattern {
|
impl RawPattern {
|
||||||
/// Returns this search pattern as Rust source code that we can feed to the Rust parser.
|
/// Returns this search pattern as Rust source code that we can feed to the Rust parser.
|
||||||
fn as_rust_code(&self) -> String {
|
fn as_rust_code(&self) -> String {
|
||||||
let mut res = String::new();
|
let mut res = String::new();
|
||||||
|
@ -95,7 +154,7 @@ impl RawSearchPattern {
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
|
|
||||||
fn placeholders_by_stand_in(&self) -> FxHashMap<SmolStr, Placeholder> {
|
pub(crate) fn placeholders_by_stand_in(&self) -> FxHashMap<SmolStr, Placeholder> {
|
||||||
let mut res = FxHashMap::default();
|
let mut res = FxHashMap::default();
|
||||||
for t in &self.tokens {
|
for t in &self.tokens {
|
||||||
if let PatternElement::Placeholder(placeholder) = t {
|
if let PatternElement::Placeholder(placeholder) = t {
|
||||||
|
@ -110,41 +169,9 @@ impl FromStr for SsrPattern {
|
||||||
type Err = SsrError;
|
type Err = SsrError;
|
||||||
|
|
||||||
fn from_str(pattern_str: &str) -> Result<SsrPattern, SsrError> {
|
fn from_str(pattern_str: &str) -> Result<SsrPattern, SsrError> {
|
||||||
let raw: RawSearchPattern = pattern_str.parse()?;
|
let raw_pattern = pattern_str.parse()?;
|
||||||
let raw_str = raw.as_rust_code();
|
let parsed_rules = ParsedRule::new(&raw_pattern, None)?;
|
||||||
let res = SsrPattern {
|
Ok(SsrPattern { raw: raw_pattern, parsed_rules })
|
||||||
expr: ast::Expr::parse(&raw_str).ok().map(|n| n.syntax().clone()),
|
|
||||||
type_ref: ast::TypeRef::parse(&raw_str).ok().map(|n| n.syntax().clone()),
|
|
||||||
item: ast::ModuleItem::parse(&raw_str).ok().map(|n| n.syntax().clone()),
|
|
||||||
path: ast::Path::parse(&raw_str).ok().map(|n| n.syntax().clone()),
|
|
||||||
pattern: ast::Pat::parse(&raw_str).ok().map(|n| n.syntax().clone()),
|
|
||||||
placeholders_by_stand_in: raw.placeholders_by_stand_in(),
|
|
||||||
raw,
|
|
||||||
};
|
|
||||||
if res.expr.is_none()
|
|
||||||
&& res.type_ref.is_none()
|
|
||||||
&& res.item.is_none()
|
|
||||||
&& res.path.is_none()
|
|
||||||
&& res.pattern.is_none()
|
|
||||||
{
|
|
||||||
bail!("Pattern is not a valid Rust expression, type, item, path or pattern");
|
|
||||||
}
|
|
||||||
Ok(res)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl FromStr for SsrTemplate {
|
|
||||||
type Err = SsrError;
|
|
||||||
|
|
||||||
fn from_str(pattern_str: &str) -> Result<SsrTemplate, SsrError> {
|
|
||||||
let tokens = parse_pattern(pattern_str)?;
|
|
||||||
// Validate that the template is a valid fragment of Rust code. We reuse the validation
|
|
||||||
// logic for search patterns since the only thing that differs is the error message.
|
|
||||||
if SsrPattern::from_str(pattern_str).is_err() {
|
|
||||||
bail!("Replacement is not a valid Rust expression, type, item, path or pattern");
|
|
||||||
}
|
|
||||||
// Our actual template needs to preserve whitespace, so we can't reuse `tokens`.
|
|
||||||
Ok(SsrTemplate { tokens })
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -173,7 +200,7 @@ fn parse_pattern(pattern_str: &str) -> Result<Vec<PatternElement>, SsrError> {
|
||||||
/// pattern didn't define.
|
/// pattern didn't define.
|
||||||
fn validate_rule(rule: &SsrRule) -> Result<(), SsrError> {
|
fn validate_rule(rule: &SsrRule) -> Result<(), SsrError> {
|
||||||
let mut defined_placeholders = FxHashSet::default();
|
let mut defined_placeholders = FxHashSet::default();
|
||||||
for p in &rule.pattern.raw.tokens {
|
for p in &rule.pattern.tokens {
|
||||||
if let PatternElement::Placeholder(placeholder) = p {
|
if let PatternElement::Placeholder(placeholder) = p {
|
||||||
defined_placeholders.insert(&placeholder.ident);
|
defined_placeholders.insert(&placeholder.ident);
|
||||||
}
|
}
|
||||||
|
@ -316,7 +343,7 @@ mod tests {
|
||||||
}
|
}
|
||||||
let result: SsrRule = "foo($a, $b) ==>> bar($b, $a)".parse().unwrap();
|
let result: SsrRule = "foo($a, $b) ==>> bar($b, $a)".parse().unwrap();
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
result.pattern.raw.tokens,
|
result.pattern.tokens,
|
||||||
vec![
|
vec![
|
||||||
token(SyntaxKind::IDENT, "foo"),
|
token(SyntaxKind::IDENT, "foo"),
|
||||||
token(T!['('], "("),
|
token(T!['('], "("),
|
||||||
|
|
|
@ -1,66 +1,124 @@
|
||||||
//! Code for applying replacement templates for matches that have previously been found.
|
//! Code for applying replacement templates for matches that have previously been found.
|
||||||
|
|
||||||
use crate::matching::Var;
|
use crate::matching::Var;
|
||||||
use crate::parsing::PatternElement;
|
use crate::{resolving::ResolvedRule, Match, SsrMatches};
|
||||||
use crate::{Match, SsrMatches};
|
use ra_syntax::ast::{self, AstToken};
|
||||||
use ra_syntax::ast::AstToken;
|
use ra_syntax::{SyntaxElement, SyntaxKind, SyntaxNode, SyntaxToken, TextSize};
|
||||||
use ra_syntax::TextSize;
|
|
||||||
use ra_text_edit::TextEdit;
|
use ra_text_edit::TextEdit;
|
||||||
|
|
||||||
/// Returns a text edit that will replace each match in `matches` with its corresponding replacement
|
/// Returns a text edit that will replace each match in `matches` with its corresponding replacement
|
||||||
/// template. Placeholders in the template will have been substituted with whatever they matched to
|
/// template. Placeholders in the template will have been substituted with whatever they matched to
|
||||||
/// in the original code.
|
/// in the original code.
|
||||||
pub(crate) fn matches_to_edit(matches: &SsrMatches, file_src: &str) -> TextEdit {
|
pub(crate) fn matches_to_edit(
|
||||||
matches_to_edit_at_offset(matches, file_src, 0.into())
|
matches: &SsrMatches,
|
||||||
|
file_src: &str,
|
||||||
|
rules: &[ResolvedRule],
|
||||||
|
) -> TextEdit {
|
||||||
|
matches_to_edit_at_offset(matches, file_src, 0.into(), rules)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn matches_to_edit_at_offset(
|
fn matches_to_edit_at_offset(
|
||||||
matches: &SsrMatches,
|
matches: &SsrMatches,
|
||||||
file_src: &str,
|
file_src: &str,
|
||||||
relative_start: TextSize,
|
relative_start: TextSize,
|
||||||
|
rules: &[ResolvedRule],
|
||||||
) -> TextEdit {
|
) -> TextEdit {
|
||||||
let mut edit_builder = ra_text_edit::TextEditBuilder::default();
|
let mut edit_builder = ra_text_edit::TextEditBuilder::default();
|
||||||
for m in &matches.matches {
|
for m in &matches.matches {
|
||||||
edit_builder.replace(
|
edit_builder.replace(
|
||||||
m.range.range.checked_sub(relative_start).unwrap(),
|
m.range.range.checked_sub(relative_start).unwrap(),
|
||||||
render_replace(m, file_src),
|
render_replace(m, file_src, rules),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
edit_builder.finish()
|
edit_builder.finish()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn render_replace(match_info: &Match, file_src: &str) -> String {
|
struct ReplacementRenderer<'a> {
|
||||||
|
match_info: &'a Match,
|
||||||
|
file_src: &'a str,
|
||||||
|
rules: &'a [ResolvedRule],
|
||||||
|
rule: &'a ResolvedRule,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_replace(match_info: &Match, file_src: &str, rules: &[ResolvedRule]) -> String {
|
||||||
let mut out = String::new();
|
let mut out = String::new();
|
||||||
for r in &match_info.template.tokens {
|
let rule = &rules[match_info.rule_index];
|
||||||
match r {
|
let template = rule
|
||||||
PatternElement::Token(t) => out.push_str(t.text.as_str()),
|
.template
|
||||||
PatternElement::Placeholder(p) => {
|
.as_ref()
|
||||||
if let Some(placeholder_value) =
|
.expect("You called MatchFinder::edits after calling MatchFinder::add_search_pattern");
|
||||||
match_info.placeholder_values.get(&Var(p.ident.to_string()))
|
let renderer = ReplacementRenderer { match_info, file_src, rules, rule };
|
||||||
{
|
renderer.render_node(&template.node, &mut out);
|
||||||
let range = &placeholder_value.range.range;
|
|
||||||
let mut matched_text =
|
|
||||||
file_src[usize::from(range.start())..usize::from(range.end())].to_owned();
|
|
||||||
let edit = matches_to_edit_at_offset(
|
|
||||||
&placeholder_value.inner_matches,
|
|
||||||
file_src,
|
|
||||||
range.start(),
|
|
||||||
);
|
|
||||||
edit.apply(&mut matched_text);
|
|
||||||
out.push_str(&matched_text);
|
|
||||||
} else {
|
|
||||||
// We validated that all placeholder references were valid before we
|
|
||||||
// started, so this shouldn't happen.
|
|
||||||
panic!(
|
|
||||||
"Internal error: replacement referenced unknown placeholder {}",
|
|
||||||
p.ident
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for comment in &match_info.ignored_comments {
|
for comment in &match_info.ignored_comments {
|
||||||
out.push_str(&comment.syntax().to_string());
|
out.push_str(&comment.syntax().to_string());
|
||||||
}
|
}
|
||||||
out
|
out
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ReplacementRenderer<'_> {
|
||||||
|
fn render_node_children(&self, node: &SyntaxNode, out: &mut String) {
|
||||||
|
for node_or_token in node.children_with_tokens() {
|
||||||
|
self.render_node_or_token(&node_or_token, out);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_node_or_token(&self, node_or_token: &SyntaxElement, out: &mut String) {
|
||||||
|
match node_or_token {
|
||||||
|
SyntaxElement::Token(token) => {
|
||||||
|
self.render_token(&token, out);
|
||||||
|
}
|
||||||
|
SyntaxElement::Node(child_node) => {
|
||||||
|
self.render_node(&child_node, out);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_node(&self, node: &SyntaxNode, out: &mut String) {
|
||||||
|
use ra_syntax::ast::AstNode;
|
||||||
|
if let Some(mod_path) = self.match_info.rendered_template_paths.get(&node) {
|
||||||
|
out.push_str(&mod_path.to_string());
|
||||||
|
// Emit everything except for the segment's name-ref, since we already effectively
|
||||||
|
// emitted that as part of `mod_path`.
|
||||||
|
if let Some(path) = ast::Path::cast(node.clone()) {
|
||||||
|
if let Some(segment) = path.segment() {
|
||||||
|
for node_or_token in segment.syntax().children_with_tokens() {
|
||||||
|
if node_or_token.kind() != SyntaxKind::NAME_REF {
|
||||||
|
self.render_node_or_token(&node_or_token, out);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
self.render_node_children(&node, out);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_token(&self, token: &SyntaxToken, out: &mut String) {
|
||||||
|
if let Some(placeholder) = self.rule.get_placeholder(&token) {
|
||||||
|
if let Some(placeholder_value) =
|
||||||
|
self.match_info.placeholder_values.get(&Var(placeholder.ident.to_string()))
|
||||||
|
{
|
||||||
|
let range = &placeholder_value.range.range;
|
||||||
|
let mut matched_text =
|
||||||
|
self.file_src[usize::from(range.start())..usize::from(range.end())].to_owned();
|
||||||
|
let edit = matches_to_edit_at_offset(
|
||||||
|
&placeholder_value.inner_matches,
|
||||||
|
self.file_src,
|
||||||
|
range.start(),
|
||||||
|
self.rules,
|
||||||
|
);
|
||||||
|
edit.apply(&mut matched_text);
|
||||||
|
out.push_str(&matched_text);
|
||||||
|
} else {
|
||||||
|
// We validated that all placeholder references were valid before we
|
||||||
|
// started, so this shouldn't happen.
|
||||||
|
panic!(
|
||||||
|
"Internal error: replacement referenced unknown placeholder {}",
|
||||||
|
placeholder.ident
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
out.push_str(token.text().as_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
173
crates/ra_ssr/src/resolving.rs
Normal file
173
crates/ra_ssr/src/resolving.rs
Normal file
|
@ -0,0 +1,173 @@
|
||||||
|
//! This module is responsible for resolving paths within rules.
|
||||||
|
|
||||||
|
use crate::errors::error;
|
||||||
|
use crate::{parsing, SsrError};
|
||||||
|
use parsing::Placeholder;
|
||||||
|
use ra_syntax::{ast, SmolStr, SyntaxKind, SyntaxNode, SyntaxToken};
|
||||||
|
use rustc_hash::{FxHashMap, FxHashSet};
|
||||||
|
use test_utils::mark;
|
||||||
|
|
||||||
|
pub(crate) struct ResolvedRule {
|
||||||
|
pub(crate) pattern: ResolvedPattern,
|
||||||
|
pub(crate) template: Option<ResolvedPattern>,
|
||||||
|
pub(crate) index: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) struct ResolvedPattern {
|
||||||
|
pub(crate) placeholders_by_stand_in: FxHashMap<SmolStr, parsing::Placeholder>,
|
||||||
|
pub(crate) node: SyntaxNode,
|
||||||
|
// Paths in `node` that we've resolved.
|
||||||
|
pub(crate) resolved_paths: FxHashMap<SyntaxNode, ResolvedPath>,
|
||||||
|
pub(crate) ufcs_function_calls: FxHashMap<SyntaxNode, hir::Function>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) struct ResolvedPath {
|
||||||
|
pub(crate) resolution: hir::PathResolution,
|
||||||
|
/// The depth of the ast::Path that was resolved within the pattern.
|
||||||
|
pub(crate) depth: u32,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ResolvedRule {
|
||||||
|
pub(crate) fn new(
|
||||||
|
rule: parsing::ParsedRule,
|
||||||
|
scope: &hir::SemanticsScope,
|
||||||
|
hygiene: &hir::Hygiene,
|
||||||
|
index: usize,
|
||||||
|
) -> Result<ResolvedRule, SsrError> {
|
||||||
|
let resolver =
|
||||||
|
Resolver { scope, hygiene, placeholders_by_stand_in: rule.placeholders_by_stand_in };
|
||||||
|
let resolved_template = if let Some(template) = rule.template {
|
||||||
|
Some(resolver.resolve_pattern_tree(template)?)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
Ok(ResolvedRule {
|
||||||
|
pattern: resolver.resolve_pattern_tree(rule.pattern)?,
|
||||||
|
template: resolved_template,
|
||||||
|
index,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn get_placeholder(&self, token: &SyntaxToken) -> Option<&Placeholder> {
|
||||||
|
if token.kind() != SyntaxKind::IDENT {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
self.pattern.placeholders_by_stand_in.get(token.text())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Resolver<'a, 'db> {
|
||||||
|
scope: &'a hir::SemanticsScope<'db>,
|
||||||
|
hygiene: &'a hir::Hygiene,
|
||||||
|
placeholders_by_stand_in: FxHashMap<SmolStr, parsing::Placeholder>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Resolver<'_, '_> {
|
||||||
|
fn resolve_pattern_tree(&self, pattern: SyntaxNode) -> Result<ResolvedPattern, SsrError> {
|
||||||
|
let mut resolved_paths = FxHashMap::default();
|
||||||
|
self.resolve(pattern.clone(), 0, &mut resolved_paths)?;
|
||||||
|
let ufcs_function_calls = resolved_paths
|
||||||
|
.iter()
|
||||||
|
.filter_map(|(path_node, resolved)| {
|
||||||
|
if let Some(grandparent) = path_node.parent().and_then(|parent| parent.parent()) {
|
||||||
|
if grandparent.kind() == SyntaxKind::CALL_EXPR {
|
||||||
|
if let hir::PathResolution::AssocItem(hir::AssocItem::Function(function)) =
|
||||||
|
&resolved.resolution
|
||||||
|
{
|
||||||
|
return Some((grandparent, *function));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
Ok(ResolvedPattern {
|
||||||
|
node: pattern,
|
||||||
|
resolved_paths,
|
||||||
|
placeholders_by_stand_in: self.placeholders_by_stand_in.clone(),
|
||||||
|
ufcs_function_calls,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn resolve(
|
||||||
|
&self,
|
||||||
|
node: SyntaxNode,
|
||||||
|
depth: u32,
|
||||||
|
resolved_paths: &mut FxHashMap<SyntaxNode, ResolvedPath>,
|
||||||
|
) -> Result<(), SsrError> {
|
||||||
|
use ra_syntax::ast::AstNode;
|
||||||
|
if let Some(path) = ast::Path::cast(node.clone()) {
|
||||||
|
// Check if this is an appropriate place in the path to resolve. If the path is
|
||||||
|
// something like `a::B::<i32>::c` then we want to resolve `a::B`. If the path contains
|
||||||
|
// a placeholder. e.g. `a::$b::c` then we want to resolve `a`.
|
||||||
|
if !path_contains_type_arguments(path.qualifier())
|
||||||
|
&& !self.path_contains_placeholder(&path)
|
||||||
|
{
|
||||||
|
let resolution = self
|
||||||
|
.resolve_path(&path)
|
||||||
|
.ok_or_else(|| error!("Failed to resolve path `{}`", node.text()))?;
|
||||||
|
resolved_paths.insert(node, ResolvedPath { resolution, depth });
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for node in node.children() {
|
||||||
|
self.resolve(node, depth + 1, resolved_paths)?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns whether `path` contains a placeholder, but ignores any placeholders within type
|
||||||
|
/// arguments.
|
||||||
|
fn path_contains_placeholder(&self, path: &ast::Path) -> bool {
|
||||||
|
if let Some(segment) = path.segment() {
|
||||||
|
if let Some(name_ref) = segment.name_ref() {
|
||||||
|
if self.placeholders_by_stand_in.contains_key(name_ref.text()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if let Some(qualifier) = path.qualifier() {
|
||||||
|
return self.path_contains_placeholder(&qualifier);
|
||||||
|
}
|
||||||
|
false
|
||||||
|
}
|
||||||
|
|
||||||
|
fn resolve_path(&self, path: &ast::Path) -> Option<hir::PathResolution> {
|
||||||
|
let hir_path = hir::Path::from_src(path.clone(), self.hygiene)?;
|
||||||
|
// First try resolving the whole path. This will work for things like
|
||||||
|
// `std::collections::HashMap`, but will fail for things like
|
||||||
|
// `std::collections::HashMap::new`.
|
||||||
|
if let Some(resolution) = self.scope.resolve_hir_path(&hir_path) {
|
||||||
|
return Some(resolution);
|
||||||
|
}
|
||||||
|
// Resolution failed, try resolving the qualifier (e.g. `std::collections::HashMap` and if
|
||||||
|
// that succeeds, then iterate through the candidates on the resolved type with the provided
|
||||||
|
// name.
|
||||||
|
let resolved_qualifier = self.scope.resolve_hir_path_qualifier(&hir_path.qualifier()?)?;
|
||||||
|
if let hir::PathResolution::Def(hir::ModuleDef::Adt(adt)) = resolved_qualifier {
|
||||||
|
adt.ty(self.scope.db).iterate_path_candidates(
|
||||||
|
self.scope.db,
|
||||||
|
self.scope.module()?.krate(),
|
||||||
|
&FxHashSet::default(),
|
||||||
|
Some(hir_path.segments().last()?.name),
|
||||||
|
|_ty, assoc_item| Some(hir::PathResolution::AssocItem(assoc_item)),
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns whether `path` or any of its qualifiers contains type arguments.
|
||||||
|
fn path_contains_type_arguments(path: Option<ast::Path>) -> bool {
|
||||||
|
if let Some(path) = path {
|
||||||
|
if let Some(segment) = path.segment() {
|
||||||
|
if segment.type_arg_list().is_some() {
|
||||||
|
mark::hit!(type_arguments_within_path);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return path_contains_type_arguments(path.qualifier());
|
||||||
|
}
|
||||||
|
false
|
||||||
|
}
|
232
crates/ra_ssr/src/search.rs
Normal file
232
crates/ra_ssr/src/search.rs
Normal file
|
@ -0,0 +1,232 @@
|
||||||
|
//! Searching for matches.
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
matching,
|
||||||
|
resolving::{ResolvedPath, ResolvedPattern, ResolvedRule},
|
||||||
|
Match, MatchFinder,
|
||||||
|
};
|
||||||
|
use ra_db::FileRange;
|
||||||
|
use ra_ide_db::{
|
||||||
|
defs::Definition,
|
||||||
|
search::{Reference, SearchScope},
|
||||||
|
};
|
||||||
|
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
|
||||||
|
/// same path. e.g. if the pattern was `foo::Bar` that can parse as a path, an expression, a type
|
||||||
|
/// and as a pattern. In each, the usages of `foo::Bar` are the same and we'd like to avoid finding
|
||||||
|
/// them more than once.
|
||||||
|
#[derive(Default)]
|
||||||
|
pub(crate) struct UsageCache {
|
||||||
|
usages: Vec<(Definition, Vec<Reference>)>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'db> MatchFinder<'db> {
|
||||||
|
/// Adds all matches for `rule` to `matches_out`. Matches may overlap in ways that make
|
||||||
|
/// replacement impossible, so further processing is required in order to properly nest matches
|
||||||
|
/// and remove overlapping matches. This is done in the `nesting` module.
|
||||||
|
pub(crate) fn find_matches_for_rule(
|
||||||
|
&self,
|
||||||
|
rule: &ResolvedRule,
|
||||||
|
usage_cache: &mut UsageCache,
|
||||||
|
matches_out: &mut Vec<Match>,
|
||||||
|
) {
|
||||||
|
if pick_path_for_usages(&rule.pattern).is_none() {
|
||||||
|
self.slow_scan(rule, matches_out);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.find_matches_for_pattern_tree(rule, &rule.pattern, usage_cache, matches_out);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn find_matches_for_pattern_tree(
|
||||||
|
&self,
|
||||||
|
rule: &ResolvedRule,
|
||||||
|
pattern: &ResolvedPattern,
|
||||||
|
usage_cache: &mut UsageCache,
|
||||||
|
matches_out: &mut Vec<Match>,
|
||||||
|
) {
|
||||||
|
if let Some(resolved_path) = pick_path_for_usages(pattern) {
|
||||||
|
let definition: Definition = resolved_path.resolution.clone().into();
|
||||||
|
for reference in self.find_usages(usage_cache, definition) {
|
||||||
|
if let Some(node_to_match) = self.find_node_to_match(resolved_path, reference) {
|
||||||
|
if !is_search_permitted_ancestors(&node_to_match) {
|
||||||
|
mark::hit!(use_declaration_with_braces);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if let Ok(m) =
|
||||||
|
matching::get_match(false, rule, &node_to_match, &None, &self.sema)
|
||||||
|
{
|
||||||
|
matches_out.push(m);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn find_node_to_match(
|
||||||
|
&self,
|
||||||
|
resolved_path: &ResolvedPath,
|
||||||
|
reference: &Reference,
|
||||||
|
) -> Option<SyntaxNode> {
|
||||||
|
let file = self.sema.parse(reference.file_range.file_id);
|
||||||
|
let depth = resolved_path.depth as usize;
|
||||||
|
let offset = reference.file_range.range.start();
|
||||||
|
if let Some(path) =
|
||||||
|
self.sema.find_node_at_offset_with_descend::<ast::Path>(file.syntax(), offset)
|
||||||
|
{
|
||||||
|
self.sema.ancestors_with_macros(path.syntax().clone()).skip(depth).next()
|
||||||
|
} else if let Some(path) =
|
||||||
|
self.sema.find_node_at_offset_with_descend::<ast::MethodCallExpr>(file.syntax(), offset)
|
||||||
|
{
|
||||||
|
// If the pattern contained a path and we found a reference to that path that wasn't
|
||||||
|
// itself a path, but was a method call, then we need to adjust how far up to try
|
||||||
|
// matching by how deep the path was within a CallExpr. The structure would have been
|
||||||
|
// CallExpr, PathExpr, Path - i.e. a depth offset of 2. We don't need to check if the
|
||||||
|
// path was part of a CallExpr because if it wasn't then all that will happen is we'll
|
||||||
|
// fail to match, which is the desired behavior.
|
||||||
|
const PATH_DEPTH_IN_CALL_EXPR: usize = 2;
|
||||||
|
if depth < PATH_DEPTH_IN_CALL_EXPR {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
self.sema
|
||||||
|
.ancestors_with_macros(path.syntax().clone())
|
||||||
|
.skip(depth - PATH_DEPTH_IN_CALL_EXPR)
|
||||||
|
.next()
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn find_usages<'a>(
|
||||||
|
&self,
|
||||||
|
usage_cache: &'a mut UsageCache,
|
||||||
|
definition: Definition,
|
||||||
|
) -> &'a [Reference] {
|
||||||
|
// Logically if a lookup succeeds we should just return it. Unfortunately returning it would
|
||||||
|
// extend the lifetime of the borrow, then we wouldn't be able to do the insertion on a
|
||||||
|
// cache miss. This is a limitation of NLL and is fixed with Polonius. For now we do two
|
||||||
|
// lookups in the case of a cache hit.
|
||||||
|
if usage_cache.find(&definition).is_none() {
|
||||||
|
let usages = definition.find_usages(&self.sema, Some(self.search_scope()));
|
||||||
|
usage_cache.usages.push((definition, usages));
|
||||||
|
return &usage_cache.usages.last().unwrap().1;
|
||||||
|
}
|
||||||
|
usage_cache.find(&definition).unwrap()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the scope within which we want to search. We don't want un unrestricted search
|
||||||
|
/// scope, since we don't want to find references in external dependencies.
|
||||||
|
fn search_scope(&self) -> SearchScope {
|
||||||
|
// FIXME: We should ideally have a test that checks that we edit local roots and not library
|
||||||
|
// roots. This probably would require some changes to fixtures, since currently everything
|
||||||
|
// seems to get put into a single source root.
|
||||||
|
use ra_db::SourceDatabaseExt;
|
||||||
|
use ra_ide_db::symbol_index::SymbolsDatabase;
|
||||||
|
let mut files = Vec::new();
|
||||||
|
for &root in self.sema.db.local_roots().iter() {
|
||||||
|
let sr = self.sema.db.source_root(root);
|
||||||
|
files.extend(sr.iter());
|
||||||
|
}
|
||||||
|
SearchScope::files(&files)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn slow_scan(&self, rule: &ResolvedRule, matches_out: &mut Vec<Match>) {
|
||||||
|
use ra_db::SourceDatabaseExt;
|
||||||
|
use ra_ide_db::symbol_index::SymbolsDatabase;
|
||||||
|
for &root in self.sema.db.local_roots().iter() {
|
||||||
|
let sr = self.sema.db.source_root(root);
|
||||||
|
for file_id in sr.iter() {
|
||||||
|
let file = self.sema.parse(file_id);
|
||||||
|
let code = file.syntax();
|
||||||
|
self.slow_scan_node(code, rule, &None, matches_out);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn slow_scan_node(
|
||||||
|
&self,
|
||||||
|
code: &SyntaxNode,
|
||||||
|
rule: &ResolvedRule,
|
||||||
|
restrict_range: &Option<FileRange>,
|
||||||
|
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) {
|
||||||
|
matches_out.push(m);
|
||||||
|
}
|
||||||
|
// If we've got a macro call, we already tried matching it pre-expansion, which is the only
|
||||||
|
// way to match the whole macro, now try expanding it and matching the expansion.
|
||||||
|
if let Some(macro_call) = ast::MacroCall::cast(code.clone()) {
|
||||||
|
if let Some(expanded) = self.sema.expand(¯o_call) {
|
||||||
|
if let Some(tt) = macro_call.token_tree() {
|
||||||
|
// When matching within a macro expansion, we only want to allow matches of
|
||||||
|
// nodes that originated entirely from within the token tree of the macro call.
|
||||||
|
// i.e. we don't want to match something that came from the macro itself.
|
||||||
|
self.slow_scan_node(
|
||||||
|
&expanded,
|
||||||
|
rule,
|
||||||
|
&Some(self.sema.original_range(tt.syntax())),
|
||||||
|
matches_out,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for child in code.children() {
|
||||||
|
self.slow_scan_node(&child, rule, restrict_range, matches_out);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 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 {
|
||||||
|
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
|
||||||
|
// fast enough and avoids the need to implement Hash for Definition.
|
||||||
|
for (d, refs) in &self.usages {
|
||||||
|
if d == definition {
|
||||||
|
return Some(refs);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a path that's suitable for path resolution. We exclude builtin types, since they aren't
|
||||||
|
/// something that we can find references to. We then somewhat arbitrarily pick the path that is the
|
||||||
|
/// longest as this is hopefully more likely to be less common, making it faster to find.
|
||||||
|
fn pick_path_for_usages(pattern: &ResolvedPattern) -> Option<&ResolvedPath> {
|
||||||
|
// FIXME: Take the scope of the resolved path into account. e.g. if there are any paths that are
|
||||||
|
// private to the current module, then we definitely would want to pick them over say a path
|
||||||
|
// from std. Possibly we should go further than this and intersect the search scopes for all
|
||||||
|
// resolved paths then search only in that scope.
|
||||||
|
pattern
|
||||||
|
.resolved_paths
|
||||||
|
.iter()
|
||||||
|
.filter(|(_, p)| {
|
||||||
|
!matches!(p.resolution, hir::PathResolution::Def(hir::ModuleDef::BuiltinType(_)))
|
||||||
|
})
|
||||||
|
.map(|(node, resolved)| (node.text().len(), resolved))
|
||||||
|
.max_by(|(a, _), (b, _)| a.cmp(b))
|
||||||
|
.map(|(_, resolved)| resolved)
|
||||||
|
}
|
|
@ -1,5 +1,8 @@
|
||||||
use crate::{MatchFinder, SsrRule};
|
use crate::{MatchFinder, SsrRule};
|
||||||
use ra_db::{FileId, SourceDatabaseExt};
|
use expect::{expect, Expect};
|
||||||
|
use ra_db::{salsa::Durability, FileId, FilePosition, SourceDatabaseExt};
|
||||||
|
use rustc_hash::FxHashSet;
|
||||||
|
use std::sync::Arc;
|
||||||
use test_utils::mark;
|
use test_utils::mark;
|
||||||
|
|
||||||
fn parse_error_text(query: &str) -> String {
|
fn parse_error_text(query: &str) -> String {
|
||||||
|
@ -36,7 +39,7 @@ fn parser_repeated_name() {
|
||||||
fn parser_invalid_pattern() {
|
fn parser_invalid_pattern() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
parse_error_text(" ==>> ()"),
|
parse_error_text(" ==>> ()"),
|
||||||
"Parse error: Pattern is not a valid Rust expression, type, item, path or pattern"
|
"Parse error: Not a valid Rust expression, type, item, path or pattern"
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,7 +47,7 @@ fn parser_invalid_pattern() {
|
||||||
fn parser_invalid_template() {
|
fn parser_invalid_template() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
parse_error_text("() ==>> )"),
|
parse_error_text("() ==>> )"),
|
||||||
"Parse error: Replacement is not a valid Rust expression, type, item, path or pattern"
|
"Parse error: Not a valid Rust expression, type, item, path or pattern"
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -56,39 +59,44 @@ fn parser_undefined_placeholder_in_replacement() {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn single_file(code: &str) -> (ra_ide_db::RootDatabase, FileId) {
|
/// `code` may optionally contain a cursor marker `<|>`. If it doesn't, then the position will be
|
||||||
|
/// the start of the file.
|
||||||
|
pub(crate) fn single_file(code: &str) -> (ra_ide_db::RootDatabase, FilePosition) {
|
||||||
use ra_db::fixture::WithFixture;
|
use ra_db::fixture::WithFixture;
|
||||||
ra_ide_db::RootDatabase::with_single_file(code)
|
use ra_ide_db::symbol_index::SymbolsDatabase;
|
||||||
|
let (mut db, position) = if code.contains(test_utils::CURSOR_MARKER) {
|
||||||
|
ra_ide_db::RootDatabase::with_position(code)
|
||||||
|
} else {
|
||||||
|
let (db, file_id) = ra_ide_db::RootDatabase::with_single_file(code);
|
||||||
|
(db, FilePosition { file_id, offset: 0.into() })
|
||||||
|
};
|
||||||
|
let mut local_roots = FxHashSet::default();
|
||||||
|
local_roots.insert(ra_db::fixture::WORKSPACE);
|
||||||
|
db.set_local_roots_with_durability(Arc::new(local_roots), Durability::HIGH);
|
||||||
|
(db, position)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn assert_ssr_transform(rule: &str, input: &str, result: &str) {
|
fn assert_ssr_transform(rule: &str, input: &str, expected: Expect) {
|
||||||
assert_ssr_transforms(&[rule], input, result);
|
assert_ssr_transforms(&[rule], input, expected);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn normalize_code(code: &str) -> String {
|
fn assert_ssr_transforms(rules: &[&str], input: &str, expected: Expect) {
|
||||||
let (db, file_id) = single_file(code);
|
let (db, position) = single_file(input);
|
||||||
db.file_text(file_id).to_string()
|
let mut match_finder = MatchFinder::in_context(&db, position);
|
||||||
}
|
|
||||||
|
|
||||||
fn assert_ssr_transforms(rules: &[&str], input: &str, result: &str) {
|
|
||||||
let (db, file_id) = single_file(input);
|
|
||||||
let mut match_finder = MatchFinder::new(&db);
|
|
||||||
for rule in rules {
|
for rule in rules {
|
||||||
let rule: SsrRule = rule.parse().unwrap();
|
let rule: SsrRule = rule.parse().unwrap();
|
||||||
match_finder.add_rule(rule);
|
match_finder.add_rule(rule).unwrap();
|
||||||
}
|
}
|
||||||
if let Some(edits) = match_finder.edits_for_file(file_id) {
|
let edits = match_finder.edits();
|
||||||
// Note, db.file_text is not necessarily the same as `input`, since fixture parsing alters
|
if edits.is_empty() {
|
||||||
// stuff.
|
|
||||||
let mut after = db.file_text(file_id).to_string();
|
|
||||||
edits.apply(&mut after);
|
|
||||||
// Likewise, we need to make sure that whatever transformations fixture parsing applies,
|
|
||||||
// also get applied to our expected result.
|
|
||||||
let result = normalize_code(result);
|
|
||||||
assert_eq!(after, result);
|
|
||||||
} else {
|
|
||||||
panic!("No edits were made");
|
panic!("No edits were made");
|
||||||
}
|
}
|
||||||
|
assert_eq!(edits[0].file_id, position.file_id);
|
||||||
|
// Note, db.file_text is not necessarily the same as `input`, since fixture parsing alters
|
||||||
|
// stuff.
|
||||||
|
let mut actual = db.file_text(position.file_id).to_string();
|
||||||
|
edits[0].edit.apply(&mut actual);
|
||||||
|
expected.assert_eq(&actual);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn print_match_debug_info(match_finder: &MatchFinder, file_id: FileId, snippet: &str) {
|
fn print_match_debug_info(match_finder: &MatchFinder, file_id: FileId, snippet: &str) {
|
||||||
|
@ -104,39 +112,34 @@ fn print_match_debug_info(match_finder: &MatchFinder, file_id: FileId, snippet:
|
||||||
}
|
}
|
||||||
|
|
||||||
fn assert_matches(pattern: &str, code: &str, expected: &[&str]) {
|
fn assert_matches(pattern: &str, code: &str, expected: &[&str]) {
|
||||||
let (db, file_id) = single_file(code);
|
let (db, position) = single_file(code);
|
||||||
let mut match_finder = MatchFinder::new(&db);
|
let mut match_finder = MatchFinder::in_context(&db, position);
|
||||||
match_finder.add_search_pattern(pattern.parse().unwrap());
|
match_finder.add_search_pattern(pattern.parse().unwrap()).unwrap();
|
||||||
let matched_strings: Vec<String> = match_finder
|
let matched_strings: Vec<String> =
|
||||||
.find_matches_in_file(file_id)
|
match_finder.matches().flattened().matches.iter().map(|m| m.matched_text()).collect();
|
||||||
.flattened()
|
|
||||||
.matches
|
|
||||||
.iter()
|
|
||||||
.map(|m| m.matched_text())
|
|
||||||
.collect();
|
|
||||||
if matched_strings != expected && !expected.is_empty() {
|
if matched_strings != expected && !expected.is_empty() {
|
||||||
print_match_debug_info(&match_finder, file_id, &expected[0]);
|
print_match_debug_info(&match_finder, position.file_id, &expected[0]);
|
||||||
}
|
}
|
||||||
assert_eq!(matched_strings, expected);
|
assert_eq!(matched_strings, expected);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn assert_no_match(pattern: &str, code: &str) {
|
fn assert_no_match(pattern: &str, code: &str) {
|
||||||
let (db, file_id) = single_file(code);
|
let (db, position) = single_file(code);
|
||||||
let mut match_finder = MatchFinder::new(&db);
|
let mut match_finder = MatchFinder::in_context(&db, position);
|
||||||
match_finder.add_search_pattern(pattern.parse().unwrap());
|
match_finder.add_search_pattern(pattern.parse().unwrap()).unwrap();
|
||||||
let matches = match_finder.find_matches_in_file(file_id).flattened().matches;
|
let matches = match_finder.matches().flattened().matches;
|
||||||
if !matches.is_empty() {
|
if !matches.is_empty() {
|
||||||
print_match_debug_info(&match_finder, file_id, &matches[0].matched_text());
|
print_match_debug_info(&match_finder, position.file_id, &matches[0].matched_text());
|
||||||
panic!("Got {} matches when we expected none: {:#?}", matches.len(), matches);
|
panic!("Got {} matches when we expected none: {:#?}", matches.len(), matches);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn assert_match_failure_reason(pattern: &str, code: &str, snippet: &str, expected_reason: &str) {
|
fn assert_match_failure_reason(pattern: &str, code: &str, snippet: &str, expected_reason: &str) {
|
||||||
let (db, file_id) = single_file(code);
|
let (db, position) = single_file(code);
|
||||||
let mut match_finder = MatchFinder::new(&db);
|
let mut match_finder = MatchFinder::in_context(&db, position);
|
||||||
match_finder.add_search_pattern(pattern.parse().unwrap());
|
match_finder.add_search_pattern(pattern.parse().unwrap()).unwrap();
|
||||||
let mut reasons = Vec::new();
|
let mut reasons = Vec::new();
|
||||||
for d in match_finder.debug_where_text_equal(file_id, snippet) {
|
for d in match_finder.debug_where_text_equal(position.file_id, snippet) {
|
||||||
if let Some(reason) = d.match_failure_reason() {
|
if let Some(reason) = d.match_failure_reason() {
|
||||||
reasons.push(reason.to_owned());
|
reasons.push(reason.to_owned());
|
||||||
}
|
}
|
||||||
|
@ -149,7 +152,7 @@ fn ssr_function_to_method() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"my_function($a, $b) ==>> ($a).my_method($b)",
|
"my_function($a, $b) ==>> ($a).my_method($b)",
|
||||||
"fn my_function() {} fn main() { loop { my_function( other_func(x, y), z + w) } }",
|
"fn my_function() {} fn main() { loop { my_function( other_func(x, y), z + w) } }",
|
||||||
"fn my_function() {} fn main() { loop { (other_func(x, y)).my_method(z + w) } }",
|
expect![["fn my_function() {} fn main() { loop { (other_func(x, y)).my_method(z + w) } }"]],
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -157,8 +160,19 @@ fn ssr_function_to_method() {
|
||||||
fn ssr_nested_function() {
|
fn ssr_nested_function() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"foo($a, $b, $c) ==>> bar($c, baz($a, $b))",
|
"foo($a, $b, $c) ==>> bar($c, baz($a, $b))",
|
||||||
"fn foo() {} fn main { foo (x + value.method(b), x+y-z, true && false) }",
|
r#"
|
||||||
"fn foo() {} fn main { bar(true && false, baz(x + value.method(b), x+y-z)) }",
|
//- /lib.rs crate:foo
|
||||||
|
fn foo() {}
|
||||||
|
fn bar() {}
|
||||||
|
fn baz() {}
|
||||||
|
fn main { foo (x + value.method(b), x+y-z, true && false) }
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
fn foo() {}
|
||||||
|
fn bar() {}
|
||||||
|
fn baz() {}
|
||||||
|
fn main { bar(true && false, baz(x + value.method(b), x+y-z)) }
|
||||||
|
"#]],
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -167,7 +181,7 @@ fn ssr_expected_spacing() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"foo($x) + bar() ==>> bar($x)",
|
"foo($x) + bar() ==>> bar($x)",
|
||||||
"fn foo() {} fn bar() {} fn main() { foo(5) + bar() }",
|
"fn foo() {} fn bar() {} fn main() { foo(5) + bar() }",
|
||||||
"fn foo() {} fn bar() {} fn main() { bar(5) }",
|
expect![["fn foo() {} fn bar() {} fn main() { bar(5) }"]],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -176,7 +190,7 @@ fn ssr_with_extra_space() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"foo($x ) + bar() ==>> bar($x)",
|
"foo($x ) + bar() ==>> bar($x)",
|
||||||
"fn foo() {} fn bar() {} fn main() { foo( 5 ) +bar( ) }",
|
"fn foo() {} fn bar() {} fn main() { foo( 5 ) +bar( ) }",
|
||||||
"fn foo() {} fn bar() {} fn main() { bar(5) }",
|
expect![["fn foo() {} fn bar() {} fn main() { bar(5) }"]],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -184,8 +198,8 @@ fn ssr_with_extra_space() {
|
||||||
fn ssr_keeps_nested_comment() {
|
fn ssr_keeps_nested_comment() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"foo($x) ==>> bar($x)",
|
"foo($x) ==>> bar($x)",
|
||||||
"fn foo() {} fn main() { foo(other(5 /* using 5 */)) }",
|
"fn foo() {} fn bar() {} fn main() { foo(other(5 /* using 5 */)) }",
|
||||||
"fn foo() {} fn main() { bar(other(5 /* using 5 */)) }",
|
expect![["fn foo() {} fn bar() {} fn main() { bar(other(5 /* using 5 */)) }"]],
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -193,17 +207,25 @@ fn ssr_keeps_nested_comment() {
|
||||||
fn ssr_keeps_comment() {
|
fn ssr_keeps_comment() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"foo($x) ==>> bar($x)",
|
"foo($x) ==>> bar($x)",
|
||||||
"fn foo() {} fn main() { foo(5 /* using 5 */) }",
|
"fn foo() {} fn bar() {} fn main() { foo(5 /* using 5 */) }",
|
||||||
"fn foo() {} fn main() { bar(5)/* using 5 */ }",
|
expect![["fn foo() {} fn bar() {} fn main() { bar(5)/* using 5 */ }"]],
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn ssr_struct_lit() {
|
fn ssr_struct_lit() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"foo{a: $a, b: $b} ==>> foo::new($a, $b)",
|
"Foo{a: $a, b: $b} ==>> Foo::new($a, $b)",
|
||||||
"fn foo() {} fn main() { foo{b:2, a:1} }",
|
r#"
|
||||||
"fn foo() {} fn main() { foo::new(1, 2) }",
|
struct Foo() {}
|
||||||
|
impl Foo { fn new() {} }
|
||||||
|
fn main() { Foo{b:2, a:1} }
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
struct Foo() {}
|
||||||
|
impl Foo { fn new() {} }
|
||||||
|
fn main() { Foo::new(1, 2) }
|
||||||
|
"#]],
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -315,7 +337,7 @@ fn match_struct_instantiation() {
|
||||||
fn match_path() {
|
fn match_path() {
|
||||||
let code = r#"
|
let code = r#"
|
||||||
mod foo {
|
mod foo {
|
||||||
fn bar() {}
|
pub fn bar() {}
|
||||||
}
|
}
|
||||||
fn f() {foo::bar(42)}"#;
|
fn f() {foo::bar(42)}"#;
|
||||||
assert_matches("foo::bar", code, &["foo::bar"]);
|
assert_matches("foo::bar", code, &["foo::bar"]);
|
||||||
|
@ -328,6 +350,60 @@ fn match_pattern() {
|
||||||
assert_matches("Some($a)", "struct Some(); fn f() {if let Some(x) = foo() {}}", &["Some(x)"]);
|
assert_matches("Some($a)", "struct Some(); fn f() {if let Some(x) = foo() {}}", &["Some(x)"]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If our pattern has a full path, e.g. a::b::c() and the code has c(), but c resolves to
|
||||||
|
// a::b::c, then we should match.
|
||||||
|
#[test]
|
||||||
|
fn match_fully_qualified_fn_path() {
|
||||||
|
let code = r#"
|
||||||
|
mod a {
|
||||||
|
pub mod b {
|
||||||
|
pub fn c(_: i32) {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
use a::b::c;
|
||||||
|
fn f1() {
|
||||||
|
c(42);
|
||||||
|
}
|
||||||
|
"#;
|
||||||
|
assert_matches("a::b::c($a)", code, &["c(42)"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn match_resolved_type_name() {
|
||||||
|
let code = r#"
|
||||||
|
mod m1 {
|
||||||
|
pub mod m2 {
|
||||||
|
pub trait Foo<T> {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
mod m3 {
|
||||||
|
trait Foo<T> {}
|
||||||
|
fn f1(f: Option<&dyn Foo<bool>>) {}
|
||||||
|
}
|
||||||
|
mod m4 {
|
||||||
|
use crate::m1::m2::Foo;
|
||||||
|
fn f1(f: Option<&dyn Foo<i32>>) {}
|
||||||
|
}
|
||||||
|
"#;
|
||||||
|
assert_matches("m1::m2::Foo<$t>", code, &["Foo<i32>"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn type_arguments_within_path() {
|
||||||
|
mark::check!(type_arguments_within_path);
|
||||||
|
let code = r#"
|
||||||
|
mod foo {
|
||||||
|
pub struct Bar<T> {t: T}
|
||||||
|
impl<T> Bar<T> {
|
||||||
|
pub fn baz() {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn f1() {foo::Bar::<i32>::baz();}
|
||||||
|
"#;
|
||||||
|
assert_no_match("foo::Bar::<i64>::baz()", code);
|
||||||
|
assert_matches("foo::Bar::<i32>::baz()", code, &["foo::Bar::<i32>::baz()"]);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn literal_constraint() {
|
fn literal_constraint() {
|
||||||
mark::check!(literal_constraint);
|
mark::check!(literal_constraint);
|
||||||
|
@ -416,8 +492,8 @@ fn no_match_split_expression() {
|
||||||
fn replace_function_call() {
|
fn replace_function_call() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"foo() ==>> bar()",
|
"foo() ==>> bar()",
|
||||||
"fn foo() {} fn f1() {foo(); foo();}",
|
"fn foo() {} fn bar() {} fn f1() {foo(); foo();}",
|
||||||
"fn foo() {} fn f1() {bar(); bar();}",
|
expect![["fn foo() {} fn bar() {} fn f1() {bar(); bar();}"]],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -425,8 +501,8 @@ fn replace_function_call() {
|
||||||
fn replace_function_call_with_placeholders() {
|
fn replace_function_call_with_placeholders() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"foo($a, $b) ==>> bar($b, $a)",
|
"foo($a, $b) ==>> bar($b, $a)",
|
||||||
"fn foo() {} fn f1() {foo(5, 42)}",
|
"fn foo() {} fn bar() {} fn f1() {foo(5, 42)}",
|
||||||
"fn foo() {} fn f1() {bar(42, 5)}",
|
expect![["fn foo() {} fn bar() {} fn f1() {bar(42, 5)}"]],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -434,8 +510,109 @@ fn replace_function_call_with_placeholders() {
|
||||||
fn replace_nested_function_calls() {
|
fn replace_nested_function_calls() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"foo($a) ==>> bar($a)",
|
"foo($a) ==>> bar($a)",
|
||||||
"fn foo() {} fn f1() {foo(foo(42))}",
|
"fn foo() {} fn bar() {} fn f1() {foo(foo(42))}",
|
||||||
"fn foo() {} fn f1() {bar(bar(42))}",
|
expect![["fn foo() {} fn bar() {} fn f1() {bar(bar(42))}"]],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn replace_associated_function_call() {
|
||||||
|
assert_ssr_transform(
|
||||||
|
"Foo::new() ==>> Bar::new()",
|
||||||
|
r#"
|
||||||
|
struct Foo {}
|
||||||
|
impl Foo { fn new() {} }
|
||||||
|
struct Bar {}
|
||||||
|
impl Bar { fn new() {} }
|
||||||
|
fn f1() {Foo::new();}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
struct Foo {}
|
||||||
|
impl Foo { fn new() {} }
|
||||||
|
struct Bar {}
|
||||||
|
impl Bar { fn new() {} }
|
||||||
|
fn f1() {Bar::new();}
|
||||||
|
"#]],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn replace_path_in_different_contexts() {
|
||||||
|
// Note the <|> inside module a::b which marks the point where the rule is interpreted. We
|
||||||
|
// replace foo with bar, but both need different path qualifiers in different contexts. In f4,
|
||||||
|
// foo is unqualified because of a use statement, however the replacement needs to be fully
|
||||||
|
// qualified.
|
||||||
|
assert_ssr_transform(
|
||||||
|
"c::foo() ==>> c::bar()",
|
||||||
|
r#"
|
||||||
|
mod a {
|
||||||
|
pub mod b {<|>
|
||||||
|
pub mod c {
|
||||||
|
pub fn foo() {}
|
||||||
|
pub fn bar() {}
|
||||||
|
fn f1() { foo() }
|
||||||
|
}
|
||||||
|
fn f2() { c::foo() }
|
||||||
|
}
|
||||||
|
fn f3() { b::c::foo() }
|
||||||
|
}
|
||||||
|
use a::b::c::foo;
|
||||||
|
fn f4() { foo() }
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
mod a {
|
||||||
|
pub mod b {
|
||||||
|
pub mod c {
|
||||||
|
pub fn foo() {}
|
||||||
|
pub fn bar() {}
|
||||||
|
fn f1() { bar() }
|
||||||
|
}
|
||||||
|
fn f2() { c::bar() }
|
||||||
|
}
|
||||||
|
fn f3() { b::c::bar() }
|
||||||
|
}
|
||||||
|
use a::b::c::foo;
|
||||||
|
fn f4() { a::b::c::bar() }
|
||||||
|
"#]],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn replace_associated_function_with_generics() {
|
||||||
|
assert_ssr_transform(
|
||||||
|
"c::Foo::<$a>::new() ==>> d::Bar::<$a>::default()",
|
||||||
|
r#"
|
||||||
|
mod c {
|
||||||
|
pub struct Foo<T> {v: T}
|
||||||
|
impl<T> Foo<T> { pub fn new() {} }
|
||||||
|
fn f1() {
|
||||||
|
Foo::<i32>::new();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
mod d {
|
||||||
|
pub struct Bar<T> {v: T}
|
||||||
|
impl<T> Bar<T> { pub fn default() {} }
|
||||||
|
fn f1() {
|
||||||
|
super::c::Foo::<i32>::new();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
mod c {
|
||||||
|
pub struct Foo<T> {v: T}
|
||||||
|
impl<T> Foo<T> { pub fn new() {} }
|
||||||
|
fn f1() {
|
||||||
|
crate::d::Bar::<i32>::default();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
mod d {
|
||||||
|
pub struct Bar<T> {v: T}
|
||||||
|
impl<T> Bar<T> { pub fn default() {} }
|
||||||
|
fn f1() {
|
||||||
|
Bar::<i32>::default();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"#]],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -443,17 +620,10 @@ fn replace_nested_function_calls() {
|
||||||
fn replace_type() {
|
fn replace_type() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"Result<(), $a> ==>> Option<$a>",
|
"Result<(), $a> ==>> Option<$a>",
|
||||||
"struct Result<T, E> {} fn f1() -> Result<(), Vec<Error>> {foo()}",
|
"struct Result<T, E> {} struct Option<T> {} fn f1() -> Result<(), Vec<Error>> {foo()}",
|
||||||
"struct Result<T, E> {} fn f1() -> Option<Vec<Error>> {foo()}",
|
expect![[
|
||||||
);
|
"struct Result<T, E> {} struct Option<T> {} fn f1() -> Option<Vec<Error>> {foo()}"
|
||||||
}
|
]],
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn replace_struct_init() {
|
|
||||||
assert_ssr_transform(
|
|
||||||
"Foo {a: $a, b: $b} ==>> Foo::new($a, $b)",
|
|
||||||
"struct Foo {} fn f1() {Foo{b: 1, a: 2}}",
|
|
||||||
"struct Foo {} fn f1() {Foo::new(2, 1)}",
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -462,12 +632,12 @@ fn replace_macro_invocations() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"try!($a) ==>> $a?",
|
"try!($a) ==>> $a?",
|
||||||
"macro_rules! try {() => {}} fn f1() -> Result<(), E> {bar(try!(foo()));}",
|
"macro_rules! try {() => {}} fn f1() -> Result<(), E> {bar(try!(foo()));}",
|
||||||
"macro_rules! try {() => {}} fn f1() -> Result<(), E> {bar(foo()?);}",
|
expect![["macro_rules! try {() => {}} fn f1() -> Result<(), E> {bar(foo()?);}"]],
|
||||||
);
|
);
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"foo!($a($b)) ==>> foo($b, $a)",
|
"foo!($a($b)) ==>> foo($b, $a)",
|
||||||
"macro_rules! foo {() => {}} fn f1() {foo!(abc(def() + 2));}",
|
"macro_rules! foo {() => {}} fn f1() {foo!(abc(def() + 2));}",
|
||||||
"macro_rules! foo {() => {}} fn f1() {foo(def() + 2, abc);}",
|
expect![["macro_rules! foo {() => {}} fn f1() {foo(def() + 2, abc);}"]],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -476,12 +646,12 @@ fn replace_binary_op() {
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"$a + $b ==>> $b + $a",
|
"$a + $b ==>> $b + $a",
|
||||||
"fn f() {2 * 3 + 4 * 5}",
|
"fn f() {2 * 3 + 4 * 5}",
|
||||||
"fn f() {4 * 5 + 2 * 3}",
|
expect![["fn f() {4 * 5 + 2 * 3}"]],
|
||||||
);
|
);
|
||||||
assert_ssr_transform(
|
assert_ssr_transform(
|
||||||
"$a + $b ==>> $b + $a",
|
"$a + $b ==>> $b + $a",
|
||||||
"fn f() {1 + 2 + 3 + 4}",
|
"fn f() {1 + 2 + 3 + 4}",
|
||||||
"fn f() {4 + 3 + 2 + 1}",
|
expect![["fn f() {4 + 3 + 2 + 1}"]],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -494,8 +664,23 @@ fn match_binary_op() {
|
||||||
fn multiple_rules() {
|
fn multiple_rules() {
|
||||||
assert_ssr_transforms(
|
assert_ssr_transforms(
|
||||||
&["$a + 1 ==>> add_one($a)", "$a + $b ==>> add($a, $b)"],
|
&["$a + 1 ==>> add_one($a)", "$a + $b ==>> add($a, $b)"],
|
||||||
"fn f() -> i32 {3 + 2 + 1}",
|
"fn add() {} fn add_one() {} fn f() -> i32 {3 + 2 + 1}",
|
||||||
"fn f() -> i32 {add_one(add(3, 2))}",
|
expect![["fn add() {} fn add_one() {} fn f() -> i32 {add_one(add(3, 2))}"]],
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn multiple_rules_with_nested_matches() {
|
||||||
|
assert_ssr_transforms(
|
||||||
|
&["foo1($a) ==>> bar1($a)", "foo2($a) ==>> bar2($a)"],
|
||||||
|
r#"
|
||||||
|
fn foo1() {} fn foo2() {} fn bar1() {} fn bar2() {}
|
||||||
|
fn f() {foo1(foo2(foo1(foo2(foo1(42)))))}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
fn foo1() {} fn foo2() {} fn bar1() {} fn bar2() {}
|
||||||
|
fn f() {bar1(bar2(bar1(bar2(bar1(42)))))}
|
||||||
|
"#]],
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -527,12 +712,37 @@ fn replace_within_macro_expansion() {
|
||||||
macro_rules! macro1 {
|
macro_rules! macro1 {
|
||||||
($a:expr) => {$a}
|
($a:expr) => {$a}
|
||||||
}
|
}
|
||||||
fn f() {macro1!(5.x().foo().o2())}"#,
|
fn bar() {}
|
||||||
r#"
|
fn f() {macro1!(5.x().foo().o2())}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
macro_rules! macro1 {
|
macro_rules! macro1 {
|
||||||
($a:expr) => {$a}
|
($a:expr) => {$a}
|
||||||
}
|
}
|
||||||
fn f() {macro1!(bar(5.x()).o2())}"#,
|
fn bar() {}
|
||||||
|
fn f() {macro1!(bar(5.x()).o2())}
|
||||||
|
"#]],
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn replace_outside_and_within_macro_expansion() {
|
||||||
|
assert_ssr_transform(
|
||||||
|
"foo($a) ==>> bar($a)",
|
||||||
|
r#"
|
||||||
|
fn foo() {} fn bar() {}
|
||||||
|
macro_rules! macro1 {
|
||||||
|
($a:expr) => {$a}
|
||||||
|
}
|
||||||
|
fn f() {foo(foo(macro1!(foo(foo(42)))))}
|
||||||
|
"#,
|
||||||
|
expect![[r#"
|
||||||
|
fn foo() {} fn bar() {}
|
||||||
|
macro_rules! macro1 {
|
||||||
|
($a:expr) => {$a}
|
||||||
|
}
|
||||||
|
fn f() {bar(bar(macro1!(bar(bar(42)))))}
|
||||||
|
"#]],
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -544,12 +754,14 @@ fn preserves_whitespace_within_macro_expansion() {
|
||||||
macro_rules! macro1 {
|
macro_rules! macro1 {
|
||||||
($a:expr) => {$a}
|
($a:expr) => {$a}
|
||||||
}
|
}
|
||||||
fn f() {macro1!(1 * 2 + 3 + 4}"#,
|
fn f() {macro1!(1 * 2 + 3 + 4}
|
||||||
r#"
|
"#,
|
||||||
|
expect![[r#"
|
||||||
macro_rules! macro1 {
|
macro_rules! macro1 {
|
||||||
($a:expr) => {$a}
|
($a:expr) => {$a}
|
||||||
}
|
}
|
||||||
fn f() {macro1!(4 - 3 - 1 * 2}"#,
|
fn f() {macro1!(4 - 3 - 1 * 2}
|
||||||
|
"#]],
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -580,3 +792,96 @@ fn match_failure_reasons() {
|
||||||
r#"Pattern wanted token '42' (INT_NUMBER), but code had token '43' (INT_NUMBER)"#,
|
r#"Pattern wanted token '42' (INT_NUMBER), but code had token '43' (INT_NUMBER)"#,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn overlapping_possible_matches() {
|
||||||
|
// There are three possible matches here, however the middle one, `foo(foo(foo(42)))` shouldn't
|
||||||
|
// match because it overlaps with the outer match. The inner match is permitted since it's is
|
||||||
|
// contained entirely within the placeholder of the outer match.
|
||||||
|
assert_matches(
|
||||||
|
"foo(foo($a))",
|
||||||
|
"fn foo() {} fn main() {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() }
|
||||||
|
"]],
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn ufcs_matches_method_call() {
|
||||||
|
let code = r#"
|
||||||
|
struct Foo {}
|
||||||
|
impl Foo {
|
||||||
|
fn new(_: i32) -> Foo { Foo {} }
|
||||||
|
fn do_stuff(&self, _: i32) {}
|
||||||
|
}
|
||||||
|
struct Bar {}
|
||||||
|
impl Bar {
|
||||||
|
fn new(_: i32) -> Bar { Bar {} }
|
||||||
|
fn do_stuff(&self, v: i32) {}
|
||||||
|
}
|
||||||
|
fn main() {
|
||||||
|
let b = Bar {};
|
||||||
|
let f = Foo {};
|
||||||
|
b.do_stuff(1);
|
||||||
|
f.do_stuff(2);
|
||||||
|
Foo::new(4).do_stuff(3);
|
||||||
|
// Too many / too few args - should never match
|
||||||
|
f.do_stuff(2, 10);
|
||||||
|
f.do_stuff();
|
||||||
|
}
|
||||||
|
"#;
|
||||||
|
assert_matches("Foo::do_stuff($a, $b)", code, &["f.do_stuff(2)", "Foo::new(4).do_stuff(3)"]);
|
||||||
|
// The arguments needs special handling in the case of a function call matching a method call
|
||||||
|
// and the first argument is different.
|
||||||
|
assert_matches("Foo::do_stuff($a, 2)", code, &["f.do_stuff(2)"]);
|
||||||
|
assert_matches("Foo::do_stuff(Foo::new(4), $b)", code, &["Foo::new(4).do_stuff(3)"]);
|
||||||
|
|
||||||
|
assert_ssr_transform(
|
||||||
|
"Foo::do_stuff(Foo::new($a), $b) ==>> Bar::new($b).do_stuff($a)",
|
||||||
|
code,
|
||||||
|
expect![[r#"
|
||||||
|
struct Foo {}
|
||||||
|
impl Foo {
|
||||||
|
fn new(_: i32) -> Foo { Foo {} }
|
||||||
|
fn do_stuff(&self, _: i32) {}
|
||||||
|
}
|
||||||
|
struct Bar {}
|
||||||
|
impl Bar {
|
||||||
|
fn new(_: i32) -> Bar { Bar {} }
|
||||||
|
fn do_stuff(&self, v: i32) {}
|
||||||
|
}
|
||||||
|
fn main() {
|
||||||
|
let b = Bar {};
|
||||||
|
let f = Foo {};
|
||||||
|
b.do_stuff(1);
|
||||||
|
f.do_stuff(2);
|
||||||
|
Bar::new(3).do_stuff(4);
|
||||||
|
// Too many / too few args - should never match
|
||||||
|
f.do_stuff(2, 10);
|
||||||
|
f.do_stuff();
|
||||||
|
}
|
||||||
|
"#]],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
|
@ -1,27 +1,17 @@
|
||||||
//! Applies structured search replace rules from the command line.
|
//! Applies structured search replace rules from the command line.
|
||||||
|
|
||||||
use crate::cli::{load_cargo::load_cargo, Result};
|
use crate::cli::{load_cargo::load_cargo, Result};
|
||||||
use ra_ide::SourceFileEdit;
|
|
||||||
use ra_ssr::{MatchFinder, SsrPattern, SsrRule};
|
use ra_ssr::{MatchFinder, SsrPattern, SsrRule};
|
||||||
|
|
||||||
pub fn apply_ssr_rules(rules: Vec<SsrRule>) -> Result<()> {
|
pub fn apply_ssr_rules(rules: Vec<SsrRule>) -> Result<()> {
|
||||||
use ra_db::SourceDatabaseExt;
|
use ra_db::SourceDatabaseExt;
|
||||||
use ra_ide_db::symbol_index::SymbolsDatabase;
|
|
||||||
let (host, vfs) = load_cargo(&std::env::current_dir()?, true, true)?;
|
let (host, vfs) = load_cargo(&std::env::current_dir()?, true, true)?;
|
||||||
let db = host.raw_database();
|
let db = host.raw_database();
|
||||||
let mut match_finder = MatchFinder::new(db);
|
let mut match_finder = MatchFinder::at_first_file(db)?;
|
||||||
for rule in rules {
|
for rule in rules {
|
||||||
match_finder.add_rule(rule);
|
match_finder.add_rule(rule)?;
|
||||||
}
|
|
||||||
let mut edits = Vec::new();
|
|
||||||
for &root in db.local_roots().iter() {
|
|
||||||
let sr = db.source_root(root);
|
|
||||||
for file_id in sr.iter() {
|
|
||||||
if let Some(edit) = match_finder.edits_for_file(file_id) {
|
|
||||||
edits.push(SourceFileEdit { file_id, edit });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
let edits = match_finder.edits();
|
||||||
for edit in edits {
|
for edit in edits {
|
||||||
if let Some(path) = vfs.file_path(edit.file_id).as_path() {
|
if let Some(path) = vfs.file_path(edit.file_id).as_path() {
|
||||||
let mut contents = db.file_text(edit.file_id).to_string();
|
let mut contents = db.file_text(edit.file_id).to_string();
|
||||||
|
@ -38,34 +28,27 @@ pub fn apply_ssr_rules(rules: Vec<SsrRule>) -> Result<()> {
|
||||||
pub fn search_for_patterns(patterns: Vec<SsrPattern>, debug_snippet: Option<String>) -> Result<()> {
|
pub fn search_for_patterns(patterns: Vec<SsrPattern>, debug_snippet: Option<String>) -> Result<()> {
|
||||||
use ra_db::SourceDatabaseExt;
|
use ra_db::SourceDatabaseExt;
|
||||||
use ra_ide_db::symbol_index::SymbolsDatabase;
|
use ra_ide_db::symbol_index::SymbolsDatabase;
|
||||||
let (host, vfs) = load_cargo(&std::env::current_dir()?, true, true)?;
|
let (host, _vfs) = load_cargo(&std::env::current_dir()?, true, true)?;
|
||||||
let db = host.raw_database();
|
let db = host.raw_database();
|
||||||
let mut match_finder = MatchFinder::new(db);
|
let mut match_finder = MatchFinder::at_first_file(db)?;
|
||||||
for pattern in patterns {
|
for pattern in patterns {
|
||||||
match_finder.add_search_pattern(pattern);
|
match_finder.add_search_pattern(pattern)?;
|
||||||
}
|
}
|
||||||
for &root in db.local_roots().iter() {
|
if let Some(debug_snippet) = &debug_snippet {
|
||||||
let sr = db.source_root(root);
|
for &root in db.local_roots().iter() {
|
||||||
for file_id in sr.iter() {
|
let sr = db.source_root(root);
|
||||||
if let Some(debug_snippet) = &debug_snippet {
|
for file_id in sr.iter() {
|
||||||
for debug_info in match_finder.debug_where_text_equal(file_id, debug_snippet) {
|
for debug_info in match_finder.debug_where_text_equal(file_id, debug_snippet) {
|
||||||
println!("{:#?}", debug_info);
|
println!("{:#?}", debug_info);
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
let matches = match_finder.find_matches_in_file(file_id);
|
|
||||||
if !matches.matches.is_empty() {
|
|
||||||
let matches = matches.flattened().matches;
|
|
||||||
if let Some(path) = vfs.file_path(file_id).as_path() {
|
|
||||||
println!("{} matches in '{}'", matches.len(), path.to_string_lossy());
|
|
||||||
}
|
|
||||||
// We could possibly at some point do something more useful than just printing
|
|
||||||
// the matched text. For now though, that's the easiest thing to do.
|
|
||||||
for m in matches {
|
|
||||||
println!("{}", m.matched_text());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
for m in match_finder.matches().flattened().matches {
|
||||||
|
// We could possibly at some point do something more useful than just printing
|
||||||
|
// the matched text. For now though, that's the easiest thing to do.
|
||||||
|
println!("{}", m.matched_text());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
@ -1026,8 +1026,9 @@ pub(crate) fn handle_ssr(
|
||||||
params: lsp_ext::SsrParams,
|
params: lsp_ext::SsrParams,
|
||||||
) -> Result<lsp_types::WorkspaceEdit> {
|
) -> Result<lsp_types::WorkspaceEdit> {
|
||||||
let _p = profile("handle_ssr");
|
let _p = profile("handle_ssr");
|
||||||
|
let position = from_proto::file_position(&snap, params.position)?;
|
||||||
let source_change =
|
let source_change =
|
||||||
snap.analysis.structural_search_replace(¶ms.query, params.parse_only)??;
|
snap.analysis.structural_search_replace(¶ms.query, params.parse_only, position)??;
|
||||||
to_proto::workspace_edit(&snap, source_change)
|
to_proto::workspace_edit(&snap, source_change)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -216,6 +216,11 @@ impl Request for Ssr {
|
||||||
pub struct SsrParams {
|
pub struct SsrParams {
|
||||||
pub query: String,
|
pub query: String,
|
||||||
pub parse_only: bool,
|
pub parse_only: bool,
|
||||||
|
|
||||||
|
/// File position where SSR was invoked. Paths in `query` will be resolved relative to this
|
||||||
|
/// position.
|
||||||
|
#[serde(flatten)]
|
||||||
|
pub position: lsp_types::TextDocumentPositionParams,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub enum StatusNotification {}
|
pub enum StatusNotification {}
|
||||||
|
|
|
@ -274,6 +274,11 @@ interface SsrParams {
|
||||||
query: string,
|
query: string,
|
||||||
/// If true, only check the syntax of the query and don't compute the actual edit.
|
/// If true, only check the syntax of the query and don't compute the actual edit.
|
||||||
parseOnly: bool,
|
parseOnly: bool,
|
||||||
|
/// The current text document. This and `position` will be used to determine in what scope
|
||||||
|
/// paths in `query` should be resolved.
|
||||||
|
textDocument: lc.TextDocumentIdentifier;
|
||||||
|
/// Position where SSR was invoked.
|
||||||
|
position: lc.Position;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -285,7 +290,7 @@ WorkspaceEdit
|
||||||
|
|
||||||
### Example
|
### Example
|
||||||
|
|
||||||
SSR with query `foo($a:expr, $b:expr) ==>> ($a).foo($b)` will transform, eg `foo(y + 5, z)` into `(y + 5).foo(z)`.
|
SSR with query `foo($a, $b) ==>> ($a).foo($b)` will transform, eg `foo(y + 5, z)` into `(y + 5).foo(z)`.
|
||||||
|
|
||||||
### Unresolved Question
|
### Unresolved Question
|
||||||
|
|
||||||
|
|
|
@ -185,15 +185,21 @@ export function parentModule(ctx: Ctx): Cmd {
|
||||||
|
|
||||||
export function ssr(ctx: Ctx): Cmd {
|
export function ssr(ctx: Ctx): Cmd {
|
||||||
return async () => {
|
return async () => {
|
||||||
|
const editor = vscode.window.activeTextEditor;
|
||||||
const client = ctx.client;
|
const client = ctx.client;
|
||||||
if (!client) return;
|
if (!editor || !client) return;
|
||||||
|
|
||||||
|
const position = editor.selection.active;
|
||||||
|
const textDocument = { uri: editor.document.uri.toString() };
|
||||||
|
|
||||||
const options: vscode.InputBoxOptions = {
|
const options: vscode.InputBoxOptions = {
|
||||||
value: "() ==>> ()",
|
value: "() ==>> ()",
|
||||||
prompt: "Enter request, for example 'Foo($a) ==> Foo::new($a)' ",
|
prompt: "Enter request, for example 'Foo($a) ==> Foo::new($a)' ",
|
||||||
validateInput: async (x: string) => {
|
validateInput: async (x: string) => {
|
||||||
try {
|
try {
|
||||||
await client.sendRequest(ra.ssr, { query: x, parseOnly: true });
|
await client.sendRequest(ra.ssr, {
|
||||||
|
query: x, parseOnly: true, textDocument, position,
|
||||||
|
});
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
return e.toString();
|
return e.toString();
|
||||||
}
|
}
|
||||||
|
@ -208,7 +214,9 @@ export function ssr(ctx: Ctx): Cmd {
|
||||||
title: "Structured search replace in progress...",
|
title: "Structured search replace in progress...",
|
||||||
cancellable: false,
|
cancellable: false,
|
||||||
}, async (_progress, _token) => {
|
}, async (_progress, _token) => {
|
||||||
const edit = await client.sendRequest(ra.ssr, { query: request, parseOnly: false });
|
const edit = await client.sendRequest(ra.ssr, {
|
||||||
|
query: request, parseOnly: false, textDocument, position
|
||||||
|
});
|
||||||
|
|
||||||
await vscode.workspace.applyEdit(client.protocol2CodeConverter.asWorkspaceEdit(edit));
|
await vscode.workspace.applyEdit(client.protocol2CodeConverter.asWorkspaceEdit(edit));
|
||||||
});
|
});
|
||||||
|
|
|
@ -93,6 +93,8 @@ export const inlayHints = new lc.RequestType<InlayHintsParams, InlayHint[], void
|
||||||
export interface SsrParams {
|
export interface SsrParams {
|
||||||
query: string;
|
query: string;
|
||||||
parseOnly: boolean;
|
parseOnly: boolean;
|
||||||
|
textDocument: lc.TextDocumentIdentifier;
|
||||||
|
position: lc.Position;
|
||||||
}
|
}
|
||||||
export const ssr = new lc.RequestType<SsrParams, lc.WorkspaceEdit, void>('experimental/ssr');
|
export const ssr = new lc.RequestType<SsrParams, lc.WorkspaceEdit, void>('experimental/ssr');
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue