use anyhow::Result; use async_stream::stream; use crossterm::{ terminal::{disable_raw_mode, enable_raw_mode, is_raw_mode_enabled}, tty::IsTty, }; use krata::{ events::EventStream, v1::{ common::GuestStatus, control::{watch_events_reply::Event, ConsoleDataReply, ConsoleDataRequest}, }, }; use log::debug; use tokio::{ io::{stdin, stdout, AsyncReadExt, AsyncWriteExt}, task::JoinHandle, }; use tokio_stream::{Stream, StreamExt}; use tonic::Streaming; pub struct StdioConsoleStream; impl StdioConsoleStream { pub async fn stdin_stream(guest: String) -> impl Stream { let mut stdin = stdin(); stream! { yield ConsoleDataRequest { guest_id: guest, data: vec![] }; let mut buffer = vec![0u8; 60]; loop { let size = match stdin.read(&mut buffer).await { Ok(size) => size, Err(error) => { debug!("failed to read stdin: {}", error); break; } }; let data = buffer[0..size].to_vec(); if size == 1 && buffer[0] == 0x1d { break; } yield ConsoleDataRequest { guest_id: String::default(), data }; } } } pub async fn stdout(mut stream: Streaming) -> Result<()> { if stdin().is_tty() { enable_raw_mode()?; StdioConsoleStream::register_terminal_restore_hook()?; } let mut stdout = stdout(); while let Some(reply) = stream.next().await { let reply = reply?; if reply.data.is_empty() { continue; } stdout.write_all(&reply.data).await?; stdout.flush().await?; } Ok(()) } pub async fn guest_exit_hook( id: String, events: EventStream, ) -> Result>> { Ok(tokio::task::spawn(async move { let mut stream = events.subscribe(); while let Ok(event) = stream.recv().await { if let Event::GuestChanged(changed) = event { let Some(guest) = changed.guest else { continue; }; let Some(state) = guest.state else { continue; }; if guest.id != id { continue; } if let Some(exit_info) = state.exit_info { return Some(exit_info.code); } let status = state.status(); if status == GuestStatus::Destroying || status == GuestStatus::Destroyed { return Some(10); } } } None })) } fn register_terminal_restore_hook() -> Result<()> { if stdin().is_tty() { ctrlc::set_handler(move || { StdioConsoleStream::restore_terminal_mode(); })?; } Ok(()) } pub fn restore_terminal_mode() { if is_raw_mode_enabled().unwrap_or(false) { let _ = disable_raw_mode(); } } }