2019-11-21 13:53:15 +00:00
|
|
|
use std::fs::File;
|
|
|
|
use std::io::{BufRead, BufReader};
|
2019-04-07 10:18:58 +00:00
|
|
|
|
2020-04-17 12:11:25 +00:00
|
|
|
use anyhow::{Error};
|
2019-04-07 10:18:58 +00:00
|
|
|
use serde_json::{json, Value};
|
|
|
|
|
2020-08-06 13:46:01 +00:00
|
|
|
use proxmox::api::{api, Router, RpcEnvironment, Permission};
|
2019-11-21 13:53:15 +00:00
|
|
|
use proxmox::api::router::SubdirMap;
|
2020-01-21 11:28:01 +00:00
|
|
|
use proxmox::{identity, list_subdirs_api_method, sortable};
|
2019-11-21 12:10:49 +00:00
|
|
|
|
|
|
|
use crate::tools;
|
2019-05-09 05:44:09 +00:00
|
|
|
use crate::api2::types::*;
|
2020-08-13 08:29:13 +00:00
|
|
|
use crate::server::{self, UPID, TaskState};
|
2020-04-16 15:47:21 +00:00
|
|
|
use crate::config::acl::{PRIV_SYS_AUDIT, PRIV_SYS_MODIFY};
|
2020-04-30 08:05:50 +00:00
|
|
|
use crate::config::cached_user_info::CachedUserInfo;
|
|
|
|
|
2019-04-07 12:36:57 +00:00
|
|
|
|
2020-04-16 15:47:21 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
node: {
|
|
|
|
schema: NODE_SCHEMA,
|
|
|
|
},
|
|
|
|
upid: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
|
|
|
description: "Task status nformation.",
|
|
|
|
properties: {
|
|
|
|
node: {
|
|
|
|
schema: NODE_SCHEMA,
|
|
|
|
},
|
|
|
|
upid: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
|
|
|
pid: {
|
|
|
|
type: i64,
|
|
|
|
description: "The Unix PID.",
|
|
|
|
},
|
|
|
|
pstart: {
|
|
|
|
type: u64,
|
|
|
|
description: "The Unix process start time from `/proc/pid/stat`",
|
|
|
|
},
|
|
|
|
starttime: {
|
|
|
|
type: i64,
|
|
|
|
description: "The task start time (Epoch)",
|
|
|
|
},
|
|
|
|
"type": {
|
|
|
|
type: String,
|
|
|
|
description: "Worker type (arbitrary ASCII string)",
|
|
|
|
},
|
|
|
|
id: {
|
|
|
|
type: String,
|
|
|
|
optional: true,
|
|
|
|
description: "Worker ID (arbitrary ASCII string)",
|
|
|
|
},
|
|
|
|
user: {
|
|
|
|
type: String,
|
|
|
|
description: "The user who started the task.",
|
|
|
|
},
|
|
|
|
status: {
|
|
|
|
type: String,
|
|
|
|
description: "'running' or 'stopped'",
|
|
|
|
},
|
|
|
|
exitstatus: {
|
|
|
|
type: String,
|
|
|
|
optional: true,
|
|
|
|
description: "'OK', 'Error: <msg>', or 'unkwown'.",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
access: {
|
2020-04-30 08:05:50 +00:00
|
|
|
description: "Users can access there own tasks, or need Sys.Audit on /system/tasks.",
|
|
|
|
permission: &Permission::Anybody,
|
2020-04-16 15:47:21 +00:00
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Get task status.
|
2020-05-07 06:30:38 +00:00
|
|
|
async fn get_task_status(
|
2019-04-07 12:36:57 +00:00
|
|
|
param: Value,
|
2020-04-30 08:05:50 +00:00
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
2019-04-07 12:36:57 +00:00
|
|
|
) -> Result<Value, Error> {
|
|
|
|
|
|
|
|
let upid = extract_upid(¶m)?;
|
|
|
|
|
2020-08-06 13:46:01 +00:00
|
|
|
let userid: Userid = rpcenv.get_user().unwrap().parse()?;
|
2020-04-30 08:05:50 +00:00
|
|
|
|
2020-08-06 13:46:01 +00:00
|
|
|
if userid != upid.userid {
|
2020-04-30 08:05:50 +00:00
|
|
|
let user_info = CachedUserInfo::new()?;
|
2020-08-06 13:46:01 +00:00
|
|
|
user_info.check_privs(&userid, &["system", "tasks"], PRIV_SYS_AUDIT, false)?;
|
2020-04-30 08:05:50 +00:00
|
|
|
}
|
|
|
|
|
2019-04-09 13:12:20 +00:00
|
|
|
let mut result = json!({
|
|
|
|
"upid": param["upid"],
|
|
|
|
"node": upid.node,
|
|
|
|
"pid": upid.pid,
|
|
|
|
"pstart": upid.pstart,
|
|
|
|
"starttime": upid.starttime,
|
|
|
|
"type": upid.worker_type,
|
|
|
|
"id": upid.worker_id,
|
2020-08-06 13:46:01 +00:00
|
|
|
"user": upid.userid,
|
2019-04-09 13:12:20 +00:00
|
|
|
});
|
|
|
|
|
2020-05-07 06:30:38 +00:00
|
|
|
if crate::server::worker_is_active(&upid).await? {
|
2019-04-09 13:12:20 +00:00
|
|
|
result["status"] = Value::from("running");
|
2019-04-07 12:36:57 +00:00
|
|
|
} else {
|
2020-08-13 08:29:14 +00:00
|
|
|
let (_, exitstatus) = crate::server::upid_read_status(&upid).unwrap_or((0, TaskState::Unknown));
|
2019-04-09 13:12:20 +00:00
|
|
|
result["status"] = Value::from("stopped");
|
2020-08-13 08:29:13 +00:00
|
|
|
result["exitstatus"] = Value::from(exitstatus.to_string());
|
2019-04-07 12:36:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Ok(result)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn extract_upid(param: &Value) -> Result<UPID, Error> {
|
|
|
|
|
|
|
|
let upid_str = tools::required_string_param(¶m, "upid")?;
|
|
|
|
|
2019-12-10 12:45:56 +00:00
|
|
|
upid_str.parse::<UPID>()
|
2019-04-07 12:36:57 +00:00
|
|
|
}
|
|
|
|
|
2020-04-16 15:47:21 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
node: {
|
|
|
|
schema: NODE_SCHEMA,
|
|
|
|
},
|
|
|
|
upid: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
|
|
|
"test-status": {
|
|
|
|
type: bool,
|
|
|
|
optional: true,
|
|
|
|
description: "Test task status, and set result attribute \"active\" accordingly.",
|
|
|
|
},
|
|
|
|
start: {
|
|
|
|
type: u64,
|
|
|
|
optional: true,
|
|
|
|
description: "Start at this line.",
|
|
|
|
default: 0,
|
|
|
|
},
|
|
|
|
limit: {
|
|
|
|
type: u64,
|
|
|
|
optional: true,
|
|
|
|
description: "Only list this amount of lines.",
|
|
|
|
default: 50,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
access: {
|
2020-04-30 08:05:50 +00:00
|
|
|
description: "Users can access there own tasks, or need Sys.Audit on /system/tasks.",
|
|
|
|
permission: &Permission::Anybody,
|
2020-04-16 15:47:21 +00:00
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Read task log.
|
2020-05-07 06:30:38 +00:00
|
|
|
async fn read_task_log(
|
2019-04-07 12:36:57 +00:00
|
|
|
param: Value,
|
2020-05-18 07:57:35 +00:00
|
|
|
mut rpcenv: &mut dyn RpcEnvironment,
|
2019-04-07 12:36:57 +00:00
|
|
|
) -> Result<Value, Error> {
|
|
|
|
|
|
|
|
let upid = extract_upid(¶m)?;
|
2019-12-07 14:29:42 +00:00
|
|
|
|
2020-08-06 13:46:01 +00:00
|
|
|
let userid: Userid = rpcenv.get_user().unwrap().parse()?;
|
2020-04-30 08:05:50 +00:00
|
|
|
|
2020-08-06 13:46:01 +00:00
|
|
|
if userid != upid.userid {
|
2020-04-30 08:05:50 +00:00
|
|
|
let user_info = CachedUserInfo::new()?;
|
2020-08-06 13:46:01 +00:00
|
|
|
user_info.check_privs(&userid, &["system", "tasks"], PRIV_SYS_AUDIT, false)?;
|
2020-04-30 08:05:50 +00:00
|
|
|
}
|
|
|
|
|
2019-12-07 14:29:42 +00:00
|
|
|
let test_status = param["test-status"].as_bool().unwrap_or(false);
|
|
|
|
|
2019-04-07 12:36:57 +00:00
|
|
|
let start = param["start"].as_u64().unwrap_or(0);
|
|
|
|
let mut limit = param["limit"].as_u64().unwrap_or(50);
|
2019-12-07 14:29:42 +00:00
|
|
|
|
2019-04-07 12:36:57 +00:00
|
|
|
let mut count: u64 = 0;
|
|
|
|
|
|
|
|
let path = upid.log_path();
|
|
|
|
|
|
|
|
let file = File::open(path)?;
|
|
|
|
|
|
|
|
let mut lines: Vec<Value> = vec![];
|
|
|
|
|
|
|
|
for line in BufReader::new(file).lines() {
|
|
|
|
match line {
|
|
|
|
Ok(line) => {
|
|
|
|
count += 1;
|
|
|
|
if count < start { continue };
|
2019-10-25 16:04:37 +00:00
|
|
|
if limit == 0 { continue };
|
2019-04-07 12:36:57 +00:00
|
|
|
|
|
|
|
lines.push(json!({ "n": count, "t": line }));
|
|
|
|
|
|
|
|
limit -= 1;
|
|
|
|
}
|
|
|
|
Err(err) => {
|
|
|
|
log::error!("reading task log failed: {}", err);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-18 07:57:35 +00:00
|
|
|
rpcenv["total"] = Value::from(count);
|
2019-04-09 12:43:30 +00:00
|
|
|
|
2019-12-07 14:29:42 +00:00
|
|
|
if test_status {
|
2020-05-07 06:30:38 +00:00
|
|
|
let active = crate::server::worker_is_active(&upid).await?;
|
2020-05-18 07:57:35 +00:00
|
|
|
rpcenv["active"] = Value::from(active);
|
2019-12-07 14:29:42 +00:00
|
|
|
}
|
|
|
|
|
2019-04-07 12:36:57 +00:00
|
|
|
Ok(json!(lines))
|
|
|
|
}
|
2019-04-07 10:18:58 +00:00
|
|
|
|
2020-04-16 15:47:21 +00:00
|
|
|
#[api(
|
|
|
|
protected: true,
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
node: {
|
|
|
|
schema: NODE_SCHEMA,
|
|
|
|
},
|
|
|
|
upid: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
access: {
|
2020-04-30 08:05:50 +00:00
|
|
|
description: "Users can stop there own tasks, or need Sys.Modify on /system/tasks.",
|
|
|
|
permission: &Permission::Anybody,
|
2020-04-16 15:47:21 +00:00
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Try to stop a task.
|
2019-04-10 11:55:05 +00:00
|
|
|
fn stop_task(
|
|
|
|
param: Value,
|
2020-04-30 08:05:50 +00:00
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
2019-04-10 11:55:05 +00:00
|
|
|
) -> Result<Value, Error> {
|
|
|
|
|
|
|
|
let upid = extract_upid(¶m)?;
|
|
|
|
|
2020-08-06 13:46:01 +00:00
|
|
|
let userid: Userid = rpcenv.get_user().unwrap().parse()?;
|
2020-04-30 08:05:50 +00:00
|
|
|
|
2020-08-06 13:46:01 +00:00
|
|
|
if userid != upid.userid {
|
2020-04-30 08:05:50 +00:00
|
|
|
let user_info = CachedUserInfo::new()?;
|
2020-08-06 13:46:01 +00:00
|
|
|
user_info.check_privs(&userid, &["system", "tasks"], PRIV_SYS_MODIFY, false)?;
|
2020-04-30 08:05:50 +00:00
|
|
|
}
|
|
|
|
|
2020-05-07 06:30:38 +00:00
|
|
|
server::abort_worker_async(upid);
|
2019-04-10 11:55:05 +00:00
|
|
|
|
|
|
|
Ok(Value::Null)
|
|
|
|
}
|
|
|
|
|
2020-01-28 09:57:33 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
node: {
|
|
|
|
schema: NODE_SCHEMA
|
|
|
|
},
|
|
|
|
start: {
|
|
|
|
type: u64,
|
|
|
|
description: "List tasks beginning from this offset.",
|
|
|
|
default: 0,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
limit: {
|
|
|
|
type: u64,
|
|
|
|
description: "Only list this amount of tasks.",
|
|
|
|
default: 50,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
store: {
|
|
|
|
schema: DATASTORE_SCHEMA,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
running: {
|
|
|
|
type: bool,
|
|
|
|
description: "Only list running tasks.",
|
|
|
|
optional: true,
|
2020-04-30 09:51:56 +00:00
|
|
|
default: false,
|
2020-01-28 09:57:33 +00:00
|
|
|
},
|
|
|
|
errors: {
|
|
|
|
type: bool,
|
|
|
|
description: "Only list erroneous tasks.",
|
|
|
|
optional:true,
|
2020-04-30 09:51:56 +00:00
|
|
|
default: false,
|
2020-01-28 09:57:33 +00:00
|
|
|
},
|
|
|
|
userfilter: {
|
2020-08-06 13:46:01 +00:00
|
|
|
optional: true,
|
2020-01-28 09:57:33 +00:00
|
|
|
type: String,
|
|
|
|
description: "Only list tasks from this user.",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
|
|
|
description: "A list of tasks.",
|
|
|
|
type: Array,
|
2020-01-28 10:22:06 +00:00
|
|
|
items: { type: TaskListItem },
|
2020-01-28 09:57:33 +00:00
|
|
|
},
|
2020-04-16 15:47:21 +00:00
|
|
|
access: {
|
2020-04-30 08:05:50 +00:00
|
|
|
description: "Users can only see there own tasks, unless the have Sys.Audit on /system/tasks.",
|
|
|
|
permission: &Permission::Anybody,
|
2020-04-16 15:47:21 +00:00
|
|
|
},
|
2020-01-28 09:57:33 +00:00
|
|
|
)]
|
|
|
|
/// List tasks.
|
2020-01-30 12:28:25 +00:00
|
|
|
pub fn list_tasks(
|
2020-04-30 09:51:56 +00:00
|
|
|
start: u64,
|
|
|
|
limit: u64,
|
|
|
|
errors: bool,
|
|
|
|
running: bool,
|
2019-04-07 10:18:58 +00:00
|
|
|
param: Value,
|
2020-05-18 07:57:35 +00:00
|
|
|
mut rpcenv: &mut dyn RpcEnvironment,
|
2020-01-28 10:22:06 +00:00
|
|
|
) -> Result<Vec<TaskListItem>, Error> {
|
2019-04-07 10:18:58 +00:00
|
|
|
|
2020-08-06 13:46:01 +00:00
|
|
|
let userid: Userid = rpcenv.get_user().unwrap().parse()?;
|
2020-04-30 08:05:50 +00:00
|
|
|
let user_info = CachedUserInfo::new()?;
|
2020-08-06 13:46:01 +00:00
|
|
|
let user_privs = user_info.lookup_privs(&userid, &["system", "tasks"]);
|
2020-04-30 08:05:50 +00:00
|
|
|
|
|
|
|
let list_all = (user_privs & PRIV_SYS_AUDIT) != 0;
|
|
|
|
|
2019-12-11 11:53:34 +00:00
|
|
|
let store = param["store"].as_str();
|
|
|
|
|
2019-04-07 11:17:19 +00:00
|
|
|
let userfilter = param["userfilter"].as_str();
|
|
|
|
|
2019-04-07 10:18:58 +00:00
|
|
|
let list = server::read_task_list()?;
|
|
|
|
|
|
|
|
let mut result = vec![];
|
|
|
|
|
|
|
|
let mut count = 0;
|
|
|
|
|
2020-06-12 11:34:02 +00:00
|
|
|
for info in list {
|
2020-08-06 13:46:01 +00:00
|
|
|
if !list_all && info.upid.userid != userid { continue; }
|
2020-04-30 08:05:50 +00:00
|
|
|
|
2019-04-07 10:18:58 +00:00
|
|
|
|
2020-08-06 13:46:01 +00:00
|
|
|
if let Some(userid) = userfilter {
|
|
|
|
if !info.upid.userid.as_str().contains(userid) { continue; }
|
2019-04-07 11:17:19 +00:00
|
|
|
}
|
|
|
|
|
2019-12-11 11:53:34 +00:00
|
|
|
if let Some(store) = store {
|
|
|
|
// Note: useful to select all tasks spawned by proxmox-backup-client
|
|
|
|
let worker_id = match &info.upid.worker_id {
|
|
|
|
Some(w) => w,
|
|
|
|
None => continue, // skip
|
|
|
|
};
|
|
|
|
|
2019-12-14 15:05:21 +00:00
|
|
|
if info.upid.worker_type == "backup" || info.upid.worker_type == "restore" ||
|
|
|
|
info.upid.worker_type == "prune"
|
|
|
|
{
|
2019-12-11 11:53:34 +00:00
|
|
|
let prefix = format!("{}_", store);
|
|
|
|
if !worker_id.starts_with(&prefix) { continue; }
|
2019-12-14 15:05:21 +00:00
|
|
|
} else if info.upid.worker_type == "garbage_collection" {
|
2019-12-11 11:53:34 +00:00
|
|
|
if worker_id != store { continue; }
|
|
|
|
} else {
|
|
|
|
continue; // skip
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-07 10:18:58 +00:00
|
|
|
if let Some(ref state) = info.state {
|
2019-12-11 09:52:27 +00:00
|
|
|
if running { continue; }
|
2020-08-13 08:29:13 +00:00
|
|
|
if errors && state.1 == crate::server::TaskState::OK {
|
2019-04-07 10:18:58 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-07 10:41:24 +00:00
|
|
|
if (count as u64) < start {
|
2019-04-07 10:18:58 +00:00
|
|
|
count += 1;
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
count += 1;
|
|
|
|
}
|
|
|
|
|
2020-06-12 11:34:02 +00:00
|
|
|
if (result.len() as u64) < limit { result.push(info.into()); };
|
2019-04-07 10:18:58 +00:00
|
|
|
}
|
|
|
|
|
2020-05-18 07:57:35 +00:00
|
|
|
rpcenv["total"] = Value::from(count);
|
2019-04-07 10:18:58 +00:00
|
|
|
|
2020-06-12 12:57:58 +00:00
|
|
|
Ok(result)
|
2019-04-07 10:18:58 +00:00
|
|
|
}
|
|
|
|
|
2019-11-21 12:10:49 +00:00
|
|
|
#[sortable]
|
2020-04-16 15:47:21 +00:00
|
|
|
const UPID_API_SUBDIRS: SubdirMap = &sorted!([
|
2019-11-21 08:36:41 +00:00
|
|
|
(
|
|
|
|
"log", &Router::new()
|
2020-04-16 15:47:21 +00:00
|
|
|
.get(&API_METHOD_READ_TASK_LOG)
|
2019-11-21 08:36:41 +00:00
|
|
|
),
|
|
|
|
(
|
|
|
|
"status", &Router::new()
|
2020-04-16 15:47:21 +00:00
|
|
|
.get(&API_METHOD_GET_TASK_STATUS)
|
2019-11-21 08:36:41 +00:00
|
|
|
)
|
2020-04-16 15:47:21 +00:00
|
|
|
]);
|
2019-11-21 08:36:41 +00:00
|
|
|
|
|
|
|
pub const UPID_API_ROUTER: Router = Router::new()
|
|
|
|
.get(&list_subdirs_api_method!(UPID_API_SUBDIRS))
|
2020-04-16 15:47:21 +00:00
|
|
|
.delete(&API_METHOD_STOP_TASK)
|
2019-11-21 08:36:41 +00:00
|
|
|
.subdirs(&UPID_API_SUBDIRS);
|
|
|
|
|
|
|
|
pub const ROUTER: Router = Router::new()
|
2020-01-28 09:57:33 +00:00
|
|
|
.get(&API_METHOD_LIST_TASKS)
|
2019-11-21 08:36:41 +00:00
|
|
|
.match_all("upid", &UPID_API_ROUTER);
|