2024-03-14 14:03:11 +00:00
|
|
|
use anyhow::Result;
|
2024-01-21 09:58:07 +00:00
|
|
|
use clap::{Parser, Subcommand};
|
2024-02-01 10:36:54 +00:00
|
|
|
use env_logger::Env;
|
2024-03-14 14:03:11 +00:00
|
|
|
use krata::{
|
|
|
|
common::{
|
|
|
|
guest_image_spec::Image, GuestImageSpec, GuestOciImageSpec, GuestSpec, GuestState,
|
|
|
|
GuestStatus,
|
|
|
|
},
|
|
|
|
control::{
|
|
|
|
watch_events_reply::Event, CreateGuestRequest, DestroyGuestRequest, ListGuestsRequest,
|
|
|
|
WatchEventsRequest,
|
|
|
|
},
|
2024-03-06 15:57:56 +00:00
|
|
|
};
|
2024-03-06 12:05:01 +00:00
|
|
|
use kratactl::{client::ControlClientProvider, console::StdioConsoleStream};
|
2024-03-14 14:03:11 +00:00
|
|
|
use log::error;
|
|
|
|
use tokio_stream::StreamExt;
|
2024-03-06 12:05:01 +00:00
|
|
|
use tonic::Request;
|
2024-01-17 22:29:05 +00:00
|
|
|
|
|
|
|
#[derive(Parser, Debug)]
|
|
|
|
#[command(version, about)]
|
|
|
|
struct ControllerArgs {
|
2024-03-06 04:47:53 +00:00
|
|
|
#[arg(short, long, default_value = "unix:///var/lib/krata/daemon.socket")]
|
2024-03-05 11:35:25 +00:00
|
|
|
connection: String,
|
2024-01-21 09:58:07 +00:00
|
|
|
|
|
|
|
#[command(subcommand)]
|
|
|
|
command: Commands,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Subcommand, Debug)]
|
|
|
|
enum Commands {
|
2024-01-22 01:21:19 +00:00
|
|
|
List {},
|
2024-01-21 09:58:07 +00:00
|
|
|
Launch {
|
2024-03-13 13:05:17 +00:00
|
|
|
#[arg(short, long)]
|
|
|
|
name: Option<String>,
|
2024-01-21 09:58:07 +00:00
|
|
|
#[arg(short, long, default_value_t = 1)]
|
|
|
|
cpus: u32,
|
|
|
|
#[arg(short, long, default_value_t = 512)]
|
|
|
|
mem: u64,
|
2024-01-31 02:34:47 +00:00
|
|
|
#[arg[short, long]]
|
|
|
|
env: Option<Vec<String>>,
|
2024-02-01 10:44:12 +00:00
|
|
|
#[arg(short, long)]
|
|
|
|
attach: bool,
|
2024-01-31 17:08:19 +00:00
|
|
|
#[arg()]
|
2024-03-08 08:47:18 +00:00
|
|
|
oci: String,
|
2024-01-22 13:30:02 +00:00
|
|
|
#[arg(allow_hyphen_values = true, trailing_var_arg = true)]
|
|
|
|
run: Vec<String>,
|
2024-01-21 09:58:07 +00:00
|
|
|
},
|
|
|
|
Destroy {
|
2024-01-31 17:08:19 +00:00
|
|
|
#[arg()]
|
2024-03-05 11:35:25 +00:00
|
|
|
guest: String,
|
2024-01-21 09:58:07 +00:00
|
|
|
},
|
2024-01-21 12:49:31 +00:00
|
|
|
Console {
|
2024-01-31 17:08:19 +00:00
|
|
|
#[arg()]
|
2024-03-05 11:35:25 +00:00
|
|
|
guest: String,
|
2024-01-21 12:49:31 +00:00
|
|
|
},
|
2024-03-06 15:57:56 +00:00
|
|
|
Watch {},
|
2024-01-17 22:29:05 +00:00
|
|
|
}
|
|
|
|
|
2024-02-23 03:25:06 +00:00
|
|
|
#[tokio::main]
|
|
|
|
async fn main() -> Result<()> {
|
2024-02-01 10:28:17 +00:00
|
|
|
env_logger::Builder::from_env(Env::default().default_filter_or("warn")).init();
|
2024-01-17 22:29:05 +00:00
|
|
|
|
|
|
|
let args = ControllerArgs::parse();
|
2024-03-06 12:05:01 +00:00
|
|
|
let mut client = ControlClientProvider::dial(args.connection.parse()?).await?;
|
2024-03-14 14:03:11 +00:00
|
|
|
let mut events = client
|
2024-03-13 11:34:52 +00:00
|
|
|
.watch_events(WatchEventsRequest {})
|
|
|
|
.await?
|
|
|
|
.into_inner();
|
2024-01-21 09:58:07 +00:00
|
|
|
|
|
|
|
match args.command {
|
2024-01-21 12:49:31 +00:00
|
|
|
Commands::Launch {
|
2024-03-13 13:05:17 +00:00
|
|
|
name,
|
2024-03-08 08:47:18 +00:00
|
|
|
oci,
|
2024-01-21 12:49:31 +00:00
|
|
|
cpus,
|
|
|
|
mem,
|
2024-02-01 10:44:12 +00:00
|
|
|
attach,
|
2024-01-31 02:34:47 +00:00
|
|
|
env,
|
2024-01-22 13:30:02 +00:00
|
|
|
run,
|
2024-01-21 12:49:31 +00:00
|
|
|
} => {
|
2024-03-14 14:03:11 +00:00
|
|
|
let request = CreateGuestRequest {
|
|
|
|
spec: Some(GuestSpec {
|
|
|
|
name: name.unwrap_or_default(),
|
|
|
|
image: Some(GuestImageSpec {
|
|
|
|
image: Some(Image::Oci(GuestOciImageSpec { image: oci })),
|
|
|
|
}),
|
|
|
|
vcpus: cpus,
|
|
|
|
mem,
|
|
|
|
env: env.unwrap_or_default(),
|
|
|
|
run,
|
2024-03-08 08:47:18 +00:00
|
|
|
}),
|
2024-02-13 19:48:49 +00:00
|
|
|
};
|
2024-03-06 12:05:01 +00:00
|
|
|
let response = client
|
2024-03-14 14:03:11 +00:00
|
|
|
.create_guest(Request::new(request))
|
2024-03-06 12:05:01 +00:00
|
|
|
.await?
|
|
|
|
.into_inner();
|
2024-03-14 14:03:11 +00:00
|
|
|
let id = response.guest_id;
|
2024-02-01 10:44:12 +00:00
|
|
|
if attach {
|
2024-03-14 14:03:11 +00:00
|
|
|
while let Some(event) = events.next().await {
|
|
|
|
let reply = event?;
|
|
|
|
match reply.event {
|
|
|
|
Some(Event::GuestChanged(changed)) => {
|
|
|
|
let Some(guest) = changed.guest else {
|
|
|
|
continue;
|
|
|
|
};
|
|
|
|
|
|
|
|
if guest.id != id {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
let Some(state) = guest.state else {
|
|
|
|
continue;
|
|
|
|
};
|
|
|
|
|
|
|
|
if let Some(ref error) = state.error_info {
|
|
|
|
error!("guest error: {}", error.message);
|
|
|
|
}
|
|
|
|
|
|
|
|
if state.status() == GuestStatus::Destroyed {
|
|
|
|
error!("guest destroyed");
|
|
|
|
std::process::exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if state.status() == GuestStatus::Started {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
None => {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let input = StdioConsoleStream::stdin_stream(id.clone()).await;
|
2024-03-06 12:05:01 +00:00
|
|
|
let output = client.console_data(input).await?.into_inner();
|
2024-03-13 11:34:52 +00:00
|
|
|
let exit_hook_task =
|
2024-03-14 14:03:11 +00:00
|
|
|
StdioConsoleStream::guest_exit_hook(id.clone(), events).await?;
|
2024-03-06 12:05:01 +00:00
|
|
|
StdioConsoleStream::stdout(output).await?;
|
2024-03-13 11:34:52 +00:00
|
|
|
exit_hook_task.abort();
|
2024-03-14 14:03:11 +00:00
|
|
|
} else {
|
|
|
|
println!("created guest: {}", id);
|
2024-02-01 10:44:12 +00:00
|
|
|
}
|
2024-01-21 09:58:07 +00:00
|
|
|
}
|
2024-01-21 12:49:31 +00:00
|
|
|
|
2024-03-05 11:35:25 +00:00
|
|
|
Commands::Destroy { guest } => {
|
2024-03-06 12:05:01 +00:00
|
|
|
let _ = client
|
|
|
|
.destroy_guest(Request::new(DestroyGuestRequest {
|
|
|
|
guest_id: guest.clone(),
|
|
|
|
}))
|
|
|
|
.await?
|
|
|
|
.into_inner();
|
|
|
|
println!("destroyed guest: {}", guest);
|
2024-01-21 09:58:07 +00:00
|
|
|
}
|
2024-01-21 12:49:31 +00:00
|
|
|
|
2024-03-05 11:35:25 +00:00
|
|
|
Commands::Console { guest } => {
|
2024-03-13 11:34:52 +00:00
|
|
|
let input = StdioConsoleStream::stdin_stream(guest.clone()).await;
|
2024-03-06 12:05:01 +00:00
|
|
|
let output = client.console_data(input).await?.into_inner();
|
2024-03-13 11:34:52 +00:00
|
|
|
let exit_hook_task = StdioConsoleStream::guest_exit_hook(guest.clone(), events).await?;
|
2024-03-06 12:05:01 +00:00
|
|
|
StdioConsoleStream::stdout(output).await?;
|
2024-03-13 11:34:52 +00:00
|
|
|
exit_hook_task.abort();
|
2024-01-21 12:49:31 +00:00
|
|
|
}
|
2024-01-22 01:21:19 +00:00
|
|
|
|
|
|
|
Commands::List { .. } => {
|
2024-03-06 12:05:01 +00:00
|
|
|
let response = client
|
|
|
|
.list_guests(Request::new(ListGuestsRequest {}))
|
|
|
|
.await?
|
|
|
|
.into_inner();
|
2024-01-22 01:21:19 +00:00
|
|
|
let mut table = cli_tables::Table::new();
|
2024-03-14 14:03:11 +00:00
|
|
|
let header = vec!["name", "uuid", "state", "ipv4", "ipv6", "image"];
|
2024-01-22 01:21:19 +00:00
|
|
|
table.push_row(&header)?;
|
2024-03-05 11:35:25 +00:00
|
|
|
for guest in response.guests {
|
2024-03-08 08:47:18 +00:00
|
|
|
let ipv4 = guest
|
|
|
|
.network
|
|
|
|
.as_ref()
|
|
|
|
.map(|x| x.ipv4.as_str())
|
|
|
|
.unwrap_or("unknown");
|
|
|
|
let ipv6 = guest
|
|
|
|
.network
|
|
|
|
.as_ref()
|
|
|
|
.map(|x| x.ipv6.as_str())
|
|
|
|
.unwrap_or("unknown");
|
2024-03-14 14:03:11 +00:00
|
|
|
let Some(spec) = guest.spec else {
|
|
|
|
continue;
|
|
|
|
};
|
|
|
|
let image = spec
|
2024-03-08 08:47:18 +00:00
|
|
|
.image
|
|
|
|
.map(|x| {
|
|
|
|
x.image
|
|
|
|
.map(|y| match y {
|
|
|
|
Image::Oci(oci) => oci.image,
|
|
|
|
})
|
|
|
|
.unwrap_or("unknown".to_string())
|
|
|
|
})
|
|
|
|
.unwrap_or("unknown".to_string());
|
|
|
|
table.push_row_string(&vec![
|
2024-03-14 14:03:11 +00:00
|
|
|
spec.name,
|
2024-03-08 08:47:18 +00:00
|
|
|
guest.id,
|
2024-03-14 14:03:11 +00:00
|
|
|
format!("{}", guest_state_text(guest.state.unwrap_or_default())),
|
2024-03-08 08:47:18 +00:00
|
|
|
ipv4.to_string(),
|
|
|
|
ipv6.to_string(),
|
|
|
|
image,
|
|
|
|
])?;
|
2024-01-22 01:21:19 +00:00
|
|
|
}
|
2024-01-22 10:15:53 +00:00
|
|
|
if table.num_records() == 1 {
|
2024-02-29 12:52:44 +00:00
|
|
|
println!("no guests have been launched");
|
2024-01-22 10:15:53 +00:00
|
|
|
} else {
|
|
|
|
println!("{}", table.to_string());
|
|
|
|
}
|
2024-01-22 01:21:19 +00:00
|
|
|
}
|
2024-03-06 15:57:56 +00:00
|
|
|
|
|
|
|
Commands::Watch {} => {
|
|
|
|
let response = client
|
|
|
|
.watch_events(Request::new(WatchEventsRequest {}))
|
|
|
|
.await?;
|
|
|
|
let mut stream = response.into_inner();
|
|
|
|
while let Some(reply) = stream.message().await? {
|
|
|
|
let Some(event) = reply.event else {
|
|
|
|
continue;
|
|
|
|
};
|
|
|
|
|
|
|
|
match event {
|
2024-03-14 14:03:11 +00:00
|
|
|
Event::GuestChanged(changed) => {
|
|
|
|
if let Some(guest) = changed.guest {
|
|
|
|
println!("event=guest.changed guest={}", guest.id);
|
|
|
|
}
|
2024-03-06 15:57:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-01-21 09:58:07 +00:00
|
|
|
}
|
2024-01-17 22:29:05 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
2024-03-14 14:03:11 +00:00
|
|
|
|
|
|
|
fn guest_status_text(status: GuestStatus) -> String {
|
|
|
|
match status {
|
|
|
|
GuestStatus::Unknown => "unknown",
|
|
|
|
GuestStatus::Destroyed => "destroyed",
|
|
|
|
GuestStatus::Start => "starting",
|
|
|
|
GuestStatus::Exited => "exited",
|
|
|
|
GuestStatus::Started => "started",
|
|
|
|
_ => "unknown",
|
|
|
|
}
|
|
|
|
.to_string()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn guest_state_text(state: GuestState) -> String {
|
|
|
|
let mut text = guest_status_text(state.status());
|
|
|
|
|
|
|
|
if let Some(exit) = state.exit_info {
|
|
|
|
text.push_str(&format!(" (exit code: {})", exit.code));
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(error) = state.error_info {
|
|
|
|
text.push_str(&format!(" (error: {})", error.message));
|
|
|
|
}
|
|
|
|
text
|
|
|
|
}
|