Use new language server info on remote servers (#35682)
* Straightens out the `*_ext.rs` workflow for clangd and rust-analyzer: no need to asynchronously query for the language server, as we sync that information already. * Fixes inlay hints editor menu toggle not being shown in the remote sessions Release Notes: - Fixed inlay hints editor menu toggle not being shown in the remote sessions
This commit is contained in:
parent
cc93175256
commit
9caa9d042a
14 changed files with 176 additions and 291 deletions
|
@ -340,9 +340,6 @@ impl Server {
|
||||||
.add_request_handler(forward_read_only_project_request::<proto::LspExtCancelFlycheck>)
|
.add_request_handler(forward_read_only_project_request::<proto::LspExtCancelFlycheck>)
|
||||||
.add_request_handler(forward_read_only_project_request::<proto::LspExtRunFlycheck>)
|
.add_request_handler(forward_read_only_project_request::<proto::LspExtRunFlycheck>)
|
||||||
.add_request_handler(forward_read_only_project_request::<proto::LspExtClearFlycheck>)
|
.add_request_handler(forward_read_only_project_request::<proto::LspExtClearFlycheck>)
|
||||||
.add_request_handler(
|
|
||||||
forward_read_only_project_request::<proto::LanguageServerIdForName>,
|
|
||||||
)
|
|
||||||
.add_request_handler(forward_read_only_project_request::<proto::GetDocumentDiagnostics>)
|
.add_request_handler(forward_read_only_project_request::<proto::GetDocumentDiagnostics>)
|
||||||
.add_request_handler(
|
.add_request_handler(
|
||||||
forward_mutating_project_request::<proto::RegisterBufferWithLanguageServers>,
|
forward_mutating_project_request::<proto::RegisterBufferWithLanguageServers>,
|
||||||
|
|
|
@ -24,10 +24,7 @@ use language::{
|
||||||
};
|
};
|
||||||
use project::{
|
use project::{
|
||||||
ProjectPath, SERVER_PROGRESS_THROTTLE_TIMEOUT,
|
ProjectPath, SERVER_PROGRESS_THROTTLE_TIMEOUT,
|
||||||
lsp_store::{
|
lsp_store::lsp_ext_command::{ExpandedMacro, LspExtExpandMacro},
|
||||||
lsp_ext_command::{ExpandedMacro, LspExtExpandMacro},
|
|
||||||
rust_analyzer_ext::RUST_ANALYZER_NAME,
|
|
||||||
},
|
|
||||||
project_settings::{InlineBlameSettings, ProjectSettings},
|
project_settings::{InlineBlameSettings, ProjectSettings},
|
||||||
};
|
};
|
||||||
use recent_projects::disconnected_overlay::DisconnectedOverlay;
|
use recent_projects::disconnected_overlay::DisconnectedOverlay;
|
||||||
|
@ -3786,11 +3783,18 @@ async fn test_client_can_query_lsp_ext(cx_a: &mut TestAppContext, cx_b: &mut Tes
|
||||||
cx_b.update(editor::init);
|
cx_b.update(editor::init);
|
||||||
|
|
||||||
client_a.language_registry().add(rust_lang());
|
client_a.language_registry().add(rust_lang());
|
||||||
client_b.language_registry().add(rust_lang());
|
|
||||||
let mut fake_language_servers = client_a.language_registry().register_fake_lsp(
|
let mut fake_language_servers = client_a.language_registry().register_fake_lsp(
|
||||||
"Rust",
|
"Rust",
|
||||||
FakeLspAdapter {
|
FakeLspAdapter {
|
||||||
name: RUST_ANALYZER_NAME,
|
name: "rust-analyzer",
|
||||||
|
..FakeLspAdapter::default()
|
||||||
|
},
|
||||||
|
);
|
||||||
|
client_b.language_registry().add(rust_lang());
|
||||||
|
client_b.language_registry().register_fake_lsp_adapter(
|
||||||
|
"Rust",
|
||||||
|
FakeLspAdapter {
|
||||||
|
name: "rust-analyzer",
|
||||||
..FakeLspAdapter::default()
|
..FakeLspAdapter::default()
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
|
@ -29,16 +29,14 @@ pub fn switch_source_header(
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
let server_lookup =
|
let Some((_, _, server_to_query, buffer)) =
|
||||||
find_specific_language_server_in_selection(editor, cx, is_c_language, CLANGD_SERVER_NAME);
|
find_specific_language_server_in_selection(editor, cx, is_c_language, CLANGD_SERVER_NAME)
|
||||||
|
else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
let project = project.clone();
|
let project = project.clone();
|
||||||
let upstream_client = project.read(cx).lsp_store().read(cx).upstream_client();
|
let upstream_client = project.read(cx).lsp_store().read(cx).upstream_client();
|
||||||
cx.spawn_in(window, async move |_editor, cx| {
|
cx.spawn_in(window, async move |_editor, cx| {
|
||||||
let Some((_, _, server_to_query, buffer)) =
|
|
||||||
server_lookup.await
|
|
||||||
else {
|
|
||||||
return Ok(());
|
|
||||||
};
|
|
||||||
let source_file = buffer.read_with(cx, |buffer, _| {
|
let source_file = buffer.read_with(cx, |buffer, _| {
|
||||||
buffer.file().map(|file| file.path()).map(|path| path.to_string_lossy().to_string()).unwrap_or_else(|| "Unknown".to_string())
|
buffer.file().map(|file| file.path()).map(|path| path.to_string_lossy().to_string()).unwrap_or_else(|| "Unknown".to_string())
|
||||||
})?;
|
})?;
|
||||||
|
|
|
@ -22192,7 +22192,6 @@ impl SemanticsProvider for Entity<Project> {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn supports_inlay_hints(&self, buffer: &Entity<Buffer>, cx: &mut App) -> bool {
|
fn supports_inlay_hints(&self, buffer: &Entity<Buffer>, cx: &mut App) -> bool {
|
||||||
// TODO: make this work for remote projects
|
|
||||||
self.update(cx, |project, cx| {
|
self.update(cx, |project, cx| {
|
||||||
if project
|
if project
|
||||||
.active_debug_session(cx)
|
.active_debug_session(cx)
|
||||||
|
|
|
@ -3,9 +3,8 @@ use std::time::Duration;
|
||||||
|
|
||||||
use crate::Editor;
|
use crate::Editor;
|
||||||
use collections::HashMap;
|
use collections::HashMap;
|
||||||
use futures::stream::FuturesUnordered;
|
|
||||||
use gpui::AsyncApp;
|
use gpui::AsyncApp;
|
||||||
use gpui::{App, AppContext as _, Entity, Task};
|
use gpui::{App, Entity, Task};
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
use language::Buffer;
|
use language::Buffer;
|
||||||
use language::Language;
|
use language::Language;
|
||||||
|
@ -18,7 +17,6 @@ use project::Project;
|
||||||
use project::TaskSourceKind;
|
use project::TaskSourceKind;
|
||||||
use project::lsp_store::lsp_ext_command::GetLspRunnables;
|
use project::lsp_store::lsp_ext_command::GetLspRunnables;
|
||||||
use smol::future::FutureExt as _;
|
use smol::future::FutureExt as _;
|
||||||
use smol::stream::StreamExt;
|
|
||||||
use task::ResolvedTask;
|
use task::ResolvedTask;
|
||||||
use task::TaskContext;
|
use task::TaskContext;
|
||||||
use text::BufferId;
|
use text::BufferId;
|
||||||
|
@ -29,52 +27,32 @@ pub(crate) fn find_specific_language_server_in_selection<F>(
|
||||||
editor: &Editor,
|
editor: &Editor,
|
||||||
cx: &mut App,
|
cx: &mut App,
|
||||||
filter_language: F,
|
filter_language: F,
|
||||||
language_server_name: &str,
|
language_server_name: LanguageServerName,
|
||||||
) -> Task<Option<(Anchor, Arc<Language>, LanguageServerId, Entity<Buffer>)>>
|
) -> Option<(Anchor, Arc<Language>, LanguageServerId, Entity<Buffer>)>
|
||||||
where
|
where
|
||||||
F: Fn(&Language) -> bool,
|
F: Fn(&Language) -> bool,
|
||||||
{
|
{
|
||||||
let Some(project) = &editor.project else {
|
let project = editor.project.clone()?;
|
||||||
return Task::ready(None);
|
editor
|
||||||
};
|
|
||||||
|
|
||||||
let applicable_buffers = editor
|
|
||||||
.selections
|
.selections
|
||||||
.disjoint_anchors()
|
.disjoint_anchors()
|
||||||
.iter()
|
.iter()
|
||||||
.filter_map(|selection| Some((selection.head(), selection.head().buffer_id?)))
|
.filter_map(|selection| Some((selection.head(), selection.head().buffer_id?)))
|
||||||
.unique_by(|(_, buffer_id)| *buffer_id)
|
.unique_by(|(_, buffer_id)| *buffer_id)
|
||||||
.filter_map(|(trigger_anchor, buffer_id)| {
|
.find_map(|(trigger_anchor, buffer_id)| {
|
||||||
let buffer = editor.buffer().read(cx).buffer(buffer_id)?;
|
let buffer = editor.buffer().read(cx).buffer(buffer_id)?;
|
||||||
let language = buffer.read(cx).language_at(trigger_anchor.text_anchor)?;
|
let language = buffer.read(cx).language_at(trigger_anchor.text_anchor)?;
|
||||||
if filter_language(&language) {
|
if filter_language(&language) {
|
||||||
Some((trigger_anchor, buffer, language))
|
let server_id = buffer.update(cx, |buffer, cx| {
|
||||||
|
project
|
||||||
|
.read(cx)
|
||||||
|
.language_server_id_for_name(buffer, &language_server_name, cx)
|
||||||
|
})?;
|
||||||
|
Some((trigger_anchor, language, server_id, buffer))
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
let applicable_buffer_tasks = applicable_buffers
|
|
||||||
.into_iter()
|
|
||||||
.map(|(trigger_anchor, buffer, language)| {
|
|
||||||
let task = buffer.update(cx, |buffer, cx| {
|
|
||||||
project.update(cx, |project, cx| {
|
|
||||||
project.language_server_id_for_name(buffer, language_server_name, cx)
|
|
||||||
})
|
|
||||||
});
|
|
||||||
(trigger_anchor, buffer, language, task)
|
|
||||||
})
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
cx.background_spawn(async move {
|
|
||||||
for (trigger_anchor, buffer, language, task) in applicable_buffer_tasks {
|
|
||||||
if let Some(server_id) = task.await {
|
|
||||||
return Some((trigger_anchor, language, server_id, buffer));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
None
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn lsp_task_context(
|
async fn lsp_task_context(
|
||||||
|
@ -116,9 +94,9 @@ pub fn lsp_tasks(
|
||||||
for_position: Option<text::Anchor>,
|
for_position: Option<text::Anchor>,
|
||||||
cx: &mut App,
|
cx: &mut App,
|
||||||
) -> Task<Vec<(TaskSourceKind, Vec<(Option<LocationLink>, ResolvedTask)>)>> {
|
) -> Task<Vec<(TaskSourceKind, Vec<(Option<LocationLink>, ResolvedTask)>)>> {
|
||||||
let mut lsp_task_sources = task_sources
|
let lsp_task_sources = task_sources
|
||||||
.iter()
|
.iter()
|
||||||
.map(|(name, buffer_ids)| {
|
.filter_map(|(name, buffer_ids)| {
|
||||||
let buffers = buffer_ids
|
let buffers = buffer_ids
|
||||||
.iter()
|
.iter()
|
||||||
.filter(|&&buffer_id| match for_position {
|
.filter(|&&buffer_id| match for_position {
|
||||||
|
@ -127,61 +105,63 @@ pub fn lsp_tasks(
|
||||||
})
|
})
|
||||||
.filter_map(|&buffer_id| project.read(cx).buffer_for_id(buffer_id, cx))
|
.filter_map(|&buffer_id| project.read(cx).buffer_for_id(buffer_id, cx))
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
language_server_for_buffers(project.clone(), name.clone(), buffers, cx)
|
|
||||||
|
let server_id = buffers.iter().find_map(|buffer| {
|
||||||
|
project.read_with(cx, |project, cx| {
|
||||||
|
project.language_server_id_for_name(buffer.read(cx), name, cx)
|
||||||
|
})
|
||||||
|
});
|
||||||
|
server_id.zip(Some(buffers))
|
||||||
})
|
})
|
||||||
.collect::<FuturesUnordered<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
cx.spawn(async move |cx| {
|
cx.spawn(async move |cx| {
|
||||||
cx.spawn(async move |cx| {
|
cx.spawn(async move |cx| {
|
||||||
let mut lsp_tasks = HashMap::default();
|
let mut lsp_tasks = HashMap::default();
|
||||||
while let Some(server_to_query) = lsp_task_sources.next().await {
|
for (server_id, buffers) in lsp_task_sources {
|
||||||
if let Some((server_id, buffers)) = server_to_query {
|
let mut new_lsp_tasks = Vec::new();
|
||||||
let mut new_lsp_tasks = Vec::new();
|
for buffer in buffers {
|
||||||
for buffer in buffers {
|
let source_kind = match buffer.update(cx, |buffer, _| {
|
||||||
let source_kind = match buffer.update(cx, |buffer, _| {
|
buffer.language().map(|language| language.name())
|
||||||
buffer.language().map(|language| language.name())
|
}) {
|
||||||
}) {
|
Ok(Some(language_name)) => TaskSourceKind::Lsp {
|
||||||
Ok(Some(language_name)) => TaskSourceKind::Lsp {
|
server: server_id,
|
||||||
server: server_id,
|
language_name: SharedString::from(language_name),
|
||||||
language_name: SharedString::from(language_name),
|
},
|
||||||
},
|
Ok(None) => continue,
|
||||||
Ok(None) => continue,
|
Err(_) => return Vec::new(),
|
||||||
Err(_) => return Vec::new(),
|
};
|
||||||
};
|
let id_base = source_kind.to_id_base();
|
||||||
let id_base = source_kind.to_id_base();
|
let lsp_buffer_context = lsp_task_context(&project, &buffer, cx)
|
||||||
let lsp_buffer_context = lsp_task_context(&project, &buffer, cx)
|
.await
|
||||||
.await
|
.unwrap_or_default();
|
||||||
.unwrap_or_default();
|
|
||||||
|
|
||||||
if let Ok(runnables_task) = project.update(cx, |project, cx| {
|
if let Ok(runnables_task) = project.update(cx, |project, cx| {
|
||||||
let buffer_id = buffer.read(cx).remote_id();
|
let buffer_id = buffer.read(cx).remote_id();
|
||||||
project.request_lsp(
|
project.request_lsp(
|
||||||
buffer,
|
buffer,
|
||||||
LanguageServerToQuery::Other(server_id),
|
LanguageServerToQuery::Other(server_id),
|
||||||
GetLspRunnables {
|
GetLspRunnables {
|
||||||
buffer_id,
|
buffer_id,
|
||||||
position: for_position,
|
position: for_position,
|
||||||
|
},
|
||||||
|
cx,
|
||||||
|
)
|
||||||
|
}) {
|
||||||
|
if let Some(new_runnables) = runnables_task.await.log_err() {
|
||||||
|
new_lsp_tasks.extend(new_runnables.runnables.into_iter().filter_map(
|
||||||
|
|(location, runnable)| {
|
||||||
|
let resolved_task =
|
||||||
|
runnable.resolve_task(&id_base, &lsp_buffer_context)?;
|
||||||
|
Some((location, resolved_task))
|
||||||
},
|
},
|
||||||
cx,
|
));
|
||||||
)
|
|
||||||
}) {
|
|
||||||
if let Some(new_runnables) = runnables_task.await.log_err() {
|
|
||||||
new_lsp_tasks.extend(
|
|
||||||
new_runnables.runnables.into_iter().filter_map(
|
|
||||||
|(location, runnable)| {
|
|
||||||
let resolved_task = runnable
|
|
||||||
.resolve_task(&id_base, &lsp_buffer_context)?;
|
|
||||||
Some((location, resolved_task))
|
|
||||||
},
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
lsp_tasks
|
|
||||||
.entry(source_kind)
|
|
||||||
.or_insert_with(Vec::new)
|
|
||||||
.append(&mut new_lsp_tasks);
|
|
||||||
}
|
}
|
||||||
|
lsp_tasks
|
||||||
|
.entry(source_kind)
|
||||||
|
.or_insert_with(Vec::new)
|
||||||
|
.append(&mut new_lsp_tasks);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
lsp_tasks.into_iter().collect()
|
lsp_tasks.into_iter().collect()
|
||||||
|
@ -198,27 +178,3 @@ pub fn lsp_tasks(
|
||||||
.await
|
.await
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn language_server_for_buffers(
|
|
||||||
project: Entity<Project>,
|
|
||||||
name: LanguageServerName,
|
|
||||||
candidates: Vec<Entity<Buffer>>,
|
|
||||||
cx: &mut App,
|
|
||||||
) -> Task<Option<(LanguageServerId, Vec<Entity<Buffer>>)>> {
|
|
||||||
cx.spawn(async move |cx| {
|
|
||||||
for buffer in &candidates {
|
|
||||||
let server_id = buffer
|
|
||||||
.update(cx, |buffer, cx| {
|
|
||||||
project.update(cx, |project, cx| {
|
|
||||||
project.language_server_id_for_name(buffer, &name.0, cx)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
.ok()?
|
|
||||||
.await;
|
|
||||||
if let Some(server_id) = server_id {
|
|
||||||
return Some((server_id, candidates));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
None
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
|
@ -57,21 +57,21 @@ pub fn go_to_parent_module(
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
let server_lookup = find_specific_language_server_in_selection(
|
let Some((trigger_anchor, _, server_to_query, buffer)) =
|
||||||
editor,
|
find_specific_language_server_in_selection(
|
||||||
cx,
|
editor,
|
||||||
is_rust_language,
|
cx,
|
||||||
RUST_ANALYZER_NAME,
|
is_rust_language,
|
||||||
);
|
RUST_ANALYZER_NAME,
|
||||||
|
)
|
||||||
|
else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
let project = project.clone();
|
let project = project.clone();
|
||||||
let lsp_store = project.read(cx).lsp_store();
|
let lsp_store = project.read(cx).lsp_store();
|
||||||
let upstream_client = lsp_store.read(cx).upstream_client();
|
let upstream_client = lsp_store.read(cx).upstream_client();
|
||||||
cx.spawn_in(window, async move |editor, cx| {
|
cx.spawn_in(window, async move |editor, cx| {
|
||||||
let Some((trigger_anchor, _, server_to_query, buffer)) = server_lookup.await else {
|
|
||||||
return anyhow::Ok(());
|
|
||||||
};
|
|
||||||
|
|
||||||
let location_links = if let Some((client, project_id)) = upstream_client {
|
let location_links = if let Some((client, project_id)) = upstream_client {
|
||||||
let buffer_id = buffer.read_with(cx, |buffer, _| buffer.remote_id())?;
|
let buffer_id = buffer.read_with(cx, |buffer, _| buffer.remote_id())?;
|
||||||
|
|
||||||
|
@ -121,7 +121,7 @@ pub fn go_to_parent_module(
|
||||||
)
|
)
|
||||||
})?
|
})?
|
||||||
.await?;
|
.await?;
|
||||||
Ok(())
|
anyhow::Ok(())
|
||||||
})
|
})
|
||||||
.detach_and_log_err(cx);
|
.detach_and_log_err(cx);
|
||||||
}
|
}
|
||||||
|
@ -139,21 +139,19 @@ pub fn expand_macro_recursively(
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
let server_lookup = find_specific_language_server_in_selection(
|
let Some((trigger_anchor, rust_language, server_to_query, buffer)) =
|
||||||
editor,
|
find_specific_language_server_in_selection(
|
||||||
cx,
|
editor,
|
||||||
is_rust_language,
|
cx,
|
||||||
RUST_ANALYZER_NAME,
|
is_rust_language,
|
||||||
);
|
RUST_ANALYZER_NAME,
|
||||||
|
)
|
||||||
|
else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
let project = project.clone();
|
let project = project.clone();
|
||||||
let upstream_client = project.read(cx).lsp_store().read(cx).upstream_client();
|
let upstream_client = project.read(cx).lsp_store().read(cx).upstream_client();
|
||||||
cx.spawn_in(window, async move |_editor, cx| {
|
cx.spawn_in(window, async move |_editor, cx| {
|
||||||
let Some((trigger_anchor, rust_language, server_to_query, buffer)) = server_lookup.await
|
|
||||||
else {
|
|
||||||
return Ok(());
|
|
||||||
};
|
|
||||||
|
|
||||||
let macro_expansion = if let Some((client, project_id)) = upstream_client {
|
let macro_expansion = if let Some((client, project_id)) = upstream_client {
|
||||||
let buffer_id = buffer.update(cx, |buffer, _| buffer.remote_id())?;
|
let buffer_id = buffer.update(cx, |buffer, _| buffer.remote_id())?;
|
||||||
let request = proto::LspExtExpandMacro {
|
let request = proto::LspExtExpandMacro {
|
||||||
|
@ -231,20 +229,20 @@ pub fn open_docs(editor: &mut Editor, _: &OpenDocs, window: &mut Window, cx: &mu
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
let server_lookup = find_specific_language_server_in_selection(
|
let Some((trigger_anchor, _, server_to_query, buffer)) =
|
||||||
editor,
|
find_specific_language_server_in_selection(
|
||||||
cx,
|
editor,
|
||||||
is_rust_language,
|
cx,
|
||||||
RUST_ANALYZER_NAME,
|
is_rust_language,
|
||||||
);
|
RUST_ANALYZER_NAME,
|
||||||
|
)
|
||||||
|
else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
let project = project.clone();
|
let project = project.clone();
|
||||||
let upstream_client = project.read(cx).lsp_store().read(cx).upstream_client();
|
let upstream_client = project.read(cx).lsp_store().read(cx).upstream_client();
|
||||||
cx.spawn_in(window, async move |_editor, cx| {
|
cx.spawn_in(window, async move |_editor, cx| {
|
||||||
let Some((trigger_anchor, _, server_to_query, buffer)) = server_lookup.await else {
|
|
||||||
return Ok(());
|
|
||||||
};
|
|
||||||
|
|
||||||
let docs_urls = if let Some((client, project_id)) = upstream_client {
|
let docs_urls = if let Some((client, project_id)) = upstream_client {
|
||||||
let buffer_id = buffer.read_with(cx, |buffer, _| buffer.remote_id())?;
|
let buffer_id = buffer.read_with(cx, |buffer, _| buffer.remote_id())?;
|
||||||
let request = proto::LspExtOpenDocs {
|
let request = proto::LspExtOpenDocs {
|
||||||
|
|
|
@ -3284,6 +3284,16 @@ impl InlayHints {
|
||||||
})
|
})
|
||||||
.unwrap_or(false)
|
.unwrap_or(false)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn check_capabilities(capabilities: &ServerCapabilities) -> bool {
|
||||||
|
capabilities
|
||||||
|
.inlay_hint_provider
|
||||||
|
.as_ref()
|
||||||
|
.is_some_and(|inlay_hint_provider| match inlay_hint_provider {
|
||||||
|
lsp::OneOf::Left(enabled) => *enabled,
|
||||||
|
lsp::OneOf::Right(_) => true,
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait(?Send)]
|
#[async_trait(?Send)]
|
||||||
|
@ -3297,17 +3307,7 @@ impl LspCommand for InlayHints {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn check_capabilities(&self, capabilities: AdapterServerCapabilities) -> bool {
|
fn check_capabilities(&self, capabilities: AdapterServerCapabilities) -> bool {
|
||||||
let Some(inlay_hint_provider) = &capabilities.server_capabilities.inlay_hint_provider
|
Self::check_capabilities(&capabilities.server_capabilities)
|
||||||
else {
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
match inlay_hint_provider {
|
|
||||||
lsp::OneOf::Left(enabled) => *enabled,
|
|
||||||
lsp::OneOf::Right(inlay_hint_capabilities) => match inlay_hint_capabilities {
|
|
||||||
lsp::InlayHintServerCapabilities::Options(_) => true,
|
|
||||||
lsp::InlayHintServerCapabilities::RegistrationOptions(_) => false,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn to_lsp(
|
fn to_lsp(
|
||||||
|
|
|
@ -3671,7 +3671,6 @@ impl LspStore {
|
||||||
client.add_entity_request_handler(Self::handle_apply_additional_edits_for_completion);
|
client.add_entity_request_handler(Self::handle_apply_additional_edits_for_completion);
|
||||||
client.add_entity_request_handler(Self::handle_register_buffer_with_language_servers);
|
client.add_entity_request_handler(Self::handle_register_buffer_with_language_servers);
|
||||||
client.add_entity_request_handler(Self::handle_rename_project_entry);
|
client.add_entity_request_handler(Self::handle_rename_project_entry);
|
||||||
client.add_entity_request_handler(Self::handle_language_server_id_for_name);
|
|
||||||
client.add_entity_request_handler(Self::handle_pull_workspace_diagnostics);
|
client.add_entity_request_handler(Self::handle_pull_workspace_diagnostics);
|
||||||
client.add_entity_request_handler(Self::handle_lsp_command::<GetCodeActions>);
|
client.add_entity_request_handler(Self::handle_lsp_command::<GetCodeActions>);
|
||||||
client.add_entity_request_handler(Self::handle_lsp_command::<GetCompletions>);
|
client.add_entity_request_handler(Self::handle_lsp_command::<GetCompletions>);
|
||||||
|
@ -8745,34 +8744,6 @@ impl LspStore {
|
||||||
Ok(proto::Ack {})
|
Ok(proto::Ack {})
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn handle_language_server_id_for_name(
|
|
||||||
lsp_store: Entity<Self>,
|
|
||||||
envelope: TypedEnvelope<proto::LanguageServerIdForName>,
|
|
||||||
mut cx: AsyncApp,
|
|
||||||
) -> Result<proto::LanguageServerIdForNameResponse> {
|
|
||||||
let name = &envelope.payload.name;
|
|
||||||
let buffer_id = BufferId::new(envelope.payload.buffer_id)?;
|
|
||||||
lsp_store
|
|
||||||
.update(&mut cx, |lsp_store, cx| {
|
|
||||||
let buffer = lsp_store.buffer_store.read(cx).get_existing(buffer_id)?;
|
|
||||||
let server_id = buffer.update(cx, |buffer, cx| {
|
|
||||||
lsp_store
|
|
||||||
.language_servers_for_local_buffer(buffer, cx)
|
|
||||||
.find_map(|(adapter, server)| {
|
|
||||||
if adapter.name.0.as_ref() == name {
|
|
||||||
Some(server.server_id())
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
})
|
|
||||||
});
|
|
||||||
Ok(server_id)
|
|
||||||
})?
|
|
||||||
.map(|server_id| proto::LanguageServerIdForNameResponse {
|
|
||||||
server_id: server_id.map(|id| id.to_proto()),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn handle_rename_project_entry(
|
async fn handle_rename_project_entry(
|
||||||
this: Entity<Self>,
|
this: Entity<Self>,
|
||||||
envelope: TypedEnvelope<proto::RenameProjectEntry>,
|
envelope: TypedEnvelope<proto::RenameProjectEntry>,
|
||||||
|
|
|
@ -3,12 +3,12 @@ use std::sync::Arc;
|
||||||
use ::serde::{Deserialize, Serialize};
|
use ::serde::{Deserialize, Serialize};
|
||||||
use gpui::WeakEntity;
|
use gpui::WeakEntity;
|
||||||
use language::{CachedLspAdapter, Diagnostic, DiagnosticSourceKind};
|
use language::{CachedLspAdapter, Diagnostic, DiagnosticSourceKind};
|
||||||
use lsp::LanguageServer;
|
use lsp::{LanguageServer, LanguageServerName};
|
||||||
use util::ResultExt as _;
|
use util::ResultExt as _;
|
||||||
|
|
||||||
use crate::LspStore;
|
use crate::LspStore;
|
||||||
|
|
||||||
pub const CLANGD_SERVER_NAME: &str = "clangd";
|
pub const CLANGD_SERVER_NAME: LanguageServerName = LanguageServerName::new_static("clangd");
|
||||||
const INACTIVE_REGION_MESSAGE: &str = "inactive region";
|
const INACTIVE_REGION_MESSAGE: &str = "inactive region";
|
||||||
const INACTIVE_DIAGNOSTIC_SEVERITY: lsp::DiagnosticSeverity = lsp::DiagnosticSeverity::INFORMATION;
|
const INACTIVE_DIAGNOSTIC_SEVERITY: lsp::DiagnosticSeverity = lsp::DiagnosticSeverity::INFORMATION;
|
||||||
|
|
||||||
|
@ -34,7 +34,7 @@ pub fn is_inactive_region(diag: &Diagnostic) -> bool {
|
||||||
&& diag
|
&& diag
|
||||||
.source
|
.source
|
||||||
.as_ref()
|
.as_ref()
|
||||||
.is_some_and(|v| v == CLANGD_SERVER_NAME)
|
.is_some_and(|v| v == &CLANGD_SERVER_NAME.0)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn is_lsp_inactive_region(diag: &lsp::Diagnostic) -> bool {
|
pub fn is_lsp_inactive_region(diag: &lsp::Diagnostic) -> bool {
|
||||||
|
@ -43,7 +43,7 @@ pub fn is_lsp_inactive_region(diag: &lsp::Diagnostic) -> bool {
|
||||||
&& diag
|
&& diag
|
||||||
.source
|
.source
|
||||||
.as_ref()
|
.as_ref()
|
||||||
.is_some_and(|v| v == CLANGD_SERVER_NAME)
|
.is_some_and(|v| v == &CLANGD_SERVER_NAME.0)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn register_notifications(
|
pub fn register_notifications(
|
||||||
|
@ -51,7 +51,7 @@ pub fn register_notifications(
|
||||||
language_server: &LanguageServer,
|
language_server: &LanguageServer,
|
||||||
adapter: Arc<CachedLspAdapter>,
|
adapter: Arc<CachedLspAdapter>,
|
||||||
) {
|
) {
|
||||||
if language_server.name().0 != CLANGD_SERVER_NAME {
|
if language_server.name() != CLANGD_SERVER_NAME {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let server_id = language_server.server_id();
|
let server_id = language_server.server_id();
|
||||||
|
|
|
@ -2,12 +2,12 @@ use ::serde::{Deserialize, Serialize};
|
||||||
use anyhow::Context as _;
|
use anyhow::Context as _;
|
||||||
use gpui::{App, Entity, Task, WeakEntity};
|
use gpui::{App, Entity, Task, WeakEntity};
|
||||||
use language::ServerHealth;
|
use language::ServerHealth;
|
||||||
use lsp::LanguageServer;
|
use lsp::{LanguageServer, LanguageServerName};
|
||||||
use rpc::proto;
|
use rpc::proto;
|
||||||
|
|
||||||
use crate::{LspStore, LspStoreEvent, Project, ProjectPath, lsp_store};
|
use crate::{LspStore, LspStoreEvent, Project, ProjectPath, lsp_store};
|
||||||
|
|
||||||
pub const RUST_ANALYZER_NAME: &str = "rust-analyzer";
|
pub const RUST_ANALYZER_NAME: LanguageServerName = LanguageServerName::new_static("rust-analyzer");
|
||||||
pub const CARGO_DIAGNOSTICS_SOURCE_NAME: &str = "rustc";
|
pub const CARGO_DIAGNOSTICS_SOURCE_NAME: &str = "rustc";
|
||||||
|
|
||||||
/// Experimental: Informs the end user about the state of the server
|
/// Experimental: Informs the end user about the state of the server
|
||||||
|
@ -97,13 +97,9 @@ pub fn cancel_flycheck(
|
||||||
|
|
||||||
cx.spawn(async move |cx| {
|
cx.spawn(async move |cx| {
|
||||||
let buffer = buffer.await?;
|
let buffer = buffer.await?;
|
||||||
let Some(rust_analyzer_server) = project
|
let Some(rust_analyzer_server) = project.read_with(cx, |project, cx| {
|
||||||
.update(cx, |project, cx| {
|
project.language_server_id_for_name(buffer.read(cx), &RUST_ANALYZER_NAME, cx)
|
||||||
buffer.update(cx, |buffer, cx| {
|
})?
|
||||||
project.language_server_id_for_name(buffer, RUST_ANALYZER_NAME, cx)
|
|
||||||
})
|
|
||||||
})?
|
|
||||||
.await
|
|
||||||
else {
|
else {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
};
|
};
|
||||||
|
@ -148,13 +144,9 @@ pub fn run_flycheck(
|
||||||
|
|
||||||
cx.spawn(async move |cx| {
|
cx.spawn(async move |cx| {
|
||||||
let buffer = buffer.await?;
|
let buffer = buffer.await?;
|
||||||
let Some(rust_analyzer_server) = project
|
let Some(rust_analyzer_server) = project.read_with(cx, |project, cx| {
|
||||||
.update(cx, |project, cx| {
|
project.language_server_id_for_name(buffer.read(cx), &RUST_ANALYZER_NAME, cx)
|
||||||
buffer.update(cx, |buffer, cx| {
|
})?
|
||||||
project.language_server_id_for_name(buffer, RUST_ANALYZER_NAME, cx)
|
|
||||||
})
|
|
||||||
})?
|
|
||||||
.await
|
|
||||||
else {
|
else {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
};
|
};
|
||||||
|
@ -204,13 +196,9 @@ pub fn clear_flycheck(
|
||||||
|
|
||||||
cx.spawn(async move |cx| {
|
cx.spawn(async move |cx| {
|
||||||
let buffer = buffer.await?;
|
let buffer = buffer.await?;
|
||||||
let Some(rust_analyzer_server) = project
|
let Some(rust_analyzer_server) = project.read_with(cx, |project, cx| {
|
||||||
.update(cx, |project, cx| {
|
project.language_server_id_for_name(buffer.read(cx), &RUST_ANALYZER_NAME, cx)
|
||||||
buffer.update(cx, |buffer, cx| {
|
})?
|
||||||
project.language_server_id_for_name(buffer, RUST_ANALYZER_NAME, cx)
|
|
||||||
})
|
|
||||||
})?
|
|
||||||
.await
|
|
||||||
else {
|
else {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
};
|
};
|
||||||
|
|
|
@ -5002,63 +5002,53 @@ impl Project {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn any_language_server_supports_inlay_hints(&self, buffer: &Buffer, cx: &mut App) -> bool {
|
pub fn any_language_server_supports_inlay_hints(&self, buffer: &Buffer, cx: &mut App) -> bool {
|
||||||
self.lsp_store.update(cx, |this, cx| {
|
let Some(language) = buffer.language().cloned() else {
|
||||||
this.language_servers_for_local_buffer(buffer, cx)
|
return false;
|
||||||
.any(
|
};
|
||||||
|(_, server)| match server.capabilities().inlay_hint_provider {
|
self.lsp_store.update(cx, |lsp_store, _| {
|
||||||
Some(lsp::OneOf::Left(enabled)) => enabled,
|
let relevant_language_servers = lsp_store
|
||||||
Some(lsp::OneOf::Right(_)) => true,
|
.languages
|
||||||
None => false,
|
.lsp_adapters(&language.name())
|
||||||
},
|
.into_iter()
|
||||||
)
|
.map(|lsp_adapter| lsp_adapter.name())
|
||||||
|
.collect::<HashSet<_>>();
|
||||||
|
lsp_store
|
||||||
|
.language_server_statuses()
|
||||||
|
.filter_map(|(server_id, server_status)| {
|
||||||
|
relevant_language_servers
|
||||||
|
.contains(&server_status.name)
|
||||||
|
.then_some(server_id)
|
||||||
|
})
|
||||||
|
.filter_map(|server_id| lsp_store.lsp_server_capabilities.get(&server_id))
|
||||||
|
.any(InlayHints::check_capabilities)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn language_server_id_for_name(
|
pub fn language_server_id_for_name(
|
||||||
&self,
|
&self,
|
||||||
buffer: &Buffer,
|
buffer: &Buffer,
|
||||||
name: &str,
|
name: &LanguageServerName,
|
||||||
cx: &mut App,
|
cx: &App,
|
||||||
) -> Task<Option<LanguageServerId>> {
|
) -> Option<LanguageServerId> {
|
||||||
if self.is_local() {
|
let language = buffer.language()?;
|
||||||
Task::ready(self.lsp_store.update(cx, |lsp_store, cx| {
|
let relevant_language_servers = self
|
||||||
lsp_store
|
.languages
|
||||||
.language_servers_for_local_buffer(buffer, cx)
|
.lsp_adapters(&language.name())
|
||||||
.find_map(|(adapter, server)| {
|
.into_iter()
|
||||||
if adapter.name.0 == name {
|
.map(|lsp_adapter| lsp_adapter.name())
|
||||||
Some(server.server_id())
|
.collect::<HashSet<_>>();
|
||||||
} else {
|
if !relevant_language_servers.contains(name) {
|
||||||
None
|
return None;
|
||||||
}
|
|
||||||
})
|
|
||||||
}))
|
|
||||||
} else if let Some(project_id) = self.remote_id() {
|
|
||||||
let request = self.client.request(proto::LanguageServerIdForName {
|
|
||||||
project_id,
|
|
||||||
buffer_id: buffer.remote_id().to_proto(),
|
|
||||||
name: name.to_string(),
|
|
||||||
});
|
|
||||||
cx.background_spawn(async move {
|
|
||||||
let response = request.await.log_err()?;
|
|
||||||
response.server_id.map(LanguageServerId::from_proto)
|
|
||||||
})
|
|
||||||
} else if let Some(ssh_client) = self.ssh_client.as_ref() {
|
|
||||||
let request =
|
|
||||||
ssh_client
|
|
||||||
.read(cx)
|
|
||||||
.proto_client()
|
|
||||||
.request(proto::LanguageServerIdForName {
|
|
||||||
project_id: SSH_PROJECT_ID,
|
|
||||||
buffer_id: buffer.remote_id().to_proto(),
|
|
||||||
name: name.to_string(),
|
|
||||||
});
|
|
||||||
cx.background_spawn(async move {
|
|
||||||
let response = request.await.log_err()?;
|
|
||||||
response.server_id.map(LanguageServerId::from_proto)
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
Task::ready(None)
|
|
||||||
}
|
}
|
||||||
|
self.language_server_statuses(cx)
|
||||||
|
.filter(|(_, server_status)| relevant_language_servers.contains(&server_status.name))
|
||||||
|
.find_map(|(server_id, server_status)| {
|
||||||
|
if &server_status.name == name {
|
||||||
|
Some(server_id)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn has_language_servers_for(&self, buffer: &Buffer, cx: &mut App) -> bool {
|
pub fn has_language_servers_for(&self, buffer: &Buffer, cx: &mut App) -> bool {
|
||||||
|
|
|
@ -818,16 +818,6 @@ message LspResponse {
|
||||||
uint64 server_id = 7;
|
uint64 server_id = 7;
|
||||||
}
|
}
|
||||||
|
|
||||||
message LanguageServerIdForName {
|
|
||||||
uint64 project_id = 1;
|
|
||||||
uint64 buffer_id = 2;
|
|
||||||
string name = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
message LanguageServerIdForNameResponse {
|
|
||||||
optional uint64 server_id = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message LspExtRunnables {
|
message LspExtRunnables {
|
||||||
uint64 project_id = 1;
|
uint64 project_id = 1;
|
||||||
uint64 buffer_id = 2;
|
uint64 buffer_id = 2;
|
||||||
|
|
|
@ -362,9 +362,6 @@ message Envelope {
|
||||||
GetDocumentSymbols get_document_symbols = 330;
|
GetDocumentSymbols get_document_symbols = 330;
|
||||||
GetDocumentSymbolsResponse get_document_symbols_response = 331;
|
GetDocumentSymbolsResponse get_document_symbols_response = 331;
|
||||||
|
|
||||||
LanguageServerIdForName language_server_id_for_name = 332;
|
|
||||||
LanguageServerIdForNameResponse language_server_id_for_name_response = 333;
|
|
||||||
|
|
||||||
LoadCommitDiff load_commit_diff = 334;
|
LoadCommitDiff load_commit_diff = 334;
|
||||||
LoadCommitDiffResponse load_commit_diff_response = 335;
|
LoadCommitDiffResponse load_commit_diff_response = 335;
|
||||||
|
|
||||||
|
@ -424,6 +421,7 @@ message Envelope {
|
||||||
reserved 247 to 254;
|
reserved 247 to 254;
|
||||||
reserved 255 to 256;
|
reserved 255 to 256;
|
||||||
reserved 280 to 281;
|
reserved 280 to 281;
|
||||||
|
reserved 332 to 333;
|
||||||
}
|
}
|
||||||
|
|
||||||
message Hello {
|
message Hello {
|
||||||
|
|
|
@ -121,8 +121,6 @@ messages!(
|
||||||
(GetImplementationResponse, Background),
|
(GetImplementationResponse, Background),
|
||||||
(GetLlmToken, Background),
|
(GetLlmToken, Background),
|
||||||
(GetLlmTokenResponse, Background),
|
(GetLlmTokenResponse, Background),
|
||||||
(LanguageServerIdForName, Background),
|
|
||||||
(LanguageServerIdForNameResponse, Background),
|
|
||||||
(OpenUnstagedDiff, Foreground),
|
(OpenUnstagedDiff, Foreground),
|
||||||
(OpenUnstagedDiffResponse, Foreground),
|
(OpenUnstagedDiffResponse, Foreground),
|
||||||
(OpenUncommittedDiff, Foreground),
|
(OpenUncommittedDiff, Foreground),
|
||||||
|
@ -431,7 +429,6 @@ request_messages!(
|
||||||
(UpdateWorktree, Ack),
|
(UpdateWorktree, Ack),
|
||||||
(UpdateRepository, Ack),
|
(UpdateRepository, Ack),
|
||||||
(RemoveRepository, Ack),
|
(RemoveRepository, Ack),
|
||||||
(LanguageServerIdForName, LanguageServerIdForNameResponse),
|
|
||||||
(LspExtExpandMacro, LspExtExpandMacroResponse),
|
(LspExtExpandMacro, LspExtExpandMacroResponse),
|
||||||
(LspExtOpenDocs, LspExtOpenDocsResponse),
|
(LspExtOpenDocs, LspExtOpenDocsResponse),
|
||||||
(LspExtRunnables, LspExtRunnablesResponse),
|
(LspExtRunnables, LspExtRunnablesResponse),
|
||||||
|
@ -588,7 +585,6 @@ entity_messages!(
|
||||||
OpenServerSettings,
|
OpenServerSettings,
|
||||||
GetPermalinkToLine,
|
GetPermalinkToLine,
|
||||||
LanguageServerPromptRequest,
|
LanguageServerPromptRequest,
|
||||||
LanguageServerIdForName,
|
|
||||||
GitGetBranches,
|
GitGetBranches,
|
||||||
UpdateGitBranch,
|
UpdateGitBranch,
|
||||||
ListToolchains,
|
ListToolchains,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue