From b8f47d7ca9d6499cb523cd29a588711a4570dc2b Mon Sep 17 00:00:00 2001 From: Sofus Addington Date: Sat, 8 Mar 2025 06:11:46 +0100 Subject: [PATCH] Separate timeout for language servers without interfile dependencies --- helix-term/src/handlers/diagnostics.rs | 55 +++++++++++++++++++++----- 1 file changed, 46 insertions(+), 9 deletions(-) diff --git a/helix-term/src/handlers/diagnostics.rs b/helix-term/src/handlers/diagnostics.rs index ac2df70a..39799c5f 100644 --- a/helix-term/src/handlers/diagnostics.rs +++ b/helix-term/src/handlers/diagnostics.rs @@ -57,27 +57,45 @@ pub(super) fn register_hooks(handlers: &Handlers) { } #[derive(Debug)] -pub(super) struct PullDiagnosticsHandler {} +pub(super) struct PullDiagnosticsHandler { + no_inter_file_dependency_timeout: Option, +} impl PullDiagnosticsHandler { pub fn new() -> PullDiagnosticsHandler { - PullDiagnosticsHandler {} + PullDiagnosticsHandler { + no_inter_file_dependency_timeout: None, + } } } +const TIMEOUT: Duration = Duration::from_millis(500); +const TIMEOUT_NO_INTER_FILE_DEPENDENCY: Duration = Duration::from_millis(125); + impl helix_event::AsyncHook for PullDiagnosticsHandler { type Event = PullDiagnosticsEvent; fn handle_event( &mut self, event: Self::Event, - existing_debounce: Option, + timeout: Option, ) -> Option { - if existing_debounce.is_none() { - dispatch_pull_diagnostic_for_document(event.document_id); + if timeout.is_none() { + dispatch_pull_diagnostic_for_document(event.document_id, false); + self.no_inter_file_dependency_timeout = Some(Instant::now()); } - Some(Instant::now() + Duration::from_millis(500)) + if self + .no_inter_file_dependency_timeout + .is_some_and(|nifd_timeout| { + nifd_timeout.duration_since(Instant::now()) > TIMEOUT_NO_INTER_FILE_DEPENDENCY + }) + { + dispatch_pull_diagnostic_for_document(event.document_id, true); + self.no_inter_file_dependency_timeout = Some(Instant::now()); + }; + + Some(Instant::now() + TIMEOUT) } fn finish_debounce(&mut self) { @@ -85,14 +103,33 @@ impl helix_event::AsyncHook for PullDiagnosticsHandler { } } -fn dispatch_pull_diagnostic_for_document(document_id: DocumentId) { +fn dispatch_pull_diagnostic_for_document( + document_id: DocumentId, + exclude_language_servers_without_inter_file_dependency: bool, +) { job::dispatch_blocking(move |editor, _| { let Some(doc) = editor.document(document_id) else { return; }; - let language_servers = - doc.language_servers_with_feature(LanguageServerFeature::PullDiagnostics); + let language_servers = doc + .language_servers_with_feature(LanguageServerFeature::PullDiagnostics) + .filter(|ls| { + if !exclude_language_servers_without_inter_file_dependency { + return true; + }; + ls.capabilities() + .diagnostic_provider + .as_ref() + .is_some_and(|dp| match dp { + lsp::DiagnosticServerCapabilities::Options(options) => { + options.inter_file_dependencies + } + lsp::DiagnosticServerCapabilities::RegistrationOptions(options) => { + options.diagnostic_options.inter_file_dependencies + } + }) + }); for language_server in language_servers { pull_diagnostics_for_document(doc, language_server);