flip comma

This commit is contained in:
Aleksey Kladov 2018-08-12 18:50:16 +03:00
parent 56aa6e20e0
commit 66be735aa9
12 changed files with 273 additions and 57 deletions

View file

@ -0,0 +1,7 @@
[package]
name = "assert_eq_text"
version = "0.1.0"
authors = ["Aleksey Kladov <aleksey.kladov@gmail.com>"]
[dependencies]
difference = "2.0.0"

View file

@ -0,0 +1,25 @@
extern crate difference;
pub use self::difference::Changeset as __Changeset;
#[macro_export]
macro_rules! assert_eq_text {
($expected:expr, $actual:expr) => {{
let expected = $expected;
let actual = $actual;
if expected != actual {
let changeset = $crate::__Changeset::new(actual, expected, "\n");
println!("Expected:\n{}\n\nActual:\n{}\nDiff:{}\n", expected, actual, changeset);
panic!("text differs");
}
}};
($expected:expr, $actual:expr, $($tt:tt)*) => {{
let expected = $expected;
let actual = $actual;
if expected != actual {
let changeset = $crate::__Changeset::new(actual, expected, "\n");
println!("Expected:\n{}\n\nActual:\n{}\n\nDiff:\n{}\n", expected, actual, changeset);
println!($($tt)*);
panic!("text differs");
}
}};
}

View file

@ -7,4 +7,6 @@ publish = false
[dependencies] [dependencies]
itertools = "0.7.8" itertools = "0.7.8"
superslice = "0.1.0" superslice = "0.1.0"
libsyntax2 = { path = "../libsyntax2" } libsyntax2 = { path = "../libsyntax2" }
assert_eq_text = { path = "../assert_eq_text" }

View file

@ -0,0 +1,33 @@
use {TextUnit, File, EditBuilder, Edit};
use libsyntax2::{
ast::AstNode,
SyntaxKind::COMMA,
SyntaxNodeRef,
algo::{
Direction, siblings,
find_leaf_at_offset,
},
};
pub fn flip_comma<'a>(file: &'a File, offset: TextUnit) -> Option<impl FnOnce() -> Edit + 'a> {
let syntax = file.syntax();
let syntax = syntax.as_ref();
let comma = find_leaf_at_offset(syntax, offset).find(|leaf| leaf.kind() == COMMA)?;
let left = non_trivia_sibling(comma, Direction::Backward)?;
let right = non_trivia_sibling(comma, Direction::Forward)?;
Some(move || {
let mut edit = EditBuilder::new();
edit.replace(left.range(), right.text());
edit.replace(right.range(), left.text());
edit.finish()
})
}
fn non_trivia_sibling(node: SyntaxNodeRef, direction: Direction) -> Option<SyntaxNodeRef> {
siblings(node, direction)
.skip(1)
.find(|node| !node.kind().is_trivia())
}

View file

