2020-04-14 04:35:34 -07:00
|
|
|
//! Analyze all modules in a project for diagnostics. Exits with a non-zero status
|
2020-04-13 05:44:35 -07:00
|
|
|
//! code if any errors are found.
|
|
|
|
|
|
2020-06-29 18:07:52 +03:00
|
|
|
use std::path::Path;
|
|
|
|
|
|
2020-04-13 05:44:35 -07:00
|
|
|
use anyhow::anyhow;
|
2020-06-29 18:07:52 +03:00
|
|
|
use rustc_hash::FxHashSet;
|
|
|
|
|
|
2020-12-04 19:37:37 +01:00
|
|
|
use hir::{db::HirDatabase, Crate, Module};
|
2020-08-18 16:03:15 +02:00
|
|
|
use ide::{DiagnosticsConfig, Severity};
|
2020-10-24 11:39:57 +03:00
|
|
|
use ide_db::base_db::SourceDatabaseExt;
|
2020-04-13 05:44:35 -07:00
|
|
|
|
|
|
|
|
use crate::cli::{load_cargo::load_cargo, Result};
|
|
|
|
|
|
2020-12-04 19:37:37 +01:00
|
|
|
fn all_modules(db: &dyn HirDatabase) -> Vec<Module> {
|
|
|
|
|
let mut worklist: Vec<_> =
|
|
|
|
|
Crate::all(db).into_iter().map(|krate| krate.root_module(db)).collect();
|
|
|
|
|
let mut modules = Vec::new();
|
|
|
|
|
|
|
|
|
|
while let Some(module) = worklist.pop() {
|
|
|
|
|
modules.push(module);
|
|
|
|
|
worklist.extend(module.children(db));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
modules
|
|
|
|
|
}
|
|
|
|
|
|
2020-08-26 12:52:42 +02:00
|
|
|
pub fn diagnostics(path: &Path, load_output_dirs: bool, with_proc_macro: bool) -> Result<()> {
|
2020-06-11 11:04:09 +02:00
|
|
|
let (host, _vfs) = load_cargo(path, load_output_dirs, with_proc_macro)?;
|
2020-04-13 05:44:35 -07:00
|
|
|
let db = host.raw_database();
|
|
|
|
|
let analysis = host.analysis();
|
|
|
|
|
|
|
|
|
|
let mut found_error = false;
|
2020-06-29 18:07:52 +03:00
|
|
|
let mut visited_files = FxHashSet::default();
|
2020-04-14 05:32:32 -07:00
|
|
|
|
2020-12-04 19:37:37 +01:00
|
|
|
let work = all_modules(db).into_iter().filter(|module| {
|
|
|
|
|
let file_id = module.definition_source(db).file_id.original_file(db);
|
2020-06-11 11:04:09 +02:00
|
|
|
let source_root = db.file_source_root(file_id);
|
|
|
|
|
let source_root = db.source_root(source_root);
|
2020-12-04 19:37:37 +01:00
|
|
|
!source_root.is_library
|
|
|
|
|
});
|
2020-04-14 04:35:34 -07:00
|
|
|
|
2020-06-11 11:04:09 +02:00
|
|
|
for module in work {
|
|
|
|
|
let file_id = module.definition_source(db).file_id.original_file(db);
|
|
|
|
|
if !visited_files.contains(&file_id) {
|
2020-10-20 15:38:11 +02:00
|
|
|
let crate_name =
|
|
|
|
|
module.krate().display_name(db).as_deref().unwrap_or("unknown").to_string();
|
2020-06-11 11:04:09 +02:00
|
|
|
println!("processing crate: {}, module: {}", crate_name, _vfs.file_path(file_id));
|
2020-08-18 16:03:15 +02:00
|
|
|
for diagnostic in analysis.diagnostics(&DiagnosticsConfig::default(), file_id).unwrap()
|
|
|
|
|
{
|
2020-06-11 11:04:09 +02:00
|
|
|
if matches!(diagnostic.severity, Severity::Error) {
|
|
|
|
|
found_error = true;
|
2020-04-14 04:35:34 -07:00
|
|
|
}
|
2020-06-11 11:04:09 +02:00
|
|
|
|
|
|
|
|
println!("{:?}", diagnostic);
|
2020-04-14 04:35:34 -07:00
|
|
|
}
|
2020-06-11 11:04:09 +02:00
|
|
|
|
|
|
|
|
visited_files.insert(file_id);
|
2020-04-14 04:35:34 -07:00
|
|
|
}
|
2020-04-13 05:44:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
println!();
|
|
|
|
|
println!("diagnostic scan complete");
|
|
|
|
|
|
|
|
|
|
if found_error {
|
|
|
|
|
println!();
|
|
|
|
|
Err(anyhow!("diagnostic error detected"))
|
|
|
|
|
} else {
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
}
|