2018-08-28 10:22:52 -05:00
|
|
|
extern crate relative_path;
|
2018-09-16 04:54:24 -05:00
|
|
|
extern crate ra_analysis;
|
2018-10-11 13:07:44 -05:00
|
|
|
extern crate rustc_hash;
|
2018-10-09 09:08:17 -05:00
|
|
|
extern crate ra_editor;
|
|
|
|
extern crate ra_syntax;
|
2018-08-25 06:26:34 -05:00
|
|
|
extern crate test_utils;
|
2018-08-17 08:04:34 -05:00
|
|
|
|
2018-08-31 11:14:08 -05:00
|
|
|
use std::{
|
2018-09-10 04:57:40 -05:00
|
|
|
sync::Arc,
|
2018-08-31 11:14:08 -05:00
|
|
|
};
|
2018-08-17 08:04:34 -05:00
|
|
|
|
2018-10-11 13:07:44 -05:00
|
|
|
use rustc_hash::FxHashMap;
|
2018-09-10 04:57:40 -05:00
|
|
|
use relative_path::{RelativePath, RelativePathBuf};
|
2018-10-09 09:08:17 -05:00
|
|
|
use ra_analysis::{Analysis, AnalysisHost, FileId, FileResolver, JobHandle, CrateGraph, CrateId, FnDescriptor};
|
|
|
|
use test_utils::{assert_eq_dbg, extract_offset};
|
2018-08-17 08:04:34 -05:00
|
|
|
|
2018-09-10 04:57:40 -05:00
|
|
|
#[derive(Debug)]
|
|
|
|
struct FileMap(Vec<(FileId, RelativePathBuf)>);
|
|
|
|
|
2018-08-28 10:22:52 -05:00
|
|
|
impl FileMap {
|
2018-09-05 12:22:52 -05:00
|
|
|
fn iter<'a>(&'a self) -> impl Iterator<Item=(FileId, &'a RelativePath)> + 'a {
|
2018-09-10 04:57:40 -05:00
|
|
|
self.0.iter().map(|(id, path)| (*id, path.as_relative_path()))
|
2018-09-05 12:22:52 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn path(&self, id: FileId) -> &RelativePath {
|
|
|
|
self.iter()
|
|
|
|
.find(|&(it, _)| it == id)
|
2018-08-28 10:22:52 -05:00
|
|
|
.unwrap()
|
2018-09-05 12:22:52 -05:00
|
|
|
.1
|
2018-08-28 10:22:52 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FileResolver for FileMap {
|
|
|
|
fn file_stem(&self, id: FileId) -> String {
|
2018-09-05 12:22:52 -05:00
|
|
|
self.path(id).file_stem().unwrap().to_string()
|
2018-08-28 10:22:52 -05:00
|
|
|
}
|
|
|
|
fn resolve(&self, id: FileId, rel: &RelativePath) -> Option<FileId> {
|
2018-09-05 12:22:52 -05:00
|
|
|
let path = self.path(id).join(rel).normalize();
|
|
|
|
let id = self.iter().find(|&(_, p)| path == p)?.0;
|
|
|
|
Some(id)
|
2018-08-28 10:22:52 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-09 09:08:17 -05:00
|
|
|
fn analysis_host(files: &[(&str, &str)]) -> AnalysisHost {
|
2018-09-10 12:14:31 -05: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
|
|
|
|
}
|
|
|
|
|
2018-10-09 09:08:17 -05:00
|
|
|
fn analysis(files: &[(&str, &str)]) -> Analysis {
|
2018-09-10 12:14:31 -05:00
|
|
|
analysis_host(files).analysis()
|
|
|
|
}
|
2018-08-17 08:04:34 -05:00
|
|
|
|
2018-10-09 09:08:17 -05: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 08:04:34 -05:00
|
|
|
#[test]
|
|
|
|
fn test_resolve_module() {
|
2018-09-10 04:57:40 -05:00
|
|
|
let snap = analysis(&[
|
|
|
|
("/lib.rs", "mod foo;"),
|
|
|
|
("/foo.rs", "")
|
|
|
|
]);
|
2018-08-31 04:13:02 -05:00
|
|
|
let (_handle, token) = JobHandle::new();
|
|
|
|
let symbols = snap.approximately_resolve_symbol(FileId(1), 4.into(), &token);
|
2018-08-17 08:04:34 -05:00
|
|
|
assert_eq_dbg(
|
|
|
|
r#"[(FileId(2), FileSymbol { name: "foo", node_range: [0; 0), kind: MODULE })]"#,
|
|
|
|
&symbols,
|
|
|
|
);
|
|
|
|
|
2018-09-10 04:57:40 -05:00
|
|
|
let snap = analysis(&[
|
|
|
|
("/lib.rs", "mod foo;"),
|
|
|
|
("/foo/mod.rs", "")
|
|
|
|
]);
|
2018-08-31 04:13:02 -05:00
|
|
|
let symbols = snap.approximately_resolve_symbol(FileId(1), 4.into(), &token);
|
2018-08-17 08:04:34 -05:00
|
|
|
assert_eq_dbg(
|
|
|
|
r#"[(FileId(2), FileSymbol { name: "foo", node_range: [0; 0), kind: MODULE })]"#,
|
|
|
|
&symbols,
|
|
|
|
);
|
|
|
|
}
|
2018-08-21 10:30:10 -05:00
|
|
|
|
2018-08-27 12:58:38 -05:00
|
|
|
#[test]
|
|
|
|
fn test_unresolved_module_diagnostic() {
|
2018-09-10 04:57:40 -05:00
|
|
|
let snap = analysis(&[("/lib.rs", "mod foo;")]);
|
2018-08-29 10:35:28 -05:00
|
|
|
let diagnostics = snap.diagnostics(FileId(1));
|
2018-08-27 12:58:38 -05:00
|
|
|
assert_eq_dbg(
|
2018-08-29 10:35:28 -05: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 12:58:38 -05:00
|
|
|
&diagnostics,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-08-28 12:29:36 -05:00
|
|
|
#[test]
|
|
|
|
fn test_unresolved_module_diagnostic_no_diag_for_inline_mode() {
|
2018-09-10 04:57:40 -05:00
|
|
|
let snap = analysis(&[("/lib.rs", "mod foo {}")]);
|
2018-08-29 10:35:28 -05:00
|
|
|
let diagnostics = snap.diagnostics(FileId(1));
|
2018-08-28 12:29:36 -05:00
|
|
|
assert_eq_dbg(
|
|
|
|
r#"[]"#,
|
|
|
|
&diagnostics,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-08-21 10:30:10 -05:00
|
|
|
#[test]
|
|
|
|
fn test_resolve_parent_module() {
|
2018-09-10 04:57:40 -05:00
|
|
|
let snap = analysis(&[
|
|
|
|
("/lib.rs", "mod foo;"),
|
|
|
|
("/foo.rs", ""),
|
|
|
|
]);
|
2018-08-21 10:30:10 -05: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 11:14:08 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_resolve_crate_root() {
|
2018-09-10 04:57:40 -05:00
|
|
|
let mut host = analysis_host(&[
|
|
|
|
("/lib.rs", "mod foo;"),
|
|
|
|
("/foo.rs", ""),
|
|
|
|
]);
|
|
|
|
let snap = host.analysis();
|
2018-09-02 08:36:03 -05:00
|
|
|
assert!(snap.crate_for(FileId(2)).is_empty());
|
2018-08-31 11:14:08 -05:00
|
|
|
|
|
|
|
let crate_graph = CrateGraph {
|
|
|
|
crate_roots: {
|
2018-10-11 13:07:44 -05:00
|
|
|
let mut m = FxHashMap::default();
|
2018-08-31 11:14:08 -05:00
|
|
|
m.insert(CrateId(1), FileId(1));
|
|
|
|
m
|
|
|
|
},
|
|
|
|
};
|
2018-09-10 04:57:40 -05:00
|
|
|
host.set_crate_graph(crate_graph);
|
|
|
|
let snap = host.analysis();
|
2018-08-31 11:14:08 -05:00
|
|
|
|
|
|
|
assert_eq!(
|
2018-09-02 08:36:03 -05:00
|
|
|
snap.crate_for(FileId(2)),
|
2018-08-31 11:14:08 -05:00
|
|
|
vec![CrateId(1)],
|
|
|
|
);
|
|
|
|
}
|
2018-10-09 09:08:17 -05: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, ); }"#);
|
|
|
|
|
2018-10-12 06:54:57 -05:00
|
|
|
assert_eq!(desc.name, "foo".to_string());
|
2018-10-09 09:08:17 -05:00
|
|
|
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, <|>); }"#);
|
|
|
|
|
2018-10-12 06:54:57 -05:00
|
|
|
assert_eq!(desc.name, "foo".to_string());
|
2018-10-09 09:08:17 -05:00
|
|
|
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(<|>);}"#);
|
|
|
|
|
2018-10-12 06:54:57 -05:00
|
|
|
assert_eq!(desc.name, "new".to_string());
|
2018-10-09 09:08:17 -05:00
|
|
|
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(<|>);
|
|
|
|
}"#);
|
|
|
|
|
2018-10-12 06:54:57 -05:00
|
|
|
assert_eq!(desc.name, "do_it".to_string());
|
2018-10-09 09:08:17 -05:00
|
|
|
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(<|>);
|
|
|
|
}"#);
|
|
|
|
|
2018-10-12 06:54:57 -05:00
|
|
|
assert_eq!(desc.name, "do_it".to_string());
|
2018-10-09 09:08:17 -05:00
|
|
|
assert_eq!(desc.params, vec!["&self".to_string(), "x".to_string()]);
|
|
|
|
assert_eq!(desc.ret_type, None);
|
|
|
|
assert_eq!(param, Some(1));
|
|
|
|
}
|