rust-analyzer/crates/ra_analysis/tests/tests.rs

269 lines
6.9 KiB
Rust
Raw Normal View History

2018-09-16 09:54:24 +00:00
extern crate ra_analysis;
extern crate ra_editor;
extern crate ra_syntax;
extern crate relative_path;
extern crate rustc_hash;
2018-08-25 11:26:34 +00:00
extern crate test_utils;
2018-08-17 13:04:34 +00:00
use std::sync::Arc;
2018-08-17 13:04:34 +00:00
use ra_analysis::{
Analysis, AnalysisHost, CrateGraph, CrateId, FileId, FileResolver, FnDescriptor, JobHandle,
};
use ra_syntax::TextRange;
2018-09-10 09:57:40 +00:00
use relative_path::{RelativePath, RelativePathBuf};
use rustc_hash::FxHashMap;
use test_utils::{assert_eq_dbg, extract_offset};
2018-08-17 13:04:34 +00:00
2018-09-10 09:57:40 +00:00
#[derive(Debug)]
struct FileMap(Vec<(FileId, RelativePathBuf)>);
2018-08-28 15:22:52 +00:00
impl FileMap {
fn iter<'a>(&'a self) -> impl Iterator<Item = (FileId, &'a RelativePath)> + 'a {
self.0
.iter()
.map(|(id, path)| (*id, path.as_relative_path()))
2018-09-05 17:22:52 +00:00
}
fn path(&self, id: FileId) -> &RelativePath {
self.iter().find(|&(it, _)| it == id).unwrap().1
2018-08-28 15:22:52 +00:00
}
}
impl FileResolver for FileMap {
fn file_stem(&self, id: FileId) -> String {
2018-09-05 17:22:52 +00:00
self.path(id).file_stem().unwrap().to_string()
2018-08-28 15:22:52 +00:00
}
fn resolve(&self, id: FileId, rel: &RelativePath) -> Option<FileId> {
2018-09-05 17:22:52 +00:00
let path = self.path(id).join(rel).normalize();
let id = self.iter().find(|&(_, p)| path == p)?.0;
Some(id)
2018-08-28 15:22:52 +00:00
}
}
fn analysis_host(files: &[(&str, &str)]) -> AnalysisHost {
2018-09-10 17:14:31 +00:00
let mut host = AnalysisHost::new();
let mut file_map = Vec::new();
for (id, &(path, contents)) in files.iter().enumerate() {
let file_id = FileId((id + 1) as u32);
assert!(path.starts_with('/'));
let path = RelativePathBuf::from_path(&path[1..]).unwrap();
host.change_file(file_id, Some(contents.to_string()));
file_map.push((file_id, path));
}
host.set_file_resolver(Arc::new(FileMap(file_map)));
host
}
fn analysis(files: &[(&str, &str)]) -> Analysis {
2018-09-10 17:14:31 +00:00
analysis_host(files).analysis()
}
2018-08-17 13:04:34 +00:00
fn get_signature(text: &str) -> (FnDescriptor, Option<usize>) {
let (offset, code) = extract_offset(text);
let code = code.as_str();
let (_handle, token) = JobHandle::new();
let snap = analysis(&[("/lib.rs", code)]);
snap.resolve_callable(FileId(1), offset, &token).unwrap()
}
2018-08-17 13:04:34 +00:00
#[test]
fn test_resolve_module() {
let snap = analysis(&[("/lib.rs", "mod foo;"), ("/foo.rs", "")]);
2018-08-31 09:13:02 +00:00
let (_handle, token) = JobHandle::new();
let symbols = snap.approximately_resolve_symbol(FileId(1), 4.into(), &token);
2018-08-17 13:04:34 +00:00
assert_eq_dbg(
r#"[(FileId(2), FileSymbol { name: "foo", node_range: [0; 0), kind: MODULE })]"#,
&symbols,
);
let snap = analysis(&[("/lib.rs", "mod foo;"), ("/foo/mod.rs", "")]);
2018-08-31 09:13:02 +00:00
let symbols = snap.approximately_resolve_symbol(FileId(1), 4.into(), &token);
2018-08-17 13:04:34 +00:00
assert_eq_dbg(
r#"[(FileId(2), FileSymbol { name: "foo", node_range: [0; 0), kind: MODULE })]"#,
&symbols,
);
}
2018-08-21 15:30:10 +00:00
2018-08-27 17:58:38 +00:00
#[test]
fn test_unresolved_module_diagnostic() {
2018-09-10 09:57:40 +00:00
let snap = analysis(&[("/lib.rs", "mod foo;")]);
2018-08-29 15:35:28 +00:00
let diagnostics = snap.diagnostics(FileId(1));
2018-08-27 17:58:38 +00:00
assert_eq_dbg(
2018-08-29 15:35:28 +00:00
r#"[Diagnostic {
message: "unresolved module",
range: [4; 7),
fix: Some(SourceChange {
label: "create module",
source_file_edits: [],
file_system_edits: [CreateFile { anchor: FileId(1), path: "../foo.rs" }],
cursor_position: None }) }]"#,
2018-08-27 17:58:38 +00:00
&diagnostics,
);
}
2018-08-28 17:29:36 +00:00
#[test]
fn test_unresolved_module_diagnostic_no_diag_for_inline_mode() {
2018-09-10 09:57:40 +00:00
let snap = analysis(&[("/lib.rs", "mod foo {}")]);
2018-08-29 15:35:28 +00:00
let diagnostics = snap.diagnostics(FileId(1));
assert_eq_dbg(r#"[]"#, &diagnostics);
2018-08-28 17:29:36 +00:00
}
2018-08-21 15:30:10 +00:00
#[test]
fn test_resolve_parent_module() {
let snap = analysis(&[("/lib.rs", "mod foo;"), ("/foo.rs", "")]);
2018-08-21 15:30:10 +00:00
let symbols = snap.parent_module(FileId(2));
assert_eq_dbg(
r#"[(FileId(1), FileSymbol { name: "foo", node_range: [0; 8), kind: MODULE })]"#,
&symbols,
);
}
2018-08-31 16:14:08 +00:00
#[test]
fn test_resolve_crate_root() {
let mut host = analysis_host(&[("/lib.rs", "mod foo;"), ("/foo.rs", "")]);
2018-09-10 09:57:40 +00:00
let snap = host.analysis();
2018-09-02 13:36:03 +00:00
assert!(snap.crate_for(FileId(2)).is_empty());
2018-08-31 16:14:08 +00:00
let crate_graph = CrateGraph {
crate_roots: {
let mut m = FxHashMap::default();
2018-08-31 16:14:08 +00:00
m.insert(CrateId(1), FileId(1));
m
},
};
2018-09-10 09:57:40 +00:00
host.set_crate_graph(crate_graph);
let snap = host.analysis();
2018-08-31 16:14:08 +00:00
assert_eq!(snap.crate_for(FileId(2)), vec![CrateId(1)],);
2018-08-31 16:14:08 +00:00
}
#[test]
fn test_fn_signature_two_args_first() {
let (desc, param) = get_signature(
r#"fn foo(x: u32, y: u32) -> u32 {x + y}
fn bar() { foo(<|>3, ); }"#,
);
assert_eq!(desc.name, "foo".to_string());
assert_eq!(desc.params, vec!("x".to_string(), "y".to_string()));
assert_eq!(desc.ret_type, Some("-> u32".into()));
assert_eq!(param, Some(0));
}
#[test]
fn test_fn_signature_two_args_second() {
let (desc, param) = get_signature(
r#"fn foo(x: u32, y: u32) -> u32 {x + y}
fn bar() { foo(3, <|>); }"#,
);
assert_eq!(desc.name, "foo".to_string());
assert_eq!(desc.params, vec!("x".to_string(), "y".to_string()));
assert_eq!(desc.ret_type, Some("-> u32".into()));
assert_eq!(param, Some(1));
}
#[test]
fn test_fn_signature_for_impl() {
let (desc, param) = get_signature(
r#"struct F; impl F { pub fn new() { F{}} }
fn bar() {let _ : F = F::new(<|>);}"#,
);
assert_eq!(desc.name, "new".to_string());
assert_eq!(desc.params, Vec::<String>::new());
assert_eq!(desc.ret_type, None);
assert_eq!(param, None);
}
#[test]
fn test_fn_signature_for_method_self() {
let (desc, param) = get_signature(
r#"struct F;
impl F {
pub fn new() -> F{
F{}
}
pub fn do_it(&self) {}
}
fn bar() {
let f : F = F::new();
f.do_it(<|>);
}"#,
);
assert_eq!(desc.name, "do_it".to_string());
assert_eq!(desc.params, vec!["&self".to_string()]);
assert_eq!(desc.ret_type, None);
assert_eq!(param, None);
}
#[test]
fn test_fn_signature_for_method_with_arg() {
let (desc, param) = get_signature(
r#"struct F;
impl F {
pub fn new() -> F{
F{}
}
pub fn do_it(&self, x: i32) {}
}
fn bar() {
let f : F = F::new();
f.do_it(<|>);
}"#,
);
assert_eq!(desc.name, "do_it".to_string());
assert_eq!(desc.params, vec!["&self".to_string(), "x".to_string()]);
assert_eq!(desc.ret_type, None);
assert_eq!(param, Some(1));
2018-10-15 18:54:12 +00:00
}
fn get_all_refs(text: &str) -> Vec<(FileId, TextRange)> {
let (offset, code) = extract_offset(text);
let code = code.as_str();
let (_handle, token) = JobHandle::new();
let snap = analysis(&[("/lib.rs", code)]);
snap.find_all_refs(FileId(1), offset, &token)
}
#[test]
fn test_find_all_refs_for_local() {
let code = r#"
fn main() {
let mut i = 1;
let j = 1;
i = i<|> + j;
{
i = 0;
}
i = 5;
}"#;
let refs = get_all_refs(code);
assert_eq!(refs.len(), 5);
}
#[test]
fn test_find_all_refs_for_param_inside() {
let code = r#"
fn foo(i : u32) -> u32 {
i<|>
}"#;
let refs = get_all_refs(code);
assert_eq!(refs.len(), 2);
}