From d285a8a9e5e2e9c0576c21a7bdc5e439b80a636c Mon Sep 17 00:00:00 2001 From: Michael Davis Date: Thu, 30 Jan 2025 11:54:36 -0500 Subject: [PATCH] LSP: Fix option handling in goto definition/references commands The language server may return `None` for a definition/reference request. The parent commits introduced a regression for these commands when a server did not provide locations. With this change a server may respond with `null` and its locations will instead not be considered. Fixes #12732 --- helix-term/src/commands/lsp.rs | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/helix-term/src/commands/lsp.rs b/helix-term/src/commands/lsp.rs index 2046589f6..130428d4d 100644 --- a/helix-term/src/commands/lsp.rs +++ b/helix-term/src/commands/lsp.rs @@ -903,7 +903,7 @@ where let future = request_provider(language_server, pos, doc.identifier()).unwrap(); async move { let json = future.await?; - let response: lsp::GotoDefinitionResponse = serde_json::from_value(json)?; + let response: Option = serde_json::from_value(json)?; anyhow::Ok((response, offset_encoding)) } }) @@ -913,17 +913,17 @@ where let mut locations = Vec::new(); while let Some((response, offset_encoding)) = futures.try_next().await? { match response { - lsp::GotoDefinitionResponse::Scalar(lsp_location) => { + Some(lsp::GotoDefinitionResponse::Scalar(lsp_location)) => { locations.extend(lsp_location_to_location(lsp_location, offset_encoding)); } - lsp::GotoDefinitionResponse::Array(lsp_locations) => { + Some(lsp::GotoDefinitionResponse::Array(lsp_locations)) => { locations.extend( lsp_locations.into_iter().flat_map(|location| { lsp_location_to_location(location, offset_encoding) }), ); } - lsp::GotoDefinitionResponse::Link(lsp_locations) => { + Some(lsp::GotoDefinitionResponse::Link(lsp_locations)) => { locations.extend( lsp_locations .into_iter() @@ -938,6 +938,7 @@ where }), ); } + None => (), } } let call = move |editor: &mut Editor, compositor: &mut Compositor| { @@ -1002,7 +1003,7 @@ pub fn goto_reference(cx: &mut Context) { .unwrap(); async move { let json = future.await?; - let locations: Vec = serde_json::from_value(json)?; + let locations: Option> = serde_json::from_value(json)?; anyhow::Ok((locations, offset_encoding)) } }) @@ -1014,6 +1015,7 @@ pub fn goto_reference(cx: &mut Context) { locations.extend( lsp_locations .into_iter() + .flatten() .flat_map(|location| lsp_location_to_location(location, offset_encoding)), ); }