2023-09-29 10:37:57 +00:00
|
|
|
use hir::{db::ExpandDatabase, HirFileIdExt};
|
2021-06-14 10:27:11 +00:00
|
|
|
use ide_db::{assists::Assist, base_db::AnchoredPathBuf, source_change::FileSystemEdit};
|
2022-03-11 15:49:41 +00:00
|
|
|
use itertools::Itertools;
|
2021-05-17 09:04:17 +00:00
|
|
|
use syntax::AstNode;
|
|
|
|
|
2023-06-14 22:17:22 +00:00
|
|
|
use crate::{fix, Diagnostic, DiagnosticCode, DiagnosticsContext};
|
2021-05-17 09:04:17 +00:00
|
|
|
|
2021-06-13 11:41:19 +00:00
|
|
|
// Diagnostic: unresolved-module
|
|
|
|
//
|
|
|
|
// This diagnostic is triggered if rust-analyzer is unable to discover referred module.
|
2021-06-14 16:32:39 +00:00
|
|
|
pub(crate) fn unresolved_module(
|
2021-06-13 12:27:15 +00:00
|
|
|
ctx: &DiagnosticsContext<'_>,
|
|
|
|
d: &hir::UnresolvedModule,
|
|
|
|
) -> Diagnostic {
|
2023-06-14 22:17:22 +00:00
|
|
|
Diagnostic::new_with_syntax_node_ptr(
|
|
|
|
ctx,
|
|
|
|
DiagnosticCode::RustcHardError("E0583"),
|
2022-03-11 15:49:41 +00:00
|
|
|
match &*d.candidates {
|
|
|
|
[] => "unresolved module".to_string(),
|
2022-12-23 18:42:58 +00:00
|
|
|
[candidate] => format!("unresolved module, can't find module file: {candidate}"),
|
2022-03-11 15:49:41 +00:00
|
|
|
[candidates @ .., last] => {
|
|
|
|
format!(
|
|
|
|
"unresolved module, can't find module file: {}, or {}",
|
|
|
|
candidates.iter().format(", "),
|
|
|
|
last
|
|
|
|
)
|
|
|
|
}
|
|
|
|
},
|
2023-06-14 22:17:22 +00:00
|
|
|
d.decl.clone().map(|it| it.into()),
|
2021-06-13 11:41:19 +00:00
|
|
|
)
|
|
|
|
.with_fixes(fixes(ctx, d))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fixes(ctx: &DiagnosticsContext<'_>, d: &hir::UnresolvedModule) -> Option<Vec<Assist>> {
|
2023-04-16 17:20:48 +00:00
|
|
|
let root = ctx.sema.db.parse_or_expand(d.decl.file_id);
|
2021-06-13 11:41:19 +00:00
|
|
|
let unresolved_module = d.decl.value.to_node(&root);
|
2022-03-11 15:49:41 +00:00
|
|
|
Some(
|
|
|
|
d.candidates
|
|
|
|
.iter()
|
|
|
|
.map(|candidate| {
|
|
|
|
fix(
|
|
|
|
"create_module",
|
2022-04-04 13:54:29 +00:00
|
|
|
&format!("Create module at `{candidate}`"),
|
2022-03-11 15:49:41 +00:00
|
|
|
FileSystemEdit::CreateFile {
|
|
|
|
dst: AnchoredPathBuf {
|
|
|
|
anchor: d.decl.file_id.original_file(ctx.sema.db),
|
|
|
|
path: candidate.clone(),
|
|
|
|
},
|
|
|
|
initial_contents: "".to_string(),
|
|
|
|
}
|
|
|
|
.into(),
|
|
|
|
unresolved_module.syntax().text_range(),
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.collect(),
|
|
|
|
)
|
2021-05-17 09:04:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use expect_test::expect;
|
|
|
|
|
2021-06-14 10:15:05 +00:00
|
|
|
use crate::tests::{check_diagnostics, check_expect};
|
2021-06-13 11:41:19 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn unresolved_module() {
|
|
|
|
check_diagnostics(
|
|
|
|
r#"
|
|
|
|
//- /lib.rs
|
|
|
|
mod foo;
|
|
|
|
mod bar;
|
2022-03-11 15:49:41 +00:00
|
|
|
//^^^^^^^^ 💡 error: unresolved module, can't find module file: bar.rs, or bar/mod.rs
|
2021-06-13 11:41:19 +00:00
|
|
|
mod baz {}
|
|
|
|
//- /foo.rs
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
}
|
2021-05-17 09:04:17 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_unresolved_module_diagnostic() {
|
|
|
|
check_expect(
|
|
|
|
r#"mod foo;"#,
|
|
|
|
expect![[r#"
|
|
|
|
[
|
|
|
|
Diagnostic {
|
2023-06-14 22:17:22 +00:00
|
|
|
code: RustcHardError(
|
|
|
|
"E0583",
|
2021-06-13 19:17:05 +00:00
|
|
|
),
|
2022-03-11 15:49:41 +00:00
|
|
|
message: "unresolved module, can't find module file: foo.rs, or foo/mod.rs",
|
2023-12-06 13:36:45 +00:00
|
|
|
range: FileRange {
|
|
|
|
file_id: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
range: 0..8,
|
|
|
|
},
|
2021-05-17 09:04:17 +00:00
|
|
|
severity: Error,
|
2021-06-13 19:17:05 +00:00
|
|
|
unused: false,
|
|
|
|
experimental: false,
|
2021-05-17 23:11:07 +00:00
|
|
|
fixes: Some(
|
|
|
|
[
|
|
|
|
Assist {
|
|
|
|
id: AssistId(
|
|
|
|
"create_module",
|
|
|
|
QuickFix,
|
|
|
|
),
|
2022-04-04 13:54:29 +00:00
|
|
|
label: "Create module at `foo.rs`",
|
2021-05-17 23:11:07 +00:00
|
|
|
group: None,
|
|
|
|
target: 0..8,
|
|
|
|
source_change: Some(
|
|
|
|
SourceChange {
|
|
|
|
source_file_edits: {},
|
|
|
|
file_system_edits: [
|
|
|
|
CreateFile {
|
|
|
|
dst: AnchoredPathBuf {
|
|
|
|
anchor: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
path: "foo.rs",
|
|
|
|
},
|
|
|
|
initial_contents: "",
|
2021-05-17 09:04:17 +00:00
|
|
|
},
|
2021-05-17 23:11:07 +00:00
|
|
|
],
|
|
|
|
is_snippet: false,
|
|
|
|
},
|
|
|
|
),
|
2022-04-19 16:37:18 +00:00
|
|
|
trigger_signature_help: false,
|
2021-05-17 23:11:07 +00:00
|
|
|
},
|
2022-03-11 15:49:41 +00:00
|
|
|
Assist {
|
|
|
|
id: AssistId(
|
|
|
|
"create_module",
|
|
|
|
QuickFix,
|
|
|
|
),
|
2022-04-04 13:54:29 +00:00
|
|
|
label: "Create module at `foo/mod.rs`",
|
2022-03-11 15:49:41 +00:00
|
|
|
group: None,
|
|
|
|
target: 0..8,
|
|
|
|
source_change: Some(
|
|
|
|
SourceChange {
|
|
|
|
source_file_edits: {},
|
|
|
|
file_system_edits: [
|
|
|
|
CreateFile {
|
|
|
|
dst: AnchoredPathBuf {
|
|
|
|
anchor: FileId(
|
|
|
|
0,
|
|
|
|
),
|
|
|
|
path: "foo/mod.rs",
|
|
|
|
},
|
|
|
|
initial_contents: "",
|
|
|
|
},
|
|
|
|
],
|
|
|
|
is_snippet: false,
|
|
|
|
},
|
|
|
|
),
|
2022-04-19 16:37:18 +00:00
|
|
|
trigger_signature_help: false,
|
2022-03-11 15:49:41 +00:00
|
|
|
},
|
2021-05-17 23:11:07 +00:00
|
|
|
],
|
2021-05-17 09:04:17 +00:00
|
|
|
),
|
2023-06-14 22:17:22 +00:00
|
|
|
main_node: Some(
|
2023-11-29 14:48:40 +00:00
|
|
|
InFileWrapper {
|
2023-06-14 22:17:22 +00:00
|
|
|
file_id: FileId(
|
2023-11-24 15:38:48 +00:00
|
|
|
0,
|
2023-06-14 22:17:22 +00:00
|
|
|
),
|
|
|
|
value: MODULE@0..8
|
|
|
|
MOD_KW@0..3 "mod"
|
|
|
|
WHITESPACE@3..4 " "
|
|
|
|
NAME@4..7
|
|
|
|
IDENT@4..7 "foo"
|
|
|
|
SEMICOLON@7..8 ";"
|
|
|
|
,
|
|
|
|
},
|
|
|
|
),
|
2021-05-17 09:04:17 +00:00
|
|
|
},
|
|
|
|
]
|
|
|
|
"#]],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|