mirror of
https://github.com/helix-editor/helix.git
synced 2025-04-03 10:57:48 +03:00
LSP: Eagerly decode request results in the client
Previously the `call` helper (and its related functions) returned a `serde_json::Value` which was then decoded either later in the client (see signature help and hover) or by the client's caller. This led to some unnecessary boilerplate in the client: let resp = self.call::<MyRequest>(params); Some(async move { Ok(serde_json::from_value(resp.await?)?) }) and in the caller. It also allowed for mistakes with the types. The workspace symbol request's calling code for example mistakenly decoded a `lsp::WorkspaceSymbolResponse` as `Vec<lsp::SymbolInformation>` - one of the untagged enum members (so it parsed successfully) but not the correct type. With this change, the `call` helper eagerly decodes the response to a request as the `lsp::request::Request::Result` trait item. This is similar to the old helper `request` (which has become redundant and has been eliminated) but all work is done within the same async block which avoids some awkward lifetimes. The return types of functions like `Client::text_document_range_inlay_hints` are now more verbose but it is no longer possible to accidentally decode as an incorrect type. Additionally `Client::resolve_code_action` now uses the `call_with_ref` helper to avoid an unnecessary clone.
This commit is contained in:
parent
6da1a79d80
commit
7e7a98560e
6 changed files with 96 additions and 137 deletions
|
@ -385,23 +385,11 @@ impl Client {
|
||||||
self.workspace_folders.lock()
|
self.workspace_folders.lock()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Execute a RPC request on the language server.
|
|
||||||
async fn request<R: lsp::request::Request>(&self, params: R::Params) -> Result<R::Result>
|
|
||||||
where
|
|
||||||
R::Params: serde::Serialize,
|
|
||||||
R::Result: core::fmt::Debug, // TODO: temporary
|
|
||||||
{
|
|
||||||
// a future that resolves into the response
|
|
||||||
let json = self.call::<R>(params).await?;
|
|
||||||
let response = serde_json::from_value(json)?;
|
|
||||||
Ok(response)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Execute a RPC request on the language server.
|
/// Execute a RPC request on the language server.
|
||||||
fn call<R: lsp::request::Request>(
|
fn call<R: lsp::request::Request>(
|
||||||
&self,
|
&self,
|
||||||
params: R::Params,
|
params: R::Params,
|
||||||
) -> impl Future<Output = Result<Value>>
|
) -> impl Future<Output = Result<R::Result>>
|
||||||
where
|
where
|
||||||
R::Params: serde::Serialize,
|
R::Params: serde::Serialize,
|
||||||
{
|
{
|
||||||
|
@ -411,7 +399,7 @@ impl Client {
|
||||||
fn call_with_ref<R: lsp::request::Request>(
|
fn call_with_ref<R: lsp::request::Request>(
|
||||||
&self,
|
&self,
|
||||||
params: &R::Params,
|
params: &R::Params,
|
||||||
) -> impl Future<Output = Result<Value>>
|
) -> impl Future<Output = Result<R::Result>>
|
||||||
where
|
where
|
||||||
R::Params: serde::Serialize,
|
R::Params: serde::Serialize,
|
||||||
{
|
{
|
||||||
|
@ -422,7 +410,7 @@ impl Client {
|
||||||
&self,
|
&self,
|
||||||
params: &R::Params,
|
params: &R::Params,
|
||||||
timeout_secs: u64,
|
timeout_secs: u64,
|
||||||
) -> impl Future<Output = Result<Value>>
|
) -> impl Future<Output = Result<R::Result>>
|
||||||
where
|
where
|
||||||
R::Params: serde::Serialize,
|
R::Params: serde::Serialize,
|
||||||
{
|
{
|
||||||
|
@ -458,6 +446,7 @@ impl Client {
|
||||||
.await
|
.await
|
||||||
.map_err(|_| Error::Timeout(id))? // return Timeout
|
.map_err(|_| Error::Timeout(id))? // return Timeout
|
||||||
.ok_or(Error::StreamClosed)?
|
.ok_or(Error::StreamClosed)?
|
||||||
|
.and_then(|value| serde_json::from_value(value).map_err(Into::into))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -697,11 +686,11 @@ impl Client {
|
||||||
work_done_progress_params: lsp::WorkDoneProgressParams::default(),
|
work_done_progress_params: lsp::WorkDoneProgressParams::default(),
|
||||||
};
|
};
|
||||||
|
|
||||||
self.request::<lsp::request::Initialize>(params).await
|
self.call::<lsp::request::Initialize>(params).await
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn shutdown(&self) -> Result<()> {
|
pub async fn shutdown(&self) -> Result<()> {
|
||||||
self.request::<lsp::request::Shutdown>(()).await
|
self.call::<lsp::request::Shutdown>(()).await
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn exit(&self) {
|
pub fn exit(&self) {
|
||||||
|
@ -746,7 +735,7 @@ impl Client {
|
||||||
old_path: &Path,
|
old_path: &Path,
|
||||||
new_path: &Path,
|
new_path: &Path,
|
||||||
is_dir: bool,
|
is_dir: bool,
|
||||||
) -> Option<impl Future<Output = Result<lsp::WorkspaceEdit>>> {
|
) -> Option<impl Future<Output = Result<Option<lsp::WorkspaceEdit>>>> {
|
||||||
let capabilities = self.file_operations_intests();
|
let capabilities = self.file_operations_intests();
|
||||||
if !capabilities.will_rename.has_interest(old_path, is_dir) {
|
if !capabilities.will_rename.has_interest(old_path, is_dir) {
|
||||||
return None;
|
return None;
|
||||||
|
@ -763,16 +752,10 @@ impl Client {
|
||||||
old_uri: url_from_path(old_path)?,
|
old_uri: url_from_path(old_path)?,
|
||||||
new_uri: url_from_path(new_path)?,
|
new_uri: url_from_path(new_path)?,
|
||||||
}];
|
}];
|
||||||
let request = self.call_with_timeout::<lsp::request::WillRenameFiles>(
|
Some(self.call_with_timeout::<lsp::request::WillRenameFiles>(
|
||||||
&lsp::RenameFilesParams { files },
|
&lsp::RenameFilesParams { files },
|
||||||
5,
|
5,
|
||||||
);
|
))
|
||||||
|
|
||||||
Some(async move {
|
|
||||||
let json = request.await?;
|
|
||||||
let response: Option<lsp::WorkspaceEdit> = serde_json::from_value(json)?;
|
|
||||||
Ok(response.unwrap_or_default())
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn did_rename(&self, old_path: &Path, new_path: &Path, is_dir: bool) -> Option<()> {
|
pub fn did_rename(&self, old_path: &Path, new_path: &Path, is_dir: bool) -> Option<()> {
|
||||||
|
@ -1016,7 +999,7 @@ impl Client {
|
||||||
position: lsp::Position,
|
position: lsp::Position,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
context: lsp::CompletionContext,
|
context: lsp::CompletionContext,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<lsp::CompletionResponse>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support completion.
|
// Return early if the server does not support completion.
|
||||||
|
@ -1042,14 +1025,13 @@ impl Client {
|
||||||
&self,
|
&self,
|
||||||
completion_item: &lsp::CompletionItem,
|
completion_item: &lsp::CompletionItem,
|
||||||
) -> impl Future<Output = Result<lsp::CompletionItem>> {
|
) -> impl Future<Output = Result<lsp::CompletionItem>> {
|
||||||
let res = self.call_with_ref::<lsp::request::ResolveCompletionItem>(completion_item);
|
self.call_with_ref::<lsp::request::ResolveCompletionItem>(completion_item)
|
||||||
async move { Ok(serde_json::from_value(res.await?)?) }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn resolve_code_action(
|
pub fn resolve_code_action(
|
||||||
&self,
|
&self,
|
||||||
code_action: lsp::CodeAction,
|
code_action: &lsp::CodeAction,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<lsp::CodeAction>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support resolving code actions.
|
// Return early if the server does not support resolving code actions.
|
||||||
|
@ -1061,7 +1043,7 @@ impl Client {
|
||||||
_ => return None,
|
_ => return None,
|
||||||
}
|
}
|
||||||
|
|
||||||
Some(self.call::<lsp::request::CodeActionResolveRequest>(code_action))
|
Some(self.call_with_ref::<lsp::request::CodeActionResolveRequest>(code_action))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn text_document_signature_help(
|
pub fn text_document_signature_help(
|
||||||
|
@ -1085,8 +1067,7 @@ impl Client {
|
||||||
// lsp::SignatureHelpContext
|
// lsp::SignatureHelpContext
|
||||||
};
|
};
|
||||||
|
|
||||||
let res = self.call::<lsp::request::SignatureHelpRequest>(params);
|
Some(self.call::<lsp::request::SignatureHelpRequest>(params))
|
||||||
Some(async move { Ok(serde_json::from_value(res.await?)?) })
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn text_document_range_inlay_hints(
|
pub fn text_document_range_inlay_hints(
|
||||||
|
@ -1094,7 +1075,7 @@ impl Client {
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
range: lsp::Range,
|
range: lsp::Range,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<Vec<lsp::InlayHint>>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
match capabilities.inlay_hint_provider {
|
match capabilities.inlay_hint_provider {
|
||||||
|
@ -1140,8 +1121,7 @@ impl Client {
|
||||||
// lsp::SignatureHelpContext
|
// lsp::SignatureHelpContext
|
||||||
};
|
};
|
||||||
|
|
||||||
let res = self.call::<lsp::request::HoverRequest>(params);
|
Some(self.call::<lsp::request::HoverRequest>(params))
|
||||||
Some(async move { Ok(serde_json::from_value(res.await?)?) })
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// formatting
|
// formatting
|
||||||
|
@ -1151,7 +1131,7 @@ impl Client {
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
options: lsp::FormattingOptions,
|
options: lsp::FormattingOptions,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
) -> Option<impl Future<Output = Result<Vec<lsp::TextEdit>>>> {
|
) -> Option<impl Future<Output = Result<Option<Vec<lsp::TextEdit>>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support formatting.
|
// Return early if the server does not support formatting.
|
||||||
|
@ -1184,13 +1164,7 @@ impl Client {
|
||||||
work_done_progress_params: lsp::WorkDoneProgressParams { work_done_token },
|
work_done_progress_params: lsp::WorkDoneProgressParams { work_done_token },
|
||||||
};
|
};
|
||||||
|
|
||||||
let request = self.call::<lsp::request::Formatting>(params);
|
Some(self.call::<lsp::request::Formatting>(params))
|
||||||
|
|
||||||
Some(async move {
|
|
||||||
let json = request.await?;
|
|
||||||
let response: Option<Vec<lsp::TextEdit>> = serde_json::from_value(json)?;
|
|
||||||
Ok(response.unwrap_or_default())
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn text_document_range_formatting(
|
pub fn text_document_range_formatting(
|
||||||
|
@ -1199,7 +1173,7 @@ impl Client {
|
||||||
range: lsp::Range,
|
range: lsp::Range,
|
||||||
options: lsp::FormattingOptions,
|
options: lsp::FormattingOptions,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
) -> Option<impl Future<Output = Result<Vec<lsp::TextEdit>>>> {
|
) -> Option<impl Future<Output = Result<Option<Vec<lsp::TextEdit>>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support range formatting.
|
// Return early if the server does not support range formatting.
|
||||||
|
@ -1215,13 +1189,7 @@ impl Client {
|
||||||
work_done_progress_params: lsp::WorkDoneProgressParams { work_done_token },
|
work_done_progress_params: lsp::WorkDoneProgressParams { work_done_token },
|
||||||
};
|
};
|
||||||
|
|
||||||
let request = self.call::<lsp::request::RangeFormatting>(params);
|
Some(self.call::<lsp::request::RangeFormatting>(params))
|
||||||
|
|
||||||
Some(async move {
|
|
||||||
let json = request.await?;
|
|
||||||
let response: Option<Vec<lsp::TextEdit>> = serde_json::from_value(json)?;
|
|
||||||
Ok(response.unwrap_or_default())
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn text_document_document_highlight(
|
pub fn text_document_document_highlight(
|
||||||
|
@ -1229,7 +1197,7 @@ impl Client {
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
position: lsp::Position,
|
position: lsp::Position,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<Vec<lsp::DocumentHighlight>>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support document highlight.
|
// Return early if the server does not support document highlight.
|
||||||
|
@ -1262,7 +1230,7 @@ impl Client {
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
position: lsp::Position,
|
position: lsp::Position,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
) -> impl Future<Output = Result<Value>> {
|
) -> impl Future<Output = Result<T::Result>> {
|
||||||
let params = lsp::GotoDefinitionParams {
|
let params = lsp::GotoDefinitionParams {
|
||||||
text_document_position_params: lsp::TextDocumentPositionParams {
|
text_document_position_params: lsp::TextDocumentPositionParams {
|
||||||
text_document,
|
text_document,
|
||||||
|
@ -1282,7 +1250,7 @@ impl Client {
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
position: lsp::Position,
|
position: lsp::Position,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<lsp::GotoDefinitionResponse>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support goto-definition.
|
// Return early if the server does not support goto-definition.
|
||||||
|
@ -1303,7 +1271,7 @@ impl Client {
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
position: lsp::Position,
|
position: lsp::Position,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<lsp::GotoDefinitionResponse>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support goto-declaration.
|
// Return early if the server does not support goto-declaration.
|
||||||
|
@ -1328,7 +1296,7 @@ impl Client {
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
position: lsp::Position,
|
position: lsp::Position,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<lsp::GotoDefinitionResponse>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support goto-type-definition.
|
// Return early if the server does not support goto-type-definition.
|
||||||
|
@ -1352,7 +1320,7 @@ impl Client {
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
position: lsp::Position,
|
position: lsp::Position,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<lsp::GotoDefinitionResponse>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support goto-definition.
|
// Return early if the server does not support goto-definition.
|
||||||
|
@ -1377,7 +1345,7 @@ impl Client {
|
||||||
position: lsp::Position,
|
position: lsp::Position,
|
||||||
include_declaration: bool,
|
include_declaration: bool,
|
||||||
work_done_token: Option<lsp::ProgressToken>,
|
work_done_token: Option<lsp::ProgressToken>,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<Vec<lsp::Location>>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support goto-reference.
|
// Return early if the server does not support goto-reference.
|
||||||
|
@ -1406,7 +1374,7 @@ impl Client {
|
||||||
pub fn document_symbols(
|
pub fn document_symbols(
|
||||||
&self,
|
&self,
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<lsp::DocumentSymbolResponse>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support document symbols.
|
// Return early if the server does not support document symbols.
|
||||||
|
@ -1428,7 +1396,7 @@ impl Client {
|
||||||
&self,
|
&self,
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
position: lsp::Position,
|
position: lsp::Position,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<lsp::PrepareRenameResponse>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
match capabilities.rename_provider {
|
match capabilities.rename_provider {
|
||||||
|
@ -1448,7 +1416,10 @@ impl Client {
|
||||||
}
|
}
|
||||||
|
|
||||||
// empty string to get all symbols
|
// empty string to get all symbols
|
||||||
pub fn workspace_symbols(&self, query: String) -> Option<impl Future<Output = Result<Value>>> {
|
pub fn workspace_symbols(
|
||||||
|
&self,
|
||||||
|
query: String,
|
||||||
|
) -> Option<impl Future<Output = Result<Option<lsp::WorkspaceSymbolResponse>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support workspace symbols.
|
// Return early if the server does not support workspace symbols.
|
||||||
|
@ -1471,7 +1442,7 @@ impl Client {
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
range: lsp::Range,
|
range: lsp::Range,
|
||||||
context: lsp::CodeActionContext,
|
context: lsp::CodeActionContext,
|
||||||
) -> Option<impl Future<Output = Result<Value>>> {
|
) -> Option<impl Future<Output = Result<Option<Vec<lsp::CodeActionOrCommand>>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the server does not support code actions.
|
// Return early if the server does not support code actions.
|
||||||
|
@ -1499,7 +1470,7 @@ impl Client {
|
||||||
text_document: lsp::TextDocumentIdentifier,
|
text_document: lsp::TextDocumentIdentifier,
|
||||||
position: lsp::Position,
|
position: lsp::Position,
|
||||||
new_name: String,
|
new_name: String,
|
||||||
) -> Option<impl Future<Output = Result<lsp::WorkspaceEdit>>> {
|
) -> Option<impl Future<Output = Result<Option<lsp::WorkspaceEdit>>>> {
|
||||||
if !self.supports_feature(LanguageServerFeature::RenameSymbol) {
|
if !self.supports_feature(LanguageServerFeature::RenameSymbol) {
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
@ -1515,16 +1486,13 @@ impl Client {
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
let request = self.call::<lsp::request::Rename>(params);
|
Some(self.call::<lsp::request::Rename>(params))
|
||||||
|
|
||||||
Some(async move {
|
|
||||||
let json = request.await?;
|
|
||||||
let response: Option<lsp::WorkspaceEdit> = serde_json::from_value(json)?;
|
|
||||||
Ok(response.unwrap_or_default())
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn command(&self, command: lsp::Command) -> Option<impl Future<Output = Result<Value>>> {
|
pub fn command(
|
||||||
|
&self,
|
||||||
|
command: lsp::Command,
|
||||||
|
) -> Option<impl Future<Output = Result<Option<Value>>>> {
|
||||||
let capabilities = self.capabilities.get().unwrap();
|
let capabilities = self.capabilities.get().unwrap();
|
||||||
|
|
||||||
// Return early if the language server does not support executing commands.
|
// Return early if the language server does not support executing commands.
|
||||||
|
|
|
@ -146,10 +146,10 @@ impl Context<'_> {
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn callback<T, F>(
|
pub fn callback<T, F>(
|
||||||
&mut self,
|
&mut self,
|
||||||
call: impl Future<Output = helix_lsp::Result<serde_json::Value>> + 'static + Send,
|
call: impl Future<Output = helix_lsp::Result<T>> + 'static + Send,
|
||||||
callback: F,
|
callback: F,
|
||||||
) where
|
) where
|
||||||
T: for<'de> serde::Deserialize<'de> + Send + 'static,
|
T: Send + 'static,
|
||||||
F: FnOnce(&mut Editor, &mut Compositor, T) + Send + 'static,
|
F: FnOnce(&mut Editor, &mut Compositor, T) + Send + 'static,
|
||||||
{
|
{
|
||||||
self.jobs.callback(make_job_callback(call, callback));
|
self.jobs.callback(make_job_callback(call, callback));
|
||||||
|
@ -175,16 +175,15 @@ impl Context<'_> {
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn make_job_callback<T, F>(
|
fn make_job_callback<T, F>(
|
||||||
call: impl Future<Output = helix_lsp::Result<serde_json::Value>> + 'static + Send,
|
call: impl Future<Output = helix_lsp::Result<T>> + 'static + Send,
|
||||||
callback: F,
|
callback: F,
|
||||||
) -> std::pin::Pin<Box<impl Future<Output = Result<Callback, anyhow::Error>>>>
|
) -> std::pin::Pin<Box<impl Future<Output = Result<Callback, anyhow::Error>>>>
|
||||||
where
|
where
|
||||||
T: for<'de> serde::Deserialize<'de> + Send + 'static,
|
T: Send + 'static,
|
||||||
F: FnOnce(&mut Editor, &mut Compositor, T) + Send + 'static,
|
F: FnOnce(&mut Editor, &mut Compositor, T) + Send + 'static,
|
||||||
{
|
{
|
||||||
Box::pin(async move {
|
Box::pin(async move {
|
||||||
let json = call.await?;
|
let response = call.await?;
|
||||||
let response = serde_json::from_value(json)?;
|
|
||||||
let call: job::Callback = Callback::EditorCompositor(Box::new(
|
let call: job::Callback = Callback::EditorCompositor(Box::new(
|
||||||
move |editor: &mut Editor, compositor: &mut Compositor| {
|
move |editor: &mut Editor, compositor: &mut Compositor| {
|
||||||
callback(editor, compositor, response)
|
callback(editor, compositor, response)
|
||||||
|
@ -4905,7 +4904,10 @@ fn format_selections(cx: &mut Context) {
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
let edits = tokio::task::block_in_place(|| helix_lsp::block_on(future)).unwrap_or_default();
|
let edits = tokio::task::block_in_place(|| helix_lsp::block_on(future))
|
||||||
|
.ok()
|
||||||
|
.flatten()
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
let transaction =
|
let transaction =
|
||||||
helix_lsp::util::generate_transaction_from_edits(doc.text(), edits, offset_encoding);
|
helix_lsp::util::generate_transaction_from_edits(doc.text(), edits, offset_encoding);
|
||||||
|
|
|
@ -343,9 +343,7 @@ pub fn symbol_picker(cx: &mut Context) {
|
||||||
.expect("docs with active language servers must be backed by paths");
|
.expect("docs with active language servers must be backed by paths");
|
||||||
|
|
||||||
async move {
|
async move {
|
||||||
let json = request.await?;
|
let symbols = match request.await? {
|
||||||
let response: Option<lsp::DocumentSymbolResponse> = serde_json::from_value(json)?;
|
|
||||||
let symbols = match response {
|
|
||||||
Some(symbols) => symbols,
|
Some(symbols) => symbols,
|
||||||
None => return anyhow::Ok(vec![]),
|
None => return anyhow::Ok(vec![]),
|
||||||
};
|
};
|
||||||
|
@ -461,30 +459,34 @@ pub fn workspace_symbol_picker(cx: &mut Context) {
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let offset_encoding = language_server.offset_encoding();
|
let offset_encoding = language_server.offset_encoding();
|
||||||
async move {
|
async move {
|
||||||
let json = request.await?;
|
let symbols = request
|
||||||
|
.await?
|
||||||
|
.and_then(|resp| match resp {
|
||||||
|
lsp::WorkspaceSymbolResponse::Flat(symbols) => Some(symbols),
|
||||||
|
lsp::WorkspaceSymbolResponse::Nested(_) => None,
|
||||||
|
})
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
let response: Vec<_> =
|
let response: Vec<_> = symbols
|
||||||
serde_json::from_value::<Option<Vec<lsp::SymbolInformation>>>(json)?
|
.into_iter()
|
||||||
.unwrap_or_default()
|
.filter_map(|symbol| {
|
||||||
.into_iter()
|
let uri = match Uri::try_from(&symbol.location.uri) {
|
||||||
.filter_map(|symbol| {
|
Ok(uri) => uri,
|
||||||
let uri = match Uri::try_from(&symbol.location.uri) {
|
Err(err) => {
|
||||||
Ok(uri) => uri,
|
log::warn!("discarding symbol with invalid URI: {err}");
|
||||||
Err(err) => {
|
return None;
|
||||||
log::warn!("discarding symbol with invalid URI: {err}");
|
}
|
||||||
return None;
|
};
|
||||||
}
|
Some(SymbolInformationItem {
|
||||||
};
|
location: Location {
|
||||||
Some(SymbolInformationItem {
|
uri,
|
||||||
location: Location {
|
range: symbol.location.range,
|
||||||
uri,
|
offset_encoding,
|
||||||
range: symbol.location.range,
|
},
|
||||||
offset_encoding,
|
symbol,
|
||||||
},
|
|
||||||
symbol,
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
.collect();
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
anyhow::Ok(response)
|
anyhow::Ok(response)
|
||||||
}
|
}
|
||||||
|
@ -676,11 +678,8 @@ pub fn code_action(cx: &mut Context) {
|
||||||
Some((code_action_request, language_server_id))
|
Some((code_action_request, language_server_id))
|
||||||
})
|
})
|
||||||
.map(|(request, ls_id)| async move {
|
.map(|(request, ls_id)| async move {
|
||||||
let json = request.await?;
|
let Some(mut actions) = request.await? else {
|
||||||
let response: Option<lsp::CodeActionResponse> = serde_json::from_value(json)?;
|
return anyhow::Ok(Vec::new());
|
||||||
let mut actions = match response {
|
|
||||||
Some(a) => a,
|
|
||||||
None => return anyhow::Ok(Vec::new()),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// remove disabled code actions
|
// remove disabled code actions
|
||||||
|
@ -782,15 +781,9 @@ pub fn code_action(cx: &mut Context) {
|
||||||
// we support lsp "codeAction/resolve" for `edit` and `command` fields
|
// we support lsp "codeAction/resolve" for `edit` and `command` fields
|
||||||
let mut resolved_code_action = None;
|
let mut resolved_code_action = None;
|
||||||
if code_action.edit.is_none() || code_action.command.is_none() {
|
if code_action.edit.is_none() || code_action.command.is_none() {
|
||||||
if let Some(future) =
|
if let Some(future) = language_server.resolve_code_action(code_action) {
|
||||||
language_server.resolve_code_action(code_action.clone())
|
if let Ok(code_action) = helix_lsp::block_on(future) {
|
||||||
{
|
resolved_code_action = Some(code_action);
|
||||||
if let Ok(response) = helix_lsp::block_on(future) {
|
|
||||||
if let Ok(code_action) =
|
|
||||||
serde_json::from_value::<CodeAction>(response)
|
|
||||||
{
|
|
||||||
resolved_code_action = Some(code_action);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -882,7 +875,7 @@ fn goto_impl(editor: &mut Editor, compositor: &mut Compositor, locations: Vec<Lo
|
||||||
fn goto_single_impl<P, F>(cx: &mut Context, feature: LanguageServerFeature, request_provider: P)
|
fn goto_single_impl<P, F>(cx: &mut Context, feature: LanguageServerFeature, request_provider: P)
|
||||||
where
|
where
|
||||||
P: Fn(&Client, lsp::Position, lsp::TextDocumentIdentifier) -> Option<F>,
|
P: Fn(&Client, lsp::Position, lsp::TextDocumentIdentifier) -> Option<F>,
|
||||||
F: Future<Output = helix_lsp::Result<serde_json::Value>> + 'static + Send,
|
F: Future<Output = helix_lsp::Result<Option<lsp::GotoDefinitionResponse>>> + 'static + Send,
|
||||||
{
|
{
|
||||||
let (view, doc) = current_ref!(cx.editor);
|
let (view, doc) = current_ref!(cx.editor);
|
||||||
let mut futures: FuturesOrdered<_> = doc
|
let mut futures: FuturesOrdered<_> = doc
|
||||||
|
@ -891,11 +884,7 @@ where
|
||||||
let offset_encoding = language_server.offset_encoding();
|
let offset_encoding = language_server.offset_encoding();
|
||||||
let pos = doc.position(view.id, offset_encoding);
|
let pos = doc.position(view.id, offset_encoding);
|
||||||
let future = request_provider(language_server, pos, doc.identifier()).unwrap();
|
let future = request_provider(language_server, pos, doc.identifier()).unwrap();
|
||||||
async move {
|
async move { anyhow::Ok((future.await?, offset_encoding)) }
|
||||||
let json = future.await?;
|
|
||||||
let response: Option<lsp::GotoDefinitionResponse> = serde_json::from_value(json)?;
|
|
||||||
anyhow::Ok((response, offset_encoding))
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
|
@ -992,11 +981,7 @@ pub fn goto_reference(cx: &mut Context) {
|
||||||
None,
|
None,
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
async move {
|
async move { anyhow::Ok((future.await?, offset_encoding)) }
|
||||||
let json = future.await?;
|
|
||||||
let locations: Option<Vec<lsp::Location>> = serde_json::from_value(json)?;
|
|
||||||
anyhow::Ok((locations, offset_encoding))
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
|
@ -1158,7 +1143,9 @@ pub fn rename_symbol(cx: &mut Context) {
|
||||||
|
|
||||||
match block_on(future) {
|
match block_on(future) {
|
||||||
Ok(edits) => {
|
Ok(edits) => {
|
||||||
let _ = cx.editor.apply_workspace_edit(offset_encoding, &edits);
|
let _ = cx
|
||||||
|
.editor
|
||||||
|
.apply_workspace_edit(offset_encoding, &edits.unwrap_or_default());
|
||||||
}
|
}
|
||||||
Err(err) => cx.editor.set_error(err.to_string()),
|
Err(err) => cx.editor.set_error(err.to_string()),
|
||||||
}
|
}
|
||||||
|
|
|
@ -303,7 +303,6 @@ fn request_completions_from_language_server(
|
||||||
async move {
|
async move {
|
||||||
let response: Option<lsp::CompletionResponse> = completion_response
|
let response: Option<lsp::CompletionResponse> = completion_response
|
||||||
.await
|
.await
|
||||||
.and_then(|json| serde_json::from_value(json).map_err(helix_lsp::Error::Parse))
|
|
||||||
.inspect_err(|err| log::error!("completion request failed: {err}"))
|
.inspect_err(|err| log::error!("completion request failed: {err}"))
|
||||||
.ok()
|
.ok()
|
||||||
.flatten();
|
.flatten();
|
||||||
|
|
|
@ -867,10 +867,13 @@ impl Document {
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
let fut = async move {
|
let fut = async move {
|
||||||
let edits = request.await.unwrap_or_else(|e| {
|
let edits = request
|
||||||
log::warn!("LSP formatting failed: {}", e);
|
.await
|
||||||
Default::default()
|
.unwrap_or_else(|e| {
|
||||||
});
|
log::warn!("LSP formatting failed: {}", e);
|
||||||
|
Default::default()
|
||||||
|
})
|
||||||
|
.unwrap_or_default();
|
||||||
Ok(helix_lsp::util::generate_transaction_from_edits(
|
Ok(helix_lsp::util::generate_transaction_from_edits(
|
||||||
&text,
|
&text,
|
||||||
edits,
|
edits,
|
||||||
|
|
|
@ -1410,7 +1410,7 @@ impl Editor {
|
||||||
continue;
|
continue;
|
||||||
};
|
};
|
||||||
let edit = match helix_lsp::block_on(request) {
|
let edit = match helix_lsp::block_on(request) {
|
||||||
Ok(edit) => edit,
|
Ok(edit) => edit.unwrap_or_default(),
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
log::error!("invalid willRename response: {err:?}");
|
log::error!("invalid willRename response: {err:?}");
|
||||||
continue;
|
continue;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue