/* * This file is part of espanso. * * Copyright (C) 2019-2021 Federico Terzi * * espanso is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * espanso is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with espanso. If not, see . */ use anyhow::Result; use log::{error, info}; use named_pipe::{ConnectingServer, PipeClient, PipeOptions}; use serde::{de::DeserializeOwned, Serialize}; use std::{io::{Write}}; use crate::{ EventHandler, EventHandlerResponse, IPCClient, IPCClientError, IPCServer, }; const DEFAULT_CLIENT_TIMEOUT: u32 = 2000; pub struct WinIPCServer { server: Option, } impl WinIPCServer { pub fn new(id: &str) -> Result { let pipe_name = format!("\\\\.\\pipe\\{}", id); let options = PipeOptions::new(&pipe_name); let server = Some(options.single()?); info!("binded to named pipe: {}", pipe_name); Ok(Self { server }) } } impl IPCServer for WinIPCServer { fn run(mut self, handler: EventHandler) -> anyhow::Result<()> { let server = self .server .take() .expect("unable to extract IPC server handle"); let mut stream = server.wait()?; loop { // Read multiple commands from the client loop { match read_line(&mut stream) { Ok(Some(line)) => { let event: Result = serde_json::from_str(&line); match event { Ok(event) => match handler(event) { EventHandlerResponse::Response(response) => { let mut json_event = serde_json::to_string(&response)?; json_event.push('\n'); stream.write_all(json_event.as_bytes())?; stream.flush()?; } EventHandlerResponse::NoResponse => { // Async event, no need to reply } EventHandlerResponse::Error(err) => { error!("ipc handler reported an error: {}", err); } EventHandlerResponse::Exit => { return Ok(()); } }, Err(error) => { error!("received malformed event from ipc stream: {}", error); break; } } } Ok(None) => { // EOF reached break; } Err(error) => { error!("error reading ipc stream: {}", error); break; } } } stream = stream.disconnect()?.wait()?; } } } // Unbuffered version, necessary to concurrently write // to the buffer if necessary (when receiving sync messages) fn read_line(stream: R) -> Result> { let mut buffer = Vec::new(); let mut is_eof = true; for byte_res in stream.bytes() { let byte = byte_res?; if byte == 10 { // Newline break; } else { buffer.push(byte); } is_eof = false; } if is_eof { Ok(None) } else { Ok(Some(String::from_utf8(buffer)?)) } } pub struct WinIPCClient { stream: PipeClient, } impl WinIPCClient { pub fn new(id: &str) -> Result { let pipe_name = format!("\\\\.\\pipe\\{}", id); let stream = PipeClient::connect_ms(&pipe_name, DEFAULT_CLIENT_TIMEOUT)?; Ok(Self { stream }) } } impl IPCClient for WinIPCClient { fn send_sync(&mut self, event: Event) -> Result { { let mut json_event = serde_json::to_string(&event)?; json_event.push('\n'); self.stream.write_all(json_event.as_bytes())?; self.stream.flush()?; } // Read the response if let Some(line) = read_line(&mut self.stream)? { let event: Result = serde_json::from_str(&line); match event { Ok(response) => Ok(response), Err(err) => Err(IPCClientError::MalformedResponse(err.into()).into()), } } else { Err(IPCClientError::EmptyResponse.into()) } } fn send_async(&mut self, event: Event) -> Result<()> { let mut json_event = serde_json::to_string(&event)?; json_event.push('\n'); self.stream.write_all(json_event.as_bytes())?; self.stream.flush()?; Ok(()) } }