@ -0,0 +1,93 @@
use {TextRange, TextUnit};
#[derive(Debug)]
pub struct Edit {
pub atoms: Vec<AtomEdit>,
}
#[derive(Debug)]
pub struct AtomEdit {
pub delete: TextRange,
pub insert: String,
}
#[derive(Debug)]
pub struct EditBuilder {
atoms: Vec<AtomEdit>
}
impl EditBuilder {
pub fn new() -> EditBuilder {
EditBuilder { atoms: Vec::new() }
}
pub fn replace(&mut self, range: TextRange, replacement: String) {
let range = self.translate(range);
self.atoms.push(AtomEdit { delete: range, insert: replacement })
}
pub fn delete(&mut self, range: TextRange) {
self.replace(range, String::new());
}
pub fn insert(&mut self, offset: TextUnit, text: String) {
self.replace(TextRange::offset_len(offset, 0.into()), text)
}
pub fn finish(self) -> Edit {
Edit { atoms: self.atoms }
}
fn translate(&self, range: TextRange) -> TextRange {
let mut range = range;
for atom in self.atoms.iter() {
range = atom.apply_to_range(range)
.expect("conflicting edits");
}
range
}
}
impl Edit {
pub fn apply(&self, text: &str) -> String {
let mut text = text.to_owned();
for atom in self.atoms.iter() {
text = atom.apply(&text);
}
text
}
}
impl AtomEdit {
fn apply(&self, text: &str) -> String {
let prefix = &text[
TextRange::from_to(0.into(), self.delete.start())
];
let suffix = &text[
TextRange::from_to(self.delete.end(), TextUnit::of_str(text))
];
let mut res = String::with_capacity(prefix.len() + self.insert.len() + suffix.len());
res.push_str(prefix);
res.push_str(&self.insert);
res.push_str(suffix);
res
}
fn apply_to_position(&self, pos: TextUnit) -> Option<TextUnit> {
if pos <= self.delete.start() {
return Some(pos);
}
if pos < self.delete.end() {
return None;
}
Some(pos - self.delete.len() + TextUnit::of_str(&self.insert))
}
fn apply_to_range(&self, range: TextRange) -> Option<TextRange> {
Some(TextRange::from_to(
self.apply_to_position(range.start())?,
self.apply_to_position(range.end())?,
))
}
}

View file

