use std::sync::Arc;
pub use super::checker::DiagnosticContext;
use super::{checker::check_file, lua_diagnostic_config::LuaDiagnosticConfig};
use crate::{DiagnosticCode, Emmyrc, FileId, LuaCompilation};
use lsp_types::Diagnostic;
use tokio_util::sync::CancellationToken;
#[derive(Debug)]
pub struct LuaDiagnostic {
enable: bool,
config: Arc<LuaDiagnosticConfig>,
}
impl Default for LuaDiagnostic {
fn default() -> Self {
Self::new()
}
}
impl LuaDiagnostic {
pub fn new() -> Self {
Self {
enable: true,
config: Arc::new(LuaDiagnosticConfig::default()),
}
}
pub fn update_config(&mut self, emmyrc: Arc<Emmyrc>) {
self.enable = emmyrc.diagnostics.enable;
self.config = LuaDiagnosticConfig::new(&emmyrc).into();
}
pub fn enable_only(&mut self, code: DiagnosticCode) {
let mut emmyrc = Emmyrc::default();
emmyrc.diagnostics.enables.push(code);
for diagnostic_code in DiagnosticCode::all().iter() {
if *diagnostic_code != code {
emmyrc.diagnostics.disable.push(*diagnostic_code);
}
}
self.config = LuaDiagnosticConfig::new(&emmyrc).into();
}
pub fn diagnose_file(
&self,
compilation: &LuaCompilation,
file_id: FileId,
cancel_token: CancellationToken,
) -> Option<Vec<Diagnostic>> {
if !self.enable {
return None;
}
if cancel_token.is_cancelled() {
return None;
}
let db = compilation.get_db();
if let Some(module_info) = db.get_module_index().get_workspace_id(file_id)
&& !module_info.is_main()
{
return None;
}
let semantic_model = compilation.get_semantic_model(file_id)?;
let mut context = DiagnosticContext::new(file_id, db, self.config.clone());
check_file(&mut context, &semantic_model);
Some(context.get_diagnostics())
}
}