
(and any other LSP server in theory, if it exposes any LSP-ext endpoint for the same) Closes https://github.com/zed-industries/zed/issues/16160 * adds a way to disable tree-sitter tasks (the ones from the plugins, enabled by default) with ```json5 "languages": { "Rust": "tasks": { "enabled": false } } } ``` language settings * adds a way to disable LSP tasks (the ones from the rust-analyzer language server, enabled by default) with ```json5 "lsp": { "rust-analyzer": { "enable_lsp_tasks": false, } } ``` * adds rust-analyzer tasks into tasks modal and gutter: <img width="1728" alt="modal" src="https://github.com/user-attachments/assets/22b9cee1-4ffb-4c9e-b1f1-d01e80e72508" /> <img width="396" alt="gutter" src="https://github.com/user-attachments/assets/bd818079-e247-4332-bdb5-1b7cb1cce768" /> Release Notes: - Added tasks from rust-analyzer
80 lines
3.2 KiB
Rust
80 lines
3.2 KiB
Rust
use ::serde::{Deserialize, Serialize};
|
|
use gpui::{PromptLevel, WeakEntity};
|
|
use lsp::LanguageServer;
|
|
|
|
use crate::{LanguageServerPromptRequest, LspStore, LspStoreEvent};
|
|
|
|
pub const RUST_ANALYZER_NAME: &str = "rust-analyzer";
|
|
|
|
/// Experimental: Informs the end user about the state of the server
|
|
///
|
|
/// [Rust Analyzer Specification](https://rust-analyzer.github.io/book/contributing/lsp-extensions.html#server-status)
|
|
#[derive(Debug)]
|
|
enum ServerStatus {}
|
|
|
|
/// Other(String) variant to handle unknown values due to this still being experimental
|
|
#[derive(Debug, PartialEq, Deserialize, Serialize, Clone)]
|
|
#[serde(rename_all = "camelCase")]
|
|
enum ServerHealthStatus {
|
|
Ok,
|
|
Warning,
|
|
Error,
|
|
Other(String),
|
|
}
|
|
|
|
#[derive(Debug, PartialEq, Deserialize, Serialize, Clone)]
|
|
#[serde(rename_all = "camelCase")]
|
|
struct ServerStatusParams {
|
|
pub health: ServerHealthStatus,
|
|
pub message: Option<String>,
|
|
}
|
|
|
|
impl lsp::notification::Notification for ServerStatus {
|
|
type Params = ServerStatusParams;
|
|
const METHOD: &'static str = "experimental/serverStatus";
|
|
}
|
|
|
|
pub fn register_notifications(lsp_store: WeakEntity<LspStore>, language_server: &LanguageServer) {
|
|
let name = language_server.name();
|
|
let server_id = language_server.server_id();
|
|
|
|
language_server
|
|
.on_notification::<ServerStatus, _>({
|
|
let name = name.to_string();
|
|
move |params, cx| {
|
|
let name = name.to_string();
|
|
if let Some(ref message) = params.message {
|
|
let message = message.trim();
|
|
if !message.is_empty() {
|
|
let formatted_message = format!(
|
|
"Language server {name} (id {server_id}) status update: {message}"
|
|
);
|
|
match params.health {
|
|
ServerHealthStatus::Ok => log::info!("{formatted_message}"),
|
|
ServerHealthStatus::Warning => log::warn!("{formatted_message}"),
|
|
ServerHealthStatus::Error => {
|
|
log::error!("{formatted_message}");
|
|
let (tx, _rx) = smol::channel::bounded(1);
|
|
let request = LanguageServerPromptRequest {
|
|
level: PromptLevel::Critical,
|
|
message: params.message.unwrap_or_default(),
|
|
actions: Vec::new(),
|
|
response_channel: tx,
|
|
lsp_name: name.clone(),
|
|
};
|
|
lsp_store
|
|
.update(cx, |_, cx| {
|
|
cx.emit(LspStoreEvent::LanguageServerPrompt(request));
|
|
})
|
|
.ok();
|
|
}
|
|
ServerHealthStatus::Other(status) => {
|
|
log::info!("Unknown server health: {status}\n{formatted_message}")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
})
|
|
.detach();
|
|
}
|