2021-03-14 15:12:38 +00:00
|
|
|
use either::Either;
|
|
|
|
use hir::{HasSource, Semantics};
|
2021-02-13 11:07:47 +00:00
|
|
|
use ide_db::{
|
2021-03-14 15:12:38 +00:00
|
|
|
base_db::{FileId, FilePosition, FileRange},
|
2021-03-15 11:18:52 +00:00
|
|
|
helpers::visit_file_defs,
|
2021-03-14 15:12:38 +00:00
|
|
|
RootDatabase,
|
2021-02-13 11:07:47 +00:00
|
|
|
};
|
2021-03-14 15:12:38 +00:00
|
|
|
use syntax::{ast::NameOwner, AstNode, TextRange, TextSize};
|
2021-02-13 11:07:47 +00:00
|
|
|
|
|
|
|
use crate::{
|
|
|
|
fn_references::find_all_methods,
|
|
|
|
goto_implementation::goto_implementation,
|
|
|
|
references::find_all_refs,
|
|
|
|
runnables::{runnables, Runnable},
|
2021-03-14 15:12:38 +00:00
|
|
|
NavigationTarget, RunnableKind,
|
2021-02-13 11:07:47 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Feature: Annotations
|
|
|
|
//
|
|
|
|
// Provides user with annotations above items for looking up references or impl blocks
|
|
|
|
// and running/debugging binaries.
|
2021-02-13 13:27:04 +00:00
|
|
|
#[derive(Debug)]
|
2021-02-13 11:07:47 +00:00
|
|
|
pub struct Annotation {
|
|
|
|
pub range: TextRange,
|
|
|
|
pub kind: AnnotationKind,
|
|
|
|
}
|
|
|
|
|
2021-02-13 13:27:04 +00:00
|
|
|
#[derive(Debug)]
|
2021-02-13 11:07:47 +00:00
|
|
|
pub enum AnnotationKind {
|
|
|
|
Runnable { debug: bool, runnable: Runnable },
|
|
|
|
HasImpls { position: FilePosition, data: Option<Vec<NavigationTarget>> },
|
|
|
|
HasReferences { position: FilePosition, data: Option<Vec<FileRange>> },
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct AnnotationConfig {
|
|
|
|
pub binary_target: bool,
|
|
|
|
pub annotate_runnables: bool,
|
|
|
|
pub annotate_impls: bool,
|
|
|
|
pub annotate_references: bool,
|
|
|
|
pub annotate_method_references: bool,
|
|
|
|
pub run: bool,
|
|
|
|
pub debug: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn annotations(
|
|
|
|
db: &RootDatabase,
|
|
|
|
file_id: FileId,
|
|
|
|
config: AnnotationConfig,
|
|
|
|
) -> Vec<Annotation> {
|
|
|
|
let mut annotations = Vec::default();
|
|
|
|
|
|
|
|
if config.annotate_runnables {
|
|
|
|
for runnable in runnables(db, file_id) {
|
2021-02-13 13:47:53 +00:00
|
|
|
if should_skip_runnable(&runnable.kind, config.binary_target) {
|
2021-02-13 11:07:47 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
let action = runnable.action();
|
|
|
|
let range = runnable.nav.full_range;
|
|
|
|
|
2021-02-15 14:09:50 +00:00
|
|
|
if config.run {
|
2021-02-13 11:07:47 +00:00
|
|
|
annotations.push(Annotation {
|
|
|
|
range,
|
2021-02-13 13:47:53 +00:00
|
|
|
|
2021-02-13 11:07:47 +00:00
|
|
|
// FIXME: This one allocates without reason if run is enabled, but debug is disabled
|
2021-02-15 14:09:50 +00:00
|
|
|
kind: AnnotationKind::Runnable { debug: false, runnable: runnable.clone() },
|
2021-02-13 11:07:47 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-02-15 14:09:50 +00:00
|
|
|
if action.debugee && config.debug {
|
2021-02-13 11:07:47 +00:00
|
|
|
annotations.push(Annotation {
|
|
|
|
range,
|
2021-02-15 14:09:50 +00:00
|
|
|
kind: AnnotationKind::Runnable { debug: true, runnable },
|
2021-02-13 11:07:47 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-15 11:18:52 +00:00
|
|
|
visit_file_defs(&Semantics::new(db), file_id, &mut |def| match def {
|
2021-03-14 15:12:38 +00:00
|
|
|
Either::Left(def) => {
|
|
|
|
let node = match def {
|
|
|
|
hir::ModuleDef::Const(konst) => {
|
|
|
|
konst.source(db).and_then(|node| range_and_position_of(&node.value))
|
|
|
|
}
|
|
|
|
hir::ModuleDef::Trait(trait_) => {
|
|
|
|
trait_.source(db).and_then(|node| range_and_position_of(&node.value))
|
|
|
|
}
|
|
|
|
hir::ModuleDef::Adt(hir::Adt::Struct(strukt)) => {
|
|
|
|
strukt.source(db).and_then(|node| range_and_position_of(&node.value))
|
|
|
|
}
|
|
|
|
hir::ModuleDef::Adt(hir::Adt::Enum(enum_)) => {
|
|
|
|
enum_.source(db).and_then(|node| range_and_position_of(&node.value))
|
|
|
|
}
|
|
|
|
hir::ModuleDef::Adt(hir::Adt::Union(union)) => {
|
|
|
|
union.source(db).and_then(|node| range_and_position_of(&node.value))
|
|
|
|
}
|
|
|
|
_ => None,
|
|
|
|
};
|
|
|
|
let (offset, range) = match node {
|
|
|
|
Some(node) => node,
|
|
|
|
None => return,
|
|
|
|
};
|
|
|
|
|
|
|
|
if config.annotate_impls && !matches!(def, hir::ModuleDef::Const(_)) {
|
2021-02-13 11:07:47 +00:00
|
|
|
annotations.push(Annotation {
|
2021-03-14 15:12:38 +00:00
|
|
|
range,
|
2021-02-13 11:07:47 +00:00
|
|
|
kind: AnnotationKind::HasImpls {
|
2021-03-14 15:12:38 +00:00
|
|
|
position: FilePosition { file_id, offset },
|
2021-02-13 11:07:47 +00:00
|
|
|
data: None,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if config.annotate_references {
|
|
|
|
annotations.push(Annotation {
|
2021-03-14 15:12:38 +00:00
|
|
|
range,
|
2021-02-13 11:07:47 +00:00
|
|
|
kind: AnnotationKind::HasReferences {
|
2021-03-14 15:12:38 +00:00
|
|
|
position: FilePosition { file_id, offset },
|
2021-02-13 11:07:47 +00:00
|
|
|
data: None,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2021-03-14 15:12:38 +00:00
|
|
|
|
|
|
|
fn range_and_position_of(node: &dyn NameOwner) -> Option<(TextSize, TextRange)> {
|
|
|
|
Some((node.name()?.syntax().text_range().start(), node.syntax().text_range()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Either::Right(_) => (),
|
|
|
|
});
|
2021-02-13 11:07:47 +00:00
|
|
|
|
|
|
|
if config.annotate_method_references {
|
|
|
|
annotations.extend(find_all_methods(db, file_id).into_iter().map(|method| Annotation {
|
|
|
|
range: method.range,
|
|
|
|
kind: AnnotationKind::HasReferences {
|
|
|
|
position: FilePosition { file_id, offset: method.range.start() },
|
|
|
|
data: None,
|
|
|
|
},
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
annotations
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn resolve_annotation(db: &RootDatabase, mut annotation: Annotation) -> Annotation {
|
|
|
|
match annotation.kind {
|
|
|
|
AnnotationKind::HasImpls { position, ref mut data } => {
|
|
|
|
*data = goto_implementation(db, position).map(|range| range.info);
|
|
|
|
}
|
|
|
|
AnnotationKind::HasReferences { position, ref mut data } => {
|
|
|
|
*data = find_all_refs(&Semantics::new(db), position, None).map(|result| {
|
|
|
|
result
|
|
|
|
.references
|
|
|
|
.into_iter()
|
2021-02-13 11:22:12 +00:00
|
|
|
.map(|(file_id, access)| {
|
|
|
|
access.into_iter().map(move |(range, _)| FileRange { file_id, range })
|
|
|
|
})
|
2021-02-13 11:07:47 +00:00
|
|
|
.flatten()
|
|
|
|
.collect()
|
|
|
|
});
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
};
|
|
|
|
|
|
|
|
annotation
|
|
|
|
}
|
2021-02-13 13:27:04 +00:00
|
|
|
|
2021-02-13 13:47:53 +00:00
|
|
|
fn should_skip_runnable(kind: &RunnableKind, binary_target: bool) -> bool {
|
|
|
|
match kind {
|
|
|
|
RunnableKind::Bin => !binary_target,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-13 13:27:04 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2021-02-13 17:18:13 +00:00
|
|
|
use expect_test::{expect, Expect};
|
2021-02-13 13:27:04 +00:00
|
|
|
|
2021-02-13 17:18:13 +00:00
|
|
|
use crate::{fixture, Annotation, AnnotationConfig};
|
2021-02-13 13:27:04 +00:00
|
|
|
|
2021-02-13 17:18:13 +00:00
|
|
|
fn check(ra_fixture: &str, expect: Expect) {
|
2021-02-13 13:27:04 +00:00
|
|
|
let (analysis, file_id) = fixture::file(ra_fixture);
|
|
|
|
|
|
|
|
let annotations: Vec<Annotation> = analysis
|
2021-02-13 17:18:13 +00:00
|
|
|
.annotations(
|
|
|
|
file_id,
|
|
|
|
AnnotationConfig {
|
|
|
|
binary_target: true,
|
|
|
|
annotate_runnables: true,
|
|
|
|
annotate_impls: true,
|
|
|
|
annotate_references: true,
|
|
|
|
annotate_method_references: true,
|
|
|
|
run: true,
|
|
|
|
debug: true,
|
|
|
|
},
|
|
|
|
)
|
2021-02-13 13:27:04 +00:00
|
|
|
.unwrap()
|
|
|
|
.into_iter()
|
2021-02-13 17:18:13 +00:00
|
|
|
.map(|annotation| analysis.resolve_annotation(annotation).unwrap())
|
2021-02-13 13:27:04 +00:00
|
|
|
.collect();
|
|
|
|
|
2021-02-13 17:18:13 +00:00
|
|
|
expect.assert_debug_eq(&annotations);
|
2021-02-13 13:27:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn const_annotations() {
|
2021-02-13 17:18:13 +00:00
|
|
|
check(
|
2021-02-13 13:27:04 +00:00
|
|
|
r#"
|
|
|
|
const DEMO: i32 = 123;
|
|
|
|
|
2021-02-13 17:18:13 +00:00
|
|
|
const UNUSED: i32 = 123;
|
|
|
|
|
2021-02-13 13:27:04 +00:00
|
|
|
fn main() {
|
|
|
|
let hello = DEMO;
|
|
|
|
}
|
|
|
|
"#,
|
2021-02-13 17:18:13 +00:00
|
|
|
expect![[r#"
|
|
|
|
[
|
|
|
|
Annotation {
|
|
|
|
range: 50..85,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: false,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 50..85,
|
|
|
|
focus_range: 53..57,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 50..85,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: true,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 50..85,
|
|
|
|
focus_range: 53..57,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 0..22,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 6,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[
|
|
|
|
FileRange {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
range: 78..82,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 24..48,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 30,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 53..57,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 53,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
]
|
|
|
|
"#]],
|
2021-02-13 13:27:04 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn struct_references_annotations() {
|
2021-02-13 17:18:13 +00:00
|
|
|
check(
|
2021-02-13 13:27:04 +00:00
|
|
|
r#"
|
|
|
|
struct Test;
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let test = Test;
|
|
|
|
}
|
|
|
|
"#,
|
2021-02-13 17:18:13 +00:00
|
|
|
expect![[r#"
|
|
|
|
[
|
|
|
|
Annotation {
|
|
|
|
range: 14..48,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: false,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 14..48,
|
|
|
|
focus_range: 17..21,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 14..48,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: true,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 14..48,
|
|
|
|
focus_range: 17..21,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 0..12,
|
|
|
|
kind: HasImpls {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 7,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 0..12,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 7,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[
|
|
|
|
FileRange {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
range: 41..45,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 17..21,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 17,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
]
|
|
|
|
"#]],
|
2021-02-13 13:27:04 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn struct_and_trait_impls_annotations() {
|
2021-02-13 17:18:13 +00:00
|
|
|
check(
|
2021-02-13 13:27:04 +00:00
|
|
|
r#"
|
|
|
|
struct Test;
|
|
|
|
|
|
|
|
trait MyCoolTrait {}
|
|
|
|
|
|
|
|
impl MyCoolTrait for Test {}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let test = Test;
|
|
|
|
}
|
|
|
|
"#,
|
2021-02-13 17:18:13 +00:00
|
|
|
expect![[r#"
|
|
|
|
[
|
|
|
|
Annotation {
|
|
|
|
range: 66..100,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: false,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 66..100,
|
|
|
|
focus_range: 69..73,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 66..100,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: true,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 66..100,
|
|
|
|
focus_range: 69..73,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 0..12,
|
|
|
|
kind: HasImpls {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 7,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[
|
|
|
|
NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 36..64,
|
|
|
|
focus_range: 57..61,
|
|
|
|
name: "impl",
|
|
|
|
kind: Impl,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 0..12,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 7,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[
|
|
|
|
FileRange {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
range: 57..61,
|
|
|
|
},
|
|
|
|
FileRange {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
range: 93..97,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 14..34,
|
|
|
|
kind: HasImpls {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 20,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[
|
|
|
|
NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 36..64,
|
|
|
|
focus_range: 57..61,
|
|
|
|
name: "impl",
|
|
|
|
kind: Impl,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 14..34,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 20,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[
|
|
|
|
FileRange {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
range: 41..52,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 69..73,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 69,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
]
|
|
|
|
"#]],
|
2021-02-13 13:27:04 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2021-02-13 17:18:13 +00:00
|
|
|
fn runnable_annotation() {
|
|
|
|
check(
|
2021-02-13 13:27:04 +00:00
|
|
|
r#"
|
|
|
|
fn main() {}
|
|
|
|
"#,
|
2021-02-13 17:18:13 +00:00
|
|
|
expect![[r#"
|
|
|
|
[
|
|
|
|
Annotation {
|
|
|
|
range: 0..12,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: false,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 0..12,
|
|
|
|
focus_range: 3..7,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 0..12,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: true,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 0..12,
|
|
|
|
focus_range: 3..7,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 3..7,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 3,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
]
|
|
|
|
"#]],
|
2021-02-13 13:27:04 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn method_annotations() {
|
2021-02-13 17:18:13 +00:00
|
|
|
check(
|
2021-02-13 13:27:04 +00:00
|
|
|
r#"
|
|
|
|
struct Test;
|
|
|
|
|
|
|
|
impl Test {
|
|
|
|
fn self_by_ref(&self) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
Test.self_by_ref();
|
|
|
|
}
|
|
|
|
"#,
|
2021-02-13 17:18:13 +00:00
|
|
|
expect![[r#"
|
|
|
|
[
|
|
|
|
Annotation {
|
|
|
|
range: 58..95,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: false,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 58..95,
|
|
|
|
focus_range: 61..65,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 58..95,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: true,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 58..95,
|
|
|
|
focus_range: 61..65,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 0..12,
|
|
|
|
kind: HasImpls {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 7,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[
|
|
|
|
NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 14..56,
|
|
|
|
focus_range: 19..23,
|
|
|
|
name: "impl",
|
|
|
|
kind: Impl,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 0..12,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 7,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[
|
|
|
|
FileRange {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
range: 19..23,
|
|
|
|
},
|
|
|
|
FileRange {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
range: 74..78,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 33..44,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 33,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[
|
|
|
|
FileRange {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
range: 79..90,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 61..65,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 61,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
]
|
|
|
|
"#]],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_annotations() {
|
|
|
|
check(
|
|
|
|
r#"
|
|
|
|
fn main() {}
|
|
|
|
|
|
|
|
mod tests {
|
|
|
|
#[test]
|
|
|
|
fn my_cool_test() {}
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
expect![[r#"
|
|
|
|
[
|
|
|
|
Annotation {
|
|
|
|
range: 0..12,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: false,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 0..12,
|
|
|
|
focus_range: 3..7,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 0..12,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: true,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 0..12,
|
|
|
|
focus_range: 3..7,
|
|
|
|
name: "main",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Bin,
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 14..64,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: false,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 14..64,
|
|
|
|
focus_range: 18..23,
|
|
|
|
name: "tests",
|
|
|
|
kind: Module,
|
|
|
|
},
|
|
|
|
kind: TestMod {
|
|
|
|
path: "tests",
|
|
|
|
},
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 14..64,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: true,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 14..64,
|
|
|
|
focus_range: 18..23,
|
|
|
|
name: "tests",
|
|
|
|
kind: Module,
|
|
|
|
},
|
|
|
|
kind: TestMod {
|
|
|
|
path: "tests",
|
|
|
|
},
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 30..62,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: false,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 30..62,
|
|
|
|
focus_range: 45..57,
|
|
|
|
name: "my_cool_test",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Test {
|
|
|
|
test_id: Path(
|
|
|
|
"tests::my_cool_test",
|
|
|
|
),
|
|
|
|
attr: TestAttr {
|
|
|
|
ignore: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 30..62,
|
|
|
|
kind: Runnable {
|
2021-02-15 14:09:50 +00:00
|
|
|
debug: true,
|
2021-02-13 17:18:13 +00:00
|
|
|
runnable: Runnable {
|
|
|
|
nav: NavigationTarget {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
full_range: 30..62,
|
|
|
|
focus_range: 45..57,
|
|
|
|
name: "my_cool_test",
|
|
|
|
kind: Function,
|
|
|
|
},
|
|
|
|
kind: Test {
|
|
|
|
test_id: Path(
|
|
|
|
"tests::my_cool_test",
|
|
|
|
),
|
|
|
|
attr: TestAttr {
|
|
|
|
ignore: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
cfg: None,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Annotation {
|
|
|
|
range: 3..7,
|
|
|
|
kind: HasReferences {
|
|
|
|
position: FilePosition {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
offset: 3,
|
|
|
|
},
|
|
|
|
data: Some(
|
|
|
|
[],
|
|
|
|
),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
]
|
|
|
|
"#]],
|
2021-02-13 13:27:04 +00:00
|
|
|
);
|
|
|
|
}
|
2021-03-14 15:12:38 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_no_annotations_outside_module_tree() {
|
|
|
|
check(
|
|
|
|
r#"
|
|
|
|
//- /foo.rs
|
|
|
|
struct Foo;
|
|
|
|
//- /lib.rs
|
|
|
|
// this file comes last since `check` checks the first file only
|
|
|
|
"#,
|
|
|
|
expect![[r#"
|
|
|
|
[]
|
|
|
|
"#]],
|
|
|
|
);
|
|
|
|
}
|
2021-02-13 13:27:04 +00:00
|
|
|
}
|