mirror of
https://github.com/rust-lang/rust-analyzer
synced 2024-12-26 13:03:31 +00:00
flip comma
This commit is contained in:
parent
56aa6e20e0
commit
66be735aa9
12 changed files with 273 additions and 57 deletions
7
crates/assert_eq_text/Cargo.toml
Normal file
7
crates/assert_eq_text/Cargo.toml
Normal 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"
|
25
crates/assert_eq_text/src/lib.rs
Normal file
25
crates/assert_eq_text/src/lib.rs
Normal 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");
|
||||||
|
}
|
||||||
|
}};
|
||||||
|
}
|
|
@ -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" }
|
||||||
|
|
33
crates/libeditor/src/code_actions.rs
Normal file
33
crates/libeditor/src/code_actions.rs
Normal 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())
|
||||||
|
}
|
||||||
|
|
||||||
|
|
93
crates/libeditor/src/edit.rs
Normal file
93
crates/libeditor/src/edit.rs
Normal 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())?,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -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)]
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
|
@ -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" }
|
||||||
|
|
|
@ -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
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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()
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
||||||
|
|
Loading…
Reference in a new issue