2018-12-09 09:48:55 +00:00
|
|
|
use std::sync::Arc;
|
|
|
|
|
|
|
|
use salsa::Database;
|
2018-12-09 10:49:54 +00:00
|
|
|
use ra_db::{FilesDatabase, CrateGraph};
|
2018-12-19 09:40:41 +00:00
|
|
|
use relative_path::RelativePath;
|
2018-12-27 18:02:08 +00:00
|
|
|
use test_utils::assert_eq_text;
|
2018-12-09 09:48:55 +00:00
|
|
|
|
|
|
|
use crate::{
|
|
|
|
self as hir,
|
|
|
|
db::HirDatabase,
|
|
|
|
mock::MockDatabase,
|
|
|
|
};
|
|
|
|
|
|
|
|
fn item_map(fixture: &str) -> (Arc<hir::ItemMap>, hir::ModuleId) {
|
|
|
|
let (db, pos) = MockDatabase::with_position(fixture);
|
|
|
|
let source_root = db.file_source_root(pos.file_id);
|
|
|
|
let module = hir::source_binder::module_from_position(&db, pos)
|
|
|
|
.unwrap()
|
|
|
|
.unwrap();
|
2019-01-06 10:41:12 +00:00
|
|
|
let module_id = module.def_id.loc(&db).module_id;
|
2018-12-09 09:48:55 +00:00
|
|
|
(db.item_map(source_root).unwrap(), module_id)
|
|
|
|
}
|
|
|
|
|
2018-12-27 18:02:08 +00:00
|
|
|
fn check_module_item_map(map: &hir::ItemMap, module_id: hir::ModuleId, expected: &str) {
|
|
|
|
let mut lines = map.per_module[&module_id]
|
|
|
|
.items
|
|
|
|
.iter()
|
|
|
|
.map(|(name, res)| format!("{}: {}", name, dump_resolution(res)))
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
lines.sort();
|
|
|
|
let actual = lines.join("\n");
|
|
|
|
let expected = expected
|
|
|
|
.trim()
|
|
|
|
.lines()
|
|
|
|
.map(|it| it.trim())
|
|
|
|
.collect::<Vec<_>>()
|
|
|
|
.join("\n");
|
|
|
|
assert_eq_text!(&actual, &expected);
|
|
|
|
|
|
|
|
fn dump_resolution(resolution: &hir::Resolution) -> &'static str {
|
|
|
|
match (
|
|
|
|
resolution.def_id.types.is_some(),
|
|
|
|
resolution.def_id.values.is_some(),
|
|
|
|
) {
|
|
|
|
(true, true) => "t v",
|
|
|
|
(true, false) => "t",
|
|
|
|
(false, true) => "v",
|
|
|
|
(false, false) => "_",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-09 09:48:55 +00:00
|
|
|
#[test]
|
2018-12-09 10:49:54 +00:00
|
|
|
fn item_map_smoke_test() {
|
2018-12-09 09:48:55 +00:00
|
|
|
let (item_map, module_id) = item_map(
|
|
|
|
"
|
|
|
|
//- /lib.rs
|
|
|
|
mod foo;
|
|
|
|
|
|
|
|
use crate::foo::bar::Baz;
|
|
|
|
<|>
|
|
|
|
|
|
|
|
//- /foo/mod.rs
|
|
|
|
pub mod bar;
|
|
|
|
|
|
|
|
//- /foo/bar.rs
|
|
|
|
pub struct Baz;
|
|
|
|
",
|
|
|
|
);
|
2018-12-27 18:02:08 +00:00
|
|
|
check_module_item_map(
|
|
|
|
&item_map,
|
|
|
|
module_id,
|
|
|
|
"
|
|
|
|
Baz: t v
|
|
|
|
foo: t
|
|
|
|
",
|
|
|
|
);
|
2018-12-09 09:48:55 +00:00
|
|
|
}
|
|
|
|
|
2019-01-01 19:12:05 +00:00
|
|
|
#[test]
|
|
|
|
fn item_map_contains_items_from_expansions() {
|
|
|
|
let (item_map, module_id) = item_map(
|
|
|
|
"
|
|
|
|
//- /lib.rs
|
|
|
|
mod foo;
|
|
|
|
|
|
|
|
use crate::foo::bar::Baz;
|
|
|
|
<|>
|
|
|
|
|
|
|
|
//- /foo/mod.rs
|
|
|
|
pub mod bar;
|
|
|
|
|
|
|
|
//- /foo/bar.rs
|
|
|
|
salsa::query_group! {
|
|
|
|
trait Baz {}
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
check_module_item_map(
|
|
|
|
&item_map,
|
|
|
|
module_id,
|
|
|
|
"
|
|
|
|
Baz: t
|
|
|
|
foo: t
|
|
|
|
",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-12-21 22:29:59 +00:00
|
|
|
#[test]
|
2018-12-27 18:02:08 +00:00
|
|
|
fn item_map_using_self() {
|
2018-12-21 22:29:59 +00:00
|
|
|
let (item_map, module_id) = item_map(
|
|
|
|
"
|
|
|
|
//- /lib.rs
|
|
|
|
mod foo;
|
|
|
|
use crate::foo::bar::Baz::{self};
|
|
|
|
<|>
|
|
|
|
//- /foo/mod.rs
|
|
|
|
pub mod bar;
|
|
|
|
//- /foo/bar.rs
|
|
|
|
pub struct Baz;
|
|
|
|
",
|
|
|
|
);
|
2018-12-27 18:02:08 +00:00
|
|
|
check_module_item_map(
|
|
|
|
&item_map,
|
|
|
|
module_id,
|
|
|
|
"
|
|
|
|
Baz: t v
|
|
|
|
foo: t
|
|
|
|
",
|
|
|
|
);
|
2018-12-21 22:29:59 +00:00
|
|
|
}
|
|
|
|
|
2018-12-09 10:49:54 +00:00
|
|
|
#[test]
|
|
|
|
fn item_map_across_crates() {
|
2018-12-19 09:40:41 +00:00
|
|
|
let (mut db, sr) = MockDatabase::with_files(
|
2018-12-09 10:49:54 +00:00
|
|
|
"
|
|
|
|
//- /main.rs
|
|
|
|
use test_crate::Baz;
|
|
|
|
|
|
|
|
//- /lib.rs
|
|
|
|
pub struct Baz;
|
|
|
|
",
|
|
|
|
);
|
2018-12-19 09:40:41 +00:00
|
|
|
let main_id = sr.files[RelativePath::new("/main.rs")];
|
|
|
|
let lib_id = sr.files[RelativePath::new("/lib.rs")];
|
2018-12-09 10:49:54 +00:00
|
|
|
|
|
|
|
let mut crate_graph = CrateGraph::default();
|
|
|
|
let main_crate = crate_graph.add_crate_root(main_id);
|
|
|
|
let lib_crate = crate_graph.add_crate_root(lib_id);
|
|
|
|
crate_graph.add_dep(main_crate, "test_crate".into(), lib_crate);
|
|
|
|
|
|
|
|
db.set_crate_graph(crate_graph);
|
|
|
|
|
|
|
|
let source_root = db.file_source_root(main_id);
|
|
|
|
let module = hir::source_binder::module_from_file_id(&db, main_id)
|
|
|
|
.unwrap()
|
|
|
|
.unwrap();
|
2019-01-06 10:41:12 +00:00
|
|
|
let module_id = module.def_id.loc(&db).module_id;
|
2018-12-09 10:49:54 +00:00
|
|
|
let item_map = db.item_map(source_root).unwrap();
|
|
|
|
|
2018-12-27 18:02:08 +00:00
|
|
|
check_module_item_map(
|
|
|
|
&item_map,
|
|
|
|
module_id,
|
|
|
|
"
|
|
|
|
Baz: t v
|
|
|
|
test_crate: t
|
|
|
|
",
|
|
|
|
);
|
2018-12-09 10:49:54 +00:00
|
|
|
}
|
|
|
|
|
2018-12-09 09:48:55 +00:00
|
|
|
#[test]
|
|
|
|
fn typing_inside_a_function_should_not_invalidate_item_map() {
|
2019-01-01 19:12:05 +00:00
|
|
|
let (mut db, pos) = MockDatabase::with_position(
|
|
|
|
"
|
|
|
|
//- /lib.rs
|
|
|
|
mod foo;
|
|
|
|
|
|
|
|
use crate::foo::bar::Baz;
|
|
|
|
|
|
|
|
//- /foo/mod.rs
|
|
|
|
pub mod bar;
|
|
|
|
|
|
|
|
//- /foo/bar.rs
|
|
|
|
<|>
|
|
|
|
salsa::query_group! {
|
|
|
|
trait Baz {
|
|
|
|
fn foo() -> i32 { 1 + 1 }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
let source_root = db.file_source_root(pos.file_id);
|
|
|
|
{
|
|
|
|
let events = db.log_executed(|| {
|
|
|
|
db.item_map(source_root).unwrap();
|
|
|
|
});
|
|
|
|
assert!(format!("{:?}", events).contains("item_map"))
|
|
|
|
}
|
|
|
|
|
|
|
|
let new_text = "
|
|
|
|
salsa::query_group! {
|
|
|
|
trait Baz {
|
|
|
|
fn foo() -> i32 { 92 }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"
|
|
|
|
.to_string();
|
|
|
|
|
|
|
|
db.query_mut(ra_db::FileTextQuery)
|
|
|
|
.set(pos.file_id, Arc::new(new_text));
|
|
|
|
|
|
|
|
{
|
|
|
|
let events = db.log_executed(|| {
|
|
|
|
db.item_map(source_root).unwrap();
|
|
|
|
});
|
|
|
|
assert!(
|
|
|
|
!format!("{:?}", events).contains("item_map"),
|
|
|
|
"{:#?}",
|
|
|
|
events
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn typing_inside_a_function_inside_a_macro_should_not_invalidate_item_map() {
|
2018-12-09 09:48:55 +00:00
|
|
|
let (mut db, pos) = MockDatabase::with_position(
|
|
|
|
"
|
|
|
|
//- /lib.rs
|
|
|
|
mod foo;<|>
|
|
|
|
|
|
|
|
use crate::foo::bar::Baz;
|
|
|
|
|
|
|
|
fn foo() -> i32 {
|
|
|
|
1 + 1
|
|
|
|
}
|
|
|
|
//- /foo/mod.rs
|
|
|
|
pub mod bar;
|
|
|
|
|
|
|
|
//- /foo/bar.rs
|
|
|
|
pub struct Baz;
|
|
|
|
",
|
|
|
|
);
|
|
|
|
let source_root = db.file_source_root(pos.file_id);
|
|
|
|
{
|
|
|
|
let events = db.log_executed(|| {
|
|
|
|
db.item_map(source_root).unwrap();
|
|
|
|
});
|
|
|
|
assert!(format!("{:?}", events).contains("item_map"))
|
|
|
|
}
|
|
|
|
|
|
|
|
let new_text = "
|
|
|
|
mod foo;
|
|
|
|
|
|
|
|
use crate::foo::bar::Baz;
|
|
|
|
|
|
|
|
fn foo() -> i32 { 92 }
|
|
|
|
"
|
|
|
|
.to_string();
|
|
|
|
|
|
|
|
db.query_mut(ra_db::FileTextQuery)
|
|
|
|
.set(pos.file_id, Arc::new(new_text));
|
|
|
|
|
|
|
|
{
|
|
|
|
let events = db.log_executed(|| {
|
|
|
|
db.item_map(source_root).unwrap();
|
|
|
|
});
|
|
|
|
assert!(
|
2019-01-01 19:12:05 +00:00
|
|
|
!format!("{:?}", events).contains("item_map"),
|
2018-12-09 09:48:55 +00:00
|
|
|
"{:#?}",
|
|
|
|
events
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|