2022-04-14 12:08:48 +00:00
|
|
|
use std::sync::{
|
|
|
|
atomic::{AtomicUsize, Ordering},
|
|
|
|
Arc,
|
|
|
|
};
|
2021-01-29 08:06:15 +00:00
|
|
|
|
2020-04-17 12:11:25 +00:00
|
|
|
use anyhow::{bail, Error};
|
2022-04-14 12:08:48 +00:00
|
|
|
use futures::*;
|
2021-01-29 09:10:04 +00:00
|
|
|
use serde_json::{json, Value};
|
2021-01-29 08:06:15 +00:00
|
|
|
use tokio::signal::unix::{signal, SignalKind};
|
2019-12-08 10:27:15 +00:00
|
|
|
|
2021-10-08 09:19:37 +00:00
|
|
|
use proxmox_router::cli::format_and_print_result;
|
2021-01-29 09:10:04 +00:00
|
|
|
|
2021-11-25 10:48:52 +00:00
|
|
|
use pbs_api_types::percent_encoding::percent_encode_component;
|
2021-01-29 09:10:04 +00:00
|
|
|
|
2021-07-15 10:15:50 +00:00
|
|
|
use super::HttpClient;
|
2019-12-08 10:27:15 +00:00
|
|
|
|
2021-01-29 08:06:15 +00:00
|
|
|
/// Display task log on console
|
|
|
|
///
|
|
|
|
/// This polls the task API and prints the log to the console. It also
|
|
|
|
/// catches interrupt signals, and sends a abort request to the task if
|
|
|
|
/// the user presses CTRL-C. Two interrupts cause an immediate end of
|
|
|
|
/// the loop. The task may still run in that case.
|
2019-12-08 10:27:15 +00:00
|
|
|
pub async fn display_task_log(
|
2021-12-04 13:11:55 +00:00
|
|
|
client: &HttpClient,
|
2019-12-08 10:27:15 +00:00
|
|
|
upid_str: &str,
|
|
|
|
strip_date: bool,
|
|
|
|
) -> Result<(), Error> {
|
2021-01-29 08:06:15 +00:00
|
|
|
let mut signal_stream = signal(SignalKind::interrupt())?;
|
|
|
|
let abort_count = Arc::new(AtomicUsize::new(0));
|
|
|
|
let abort_count2 = Arc::clone(&abort_count);
|
2019-12-08 10:27:15 +00:00
|
|
|
|
2021-01-29 08:06:15 +00:00
|
|
|
let abort_future = async move {
|
|
|
|
while signal_stream.recv().await.is_some() {
|
|
|
|
println!("got shutdown request (SIGINT)");
|
|
|
|
let prev_count = abort_count2.fetch_add(1, Ordering::SeqCst);
|
|
|
|
if prev_count >= 1 {
|
|
|
|
println!("forced exit (task still running)");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok::<_, Error>(())
|
|
|
|
};
|
2019-12-08 10:27:15 +00:00
|
|
|
|
2021-01-29 08:06:15 +00:00
|
|
|
let request_future = async move {
|
|
|
|
let mut start = 1;
|
|
|
|
let limit = 500;
|
2019-12-08 10:27:15 +00:00
|
|
|
|
2021-01-29 08:06:15 +00:00
|
|
|
loop {
|
|
|
|
let abort = abort_count.load(Ordering::Relaxed);
|
|
|
|
if abort > 0 {
|
2022-04-14 12:08:48 +00:00
|
|
|
let path = format!(
|
|
|
|
"api2/json/nodes/localhost/tasks/{}",
|
|
|
|
percent_encode_component(upid_str)
|
|
|
|
);
|
2021-01-29 08:06:15 +00:00
|
|
|
let _ = client.delete(&path, None).await?;
|
2019-12-08 10:27:15 +00:00
|
|
|
}
|
|
|
|
|
2021-01-29 08:06:15 +00:00
|
|
|
let param = json!({ "start": start, "limit": limit, "test-status": true });
|
|
|
|
|
2022-04-14 12:08:48 +00:00
|
|
|
let path = format!(
|
|
|
|
"api2/json/nodes/localhost/tasks/{}/log",
|
|
|
|
percent_encode_component(upid_str)
|
|
|
|
);
|
2021-01-29 08:06:15 +00:00
|
|
|
let result = client.get(&path, Some(param)).await?;
|
|
|
|
|
|
|
|
let active = result["active"].as_bool().unwrap();
|
|
|
|
let total = result["total"].as_u64().unwrap();
|
|
|
|
let data = result["data"].as_array().unwrap();
|
|
|
|
|
|
|
|
let lines = data.len();
|
|
|
|
|
|
|
|
for item in data {
|
|
|
|
let n = item["n"].as_u64().unwrap();
|
|
|
|
let t = item["t"].as_str().unwrap();
|
2022-04-14 12:08:48 +00:00
|
|
|
if n != start {
|
|
|
|
bail!("got wrong line number in response data ({} != {}", n, start);
|
|
|
|
}
|
2021-01-29 08:06:15 +00:00
|
|
|
if strip_date && t.len() > 27 && &t[25..27] == ": " {
|
|
|
|
let line = &t[27..];
|
|
|
|
println!("{}", line);
|
|
|
|
} else {
|
|
|
|
println!("{}", t);
|
|
|
|
}
|
|
|
|
start += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if start > total {
|
|
|
|
if active {
|
|
|
|
tokio::time::sleep(tokio::time::Duration::from_millis(1000)).await;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if lines != limit {
|
2022-04-14 12:08:48 +00:00
|
|
|
bail!(
|
|
|
|
"got wrong number of lines from server ({} != {})",
|
|
|
|
lines,
|
|
|
|
limit
|
|
|
|
);
|
2019-12-08 10:27:15 +00:00
|
|
|
}
|
|
|
|
}
|
2021-01-29 08:06:15 +00:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
};
|
|
|
|
|
2022-04-14 12:08:48 +00:00
|
|
|
futures::select! {
|
2021-01-29 08:06:15 +00:00
|
|
|
request = request_future.fuse() => request?,
|
|
|
|
abort = abort_future.fuse() => abort?,
|
|
|
|
};
|
2019-12-08 10:27:15 +00:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2021-01-29 09:10:04 +00:00
|
|
|
|
|
|
|
/// Display task result (upid), or view task log - depending on output format
|
|
|
|
pub async fn view_task_result(
|
2021-12-04 13:11:55 +00:00
|
|
|
client: &HttpClient,
|
2021-01-29 09:10:04 +00:00
|
|
|
result: Value,
|
|
|
|
output_format: &str,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
let data = &result["data"];
|
|
|
|
if output_format == "text" {
|
|
|
|
if let Some(upid) = data.as_str() {
|
|
|
|
display_task_log(client, upid, true).await?;
|
|
|
|
}
|
|
|
|
} else {
|
2021-12-30 11:57:37 +00:00
|
|
|
format_and_print_result(data, output_format);
|
2021-01-29 09:10:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|