helix-mirror/helix-lsp/src/lib.rs

246 lines
7.7 KiB
Rust
Raw Normal View History

2020-10-21 11:42:45 +04:00
mod client;
mod transport;
2020-10-21 11:42:45 +04:00
pub use client::Client;
2021-06-07 23:11:17 +04:00
pub use futures_executor::block_on;
pub use jsonrpc::Call;
pub use jsonrpc_core as jsonrpc;
2020-10-21 11:42:45 +04:00
pub use lsp::{Position, Url};
2021-06-07 23:11:17 +04:00
pub use lsp_types as lsp;
2021-06-07 23:11:17 +04:00
use futures_util::stream::select_all::SelectAll;
use helix_core::syntax::LanguageConfiguration;
2021-06-07 22:50:20 +04:00
use std::{
collections::{hash_map::Entry, HashMap},
sync::Arc,
};
2020-11-05 10:15:19 +04:00
use serde::{Deserialize, Serialize};
2021-06-07 23:11:17 +04:00
use thiserror::Error;
2021-05-06 08:56:34 +04:00
use tokio_stream::wrappers::UnboundedReceiverStream;
2021-06-07 23:11:17 +04:00
pub type Result<T> = core::result::Result<T, Error>;
type LanguageId = String;
2021-05-12 12:24:55 +04:00
2020-10-21 11:42:45 +04:00
#[derive(Error, Debug)]
pub enum Error {
#[error("protocol error: {0}")]
Rpc(#[from] jsonrpc::Error),
#[error("failed to parse: {0}")]
Parse(#[from] serde_json::Error),
2021-06-07 23:11:17 +04:00
#[error("IO Error: {0}")]
IO(#[from] std::io::Error),
2020-10-21 11:42:45 +04:00
#[error("request timed out")]
Timeout,
2021-06-07 22:50:20 +04:00
#[error("server closed the stream")]
StreamClosed,
#[error("LSP not defined")]
LspNotDefined,
2020-10-21 11:42:45 +04:00
#[error(transparent)]
Other(#[from] anyhow::Error),
}
#[derive(Clone, Copy, Debug, Serialize, Deserialize)]
pub enum OffsetEncoding {
/// UTF-8 code units aka bytes
#[serde(rename = "utf-8")]
Utf8,
/// UTF-16 code units
#[serde(rename = "utf-16")]
Utf16,
}
2020-10-21 08:47:20 +04:00
pub mod util {
use super::*;
use helix_core::{Range, Rope, Transaction};
2020-10-21 08:47:20 +04:00
pub fn lsp_pos_to_pos(
doc: &Rope,
pos: lsp::Position,
offset_encoding: OffsetEncoding,
) -> usize {
match offset_encoding {
OffsetEncoding::Utf8 => {
let line = doc.line_to_char(pos.line as usize);
line + pos.character as usize
}
OffsetEncoding::Utf16 => {
let line = doc.line_to_char(pos.line as usize);
let line_start = doc.char_to_utf16_cu(line);
doc.utf16_cu_to_char(line_start + pos.character as usize)
}
}
2020-10-21 08:47:20 +04:00
}
pub fn pos_to_lsp_pos(
doc: &Rope,
pos: usize,
offset_encoding: OffsetEncoding,
) -> lsp::Position {
match offset_encoding {
OffsetEncoding::Utf8 => {
let line = doc.char_to_line(pos);
let line_start = doc.line_to_char(line);
let col = pos - line_start;
lsp::Position::new(line as u32, col as u32)
}
OffsetEncoding::Utf16 => {
let line = doc.char_to_line(pos);
let line_start = doc.char_to_utf16_cu(doc.line_to_char(line));
let col = doc.char_to_utf16_cu(pos) - line_start;
lsp::Position::new(line as u32, col as u32)
}
}
}
pub fn range_to_lsp_range(
doc: &Rope,
range: Range,
offset_encoding: OffsetEncoding,
) -> lsp::Range {
let start = pos_to_lsp_pos(doc, range.from(), offset_encoding);
let end = pos_to_lsp_pos(doc, range.to(), offset_encoding);
lsp::Range::new(start, end)
}
pub fn generate_transaction_from_edits(
doc: &Rope,
edits: Vec<lsp::TextEdit>,
offset_encoding: OffsetEncoding,
) -> Transaction {
Transaction::change(
doc,
edits.into_iter().map(|edit| {
// simplify "" into None for cleaner changesets
let replacement = if !edit.new_text.is_empty() {
Some(edit.new_text.into())
} else {
None
};
let start = lsp_pos_to_pos(doc, edit.range.start, offset_encoding);
let end = lsp_pos_to_pos(doc, edit.range.end, offset_encoding);
(start, end, replacement)
}),
)
}
}
#[derive(Debug, PartialEq, Clone)]
pub enum Notification {
PublishDiagnostics(lsp::PublishDiagnosticsParams),
ShowMessage(lsp::ShowMessageParams),
LogMessage(lsp::LogMessageParams),
}
impl Notification {
pub fn parse(method: &str, params: jsonrpc::Params) -> Option<Notification> {
use lsp::notification::Notification as _;
let notification = match method {
lsp::notification::PublishDiagnostics::METHOD => {
let params: lsp::PublishDiagnosticsParams = params
.parse()
.expect("Failed to parse PublishDiagnostics params");
// TODO: need to loop over diagnostics and distinguish them by URI
2021-05-09 12:13:59 +04:00
Self::PublishDiagnostics(params)
}
lsp::notification::ShowMessage::METHOD => {
let params: lsp::ShowMessageParams =
params.parse().expect("Failed to parse ShowMessage params");
2021-05-09 12:13:59 +04:00
Self::ShowMessage(params)
}
lsp::notification::LogMessage::METHOD => {
let params: lsp::LogMessageParams =
params.parse().expect("Failed to parse ShowMessage params");
2021-05-09 12:13:59 +04:00
Self::LogMessage(params)
}
_ => {
log::error!("unhandled LSP notification: {}", method);
return None;
}
};
Some(notification)
}
}
2020-11-05 10:15:19 +04:00
pub struct Registry {
2021-06-07 22:50:20 +04:00
inner: HashMap<LanguageId, Arc<Client>>,
2020-12-23 10:50:16 +04:00
2021-05-06 08:56:34 +04:00
pub incoming: SelectAll<UnboundedReceiverStream<Call>>,
2020-11-05 10:15:19 +04:00
}
2021-01-08 11:31:19 +04:00
impl Default for Registry {
fn default() -> Self {
Self::new()
}
}
2020-11-05 10:15:19 +04:00
impl Registry {
2020-12-23 10:50:16 +04:00
pub fn new() -> Self {
2020-11-05 10:15:19 +04:00
Self {
inner: HashMap::new(),
2020-12-23 10:50:16 +04:00
incoming: SelectAll::new(),
2020-11-05 10:15:19 +04:00
}
}
2021-06-07 22:50:20 +04:00
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;
2021-06-03 20:48:59 +04:00
let s_incoming = &mut self.incoming;
2021-06-07 22:50:20 +04:00
match inner.entry(language_config.scope.clone()) {
Entry::Occupied(language_server) => Ok(language_server.get().clone()),
Entry::Vacant(entry) => {
2020-11-05 10:15:19 +04:00
// initialize a new client
2021-06-07 22:50:20 +04:00
let (mut client, incoming) = Client::start(&config.command, &config.args)?;
2020-12-23 10:50:16 +04:00
// TODO: run this async without blocking
2021-06-07 22:50:20 +04:00
futures_executor::block_on(client.initialize())?;
2021-05-06 08:56:34 +04:00
s_incoming.push(UnboundedReceiverStream::new(incoming));
2021-06-07 22:50:20 +04:00
let client = Arc::new(client);
2020-12-23 10:50:16 +04:00
2021-06-07 22:50:20 +04:00
entry.insert(client.clone());
Ok(client)
}
}
} else {
Err(Error::LspNotDefined)
}
2020-11-05 10:15:19 +04:00
}
}
// REGISTRY = HashMap<LanguageId, Lazy/OnceCell<Arc<RwLock<Client>>>
// spawn one server per language type, need to spawn one per workspace if server doesn't support
// workspaces
//
// could also be a client per root dir
//
// storing a copy of Option<Arc<RwLock<Client>>> on Document would make the LSP client easily
// accessible during edit/save callbacks
//
// the event loop needs to process all incoming streams, maybe we can just have that be a separate
// task that's continually running and store the state on the client, then use read lock to
// retrieve data during render
// -> PROBLEM: how do you trigger an update on the editor side when data updates?
//
// -> The data updates should pull all events until we run out so we don't frequently re-render
2020-12-23 10:50:16 +04:00
//
//
// v2:
//
// there should be a registry of lsp clients, one per language type (or workspace).
// the clients should lazy init on first access
// the client.initialize() should be called async and we buffer any requests until that completes
// there needs to be a way to process incoming lsp messages from all clients.
// -> notifications need to be dispatched to wherever
// -> requests need to generate a reply and travel back to the same lsp!