rust-analyzer/crates/libeditor/tests/test.rs

311 lines
7.8 KiB
Rust
Raw Normal View History

2018-08-09 09:47:34 +00:00
extern crate libeditor;
2018-08-12 15:50:16 +00:00
extern crate libsyntax2;
#[macro_use]
2018-08-25 11:26:34 +00:00
extern crate test_utils;
2018-08-09 09:47:34 +00:00
2018-08-25 11:30:54 +00:00
use test_utils::{assert_eq_dbg, add_cursor, extract_offset, extract_range};
2018-08-25 08:47:24 +00:00
use libsyntax2::{File, TextUnit, TextRange};
2018-08-12 15:50:16 +00:00
use libeditor::{
2018-08-25 08:47:24 +00:00
ActionResult,
2018-08-15 20:24:20 +00:00
highlight, runnables, extend_selection, file_structure,
2018-08-22 15:05:43 +00:00
flip_comma, add_derive, add_impl, matching_brace,
2018-08-26 06:12:18 +00:00
join_lines, scope_completion,
2018-08-12 15:50:16 +00:00
};
2018-08-09 09:47:34 +00:00
#[test]
fn test_extend_selection() {
2018-08-22 08:56:36 +00:00
fn do_check(before: &str, afters: &[&str]) {
2018-08-25 11:30:54 +00:00
let (cursor, before) = extract_offset(before);
2018-08-22 08:56:36 +00:00
let file = file(&before);
let mut range = TextRange::offset_len(cursor, 0.into());
for &after in afters {
range = extend_selection(&file, range)
.unwrap();
let actual = &before[range];
assert_eq!(after, actual);
}
}
do_check(
r#"fn foo() { <|>1 + 1 }"#,
&["1", "1 + 1", "{ 1 + 1 }"],
);
do_check(
r#"
impl S {
<|> fn foo() {
}
}"#,
&["fn foo() {\n\n }"]
);
2018-08-26 07:43:03 +00:00
do_check(
r#"
fn bar(){}
// fn foo() {
// 1 + <|>1
// }
// fn foo(){}
"#,
&["// 1 + 1", "// fn foo() {\n// 1 + 1\n// }"]
);
2018-08-09 09:47:34 +00:00
}
#[test]
fn test_highlighting() {
let file = file(r#"
// comment
fn main() {}
println!("Hello, {}!", 92);
"#);
2018-08-10 14:49:45 +00:00
let hls = highlight(&file);
2018-08-17 13:04:34 +00:00
assert_eq_dbg(
2018-08-09 09:47:34 +00:00
r#"[HighlightedRange { range: [1; 11), tag: "comment" },
HighlightedRange { range: [12; 14), tag: "keyword" },
HighlightedRange { range: [15; 19), tag: "function" },
HighlightedRange { range: [29; 36), tag: "text" },
HighlightedRange { range: [38; 50), tag: "string" },
2018-08-12 15:50:16 +00:00
HighlightedRange { range: [52; 54), tag: "literal" }]"#,
&hls,
2018-08-09 09:47:34 +00:00
);
}
2018-08-09 13:03:21 +00:00
#[test]
fn test_runnables() {
let file = file(r#"
fn main() {}
#[test]
fn test_foo() {}
#[test]
#[ignore]
fn test_foo() {}
"#);
2018-08-10 14:49:45 +00:00
let runnables = runnables(&file);
2018-08-17 13:04:34 +00:00
assert_eq_dbg(
2018-08-09 13:03:21 +00:00
r#"[Runnable { range: [1; 13), kind: Bin },
Runnable { range: [15; 39), kind: Test { name: "test_foo" } },
Runnable { range: [41; 75), kind: Test { name: "test_foo" } }]"#,
2018-08-12 15:50:16 +00:00
&runnables,
2018-08-09 13:03:21 +00:00
)
}
2018-08-11 09:28:59 +00:00
#[test]
2018-08-14 09:38:20 +00:00
fn test_file_structure() {
2018-08-11 09:28:59 +00:00
let file = file(r#"
struct Foo {
x: i32
}
mod m {
fn bar() {}
}
enum E { X, Y(i32) }
2018-08-11 13:20:37 +00:00
type T = ();
static S: i32 = 92;
const C: i32 = 92;
2018-08-14 08:20:09 +00:00
impl E {}
impl fmt::Debug for E {}
2018-08-11 09:28:59 +00:00
"#);
2018-08-14 08:20:09 +00:00
let symbols = file_structure(&file);
2018-08-17 13:04:34 +00:00
assert_eq_dbg(
2018-08-14 08:20:09 +00:00
r#"[StructureNode { parent: None, label: "Foo", navigation_range: [8; 11), node_range: [1; 26), kind: STRUCT_DEF },
2018-08-24 16:27:30 +00:00
StructureNode { parent: Some(0), label: "x", navigation_range: [18; 19), node_range: [18; 24), kind: NAMED_FIELD_DEF },
2018-08-14 11:45:56 +00:00
StructureNode { parent: None, label: "m", navigation_range: [32; 33), node_range: [28; 53), kind: MODULE },
StructureNode { parent: Some(2), label: "bar", navigation_range: [43; 46), node_range: [40; 51), kind: FN_DEF },
StructureNode { parent: None, label: "E", navigation_range: [60; 61), node_range: [55; 75), kind: ENUM_DEF },
StructureNode { parent: None, label: "T", navigation_range: [81; 82), node_range: [76; 88), kind: TYPE_DEF },
StructureNode { parent: None, label: "S", navigation_range: [96; 97), node_range: [89; 108), kind: STATIC_DEF },
StructureNode { parent: None, label: "C", navigation_range: [115; 116), node_range: [109; 127), kind: CONST_DEF },
StructureNode { parent: None, label: "impl E", navigation_range: [134; 135), node_range: [129; 138), kind: IMPL_ITEM },
StructureNode { parent: None, label: "impl fmt::Debug for E", navigation_range: [160; 161), node_range: [140; 164), kind: IMPL_ITEM }]"#,
2018-08-12 15:50:16 +00:00
&symbols,
)
}
#[test]
fn test_swap_comma() {
2018-08-15 20:24:20 +00:00
check_action(
2018-08-12 15:50:16 +00:00
"fn foo(x: i32,<|> y: Result<(), ()>) {}",
2018-08-15 20:24:20 +00:00
"fn foo(y: Result<(), ()>,<|> x: i32) {}",
|file, off| flip_comma(file, off).map(|f| f()),
)
}
#[test]
fn test_add_derive() {
check_action(
"struct Foo { a: i32, <|>}",
"#[derive(<|>)]\nstruct Foo { a: i32, }",
|file, off| add_derive(file, off).map(|f| f()),
2018-08-16 10:11:20 +00:00
);
2018-08-17 16:54:08 +00:00
check_action(
"struct Foo { <|> a: i32, }",
"#[derive(<|>)]\nstruct Foo { a: i32, }",
|file, off| add_derive(file, off).map(|f| f()),
);
2018-08-16 10:11:20 +00:00
check_action(
"#[derive(Clone)]\nstruct Foo { a: i32<|>, }",
"#[derive(Clone<|>)]\nstruct Foo { a: i32, }",
|file, off| add_derive(file, off).map(|f| f()),
);
2018-08-11 09:28:59 +00:00
}
2018-08-22 15:05:43 +00:00
#[test]
fn test_add_impl() {
check_action(
"struct Foo {<|>}\n",
"struct Foo {}\n\nimpl Foo {\n<|>\n}\n",
|file, off| add_impl(file, off).map(|f| f()),
);
2018-08-22 16:02:37 +00:00
check_action(
"struct Foo<T: Clone> {<|>}",
"struct Foo<T: Clone> {}\n\nimpl<T: Clone> Foo<T> {\n<|>\n}",
|file, off| add_impl(file, off).map(|f| f()),
);
2018-08-22 15:05:43 +00:00
}
2018-08-15 21:23:22 +00:00
#[test]
fn test_matching_brace() {
fn do_check(before: &str, after: &str) {
2018-08-25 11:30:54 +00:00
let (pos, before) = extract_offset(before);
2018-08-15 21:23:22 +00:00
let file = file(&before);
let new_pos = match matching_brace(&file, pos) {
None => pos,
Some(pos) => pos,
};
let actual = add_cursor(&before, new_pos);
assert_eq_text!(after, &actual);
}
do_check(
"struct Foo { a: i32, }<|>",
"struct Foo <|>{ a: i32, }",
);
}
2018-08-23 17:55:23 +00:00
#[test]
fn test_join_lines_cursor() {
fn do_check(before: &str, after: &str) {
check_action(before, after, |file, offset| {
let range = TextRange::offset_len(offset, 0.into());
let res = join_lines(file, range);
Some(res)
})
}
do_check(r"
fn foo() {
<|>foo(1,
)
}
", r"
fn foo() {
2018-08-23 18:38:25 +00:00
<|>foo(1)
2018-08-23 17:55:23 +00:00
}
2018-08-25 10:42:40 +00:00
");
do_check(r"
pub fn reparse(&self, edit: &AtomEdit) -> File {
<|>self.incremental_reparse(edit).unwrap_or_else(|| {
self.full_reparse(edit)
})
}
", r"
pub fn reparse(&self, edit: &AtomEdit) -> File {
<|>self.incremental_reparse(edit).unwrap_or_else(|| self.full_reparse(edit))
}
2018-08-23 17:55:23 +00:00
");
}
#[test]
fn test_join_lines_selection() {
fn do_check(before: &str, after: &str) {
2018-08-25 11:30:54 +00:00
let (sel, before) = extract_range(&before);
2018-08-23 17:55:23 +00:00
let file = file(&before);
let result = join_lines(&file, sel);
let actual = result.edit.apply(&before);
assert_eq_text!(after, &actual);
}
do_check(r"
fn foo() {
<|>foo(1,
2,
3,
<|>)
}
", r"
fn foo() {
2018-08-23 18:38:25 +00:00
foo(1, 2, 3)
2018-08-23 17:55:23 +00:00
}
2018-08-23 21:13:16 +00:00
");
do_check(r"
struct Foo <|>{
f: u32,
}<|>
", r"
struct Foo { f: u32 }
2018-08-23 17:55:23 +00:00
");
}
2018-08-26 09:09:28 +00:00
#[test]
fn test_completion() {
fn do_check(code: &str, expected_completions: &str) {
let (off, code) = extract_offset(&code);
let file = file(&code);
let completions = scope_completion(&file, off).unwrap();
assert_eq_dbg(expected_completions, &completions);
}
2018-08-26 06:12:18 +00:00
2018-08-26 09:09:28 +00:00
do_check(r"
fn quux(x: i32) {
let y = 92;
2018-08-27 07:01:31 +00:00
1 + <|>;
let z = ();
2018-08-26 09:09:28 +00:00
}
", r#"[CompletionItem { name: "y" },
CompletionItem { name: "x" }]"#);
2018-08-27 09:22:09 +00:00
do_check(r"
fn quux() {
if let Some(x) = foo() {
let y = 92;
};
if let Some(a) = bar() {
let b = 62;
1 + <|>
}
}
", r#"[CompletionItem { name: "b" },
CompletionItem { name: "a" }]"#);
2018-08-26 09:09:28 +00:00
}
2018-08-26 06:12:18 +00:00
2018-08-25 08:44:58 +00:00
fn file(text: &str) -> File {
File::parse(text)
2018-08-09 09:47:34 +00:00
}
2018-08-25 08:44:58 +00:00
fn check_action<F: Fn(&File, TextUnit) -> Option<ActionResult>>(
2018-08-12 15:50:16 +00:00
before: &str,
after: &str,
2018-08-15 20:24:20 +00:00
f: F,
2018-08-12 15:50:16 +00:00
) {
2018-08-25 11:30:54 +00:00
let (before_cursor_pos, before) = extract_offset(before);
2018-08-15 20:24:20 +00:00
let file = file(&before);
let result = f(&file, before_cursor_pos).expect("code action is not applicable");
let actual = result.edit.apply(&before);
2018-08-15 21:23:22 +00:00
let actual_cursor_pos = match result.cursor_position {
2018-08-22 09:58:34 +00:00
None => result.edit.apply_to_offset(before_cursor_pos).unwrap(),
Some(off) => off,
2018-08-15 21:23:22 +00:00
};
let actual = add_cursor(&actual, actual_cursor_pos);
assert_eq_text!(after, &actual);
2018-08-15 20:24:20 +00:00
}