rust/crates/syntax/src/algo.rs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

660 lines
18 KiB
Rust
Raw Normal View History

2021-05-22 08:53:47 -05:00
//! Collection of assorted algorithms for syntax trees.
use std::hash::BuildHasherDefault;
2019-07-20 12:04:34 -05:00
use indexmap::IndexMap;
use itertools::Itertools;
use rustc_hash::FxHashMap;
2020-08-12 10:03:06 -05:00
use text_edit::TextEditBuilder;
2019-07-20 12:04:34 -05:00
use crate::{
2021-08-14 12:29:46 -05:00
AstNode, Direction, NodeOrToken, SyntaxElement, SyntaxKind, SyntaxNode, SyntaxToken, TextRange,
TextSize,
2019-07-20 12:04:34 -05:00
};
2019-01-07 07:15:47 -06:00
/// Returns ancestors of the node at the offset, sorted by length. This should
/// do the right thing at an edge, e.g. when searching for expressions at `{
2021-01-06 14:15:48 -06:00
/// $0foo }` we will get the name reference instead of the whole block, which
/// we would get if we just did `find_token_at_offset(...).flat_map(|t|
/// t.parent().ancestors())`.
pub fn ancestors_at_offset(
node: &SyntaxNode,
2020-04-24 16:40:41 -05:00
offset: TextSize,
2019-07-18 11:23:05 -05:00
) -> impl Iterator<Item = SyntaxNode> {
2019-07-21 05:28:58 -05:00
node.token_at_offset(offset)
2022-06-10 09:30:09 -05:00
.map(|token| token.parent_ancestors())
2019-07-20 04:58:27 -05:00
.kmerge_by(|node1, node2| node1.text_range().len() < node2.text_range().len())
}
2019-01-08 11:47:37 -06:00
/// Finds a node of specific Ast type at offset. Note that this is slightly
2019-01-27 07:49:02 -06:00
/// imprecise: if the cursor is strictly between two nodes of the desired type,
2019-01-08 11:47:37 -06:00
/// as in
///
2020-08-22 14:03:02 -05:00
/// ```no_run
2019-01-08 11:47:37 -06:00
/// struct Foo {}|struct Bar;
/// ```
///
/// then the shorter node will be silently preferred.
2020-04-24 16:40:41 -05:00
pub fn find_node_at_offset<N: AstNode>(syntax: &SyntaxNode, offset: TextSize) -> Option<N> {
ancestors_at_offset(syntax, offset).find_map(N::cast)
2019-01-08 11:44:31 -06:00
}
2020-06-29 10:22:47 -05:00
pub fn find_node_at_range<N: AstNode>(syntax: &SyntaxNode, range: TextRange) -> Option<N> {
2021-01-15 11:15:33 -06:00
syntax.covering_element(range).ancestors().find_map(N::cast)
2020-06-29 10:22:47 -05:00
}
2020-02-26 10:12:26 -06:00
/// Skip to next non `trivia` token
pub fn skip_trivia_token(mut token: SyntaxToken, direction: Direction) -> Option<SyntaxToken> {
while token.kind().is_trivia() {
token = match direction {
Direction::Next => token.next_token()?,
Direction::Prev => token.prev_token()?,
}
}
Some(token)
}
/// Skip to next non `whitespace` token
pub fn skip_whitespace_token(mut token: SyntaxToken, direction: Direction) -> Option<SyntaxToken> {
while token.kind() == SyntaxKind::WHITESPACE {
token = match direction {
Direction::Next => token.next_token()?,
Direction::Prev => token.prev_token()?,
}
}
Some(token)
}
2020-02-26 10:12:26 -06:00
/// Finds the first sibling in the given direction which is not `trivia`
2019-03-30 05:25:53 -05:00
pub fn non_trivia_sibling(element: SyntaxElement, direction: Direction) -> Option<SyntaxElement> {
return match element {
2019-07-20 12:04:34 -05:00
NodeOrToken::Node(node) => node.siblings_with_tokens(direction).skip(1).find(not_trivia),
NodeOrToken::Token(token) => token.siblings_with_tokens(direction).skip(1).find(not_trivia),
2019-03-30 05:25:53 -05:00
};
fn not_trivia(element: &SyntaxElement) -> bool {
match element {
2019-07-20 12:04:34 -05:00
NodeOrToken::Node(_) => true,
NodeOrToken::Token(token) => !token.kind().is_trivia(),
2019-03-30 05:25:53 -05:00
}
}
}
pub fn least_common_ancestor(u: &SyntaxNode, v: &SyntaxNode) -> Option<SyntaxNode> {
if u == v {
return Some(u.clone());
}
let u_depth = u.ancestors().count();
let v_depth = v.ancestors().count();
let keep = u_depth.min(v_depth);
let u_candidates = u.ancestors().skip(u_depth - keep);
2021-01-08 08:41:32 -06:00
let v_candidates = v.ancestors().skip(v_depth - keep);
let (res, _) = u_candidates.zip(v_candidates).find(|(x, y)| x == y)?;
Some(res)
}
2020-03-19 05:38:26 -05:00
pub fn neighbor<T: AstNode>(me: &T, direction: Direction) -> Option<T> {
me.syntax().siblings(direction).skip(1).find_map(T::cast)
}
pub fn has_errors(node: &SyntaxNode) -> bool {
node.children().any(|it| it.kind() == SyntaxKind::ERROR)
}
type FxIndexMap<K, V> = IndexMap<K, V, BuildHasherDefault<rustc_hash::FxHasher>>;
#[derive(Debug, Hash, PartialEq, Eq)]
enum TreeDiffInsertPos {
After(SyntaxElement),
AsFirstChild(SyntaxElement),
}
2020-10-22 06:51:08 -05:00
#[derive(Debug)]
2019-09-30 01:27:26 -05:00
pub struct TreeDiff {
replacements: FxHashMap<SyntaxElement, SyntaxElement>,
deletions: Vec<SyntaxElement>,
// the vec as well as the indexmap are both here to preserve order
insertions: FxIndexMap<TreeDiffInsertPos, Vec<SyntaxElement>>,
2019-09-30 01:27:26 -05:00
}
impl TreeDiff {
pub fn into_text_edit(&self, builder: &mut TextEditBuilder) {
let _p = tracing::span!(tracing::Level::INFO, "into_text_edit").entered();
2020-11-27 10:00:03 -06:00
2021-10-03 07:45:08 -05:00
for (anchor, to) in &self.insertions {
let offset = match anchor {
TreeDiffInsertPos::After(it) => it.text_range().end(),
TreeDiffInsertPos::AsFirstChild(it) => it.text_range().start(),
};
to.iter().for_each(|to| builder.insert(offset, to.to_string()));
}
2021-10-03 07:45:08 -05:00
for (from, to) in &self.replacements {
builder.replace(from.text_range(), to.to_string());
2019-09-30 01:27:26 -05:00
}
for text_range in self.deletions.iter().map(SyntaxElement::text_range) {
builder.delete(text_range);
}
2019-09-30 01:27:26 -05:00
}
2020-03-21 09:43:48 -05:00
pub fn is_empty(&self) -> bool {
self.replacements.is_empty() && self.deletions.is_empty() && self.insertions.is_empty()
2020-03-21 09:43:48 -05:00
}
2019-09-30 01:27:26 -05:00
}
2020-10-26 09:40:08 -05:00
/// Finds a (potentially minimal) diff, which, applied to `from`, will result in `to`.
2019-09-26 07:56:52 -05:00
///
/// Specifically, returns a structure that consists of a replacements, insertions and deletions
/// such that applying this map on `from` will result in `to`.
2019-09-26 07:56:52 -05:00
///
/// This function tries to find a fine-grained diff.
2019-09-30 01:27:26 -05:00
pub fn diff(from: &SyntaxNode, to: &SyntaxNode) -> TreeDiff {
let _p = tracing::span!(tracing::Level::INFO, "diff").entered();
2020-11-27 10:00:03 -06:00
let mut diff = TreeDiff {
replacements: FxHashMap::default(),
insertions: FxIndexMap::default(),
deletions: Vec::new(),
};
let (from, to) = (from.clone().into(), to.clone().into());
if !syntax_element_eq(&from, &to) {
go(&mut diff, from, to);
}
return diff;
fn syntax_element_eq(lhs: &SyntaxElement, rhs: &SyntaxElement) -> bool {
lhs.kind() == rhs.kind()
&& lhs.text_range().len() == rhs.text_range().len()
&& match (&lhs, &rhs) {
2019-09-26 07:56:52 -05:00
(NodeOrToken::Node(lhs), NodeOrToken::Node(rhs)) => {
lhs == rhs || lhs.text() == rhs.text()
2019-09-26 07:56:52 -05:00
}
(NodeOrToken::Token(lhs), NodeOrToken::Token(rhs)) => lhs.text() == rhs.text(),
_ => false,
}
}
2021-07-27 17:00:22 -05:00
// FIXME: this is horribly inefficient. I bet there's a cool algorithm to diff trees properly.
fn go(diff: &mut TreeDiff, lhs: SyntaxElement, rhs: SyntaxElement) {
let (lhs, rhs) = match lhs.as_node().zip(rhs.as_node()) {
Some((lhs, rhs)) => (lhs, rhs),
_ => {
2021-03-08 14:19:44 -06:00
cov_mark::hit!(diff_node_token_replace);
diff.replacements.insert(lhs, rhs);
2019-09-26 07:56:52 -05:00
return;
}
};
2020-10-26 09:40:08 -05:00
let mut look_ahead_scratch = Vec::default();
let mut rhs_children = rhs.children_with_tokens();
let mut lhs_children = lhs.children_with_tokens();
let mut last_lhs = None;
loop {
let lhs_child = lhs_children.next();
match (lhs_child.clone(), rhs_children.next()) {
(None, None) => break,
(None, Some(element)) => {
let insert_pos = match last_lhs.clone() {
Some(prev) => {
2021-03-08 14:19:44 -06:00
cov_mark::hit!(diff_insert);
TreeDiffInsertPos::After(prev)
}
// first iteration, insert into out parent as the first child
None => {
2021-03-08 14:19:44 -06:00
cov_mark::hit!(diff_insert_as_first_child);
TreeDiffInsertPos::AsFirstChild(lhs.clone().into())
}
};
2024-01-18 06:59:49 -06:00
diff.insertions.entry(insert_pos).or_default().push(element);
}
(Some(element), None) => {
2021-03-08 14:19:44 -06:00
cov_mark::hit!(diff_delete);
diff.deletions.push(element);
}
(Some(ref lhs_ele), Some(ref rhs_ele)) if syntax_element_eq(lhs_ele, rhs_ele) => {}
2020-10-26 09:40:08 -05:00
(Some(lhs_ele), Some(rhs_ele)) => {
// nodes differ, look for lhs_ele in rhs, if its found we can mark everything up
// until that element as insertions. This is important to keep the diff minimal
// in regards to insertions that have been actually done, this is important for
// use insertions as we do not want to replace the entire module node.
look_ahead_scratch.push(rhs_ele.clone());
let mut rhs_children_clone = rhs_children.clone();
let mut insert = false;
2021-09-19 12:00:06 -05:00
for rhs_child in &mut rhs_children_clone {
2020-10-26 09:40:08 -05:00
if syntax_element_eq(&lhs_ele, &rhs_child) {
2021-03-08 14:19:44 -06:00
cov_mark::hit!(diff_insertions);
2020-10-26 09:40:08 -05:00
insert = true;
break;
}
2021-10-03 07:45:08 -05:00
look_ahead_scratch.push(rhs_child);
2020-10-26 09:40:08 -05:00
}
let drain = look_ahead_scratch.drain(..);
if insert {
let insert_pos = if let Some(prev) = last_lhs.clone().filter(|_| insert) {
TreeDiffInsertPos::After(prev)
} else {
2021-03-08 14:19:44 -06:00
cov_mark::hit!(insert_first_child);
TreeDiffInsertPos::AsFirstChild(lhs.clone().into())
};
2024-01-18 06:59:49 -06:00
diff.insertions.entry(insert_pos).or_default().extend(drain);
2020-10-26 09:40:08 -05:00
rhs_children = rhs_children_clone;
} else {
go(diff, lhs_ele, rhs_ele);
2020-10-26 09:40:08 -05:00
}
}
}
last_lhs = lhs_child.or(last_lhs);
2019-09-26 07:56:52 -05:00
}
}
}
2020-10-22 06:51:08 -05:00
#[cfg(test)]
mod tests {
use expect_test::{expect, Expect};
use itertools::Itertools;
use parser::SyntaxKind;
use text_edit::TextEdit;
use crate::{AstNode, SyntaxElement};
#[test]
fn replace_node_token() {
2021-03-08 14:19:44 -06:00
cov_mark::check!(diff_node_token_replace);
2020-10-22 06:51:08 -05:00
check_diff(
r#"use node;"#,
r#"ident"#,
expect![[r#"
insertions:
replacements:
Line 0: Token(USE_KW@0..3 "use") -> ident
deletions:
Line 1: " "
Line 1: node
Line 1: ;
"#]],
);
}
#[test]
2020-10-26 09:40:08 -05:00
fn replace_parent() {
2021-03-08 14:19:44 -06:00
cov_mark::check!(diff_insert_as_first_child);
2020-10-26 09:40:08 -05:00
check_diff(
r#""#,
r#"use foo::bar;"#,
expect![[r#"
insertions:
Line 0: AsFirstChild(Node(SOURCE_FILE@0..0))
-> use foo::bar;
2020-10-26 09:40:08 -05:00
replacements:
2020-10-26 09:40:08 -05:00
deletions:
"#]],
);
}
#[test]
fn insert_last() {
2021-03-08 14:19:44 -06:00
cov_mark::check!(diff_insert);
2020-10-22 06:51:08 -05:00
check_diff(
2020-10-26 09:40:08 -05:00
r#"
use foo;
2020-10-22 06:51:08 -05:00
use bar;"#,
2020-10-26 09:40:08 -05:00
r#"
use foo;
use bar;
use baz;"#,
2020-10-22 06:51:08 -05:00
expect![[r#"
insertions:
Line 2: After(Node(USE@10..18))
2020-10-22 06:51:08 -05:00
-> "\n"
2020-10-26 09:40:08 -05:00
-> use baz;
2020-10-22 06:51:08 -05:00
replacements:
deletions:
"#]],
);
}
#[test]
2020-10-26 09:40:08 -05:00
fn insert_middle() {
2020-10-22 06:51:08 -05:00
check_diff(
2020-10-26 09:40:08 -05:00
r#"
use foo;
use baz;"#,
r#"
use foo;
use bar;
use baz;"#,
2020-10-22 06:51:08 -05:00
expect![[r#"
insertions:
Line 2: After(Token(WHITESPACE@9..10 "\n"))
2020-10-26 09:40:08 -05:00
-> use bar;
-> "\n"
replacements:
deletions:
"#]],
)
}
#[test]
fn insert_first() {
check_diff(
r#"
use bar;
use baz;"#,
r#"
use foo;
use bar;
use baz;"#,
expect![[r#"
insertions:
2020-10-22 06:51:08 -05:00
Line 0: After(Token(WHITESPACE@0..1 "\n"))
2020-10-26 09:40:08 -05:00
-> use foo;
-> "\n"
2020-10-22 06:51:08 -05:00
replacements:
2020-10-26 09:40:08 -05:00
2020-10-22 06:51:08 -05:00
deletions:
"#]],
2020-10-26 09:40:08 -05:00
)
2020-10-22 06:51:08 -05:00
}
#[test]
fn first_child_insertion() {
2021-03-08 14:19:44 -06:00
cov_mark::check!(insert_first_child);
check_diff(
r#"fn main() {
stdi
}"#,
r#"use foo::bar;
fn main() {
stdi
}"#,
expect![[r#"
insertions:
Line 0: AsFirstChild(Node(SOURCE_FILE@0..30))
-> use foo::bar;
-> "\n\n "
replacements:
deletions:
"#]],
);
}
2020-10-22 06:51:08 -05:00
#[test]
2020-10-26 09:40:08 -05:00
fn delete_last() {
2021-03-08 14:19:44 -06:00
cov_mark::check!(diff_delete);
2020-10-22 06:51:08 -05:00
check_diff(
r#"use foo;
use bar;"#,
r#"use foo;"#,
expect![[r#"
insertions:
replacements:
deletions:
Line 1: "\n "
Line 2: use bar;
"#]],
);
}
#[test]
2020-10-26 09:40:08 -05:00
fn delete_middle() {
2021-03-08 14:19:44 -06:00
cov_mark::check!(diff_insertions);
2020-10-22 06:51:08 -05:00
check_diff(
r#"
use expect_test::{expect, Expect};
2020-10-26 09:40:08 -05:00
use text_edit::TextEdit;
2020-10-22 06:51:08 -05:00
use crate::AstNode;
"#,
r#"
use expect_test::{expect, Expect};
use crate::AstNode;
"#,
expect![[r#"
insertions:
Line 1: After(Node(USE@1..35))
2020-10-26 09:40:08 -05:00
-> "\n\n"
2020-10-22 06:51:08 -05:00
-> use crate::AstNode;
replacements:
2020-10-26 09:40:08 -05:00
deletions:
2020-10-22 06:51:08 -05:00
2020-10-26 09:40:08 -05:00
Line 2: use text_edit::TextEdit;
Line 3: "\n\n"
Line 4: use crate::AstNode;
Line 5: "\n"
2020-10-22 06:51:08 -05:00
"#]],
)
}
#[test]
2020-10-26 09:40:08 -05:00
fn delete_first() {
2020-10-22 06:51:08 -05:00
check_diff(
r#"
use text_edit::TextEdit;
use crate::AstNode;
"#,
r#"
use crate::AstNode;
"#,
expect![[r#"
insertions:
replacements:
Line 2: Token(IDENT@5..14 "text_edit") -> crate
2020-10-26 09:40:08 -05:00
Line 2: Token(IDENT@16..24 "TextEdit") -> AstNode
Line 2: Token(WHITESPACE@25..27 "\n\n") -> "\n"
2020-10-22 06:51:08 -05:00
deletions:
2020-10-26 09:40:08 -05:00
Line 3: use crate::AstNode;
Line 4: "\n"
2020-10-22 06:51:08 -05:00
"#]],
)
}
#[test]
fn merge_use() {
check_diff(
r#"
use std::{
fmt,
hash::BuildHasherDefault,
ops::{self, RangeInclusive},
};
"#,
r#"
use std::fmt;
use std::hash::BuildHasherDefault;
use std::ops::{self, RangeInclusive};
"#,
expect![[r#"
insertions:
Line 2: After(Node(PATH_SEGMENT@5..8))
2020-10-22 06:51:08 -05:00
-> ::
-> fmt
Line 6: After(Token(WHITESPACE@86..87 "\n"))
2020-10-22 06:51:08 -05:00
-> use std::hash::BuildHasherDefault;
-> "\n"
-> use std::ops::{self, RangeInclusive};
-> "\n"
replacements:
Line 2: Token(IDENT@5..8 "std") -> std
deletions:
Line 2: ::
Line 2: {
fmt,
hash::BuildHasherDefault,
ops::{self, RangeInclusive},
}
"#]],
)
}
#[test]
fn early_return_assist() {
check_diff(
r#"
fn main() {
if let Ok(x) = Err(92) {
foo(x);
}
}
"#,
r#"
fn main() {
let x = match Err(92) {
Ok(it) => it,
_ => return,
};
foo(x);
}
"#,
expect![[r#"
insertions:
Line 3: After(Node(BLOCK_EXPR@40..63))
2020-10-22 06:51:08 -05:00
-> " "
-> match Err(92) {
Ok(it) => it,
_ => return,
}
-> ;
Line 3: After(Node(IF_EXPR@17..63))
2020-10-26 09:40:08 -05:00
-> "\n "
-> foo(x);
2020-10-22 06:51:08 -05:00
replacements:
Line 3: Token(IF_KW@17..19 "if") -> let
Line 3: Token(LET_KW@20..23 "let") -> x
Line 3: Node(BLOCK_EXPR@40..63) -> =
deletions:
Line 3: " "
Line 3: Ok(x)
Line 3: " "
Line 3: =
Line 3: " "
Line 3: Err(92)
"#]],
)
}
fn check_diff(from: &str, to: &str, expected_diff: Expect) {
let from_node = crate::SourceFile::parse(from).tree().syntax().clone();
let to_node = crate::SourceFile::parse(to).tree().syntax().clone();
let diff = super::diff(&from_node, &to_node);
let line_number =
|syn: &SyntaxElement| from[..syn.text_range().start().into()].lines().count();
let fmt_syntax = |syn: &SyntaxElement| match syn.kind() {
SyntaxKind::WHITESPACE => format!("{:?}", syn.to_string()),
_ => format!("{syn}"),
2020-10-22 06:51:08 -05:00
};
let insertions =
diff.insertions.iter().format_with("\n", |(k, v), f| -> Result<(), std::fmt::Error> {
f(&format!(
"Line {}: {:?}\n-> {}",
line_number(match k {
super::TreeDiffInsertPos::After(syn) => syn,
super::TreeDiffInsertPos::AsFirstChild(syn) => syn,
}),
k,
v.iter().format_with("\n-> ", |v, f| f(&fmt_syntax(v)))
))
});
2020-10-22 06:51:08 -05:00
let replacements = diff
.replacements
.iter()
.sorted_by_key(|(syntax, _)| syntax.text_range().start())
.format_with("\n", |(k, v), f| {
f(&format!("Line {}: {k:?} -> {}", line_number(k), fmt_syntax(v)))
2020-10-22 06:51:08 -05:00
});
let deletions = diff
.deletions
.iter()
.format_with("\n", |v, f| f(&format!("Line {}: {}", line_number(v), &fmt_syntax(v))));
let actual = format!(
2022-12-30 01:59:11 -06:00
"insertions:\n\n{insertions}\n\nreplacements:\n\n{replacements}\n\ndeletions:\n\n{deletions}\n"
2020-10-22 06:51:08 -05:00
);
expected_diff.assert_eq(&actual);
let mut from = from.to_owned();
let mut text_edit = TextEdit::builder();
diff.into_text_edit(&mut text_edit);
text_edit.finish().apply(&mut from);
assert_eq!(&*from, to, "diff did not turn `from` to `to`");
}
}