mirror of
https://github.com/helix-editor/helix.git
synced 2025-01-19 13:37:06 +04:00
Don't panic on LSP not starting
This commit is contained in:
parent
08f50310bd
commit
960bc9f134
@ -153,7 +153,7 @@ pub fn call<R: lsp::request::Request>(
|
||||
timeout(Duration::from_secs(2), rx.recv())
|
||||
.await
|
||||
.map_err(|_| Error::Timeout)? // return Timeout
|
||||
.unwrap() // TODO: None if channel closed
|
||||
.ok_or(Error::StreamClosed)?
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -13,7 +13,10 @@
|
||||
|
||||
use thiserror::Error;
|
||||
|
||||
use std::{collections::HashMap, sync::Arc};
|
||||
use std::{
|
||||
collections::{hash_map::Entry, HashMap},
|
||||
sync::Arc,
|
||||
};
|
||||
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
||||
@ -29,6 +32,10 @@ pub enum Error {
|
||||
Parse(#[from] serde_json::Error),
|
||||
#[error("request timed out")]
|
||||
Timeout,
|
||||
#[error("server closed the stream")]
|
||||
StreamClosed,
|
||||
#[error("LSP not defined")]
|
||||
LspNotDefined,
|
||||
#[error(transparent)]
|
||||
Other(#[from] anyhow::Error),
|
||||
}
|
||||
@ -173,7 +180,7 @@ pub fn parse(method: &str, params: jsonrpc::Params) -> Option<Notification> {
|
||||
use futures_util::stream::select_all::SelectAll;
|
||||
|
||||
pub struct Registry {
|
||||
inner: HashMap<LanguageId, Option<Arc<Client>>>,
|
||||
inner: HashMap<LanguageId, Arc<Client>>,
|
||||
|
||||
pub incoming: SelectAll<UnboundedReceiverStream<Call>>,
|
||||
}
|
||||
@ -192,35 +199,29 @@ pub fn new() -> Self {
|
||||
}
|
||||
}
|
||||
|
||||
pub fn get(&mut self, language_config: &LanguageConfiguration) -> Option<Arc<Client>> {
|
||||
// TODO: propagate the error
|
||||
pub fn get(&mut self, language_config: &LanguageConfiguration) -> Result<Arc<Client>> {
|
||||
if let Some(config) = &language_config.language_server {
|
||||
// avoid borrow issues
|
||||
let inner = &mut self.inner;
|
||||
let s_incoming = &mut self.incoming;
|
||||
|
||||
let language_server = inner
|
||||
.entry(language_config.scope.clone()) // can't use entry with Borrow keys: https://github.com/rust-lang/rfcs/pull/1769
|
||||
.or_insert_with(|| {
|
||||
// TODO: lookup defaults for id (name, args)
|
||||
|
||||
match inner.entry(language_config.scope.clone()) {
|
||||
Entry::Occupied(language_server) => Ok(language_server.get().clone()),
|
||||
Entry::Vacant(entry) => {
|
||||
// initialize a new client
|
||||
let (mut client, incoming) =
|
||||
Client::start(&config.command, &config.args).ok()?;
|
||||
|
||||
let (mut client, incoming) = Client::start(&config.command, &config.args)?;
|
||||
// TODO: run this async without blocking
|
||||
futures_executor::block_on(client.initialize()).unwrap();
|
||||
|
||||
futures_executor::block_on(client.initialize())?;
|
||||
s_incoming.push(UnboundedReceiverStream::new(incoming));
|
||||
let client = Arc::new(client);
|
||||
|
||||
Some(Arc::new(client))
|
||||
})
|
||||
.clone();
|
||||
|
||||
return language_server;
|
||||
entry.insert(client.clone());
|
||||
Ok(client)
|
||||
}
|
||||
}
|
||||
} else {
|
||||
Err(Error::LspNotDefined)
|
||||
}
|
||||
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -155,7 +155,7 @@ pub fn open(&mut self, path: PathBuf, action: Action) -> Result<DocumentId, Erro
|
||||
let language_server = doc
|
||||
.language
|
||||
.as_ref()
|
||||
.and_then(|language| self.language_servers.get(language));
|
||||
.and_then(|language| self.language_servers.get(language).ok());
|
||||
|
||||
if let Some(language_server) = language_server {
|
||||
doc.set_language_server(Some(language_server.clone()));
|
||||
@ -196,7 +196,7 @@ pub fn close(&mut self, id: ViewId, close_buffer: bool) {
|
||||
let language_server = doc
|
||||
.language
|
||||
.as_ref()
|
||||
.and_then(|language| language_servers.get(language));
|
||||
.and_then(|language| language_servers.get(language).ok());
|
||||
if let Some(language_server) = language_server {
|
||||
tokio::spawn(language_server.text_document_did_close(doc.identifier()));
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user