mirror of
https://github.com/helix-editor/helix.git
synced 2025-04-04 11:27:46 +03:00
feat: specify custom lang server(s) for :lsp-stop
and :lsp-restart
(#12578)
Co-authored-by: Nikita Revenco <154856872+NikitaRevenco@users.noreply.github.com>
This commit is contained in:
parent
4ded712dbd
commit
a63a2ad281
4 changed files with 93 additions and 60 deletions
|
@ -52,8 +52,8 @@
|
||||||
| `:reload-all`, `:rla` | Discard changes and reload all documents from the source files. |
|
| `:reload-all`, `:rla` | Discard changes and reload all documents from the source files. |
|
||||||
| `:update`, `:u` | Write changes only if the file has been modified. |
|
| `:update`, `:u` | Write changes only if the file has been modified. |
|
||||||
| `:lsp-workspace-command` | Open workspace command picker |
|
| `:lsp-workspace-command` | Open workspace command picker |
|
||||||
| `:lsp-restart` | Restarts the language servers used by the current doc |
|
| `:lsp-restart` | Restarts the given language servers, or all language servers that are used by the current file if no arguments are supplied |
|
||||||
| `:lsp-stop` | Stops the language servers that are used by the current doc |
|
| `:lsp-stop` | Stops the given language servers, or all language servers that are used by the current file if no arguments are supplied |
|
||||||
| `:tree-sitter-scopes` | Display tree sitter scopes, primarily for theming and development. |
|
| `:tree-sitter-scopes` | Display tree sitter scopes, primarily for theming and development. |
|
||||||
| `:tree-sitter-highlight-name` | Display name of tree-sitter highlight scope under the cursor. |
|
| `:tree-sitter-highlight-name` | Display name of tree-sitter highlight scope under the cursor. |
|
||||||
| `:debug-start`, `:dbg` | Start a debug session from a given template with given parameters. |
|
| `:debug-start`, `:dbg` | Start a debug session from a given template with given parameters. |
|
||||||
|
|
|
@ -618,51 +618,45 @@ impl Registry {
|
||||||
Ok(self.inner[id].clone())
|
Ok(self.inner[id].clone())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// If this method is called, all documents that have a reference to language servers used by the language config have to refresh their language servers,
|
/// If this method is called, all documents that have a reference to the language server have to refresh their language servers,
|
||||||
/// as it could be that language servers of these documents were stopped by this method.
|
|
||||||
/// See helix_view::editor::Editor::refresh_language_servers
|
/// See helix_view::editor::Editor::refresh_language_servers
|
||||||
pub fn restart(
|
pub fn restart_server(
|
||||||
&mut self,
|
&mut self,
|
||||||
|
name: &str,
|
||||||
language_config: &LanguageConfiguration,
|
language_config: &LanguageConfiguration,
|
||||||
doc_path: Option<&std::path::PathBuf>,
|
doc_path: Option<&std::path::PathBuf>,
|
||||||
root_dirs: &[PathBuf],
|
root_dirs: &[PathBuf],
|
||||||
enable_snippets: bool,
|
enable_snippets: bool,
|
||||||
) -> Result<Vec<Arc<Client>>> {
|
) -> Option<Result<Arc<Client>>> {
|
||||||
language_config
|
if let Some(old_clients) = self.inner_by_name.remove(name) {
|
||||||
.language_servers
|
if old_clients.is_empty() {
|
||||||
.iter()
|
log::info!("restarting client for '{name}' which was manually stopped");
|
||||||
.filter_map(|LanguageServerFeatures { name, .. }| {
|
} else {
|
||||||
if let Some(old_clients) = self.inner_by_name.remove(name) {
|
log::info!("stopping existing clients for '{name}'");
|
||||||
if old_clients.is_empty() {
|
}
|
||||||
log::info!("restarting client for '{name}' which was manually stopped");
|
for old_client in old_clients {
|
||||||
} else {
|
self.file_event_handler.remove_client(old_client.id());
|
||||||
log::info!("stopping existing clients for '{name}'");
|
self.inner.remove(old_client.id());
|
||||||
}
|
tokio::spawn(async move {
|
||||||
for old_client in old_clients {
|
let _ = old_client.force_shutdown().await;
|
||||||
self.file_event_handler.remove_client(old_client.id());
|
});
|
||||||
self.inner.remove(old_client.id());
|
}
|
||||||
tokio::spawn(async move {
|
}
|
||||||
let _ = old_client.force_shutdown().await;
|
let client = match self.start_client(
|
||||||
});
|
name.to_string(),
|
||||||
}
|
language_config,
|
||||||
}
|
doc_path,
|
||||||
let client = match self.start_client(
|
root_dirs,
|
||||||
name.clone(),
|
enable_snippets,
|
||||||
language_config,
|
) {
|
||||||
doc_path,
|
Ok(client) => client,
|
||||||
root_dirs,
|
Err(StartupError::NoRequiredRootFound) => return None,
|
||||||
enable_snippets,
|
Err(StartupError::Error(err)) => return Some(Err(err)),
|
||||||
) {
|
};
|
||||||
Ok(client) => client,
|
self.inner_by_name
|
||||||
Err(StartupError::NoRequiredRootFound) => return None,
|
.insert(name.to_owned(), vec![client.clone()]);
|
||||||
Err(StartupError::Error(err)) => return Some(Err(err)),
|
|
||||||
};
|
|
||||||
self.inner_by_name
|
|
||||||
.insert(name.to_owned(), vec![client.clone()]);
|
|
||||||
|
|
||||||
Some(Ok(client))
|
Some(Ok(client))
|
||||||
})
|
|
||||||
.collect()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn stop(&mut self, name: &str) {
|
pub fn stop(&mut self, name: &str) {
|
||||||
|
|
|
@ -1476,9 +1476,34 @@ fn lsp_workspace_command(
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns all language servers used by the current document if no servers are supplied
|
||||||
|
/// If servers are supplied, do a check to make sure that all of the servers exist
|
||||||
|
fn valid_lang_servers(doc: &Document, servers: &[Cow<str>]) -> anyhow::Result<Vec<String>> {
|
||||||
|
let valid_ls_names = doc
|
||||||
|
.language_servers()
|
||||||
|
.map(|ls| ls.name().to_string())
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
if servers.is_empty() {
|
||||||
|
Ok(valid_ls_names)
|
||||||
|
} else {
|
||||||
|
let (valid, invalid): (Vec<_>, Vec<_>) = servers
|
||||||
|
.iter()
|
||||||
|
.map(|m| m.to_string())
|
||||||
|
.partition(|ls| valid_ls_names.contains(ls));
|
||||||
|
|
||||||
|
if !invalid.is_empty() {
|
||||||
|
let s = if invalid.len() == 1 { "" } else { "s" };
|
||||||
|
bail!("Unknown language server{s}: {}", invalid.join(", "));
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(valid)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn lsp_restart(
|
fn lsp_restart(
|
||||||
cx: &mut compositor::Context,
|
cx: &mut compositor::Context,
|
||||||
_args: &[Cow<str>],
|
args: &[Cow<str>],
|
||||||
event: PromptEvent,
|
event: PromptEvent,
|
||||||
) -> anyhow::Result<()> {
|
) -> anyhow::Result<()> {
|
||||||
if event != PromptEvent::Validate {
|
if event != PromptEvent::Validate {
|
||||||
|
@ -1486,17 +1511,25 @@ fn lsp_restart(
|
||||||
}
|
}
|
||||||
|
|
||||||
let editor_config = cx.editor.config.load();
|
let editor_config = cx.editor.config.load();
|
||||||
let (_view, doc) = current!(cx.editor);
|
let doc = doc!(cx.editor);
|
||||||
let config = doc
|
let config = doc
|
||||||
.language_config()
|
.language_config()
|
||||||
.context("LSP not defined for the current document")?;
|
.context("LSP not defined for the current document")?;
|
||||||
|
|
||||||
cx.editor.language_servers.restart(
|
let ls_restart_names = valid_lang_servers(doc, args)?;
|
||||||
config,
|
|
||||||
doc.path(),
|
for server in ls_restart_names.iter() {
|
||||||
&editor_config.workspace_lsp_roots,
|
cx.editor
|
||||||
editor_config.lsp.snippets,
|
.language_servers
|
||||||
)?;
|
.restart_server(
|
||||||
|
server,
|
||||||
|
config,
|
||||||
|
doc.path(),
|
||||||
|
&editor_config.workspace_lsp_roots,
|
||||||
|
editor_config.lsp.snippets,
|
||||||
|
)
|
||||||
|
.transpose()?;
|
||||||
|
}
|
||||||
|
|
||||||
// This collect is needed because refresh_language_server would need to re-borrow editor.
|
// This collect is needed because refresh_language_server would need to re-borrow editor.
|
||||||
let document_ids_to_refresh: Vec<DocumentId> = cx
|
let document_ids_to_refresh: Vec<DocumentId> = cx
|
||||||
|
@ -1505,10 +1538,9 @@ fn lsp_restart(
|
||||||
.filter_map(|doc| match doc.language_config() {
|
.filter_map(|doc| match doc.language_config() {
|
||||||
Some(config)
|
Some(config)
|
||||||
if config.language_servers.iter().any(|ls| {
|
if config.language_servers.iter().any(|ls| {
|
||||||
config
|
ls_restart_names
|
||||||
.language_servers
|
|
||||||
.iter()
|
.iter()
|
||||||
.any(|restarted_ls| restarted_ls.name == ls.name)
|
.any(|restarted_ls| restarted_ls == &ls.name)
|
||||||
}) =>
|
}) =>
|
||||||
{
|
{
|
||||||
Some(doc.id())
|
Some(doc.id())
|
||||||
|
@ -1526,17 +1558,15 @@ fn lsp_restart(
|
||||||
|
|
||||||
fn lsp_stop(
|
fn lsp_stop(
|
||||||
cx: &mut compositor::Context,
|
cx: &mut compositor::Context,
|
||||||
_args: &[Cow<str>],
|
args: &[Cow<str>],
|
||||||
event: PromptEvent,
|
event: PromptEvent,
|
||||||
) -> anyhow::Result<()> {
|
) -> anyhow::Result<()> {
|
||||||
if event != PromptEvent::Validate {
|
if event != PromptEvent::Validate {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
let doc = doc!(cx.editor);
|
||||||
|
|
||||||
let ls_shutdown_names = doc!(cx.editor)
|
let ls_shutdown_names = valid_lang_servers(doc, args)?;
|
||||||
.language_servers()
|
|
||||||
.map(|ls| ls.name().to_string())
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
for ls_name in &ls_shutdown_names {
|
for ls_name in &ls_shutdown_names {
|
||||||
cx.editor.language_servers.stop(ls_name);
|
cx.editor.language_servers.stop(ls_name);
|
||||||
|
@ -2910,16 +2940,16 @@ pub const TYPABLE_COMMAND_LIST: &[TypableCommand] = &[
|
||||||
TypableCommand {
|
TypableCommand {
|
||||||
name: "lsp-restart",
|
name: "lsp-restart",
|
||||||
aliases: &[],
|
aliases: &[],
|
||||||
doc: "Restarts the language servers used by the current doc",
|
doc: "Restarts the given language servers, or all language servers that are used by the current file if no arguments are supplied",
|
||||||
fun: lsp_restart,
|
fun: lsp_restart,
|
||||||
signature: CommandSignature::none(),
|
signature: CommandSignature::all(completers::language_servers),
|
||||||
},
|
},
|
||||||
TypableCommand {
|
TypableCommand {
|
||||||
name: "lsp-stop",
|
name: "lsp-stop",
|
||||||
aliases: &[],
|
aliases: &[],
|
||||||
doc: "Stops the language servers that are used by the current doc",
|
doc: "Stops the given language servers, or all language servers that are used by the current file if no arguments are supplied",
|
||||||
fun: lsp_stop,
|
fun: lsp_stop,
|
||||||
signature: CommandSignature::none(),
|
signature: CommandSignature::all(completers::language_servers),
|
||||||
},
|
},
|
||||||
TypableCommand {
|
TypableCommand {
|
||||||
name: "tree-sitter-scopes",
|
name: "tree-sitter-scopes",
|
||||||
|
|
|
@ -410,6 +410,15 @@ pub mod completers {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn language_servers(editor: &Editor, input: &str) -> Vec<Completion> {
|
||||||
|
let language_servers = doc!(editor).language_servers().map(|ls| ls.name());
|
||||||
|
|
||||||
|
fuzzy_match(input, language_servers, false)
|
||||||
|
.into_iter()
|
||||||
|
.map(|(name, _)| ((0..), Span::raw(name.to_string())))
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
pub fn setting(_editor: &Editor, input: &str) -> Vec<Completion> {
|
pub fn setting(_editor: &Editor, input: &str) -> Vec<Completion> {
|
||||||
static KEYS: Lazy<Vec<String>> = Lazy::new(|| {
|
static KEYS: Lazy<Vec<String>> = Lazy::new(|| {
|
||||||
let mut keys = Vec::new();
|
let mut keys = Vec::new();
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue