2024-02-23 03:25:06 +00:00
|
|
|
use std::{
|
2024-03-06 12:05:01 +00:00
|
|
|
io::stdout,
|
2024-02-23 03:25:06 +00:00
|
|
|
os::fd::{AsRawFd, FromRawFd},
|
|
|
|
};
|
|
|
|
|
|
|
|
use anyhow::Result;
|
2024-03-06 12:05:01 +00:00
|
|
|
use async_stream::stream;
|
2024-03-14 14:03:11 +00:00
|
|
|
use krata::{
|
|
|
|
common::GuestStatus,
|
|
|
|
control::{watch_events_reply::Event, ConsoleDataReply, ConsoleDataRequest, WatchEventsReply},
|
2024-03-13 11:34:52 +00:00
|
|
|
};
|
|
|
|
use log::{debug, error, warn};
|
2024-02-23 03:25:06 +00:00
|
|
|
use termion::raw::IntoRawMode;
|
|
|
|
use tokio::{
|
|
|
|
fs::File,
|
2024-03-06 12:05:01 +00:00
|
|
|
io::{stdin, AsyncReadExt, AsyncWriteExt},
|
2024-03-13 11:34:52 +00:00
|
|
|
task::JoinHandle,
|
2024-02-23 03:25:06 +00:00
|
|
|
};
|
2024-03-06 12:05:01 +00:00
|
|
|
use tokio_stream::{Stream, StreamExt};
|
|
|
|
use tonic::Streaming;
|
2024-02-23 03:25:06 +00:00
|
|
|
|
2024-03-06 12:05:01 +00:00
|
|
|
pub struct StdioConsoleStream;
|
2024-02-23 03:25:06 +00:00
|
|
|
|
2024-03-06 12:05:01 +00:00
|
|
|
impl StdioConsoleStream {
|
|
|
|
pub async fn stdin_stream(guest: String) -> impl Stream<Item = ConsoleDataRequest> {
|
|
|
|
let mut stdin = stdin();
|
|
|
|
stream! {
|
2024-03-06 15:57:56 +00:00
|
|
|
yield ConsoleDataRequest { guest_id: guest, data: vec![] };
|
2024-03-05 11:35:25 +00:00
|
|
|
|
2024-03-06 12:05:01 +00:00
|
|
|
let mut buffer = vec![0u8; 60];
|
|
|
|
loop {
|
|
|
|
let size = match stdin.read(&mut buffer).await {
|
|
|
|
Ok(size) => size,
|
2024-03-05 11:35:25 +00:00
|
|
|
Err(error) => {
|
2024-03-06 12:05:01 +00:00
|
|
|
debug!("failed to read stdin: {}", error);
|
|
|
|
break;
|
2024-03-05 11:35:25 +00:00
|
|
|
}
|
2024-03-06 12:05:01 +00:00
|
|
|
};
|
|
|
|
let data = buffer[0..size].to_vec();
|
|
|
|
if size == 1 && buffer[0] == 0x1d {
|
|
|
|
break;
|
2024-03-05 11:35:25 +00:00
|
|
|
}
|
2024-03-06 15:57:56 +00:00
|
|
|
yield ConsoleDataRequest { guest_id: String::default(), data };
|
2024-03-06 12:05:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn stdout(mut stream: Streaming<ConsoleDataReply>) -> Result<()> {
|
|
|
|
let terminal = stdout().into_raw_mode()?;
|
|
|
|
let mut stdout = unsafe { File::from_raw_fd(terminal.as_raw_fd()) };
|
|
|
|
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?;
|
2024-02-23 03:25:06 +00:00
|
|
|
}
|
2024-03-05 11:35:25 +00:00
|
|
|
Ok(())
|
2024-02-23 03:25:06 +00:00
|
|
|
}
|
2024-03-13 11:34:52 +00:00
|
|
|
|
|
|
|
pub async fn guest_exit_hook(
|
|
|
|
id: String,
|
|
|
|
mut events: Streaming<WatchEventsReply>,
|
|
|
|
) -> Result<JoinHandle<()>> {
|
|
|
|
Ok(tokio::task::spawn(async move {
|
|
|
|
while let Some(result) = events.next().await {
|
|
|
|
match result {
|
|
|
|
Err(error) => {
|
|
|
|
error!("failed to handle events for exit hook: {}", error);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(reply) => {
|
|
|
|
let Some(event) = reply.event else {
|
|
|
|
continue;
|
|
|
|
};
|
|
|
|
|
|
|
|
match event {
|
2024-03-14 14:03:11 +00:00
|
|
|
Event::GuestChanged(changed) => {
|
|
|
|
let Some(guest) = changed.guest else {
|
|
|
|
continue;
|
|
|
|
};
|
|
|
|
|
|
|
|
let Some(state) = guest.state else {
|
|
|
|
continue;
|
|
|
|
};
|
|
|
|
|
|
|
|
if guest.id != id {
|
|
|
|
continue;
|
2024-03-13 11:34:52 +00:00
|
|
|
}
|
|
|
|
|
2024-03-14 14:03:11 +00:00
|
|
|
if let Some(exit_info) = state.exit_info {
|
|
|
|
std::process::exit(exit_info.code);
|
2024-03-13 11:34:52 +00:00
|
|
|
}
|
|
|
|
|
2024-03-14 14:03:11 +00:00
|
|
|
if state.status() == GuestStatus::Destroyed {
|
|
|
|
warn!("attached guest was destroyed");
|
|
|
|
std::process::exit(1);
|
|
|
|
}
|
2024-03-13 11:34:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}))
|
|
|
|
}
|
2024-02-23 03:25:06 +00:00
|
|
|
}
|