Add ability to specify binary path/args for rust-analyzer (#9293)

This fixes #9292 by adding a section to the language server settings
that allows users to specify the binary path and arguments with which to
start up a language server.

Example user settings for `rust-analyzer`:

```json
{
  "lsp": {
    "rust-analyzer": {
      "binary": {
        "path": "/Users/thorstenball/tmp/rust-analyzer-aarch64-apple-darwin",
        "arguments": ["--no-log-buffering"]
      }
    }
  }
}
```

Constraints:

* Right now this only allows ABSOLUTE paths.
* This is only used by `rust-analyzer` integration right now, but the
setting can be used for other language servers. We just need to update
the adapters to also respect that setting.



Release Notes:

- Added ability to specify `rust-analyzer` binary `path` (must be
absolute) and `arguments` in user settings. Example: `{"lsp":
{"rust-analyzer": {"binary": {"path": "/my/abs/path/rust-analyzer",
"arguments": ["--no-log-buffering"] }}}}`
([#9292](https://github.com/zed-industries/zed/issues/9292)).

Co-authored-by: Ricard Mallafre <rikitzzz@gmail.com>
This commit is contained in:
Thorsten Ball 2024-03-13 18:42:03 +01:00 committed by GitHub
parent 6655b964ab
commit a56a260778
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
38 changed files with 92 additions and 43 deletions

View file

@ -8320,6 +8320,7 @@ async fn test_language_server_restart_due_to_settings_change(cx: &mut gpui::Test
project_settings.lsp.insert( project_settings.lsp.insert(
"Some other server name".into(), "Some other server name".into(),
LspSettings { LspSettings {
binary: None,
settings: None, settings: None,
initialization_options: Some(json!({ initialization_options: Some(json!({
"some other init value": false "some other init value": false
@ -8338,6 +8339,7 @@ async fn test_language_server_restart_due_to_settings_change(cx: &mut gpui::Test
project_settings.lsp.insert( project_settings.lsp.insert(
language_server_name.into(), language_server_name.into(),
LspSettings { LspSettings {
binary: None,
settings: None, settings: None,
initialization_options: Some(json!({ initialization_options: Some(json!({
"anotherInitValue": false "anotherInitValue": false
@ -8356,6 +8358,7 @@ async fn test_language_server_restart_due_to_settings_change(cx: &mut gpui::Test
project_settings.lsp.insert( project_settings.lsp.insert(
language_server_name.into(), language_server_name.into(),
LspSettings { LspSettings {
binary: None,
settings: None, settings: None,
initialization_options: Some(json!({ initialization_options: Some(json!({
"anotherInitValue": false "anotherInitValue": false
@ -8374,6 +8377,7 @@ async fn test_language_server_restart_due_to_settings_change(cx: &mut gpui::Test
project_settings.lsp.insert( project_settings.lsp.insert(
language_server_name.into(), language_server_name.into(),
LspSettings { LspSettings {
binary: None,
settings: None, settings: None,
initialization_options: None, initialization_options: None,
}, },

View file

@ -19,7 +19,7 @@ pub struct ExtensionLspAdapter {
pub(crate) host: Arc<WasmHost>, pub(crate) host: Arc<WasmHost>,
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for ExtensionLspAdapter { impl LspAdapter for ExtensionLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName(self.config.name.clone().into()) LanguageServerName(self.config.name.clone().into())

View file

@ -282,7 +282,7 @@ pub trait LspAdapterDelegate: Send + Sync {
async fn read_text_file(&self, path: PathBuf) -> Result<String>; async fn read_text_file(&self, path: PathBuf) -> Result<String>;
} }
#[async_trait] #[async_trait(?Send)]
pub trait LspAdapter: 'static + Send + Sync { pub trait LspAdapter: 'static + Send + Sync {
fn name(&self) -> LanguageServerName; fn name(&self) -> LanguageServerName;
@ -306,7 +306,7 @@ pub trait LspAdapter: 'static + Send + Sync {
// We only want to cache when we fall back to the global one, // We only want to cache when we fall back to the global one,
// because we don't want to download and overwrite our global one // because we don't want to download and overwrite our global one
// for each worktree we might have open. // for each worktree we might have open.
if let Some(binary) = self.check_if_user_installed(delegate.as_ref()).await { if let Some(binary) = self.check_if_user_installed(delegate.as_ref(), cx).await {
log::info!( log::info!(
"found user-installed language server for {}. path: {:?}, arguments: {:?}", "found user-installed language server for {}. path: {:?}, arguments: {:?}",
language.name(), language.name(),
@ -380,6 +380,7 @@ pub trait LspAdapter: 'static + Send + Sync {
async fn check_if_user_installed( async fn check_if_user_installed(
&self, &self,
_: &dyn LspAdapterDelegate, _: &dyn LspAdapterDelegate,
_: &AsyncAppContext,
) -> Option<LanguageServerBinary> { ) -> Option<LanguageServerBinary> {
None None
} }
@ -1457,7 +1458,7 @@ impl Default for FakeLspAdapter {
} }
#[cfg(any(test, feature = "test-support"))] #[cfg(any(test, feature = "test-support"))]
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for FakeLspAdapter { impl LspAdapter for FakeLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName(self.name.into()) LanguageServerName(self.name.into())

View file

@ -30,7 +30,7 @@ impl AstroLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for AstroLspAdapter { impl LspAdapter for AstroLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("astro-language-server".into()) LanguageServerName("astro-language-server".into())

View file

@ -14,7 +14,7 @@ use util::{
pub struct CLspAdapter; pub struct CLspAdapter;
#[async_trait] #[async_trait(?Send)]
impl super::LspAdapter for CLspAdapter { impl super::LspAdapter for CLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("clangd".into()) LanguageServerName("clangd".into())

View file

@ -12,7 +12,7 @@ use util::{
#[derive(Copy, Clone)] #[derive(Copy, Clone)]
pub struct ClojureLspAdapter; pub struct ClojureLspAdapter;
#[async_trait] #[async_trait(?Send)]
impl super::LspAdapter for ClojureLspAdapter { impl super::LspAdapter for ClojureLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("clojure-lsp".into()) LanguageServerName("clojure-lsp".into())

View file

@ -15,7 +15,7 @@ use util::{github::GitHubLspBinaryVersion, ResultExt};
pub struct OmniSharpAdapter; pub struct OmniSharpAdapter;
#[async_trait] #[async_trait(?Send)]
impl super::LspAdapter for OmniSharpAdapter { impl super::LspAdapter for OmniSharpAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("OmniSharp".into()) LanguageServerName("OmniSharp".into())

View file

@ -31,7 +31,7 @@ impl CssLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for CssLspAdapter { impl LspAdapter for CssLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("vscode-css-language-server".into()) LanguageServerName("vscode-css-language-server".into())

View file

@ -13,7 +13,7 @@ use std::{
pub struct DartLanguageServer; pub struct DartLanguageServer;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for DartLanguageServer { impl LspAdapter for DartLanguageServer {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("dart".into()) LanguageServerName("dart".into())

View file

@ -56,7 +56,7 @@ impl DenoLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for DenoLspAdapter { impl LspAdapter for DenoLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("deno-language-server".into()) LanguageServerName("deno-language-server".into())

View file

@ -29,7 +29,7 @@ impl DockerfileLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for DockerfileLspAdapter { impl LspAdapter for DockerfileLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("docker-langserver".into()) LanguageServerName("docker-langserver".into())

View file

@ -65,7 +65,7 @@ impl Settings for ElixirSettings {
pub struct ElixirLspAdapter; pub struct ElixirLspAdapter;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for ElixirLspAdapter { impl LspAdapter for ElixirLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("elixir-ls".into()) LanguageServerName("elixir-ls".into())
@ -292,7 +292,7 @@ async fn get_cached_server_binary_elixir_ls(
pub struct NextLspAdapter; pub struct NextLspAdapter;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for NextLspAdapter { impl LspAdapter for NextLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("next-ls".into()) LanguageServerName("next-ls".into())
@ -446,7 +446,7 @@ pub struct LocalLspAdapter {
pub arguments: Vec<String>, pub arguments: Vec<String>,
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for LocalLspAdapter { impl LspAdapter for LocalLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("local-ls".into()) LanguageServerName("local-ls".into())

View file

@ -34,7 +34,7 @@ impl ElmLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for ElmLspAdapter { impl LspAdapter for ElmLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName(SERVER_NAME.into()) LanguageServerName(SERVER_NAME.into())

View file

@ -6,7 +6,7 @@ use std::{any::Any, path::PathBuf};
pub struct ErlangLspAdapter; pub struct ErlangLspAdapter;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for ErlangLspAdapter { impl LspAdapter for ErlangLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("erlang_ls".into()) LanguageServerName("erlang_ls".into())

View file

@ -21,7 +21,7 @@ fn server_binary_arguments() -> Vec<OsString> {
pub struct GleamLspAdapter; pub struct GleamLspAdapter;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for GleamLspAdapter { impl LspAdapter for GleamLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("gleam".into()) LanguageServerName("gleam".into())

View file

@ -32,7 +32,7 @@ lazy_static! {
static ref GOPLS_VERSION_REGEX: Regex = Regex::new(r"\d+\.\d+\.\d+").unwrap(); static ref GOPLS_VERSION_REGEX: Regex = Regex::new(r"\d+\.\d+\.\d+").unwrap();
} }
#[async_trait] #[async_trait(?Send)]
impl super::LspAdapter for GoLspAdapter { impl super::LspAdapter for GoLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("gopls".into()) LanguageServerName("gopls".into())
@ -57,6 +57,7 @@ impl super::LspAdapter for GoLspAdapter {
async fn check_if_user_installed( async fn check_if_user_installed(
&self, &self,
delegate: &dyn LspAdapterDelegate, delegate: &dyn LspAdapterDelegate,
_: &AsyncAppContext,
) -> Option<LanguageServerBinary> { ) -> Option<LanguageServerBinary> {
let env = delegate.shell_env().await; let env = delegate.shell_env().await;
let path = delegate.which("gopls".as_ref()).await?; let path = delegate.which("gopls".as_ref()).await?;

View file

@ -6,7 +6,7 @@ use std::{any::Any, path::PathBuf};
pub struct HaskellLanguageServer; pub struct HaskellLanguageServer;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for HaskellLanguageServer { impl LspAdapter for HaskellLanguageServer {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("hls".into()) LanguageServerName("hls".into())

View file

@ -31,7 +31,7 @@ impl HtmlLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for HtmlLspAdapter { impl LspAdapter for HtmlLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("vscode-html-language-server".into()) LanguageServerName("vscode-html-language-server".into())

View file

@ -83,7 +83,7 @@ impl JsonLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for JsonLspAdapter { impl LspAdapter for JsonLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("json-language-server".into()) LanguageServerName("json-language-server".into())

View file

@ -16,7 +16,7 @@ use util::{
#[derive(Copy, Clone)] #[derive(Copy, Clone)]
pub struct LuaLspAdapter; pub struct LuaLspAdapter;
#[async_trait] #[async_trait(?Send)]
impl super::LspAdapter for LuaLspAdapter { impl super::LspAdapter for LuaLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("lua-language-server".into()) LanguageServerName("lua-language-server".into())

View file

@ -6,7 +6,7 @@ use std::{any::Any, path::PathBuf};
pub struct NuLanguageServer; pub struct NuLanguageServer;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for NuLanguageServer { impl LspAdapter for NuLanguageServer {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("nu".into()) LanguageServerName("nu".into())

View file

@ -12,7 +12,7 @@ const OPERATOR_CHAR: [char; 17] = [
pub struct OCamlLspAdapter; pub struct OCamlLspAdapter;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for OCamlLspAdapter { impl LspAdapter for OCamlLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("ocamllsp".into()) LanguageServerName("ocamllsp".into())

View file

@ -34,7 +34,7 @@ impl IntelephenseLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for IntelephenseLspAdapter { impl LspAdapter for IntelephenseLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("intelephense".into()) LanguageServerName("intelephense".into())

View file

@ -29,7 +29,7 @@ impl PrismaLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for PrismaLspAdapter { impl LspAdapter for PrismaLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("prisma-language-server".into()) LanguageServerName("prisma-language-server".into())

View file

@ -33,7 +33,7 @@ impl PurescriptLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for PurescriptLspAdapter { impl LspAdapter for PurescriptLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("purescript-language-server".into()) LanguageServerName("purescript-language-server".into())

View file

@ -28,7 +28,7 @@ impl PythonLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for PythonLspAdapter { impl LspAdapter for PythonLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("pyright".into()) LanguageServerName("pyright".into())

View file

@ -6,7 +6,7 @@ use std::{any::Any, path::PathBuf, sync::Arc};
pub struct RubyLanguageServer; pub struct RubyLanguageServer;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for RubyLanguageServer { impl LspAdapter for RubyLanguageServer {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("solargraph".into()) LanguageServerName("solargraph".into())

View file

@ -2,12 +2,15 @@ use anyhow::{anyhow, bail, Result};
use async_compression::futures::bufread::GzipDecoder; use async_compression::futures::bufread::GzipDecoder;
use async_trait::async_trait; use async_trait::async_trait;
use futures::{io::BufReader, StreamExt}; use futures::{io::BufReader, StreamExt};
use gpui::AsyncAppContext;
pub use language::*; pub use language::*;
use lazy_static::lazy_static; use lazy_static::lazy_static;
use lsp::LanguageServerBinary; use lsp::LanguageServerBinary;
use project::project_settings::ProjectSettings;
use regex::Regex; use regex::Regex;
use settings::Settings;
use smol::fs::{self, File}; use smol::fs::{self, File};
use std::{any::Any, borrow::Cow, env::consts, path::PathBuf, str, sync::Arc}; use std::{any::Any, borrow::Cow, env::consts, path::PathBuf, sync::Arc};
use util::{ use util::{
async_maybe, async_maybe,
fs::remove_matching, fs::remove_matching,
@ -17,10 +20,41 @@ use util::{
pub struct RustLspAdapter; pub struct RustLspAdapter;
#[async_trait] impl RustLspAdapter {
const SERVER_NAME: &'static str = "rust-analyzer";
}
#[async_trait(?Send)]
impl LspAdapter for RustLspAdapter { impl LspAdapter for RustLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("rust-analyzer".into()) LanguageServerName(Self::SERVER_NAME.into())
}
async fn check_if_user_installed(
&self,
_delegate: &dyn LspAdapterDelegate,
cx: &AsyncAppContext,
) -> Option<LanguageServerBinary> {
let binary = cx
.update(|cx| {
ProjectSettings::get_global(cx)
.lsp
.get(Self::SERVER_NAME)
.and_then(|s| s.binary.clone())
})
.ok()??;
let path = binary.path?;
Some(LanguageServerBinary {
path: path.into(),
arguments: binary
.arguments
.unwrap_or_default()
.iter()
.map(|arg| arg.into())
.collect(),
env: None,
})
} }
async fn fetch_latest_server_version( async fn fetch_latest_server_version(

View file

@ -30,7 +30,7 @@ impl SvelteLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for SvelteLspAdapter { impl LspAdapter for SvelteLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("svelte-language-server".into()) LanguageServerName("svelte-language-server".into())

View file

@ -32,7 +32,7 @@ impl TailwindLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for TailwindLspAdapter { impl LspAdapter for TailwindLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("tailwindcss-language-server".into()) LanguageServerName("tailwindcss-language-server".into())

View file

@ -19,7 +19,7 @@ fn terraform_ls_binary_arguments() -> Vec<OsString> {
pub struct TerraformLspAdapter; pub struct TerraformLspAdapter;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for TerraformLspAdapter { impl LspAdapter for TerraformLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("terraform-ls".into()) LanguageServerName("terraform-ls".into())

View file

@ -12,7 +12,7 @@ use util::{github::GitHubLspBinaryVersion, ResultExt};
pub struct TaploLspAdapter; pub struct TaploLspAdapter;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for TaploLspAdapter { impl LspAdapter for TaploLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("taplo-ls".into()) LanguageServerName("taplo-ls".into())

View file

@ -50,7 +50,7 @@ struct TypeScriptVersions {
server_version: String, server_version: String,
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for TypeScriptLspAdapter { impl LspAdapter for TypeScriptLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("typescript-language-server".into()) LanguageServerName("typescript-language-server".into())
@ -224,7 +224,7 @@ impl EsLintLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for EsLintLspAdapter { impl LspAdapter for EsLintLspAdapter {
fn workspace_configuration(&self, workspace_root: &Path, cx: &mut AppContext) -> Value { fn workspace_configuration(&self, workspace_root: &Path, cx: &mut AppContext) -> Value {
let eslint_user_settings = ProjectSettings::get_global(cx) let eslint_user_settings = ProjectSettings::get_global(cx)

View file

@ -6,7 +6,7 @@ use std::{any::Any, path::PathBuf};
pub struct UiuaLanguageServer; pub struct UiuaLanguageServer;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for UiuaLanguageServer { impl LspAdapter for UiuaLanguageServer {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("uiua".into()) LanguageServerName("uiua".into())

View file

@ -38,7 +38,7 @@ impl VueLspAdapter {
} }
} }
} }
#[async_trait] #[async_trait(?Send)]
impl super::LspAdapter for VueLspAdapter { impl super::LspAdapter for VueLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("vue-language-server".into()) LanguageServerName("vue-language-server".into())

View file

@ -33,7 +33,7 @@ impl YamlLspAdapter {
} }
} }
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for YamlLspAdapter { impl LspAdapter for YamlLspAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("yaml-language-server".into()) LanguageServerName("yaml-language-server".into())

View file

@ -3,6 +3,7 @@ use async_compression::futures::bufread::GzipDecoder;
use async_tar::Archive; use async_tar::Archive;
use async_trait::async_trait; use async_trait::async_trait;
use futures::{io::BufReader, StreamExt}; use futures::{io::BufReader, StreamExt};
use gpui::AsyncAppContext;
use language::{LanguageServerName, LspAdapter, LspAdapterDelegate}; use language::{LanguageServerName, LspAdapter, LspAdapterDelegate};
use lsp::LanguageServerBinary; use lsp::LanguageServerBinary;
use smol::fs; use smol::fs;
@ -14,7 +15,7 @@ use util::{github::GitHubLspBinaryVersion, ResultExt};
pub struct ZlsAdapter; pub struct ZlsAdapter;
#[async_trait] #[async_trait(?Send)]
impl LspAdapter for ZlsAdapter { impl LspAdapter for ZlsAdapter {
fn name(&self) -> LanguageServerName { fn name(&self) -> LanguageServerName {
LanguageServerName("zls".into()) LanguageServerName("zls".into())
@ -43,6 +44,7 @@ impl LspAdapter for ZlsAdapter {
async fn check_if_user_installed( async fn check_if_user_installed(
&self, &self,
delegate: &dyn LspAdapterDelegate, delegate: &dyn LspAdapterDelegate,
_cx: &AsyncAppContext,
) -> Option<LanguageServerBinary> { ) -> Option<LanguageServerBinary> {
let env = delegate.shell_env().await; let env = delegate.shell_env().await;
let path = delegate.which("zls".as_ref()).await?; let path = delegate.which("zls".as_ref()).await?;

View file

@ -41,9 +41,16 @@ pub enum GitGutterSetting {
Hide, Hide,
} }
#[derive(Clone, Debug, Default, Serialize, Deserialize, PartialEq, Eq, JsonSchema)]
pub struct BinarySettings {
pub path: Option<String>,
pub arguments: Option<Vec<String>>,
}
#[derive(Clone, Debug, Serialize, Deserialize, PartialEq, Eq, JsonSchema)] #[derive(Clone, Debug, Serialize, Deserialize, PartialEq, Eq, JsonSchema)]
#[serde(rename_all = "snake_case")] #[serde(rename_all = "snake_case")]
pub struct LspSettings { pub struct LspSettings {
pub binary: Option<BinarySettings>,
pub initialization_options: Option<serde_json::Value>, pub initialization_options: Option<serde_json::Value>,
pub settings: Option<serde_json::Value>, pub settings: Option<serde_json::Value>,
} }