@ -1,9 +1,12 @@
extern crate libsyntax2; extern crate libsyntax2;
extern crate superslice; extern crate superslice;
extern crate itertools;
mod extend_selection; mod extend_selection;
mod symbols; mod symbols;
mod line_index; mod line_index;
mod edit;
mod code_actions;
use libsyntax2::{ use libsyntax2::{
ast::{self, NameOwner}, ast::{self, NameOwner},
@ -15,7 +18,9 @@ pub use libsyntax2::{File, TextRange, TextUnit};
pub use self::{ pub use self::{
line_index::{LineIndex, LineCol}, line_index::{LineIndex, LineCol},
extend_selection::extend_selection, extend_selection::extend_selection,
symbols::{FileSymbol, file_symbols} symbols::{FileSymbol, file_symbols},
edit::{EditBuilder, Edit},
code_actions::{flip_comma},
}; };
#[derive(Debug)] #[derive(Debug)]

View file

@ -1,9 +1,16 @@
extern crate libeditor; extern crate libeditor;
extern crate libsyntax2;
extern crate itertools; extern crate itertools;
#[macro_use]
extern crate assert_eq_text;
use std::fmt; use std::fmt;
use itertools::Itertools; use itertools::Itertools;
use libeditor::{File, highlight, runnables, extend_selection, TextRange, file_symbols}; use libsyntax2::AstNode;
use libeditor::{
File, TextUnit, TextRange,
highlight, runnables, extend_selection, file_symbols, flip_comma,
};
#[test] #[test]
fn test_extend_selection() { fn test_extend_selection() {
@ -27,13 +34,13 @@ fn main() {}
"#); "#);
let hls = highlight(&file); let hls = highlight(&file);
dbg_eq( dbg_eq(
&hls,
r#"[HighlightedRange { range: [1; 11), tag: "comment" }, r#"[HighlightedRange { range: [1; 11), tag: "comment" },
HighlightedRange { range: [12; 14), tag: "keyword" }, HighlightedRange { range: [12; 14), tag: "keyword" },
HighlightedRange { range: [15; 19), tag: "function" }, HighlightedRange { range: [15; 19), tag: "function" },
HighlightedRange { range: [29; 36), tag: "text" }, HighlightedRange { range: [29; 36), tag: "text" },
HighlightedRange { range: [38; 50), tag: "string" }, HighlightedRange { range: [38; 50), tag: "string" },
HighlightedRange { range: [52; 54), tag: "literal" }]"# HighlightedRange { range: [52; 54), tag: "literal" }]"#,
&hls,
); );
} }
@ -51,10 +58,10 @@ fn test_foo() {}
"#); "#);
let runnables = runnables(&file); let runnables = runnables(&file);
dbg_eq( dbg_eq(
&runnables,
r#"[Runnable { range: [1; 13), kind: Bin }, r#"[Runnable { range: [1; 13), kind: Bin },
Runnable { range: [15; 39), kind: Test { name: "test_foo" } }, Runnable { range: [15; 39), kind: Test { name: "test_foo" } },
Runnable { range: [41; 75), kind: Test { name: "test_foo" } }]"#, Runnable { range: [41; 75), kind: Test { name: "test_foo" } }]"#,
&runnables,
) )
} }
@ -76,7 +83,6 @@ const C: i32 = 92;
"#); "#);
let symbols = file_symbols(&file); let symbols = file_symbols(&file);
dbg_eq( dbg_eq(
&symbols,
r#"[FileSymbol { parent: None, name: "Foo", name_range: [8; 11), node_range: [1; 26), kind: STRUCT }, r#"[FileSymbol { parent: None, name: "Foo", name_range: [8; 11), node_range: [1; 26), kind: STRUCT },
FileSymbol { parent: None, name: "m", name_range: [32; 33), node_range: [28; 53), kind: MODULE }, FileSymbol { parent: None, name: "m", name_range: [32; 33), node_range: [28; 53), kind: MODULE },
FileSymbol { parent: Some(1), name: "bar", name_range: [43; 46), node_range: [40; 51), kind: FUNCTION }, FileSymbol { parent: Some(1), name: "bar", name_range: [43; 46), node_range: [40; 51), kind: FUNCTION },
@ -84,6 +90,19 @@ const C: i32 = 92;
FileSymbol { parent: None, name: "T", name_range: [81; 82), node_range: [76; 88), kind: TYPE_ITEM }, FileSymbol { parent: None, name: "T", name_range: [81; 82), node_range: [76; 88), kind: TYPE_ITEM },
FileSymbol { parent: None, name: "S", name_range: [96; 97), node_range: [89; 108), kind: STATIC_ITEM }, FileSymbol { parent: None, name: "S", name_range: [96; 97), node_range: [89; 108), kind: STATIC_ITEM },
FileSymbol { parent: None, name: "C", name_range: [115; 116), node_range: [109; 127), kind: CONST_ITEM }]"#, FileSymbol { parent: None, name: "C", name_range: [115; 116), node_range: [109; 127), kind: CONST_ITEM }]"#,
&symbols,
)
}
#[test]
fn test_swap_comma() {
check_modification(
"fn foo(x: i32,<|> y: Result<(), ()>) {}",
"fn foo(y: Result<(), ()>, x: i32) {}",
&|file, offset| {
let edit = flip_comma(file, offset).unwrap()();
edit.apply(&file.syntax().text())
},
) )
} }
@ -91,8 +110,27 @@ fn file(text: &str) -> File {
File::parse(text) File::parse(text)
} }
fn dbg_eq(actual: &impl fmt::Debug, expected: &str) { fn dbg_eq(expected: &str, actual: &impl fmt::Debug) {
let actual = format!("{:?}", actual); let actual = format!("{:?}", actual);
let expected = expected.lines().map(|l| l.trim()).join(" "); let expected = expected.lines().map(|l| l.trim()).join(" ");
assert_eq!(actual, expected); assert_eq!(expected, actual);
}
fn check_modification(
before: &str,
after: &str,
f: &impl Fn(&File, TextUnit) -> String,
) {
let cursor = "<|>";
let cursor_pos = match before.find(cursor) {
None => panic!("before text should contain cursor marker"),
Some(pos) => pos,
};
let mut text = String::with_capacity(before.len() - cursor.len());
text.push_str(&before[..cursor_pos]);
text.push_str(&before[cursor_pos + cursor.len()..]);
let cursor_pos = TextUnit::from(cursor_pos as u32);
let file = file(&text);
let actual = f(&file, cursor_pos);
assert_eq_text!(after, &actual);
} }

View file

@ -12,4 +12,4 @@ drop_bomb = "0.1.4"
parking_lot = "0.6.0" parking_lot = "0.6.0"
[dev-dependencies] [dev-dependencies]
difference = "2.0.0" assert_eq_text = { path = "../assert_eq_text" }

View file

@ -74,7 +74,6 @@ impl<'f> Iterator for LeafAtOffset<'f> {
} }
} }
pub fn find_covering_node(root: SyntaxNodeRef, range: TextRange) -> SyntaxNodeRef { pub fn find_covering_node(root: SyntaxNodeRef, range: TextRange) -> SyntaxNodeRef {
assert!(is_subrange(root.range(), range)); assert!(is_subrange(root.range(), range));
let (left, right) = match ( let (left, right) = match (
@ -88,6 +87,26 @@ pub fn find_covering_node(root: SyntaxNodeRef, range: TextRange) -> SyntaxNodeRe
common_ancestor(left, right) common_ancestor(left, right)
} }
pub fn ancestors<'a>(node: SyntaxNodeRef<'a>) -> impl Iterator<Item=SyntaxNodeRef<'a>> {
generate(Some(node), |&node| node.parent())
}
#[derive(Debug)]
pub enum Direction {
Forward,
Backward,
}
pub fn siblings<'a>(
node: SyntaxNodeRef<'a>,
direction: Direction
) -> impl Iterator<Item=SyntaxNodeRef<'a>> {
generate(Some(node), move |&node| match direction {
Direction::Forward => node.next_sibling(),
Direction::Backward => node.prev_sibling(),
})
}
fn common_ancestor<'a>(n1: SyntaxNodeRef<'a>, n2: SyntaxNodeRef<'a>) -> SyntaxNodeRef<'a> { fn common_ancestor<'a>(n1: SyntaxNodeRef<'a>, n2: SyntaxNodeRef<'a>) -> SyntaxNodeRef<'a> {
for p in ancestors(n1) { for p in ancestors(n1) {
if ancestors(n2).any(|a| a == p) { if ancestors(n2).any(|a| a == p) {
@ -97,24 +116,6 @@ fn common_ancestor<'a>(n1: SyntaxNodeRef<'a>, n2: SyntaxNodeRef<'a>) -> SyntaxNo
panic!("Can't find common ancestor of {:?} and {:?}", n1, n2) panic!("Can't find common ancestor of {:?} and {:?}", n1, n2)
} }
pub fn ancestors<'a>(node: SyntaxNodeRef<'a>) -> impl Iterator<Item=SyntaxNodeRef<'a>> {
Ancestors(Some(node))
}
#[derive(Debug)]
struct Ancestors<'a>(Option<SyntaxNodeRef<'a>>);
impl<'a> Iterator for Ancestors<'a> {
type Item = SyntaxNodeRef<'a>;
fn next(&mut self) -> Option<Self::Item> {
self.0.take().map(|n| {
self.0 = n.parent();
n
})
}
}
fn contains_offset_nonstrict(range: TextRange, offset: TextUnit) -> bool { fn contains_offset_nonstrict(range: TextRange, offset: TextUnit) -> bool {
range.start() <= offset && offset <= range.end() range.start() <= offset && offset <= range.end()
} }
@ -122,3 +123,12 @@ fn contains_offset_nonstrict(range: TextRange, offset: TextUnit) -> bool {
fn is_subrange(range: TextRange, subrange: TextRange) -> bool { fn is_subrange(range: TextRange, subrange: TextRange) -> bool {
range.start() <= subrange.start() && subrange.end() <= range.end() range.start() <= subrange.start() && subrange.end() <= range.end()
} }
fn generate<T>(seed: Option<T>, step: impl Fn(&T) -> Option<T>) -> impl Iterator<Item=T> {
::itertools::unfold(seed, move |slot| {
slot.take().map(|curr| {
*slot = step(&curr);
curr
})
})
}

View file

@ -17,7 +17,7 @@ pub(crate) struct SyntaxInfo {
} }
impl SyntaxKind { impl SyntaxKind {
pub(crate) fn is_trivia(self) -> bool { pub fn is_trivia(self) -> bool {
match self { match self {
WHITESPACE | COMMENT | DOC_COMMENT => true, WHITESPACE | COMMENT | DOC_COMMENT => true,
_ => false, _ => false,

View file

@ -101,6 +101,17 @@ impl<R: TreeRoot> SyntaxNode<R> {
}) })
} }
pub fn prev_sibling(&self) -> Option<SyntaxNode<R>> {
let red = self.red();
let parent = self.parent()?;
let prev_sibling_idx = red.index_in_parent()?.checked_sub(1)?;
let sibling_red = parent.red().get_child(prev_sibling_idx)?;
Some(SyntaxNode {
root: self.root.clone(),
red: sibling_red,
})
}
pub fn is_leaf(&self) -> bool { pub fn is_leaf(&self) -> bool {
self.first_child().is_none() self.first_child().is_none()
} }

View file

@ -1,5 +1,6 @@
extern crate libsyntax2; extern crate libsyntax2;
extern crate difference; #[macro_use]
extern crate assert_eq_text;
use std::{ use std::{
fs, fs,
@ -7,8 +8,6 @@ use std::{
fmt::Write, fmt::Write,
}; };
use difference::Changeset;
#[test] #[test]
fn lexer_tests() { fn lexer_tests() {
dir_tests(&["lexer"], |text| { dir_tests(&["lexer"], |text| {
@ -63,10 +62,26 @@ pub fn dir_tests<F>(paths: &[&str], f: F)
} }
} }
const REWRITE: bool = false;
fn assert_equal_text(expected: &str, actual: &str, path: &Path) { fn assert_equal_text(expected: &str, actual: &str, path: &Path) {
if expected != actual { if expected == actual {
print_difference(expected, actual, path) return;
} }
let dir = project_dir();
let path = path.strip_prefix(&dir).unwrap_or_else(|_| path);
if expected.trim() == actual.trim() {
println!("whitespace difference, rewriting");
println!("file: {}\n", path.display());
fs::write(path, actual).unwrap();
return;
}
if REWRITE {
println!("rewriting {}", path.display());
fs::write(path, actual).unwrap();
return;
}
assert_eq_text!(expected, actual, "file: {}", path.display());
} }
fn collect_tests(paths: &[&str]) -> Vec<PathBuf> { fn collect_tests(paths: &[&str]) -> Vec<PathBuf> {
@ -92,29 +107,6 @@ fn test_from_dir(dir: &Path) -> Vec<PathBuf> {
acc acc
} }
const REWRITE: bool = false;
fn print_difference(expected: &str, actual: &str, path: &Path) {
let dir = project_dir();
let path = path.strip_prefix(&dir).unwrap_or_else(|_| path);
if expected.trim() == actual.trim() {
println!("whitespace difference, rewriting");
println!("file: {}\n", path.display());
fs::write(path, actual).unwrap();
return;
}
if REWRITE {
println!("rewriting {}", path.display());
fs::write(path, actual).unwrap();
return;
}
let changeset = Changeset::new(actual, expected, "\n");
println!("Expected:\n{}\n\nActual:\n{}\n", expected, actual);
print!("{}", changeset);
println!("file: {}\n", path.display());
panic!("Comparison failed")
}
fn project_dir() -> PathBuf { fn project_dir() -> PathBuf {
let dir = env!("CARGO_MANIFEST_DIR"); let dir = env!("CARGO_MANIFEST_DIR");
PathBuf::from(dir) PathBuf::from(dir)