// Copyright 2018-2025 the Deno authors. MIT license. use std::cell::RefCell; use deno_core::anyhow::anyhow; use deno_core::error::AnyError; use deno_core::error::CoreError; use deno_core::serde_json; use deno_core::serde_json::Value; use deno_error::JsErrorBox; use tokio::sync::mpsc::channel; use tokio::sync::mpsc::unbounded_channel; use tokio::sync::mpsc::Receiver; use tokio::sync::mpsc::Sender; use tokio::sync::mpsc::UnboundedReceiver; use tokio::sync::mpsc::UnboundedSender; use crate::lsp::ReplCompletionItem; /// Rustyline uses synchronous methods in its interfaces, but we need to call /// async methods. To get around this, we communicate with async code by using /// a channel and blocking on the result. pub fn rustyline_channel( ) -> (RustylineSyncMessageSender, RustylineSyncMessageHandler) { let (message_tx, message_rx) = channel(1); let (response_tx, response_rx) = unbounded_channel(); ( RustylineSyncMessageSender { message_tx, response_rx: RefCell::new(response_rx), }, RustylineSyncMessageHandler { response_tx, message_rx, }, ) } pub enum RustylineSyncMessage { PostMessage { method: String, params: Option, }, LspCompletions { line_text: String, position: usize, }, } pub enum RustylineSyncResponse { PostMessage(Result), LspCompletions(Vec), } pub struct RustylineSyncMessageSender { message_tx: Sender, response_rx: RefCell>, } impl RustylineSyncMessageSender { pub fn post_message( &self, method: &str, params: Option, ) -> Result { if let Err(err) = self .message_tx .blocking_send(RustylineSyncMessage::PostMessage { method: method.to_string(), params: params .map(|params| serde_json::to_value(params)) .transpose() .map_err(JsErrorBox::from_err)?, }) { Err(JsErrorBox::from_err(err).into()) } else { match self.response_rx.borrow_mut().blocking_recv().unwrap() { RustylineSyncResponse::PostMessage(result) => result, RustylineSyncResponse::LspCompletions(_) => unreachable!(), } } } pub fn lsp_completions( &self, line_text: &str, position: usize, ) -> Vec { if self .message_tx .blocking_send(RustylineSyncMessage::LspCompletions { line_text: line_text.to_string(), position, }) .is_err() { Vec::new() } else { match self.response_rx.borrow_mut().blocking_recv().unwrap() { RustylineSyncResponse::LspCompletions(result) => result, RustylineSyncResponse::PostMessage(_) => unreachable!(), } } } } pub struct RustylineSyncMessageHandler { message_rx: Receiver, response_tx: UnboundedSender, } impl RustylineSyncMessageHandler { pub async fn recv(&mut self) -> Option { self.message_rx.recv().await } pub fn send(&self, response: RustylineSyncResponse) -> Result<(), AnyError> { self .response_tx .send(response) .map_err(|err| anyhow!("{}", err)) } }