1
Fork 0

Merge pull request #18972 from osiewicz/drop-outgoing-messages-on-background-thread

lsp-server: Drop outgoing messages on background thread
This commit is contained in:
Lukas Wirth 2025-01-20 13:39:29 +00:00 committed by GitHub
commit 2233c31531
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
3 changed files with 42 additions and 12 deletions

View file

@ -181,15 +181,15 @@ impl Message {
Ok(Some(msg)) Ok(Some(msg))
} }
pub fn write(self, w: &mut impl Write) -> io::Result<()> { pub fn write(&self, w: &mut impl Write) -> io::Result<()> {
self._write(w) self._write(w)
} }
fn _write(self, w: &mut dyn Write) -> io::Result<()> { fn _write(&self, w: &mut dyn Write) -> io::Result<()> {
#[derive(Serialize)] #[derive(Serialize)]
struct JsonRpc { struct JsonRpc<'a> {
jsonrpc: &'static str, jsonrpc: &'static str,
#[serde(flatten)] #[serde(flatten)]
msg: Message, msg: &'a Message,
} }
let text = serde_json::to_string(&JsonRpc { jsonrpc: "2.0", msg: self })?; let text = serde_json::to_string(&JsonRpc { jsonrpc: "2.0", msg: self })?;
write_msg_text(w, &text) write_msg_text(w, &text)

View file

@ -15,8 +15,11 @@ pub(crate) fn socket_transport(
stream: TcpStream, stream: TcpStream,
) -> (Sender<Message>, Receiver<Message>, IoThreads) { ) -> (Sender<Message>, Receiver<Message>, IoThreads) {
let (reader_receiver, reader) = make_reader(stream.try_clone().unwrap()); let (reader_receiver, reader) = make_reader(stream.try_clone().unwrap());
let (writer_sender, writer) = make_write(stream); let (writer_sender, writer, messages_to_drop) = make_write(stream);
let io_threads = make_io_threads(reader, writer); let dropper = std::thread::spawn(move || {
messages_to_drop.into_iter().for_each(drop);
});
let io_threads = make_io_threads(reader, writer, dropper);
(writer_sender, reader_receiver, io_threads) (writer_sender, reader_receiver, io_threads)
} }
@ -36,11 +39,21 @@ fn make_reader(stream: TcpStream) -> (Receiver<Message>, thread::JoinHandle<io::
(reader_receiver, reader) (reader_receiver, reader)
} }
fn make_write(mut stream: TcpStream) -> (Sender<Message>, thread::JoinHandle<io::Result<()>>) { fn make_write(
mut stream: TcpStream,
) -> (Sender<Message>, thread::JoinHandle<io::Result<()>>, Receiver<Message>) {
let (writer_sender, writer_receiver) = bounded::<Message>(0); let (writer_sender, writer_receiver) = bounded::<Message>(0);
let (drop_sender, drop_receiver) = bounded::<Message>(0);
let writer = thread::spawn(move || { let writer = thread::spawn(move || {
writer_receiver.into_iter().try_for_each(|it| it.write(&mut stream)).unwrap(); writer_receiver
.into_iter()
.try_for_each(|it| {
let result = it.write(&mut stream);
let _ = drop_sender.send(it);
result
})
.unwrap();
Ok(()) Ok(())
}); });
(writer_sender, writer) (writer_sender, writer, drop_receiver)
} }

View file

@ -11,15 +11,24 @@ use crate::Message;
/// Creates an LSP connection via stdio. /// Creates an LSP connection via stdio.
pub(crate) fn stdio_transport() -> (Sender<Message>, Receiver<Message>, IoThreads) { pub(crate) fn stdio_transport() -> (Sender<Message>, Receiver<Message>, IoThreads) {
let (drop_sender, drop_receiver) = bounded::<Message>(0);
let (writer_sender, writer_receiver) = bounded::<Message>(0); let (writer_sender, writer_receiver) = bounded::<Message>(0);
let writer = thread::Builder::new() let writer = thread::Builder::new()
.name("LspServerWriter".to_owned()) .name("LspServerWriter".to_owned())
.spawn(move || { .spawn(move || {
let stdout = stdout(); let stdout = stdout();
let mut stdout = stdout.lock(); let mut stdout = stdout.lock();
writer_receiver.into_iter().try_for_each(|it| it.write(&mut stdout)) writer_receiver.into_iter().try_for_each(|it| {
let result = it.write(&mut stdout);
let _ = drop_sender.send(it);
result
})
}) })
.unwrap(); .unwrap();
let dropper = thread::Builder::new()
.name("LspMessageDropper".to_owned())
.spawn(move || drop_receiver.into_iter().for_each(drop))
.unwrap();
let (reader_sender, reader_receiver) = bounded::<Message>(0); let (reader_sender, reader_receiver) = bounded::<Message>(0);
let reader = thread::Builder::new() let reader = thread::Builder::new()
.name("LspServerReader".to_owned()) .name("LspServerReader".to_owned())
@ -41,7 +50,7 @@ pub(crate) fn stdio_transport() -> (Sender<Message>, Receiver<Message>, IoThread
Ok(()) Ok(())
}) })
.unwrap(); .unwrap();
let threads = IoThreads { reader, writer }; let threads = IoThreads { reader, writer, dropper };
(writer_sender, reader_receiver, threads) (writer_sender, reader_receiver, threads)
} }
@ -49,13 +58,15 @@ pub(crate) fn stdio_transport() -> (Sender<Message>, Receiver<Message>, IoThread
pub(crate) fn make_io_threads( pub(crate) fn make_io_threads(
reader: thread::JoinHandle<io::Result<()>>, reader: thread::JoinHandle<io::Result<()>>,
writer: thread::JoinHandle<io::Result<()>>, writer: thread::JoinHandle<io::Result<()>>,
dropper: thread::JoinHandle<()>,
) -> IoThreads { ) -> IoThreads {
IoThreads { reader, writer } IoThreads { reader, writer, dropper }
} }
pub struct IoThreads { pub struct IoThreads {
reader: thread::JoinHandle<io::Result<()>>, reader: thread::JoinHandle<io::Result<()>>,
writer: thread::JoinHandle<io::Result<()>>, writer: thread::JoinHandle<io::Result<()>>,
dropper: thread::JoinHandle<()>,
} }
impl IoThreads { impl IoThreads {
@ -64,6 +75,12 @@ impl IoThreads {
Ok(r) => r?, Ok(r) => r?,
Err(err) => std::panic::panic_any(err), Err(err) => std::panic::panic_any(err),
} }
match self.dropper.join() {
Ok(_) => (),
Err(err) => {
std::panic::panic_any(err);
}
}
match self.writer.join() { match self.writer.join() {
Ok(r) => r, Ok(r) => r,
Err(err) => { Err(err) => {