mod unix;
mod windows;
use anyhow::Context as _;
use futures::AsyncRead;
use futures::AsyncWrite;
use futures::AsyncWriteExt;
use futures::executor::block_on;
use futures::io::AllowStdIo;
use futures::io::AsyncReadExt;
use pal_async::driver::Driver;
use pal_async::local::block_with_io;
use std::borrow::Cow;
use std::ffi::OsStr;
use std::path::Path;
use std::path::PathBuf;
use std::pin::Pin;
use std::process::Command;
use std::task::Context;
use term::raw_stdout;
use term::set_raw_console;
pub fn relay_console(path: &Path) -> anyhow::Result<()> {
block_with_io(async |driver| {
#[cfg(unix)]
let (read, mut write) = {
let pipe = pal_async::socket::PolledSocket::connect_unix(&driver, path)
.await
.context("failed to connect to console socket")?;
pipe.split()
};
#[cfg(windows)]
let (read, mut write) = {
let pipe = std::fs::OpenOptions::new()
.read(true)
.write(true)
.open(path)
.context("failed to connect to console pipe")?;
let pipe = pal_async::pipe::PolledPipe::new(&driver, pipe)
.context("failed to create polled pipe")?;
AsyncReadExt::split(pipe)
};
set_raw_console(true);
std::thread::Builder::new()
.name("input_thread".into())
.spawn({
move || {
block_on(futures::io::copy(
AllowStdIo::new(std::io::stdin()),
&mut write,
))
}
})
.unwrap();
futures::io::copy(read, &mut AllowStdIo::new(raw_stdout())).await?;
Ok(())
})
}
struct App<'a> {
path: Cow<'a, Path>,
args: Vec<Cow<'a, OsStr>>,
}
impl<'a, T: AsRef<OsStr> + ?Sized> From<&'a T> for App<'a> {
fn from(value: &'a T) -> Self {
Self {
path: Path::new(value).into(),
args: Vec::new(),
}
}
}
fn choose_terminal_apps(app: Option<&Path>) -> Vec<App<'_>> {
if let Some(app) = app {
return vec![app.into()];
}
let mut apps = Vec::new();
let env_set = |key| std::env::var_os(key).is_some_and(|x| !x.is_empty());
if env_set("TMUX") {
apps.push(App {
args: vec![OsStr::new("new-window").into()],
.."tmux".into()
});
}
if cfg!(unix) && env_set("DISPLAY") {
apps.push("x-terminal-emulator".into());
apps.push("xterm".into());
}
if cfg!(windows) {
apps.push("wt.exe".into());
apps.push("conhost.exe".into());
}
apps
}
pub fn launch_console(app: Option<&Path>, path: &Path) -> anyhow::Result<()> {
let apps = choose_terminal_apps(app);
for app in &apps {
let mut command = Command::new(app.path.as_ref());
command.args(&app.args);
add_argument_separator(&mut command, app.path.as_ref());
let child = command
.arg(std::env::current_exe().context("could not determine current exe path")?)
.arg("--relay-console-path")
.arg(path)
.stdin(std::process::Stdio::null())
.stdout(std::process::Stdio::null())
.spawn();
match child {
Ok(mut child) => {
std::thread::Builder::new()
.name("console_waiter".into())
.spawn(move || {
let _ = child.wait();
})
.unwrap();
return Ok(());
}
Err(err) if err.kind() == std::io::ErrorKind::NotFound && apps.len() != 1 => continue,
Err(err) => Err(err)
.with_context(|| format!("failed to launch terminal {}", app.path.display()))?,
};
}
anyhow::bail!("could not find a terminal emulator");
}
fn add_argument_separator(command: &mut Command, app: &Path) {
if let Some(file_name) = app.file_name().and_then(|s| s.to_str()) {
let arg = match file_name {
"xterm" | "rxvt" | "urxvt" | "x-terminal-emulator" => "-e",
_ => "--",
};
command.arg(arg);
};
}
pub fn random_console_path() -> PathBuf {
#[cfg(windows)]
let mut path = PathBuf::from("\\\\.\\pipe");
#[cfg(unix)]
let mut path = std::env::temp_dir();
let mut random = [0; 16];
getrandom::fill(&mut random).expect("rng failure");
path.push(u128::from_ne_bytes(random).to_string());
path
}
pub struct Console {
#[cfg(windows)]
sys: windows::WindowsNamedPipeConsole,
#[cfg(unix)]
sys: unix::UnixSocketConsole,
}
impl Console {
pub fn new(driver: impl Driver, app: Option<&Path>) -> anyhow::Result<Self> {
let path = random_console_path();
let this = Self::new_from_path(driver, &path)?;
launch_console(app, &path).context("failed to launch console")?;
Ok(this)
}
fn new_from_path(driver: impl Driver, path: &Path) -> anyhow::Result<Self> {
#[cfg(windows)]
let sys = windows::WindowsNamedPipeConsole::new(Box::new(driver), path)
.context("failed to create console pipe")?;
#[cfg(unix)]
let sys = unix::UnixSocketConsole::new(Box::new(driver), path)
.context("failed to create console socket")?;
Ok(Console { sys })
}
pub async fn relay(&mut self, io: impl AsyncRead + AsyncWrite) -> anyhow::Result<()> {
let (pipe_recv, mut pipe_send) = { AsyncReadExt::split(self) };
let (socket_recv, mut socket_send) = io.split();
let task_a = async move {
let r = futures::io::copy(pipe_recv, &mut socket_send).await;
let _ = socket_send.close().await;
r
};
let task_b = async move {
let r = futures::io::copy(socket_recv, &mut pipe_send).await;
let _ = pipe_send.close().await;
r
};
futures::future::try_join(task_a, task_b).await?;
anyhow::Result::<_>::Ok(())
}
}
impl AsyncRead for Console {
fn poll_read(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &mut [u8],
) -> std::task::Poll<std::io::Result<usize>> {
Pin::new(&mut self.get_mut().sys).poll_read(cx, buf)
}
}
impl AsyncWrite for Console {
fn poll_write(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &[u8],
) -> std::task::Poll<std::io::Result<usize>> {
Pin::new(&mut self.get_mut().sys).poll_write(cx, buf)
}
fn poll_flush(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> std::task::Poll<std::io::Result<()>> {
Pin::new(&mut self.get_mut().sys).poll_flush(cx)
}
fn poll_close(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> std::task::Poll<std::io::Result<()>> {
Pin::new(&mut self.get_mut().sys).poll_close(cx)
}
}