2019-12-17 13:06:48 +00:00
|
|
|
use std::process::{Command, Stdio};
|
|
|
|
|
2020-04-17 12:11:25 +00:00
|
|
|
use anyhow::{Error};
|
2019-12-17 13:06:48 +00:00
|
|
|
use serde_json::{json, Value};
|
|
|
|
use std::io::{BufRead,BufReader};
|
|
|
|
|
2020-04-16 10:44:59 +00:00
|
|
|
use proxmox::api::{api, ApiMethod, Router, RpcEnvironment, Permission};
|
2019-12-17 13:06:48 +00:00
|
|
|
|
|
|
|
use crate::api2::types::*;
|
2020-04-16 10:44:59 +00:00
|
|
|
use crate::config::acl::PRIV_SYS_AUDIT;
|
2019-12-17 13:06:48 +00:00
|
|
|
|
|
|
|
#[api(
|
|
|
|
protected: true,
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
node: {
|
|
|
|
schema: NODE_SCHEMA,
|
|
|
|
},
|
|
|
|
since: {
|
|
|
|
type: Integer,
|
|
|
|
optional: true,
|
|
|
|
description: "Display all log since this UNIX epoch. Conflicts with 'startcursor'.",
|
|
|
|
minimum: 0,
|
|
|
|
},
|
|
|
|
until: {
|
|
|
|
type: Integer,
|
|
|
|
optional: true,
|
|
|
|
description: "Display all log until this UNIX epoch. Conflicts with 'endcursor'.",
|
|
|
|
minimum: 0,
|
|
|
|
},
|
|
|
|
lastentries: {
|
|
|
|
type: Integer,
|
|
|
|
optional: true,
|
|
|
|
description: "Limit to the last X lines. Conflicts with a range.",
|
|
|
|
minimum: 0,
|
|
|
|
},
|
|
|
|
startcursor: {
|
|
|
|
type: String,
|
|
|
|
description: "Start after the given Cursor. Conflicts with 'since'.",
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
endcursor: {
|
|
|
|
type: String,
|
|
|
|
description: "End before the given Cursor. Conflicts with 'until'",
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
|
|
|
type: Array,
|
|
|
|
description: "Returns a list of journal entries.",
|
|
|
|
items: {
|
|
|
|
type: String,
|
|
|
|
description: "Line text.",
|
|
|
|
},
|
|
|
|
},
|
2020-04-16 10:44:59 +00:00
|
|
|
access: {
|
2020-04-30 07:30:00 +00:00
|
|
|
permission: &Permission::Privilege(&["system", "log"], PRIV_SYS_AUDIT, false),
|
2020-04-16 10:44:59 +00:00
|
|
|
},
|
2019-12-17 13:06:48 +00:00
|
|
|
)]
|
|
|
|
/// Read syslog entries.
|
|
|
|
fn get_journal(
|
2021-05-25 11:19:47 +00:00
|
|
|
since: Option<i64>,
|
|
|
|
until: Option<i64>,
|
|
|
|
lastentries: Option<u64>,
|
|
|
|
startcursor: Option<String>,
|
|
|
|
endcursor: Option<String>,
|
|
|
|
_param: Value,
|
2019-12-17 13:06:48 +00:00
|
|
|
_info: &ApiMethod,
|
|
|
|
_rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
|
|
|
|
|
|
|
let mut args = vec![];
|
|
|
|
|
2021-05-25 11:19:47 +00:00
|
|
|
if let Some(lastentries) = lastentries {
|
2019-12-17 13:06:48 +00:00
|
|
|
args.push(String::from("-n"));
|
|
|
|
args.push(format!("{}", lastentries));
|
|
|
|
}
|
|
|
|
|
2021-05-25 11:19:47 +00:00
|
|
|
if let Some(since) = since {
|
2019-12-17 13:06:48 +00:00
|
|
|
args.push(String::from("-b"));
|
2021-05-25 11:19:47 +00:00
|
|
|
args.push(since.to_string());
|
2019-12-17 13:06:48 +00:00
|
|
|
}
|
|
|
|
|
2021-05-25 11:19:47 +00:00
|
|
|
if let Some(until) = until {
|
2019-12-17 13:06:48 +00:00
|
|
|
args.push(String::from("-e"));
|
2021-05-25 11:19:47 +00:00
|
|
|
args.push(until.to_string());
|
2019-12-17 13:06:48 +00:00
|
|
|
}
|
|
|
|
|
2021-05-25 11:19:47 +00:00
|
|
|
if let Some(startcursor) = startcursor {
|
2019-12-17 13:06:48 +00:00
|
|
|
args.push(String::from("-f"));
|
2021-05-25 11:19:47 +00:00
|
|
|
args.push(startcursor);
|
2019-12-17 13:06:48 +00:00
|
|
|
}
|
|
|
|
|
2021-05-25 11:19:47 +00:00
|
|
|
if let Some(endcursor) = endcursor {
|
2019-12-17 13:06:48 +00:00
|
|
|
args.push(String::from("-t"));
|
2021-05-25 11:19:47 +00:00
|
|
|
args.push(endcursor);
|
2019-12-17 13:06:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut lines: Vec<String> = vec![];
|
|
|
|
|
2020-06-15 09:16:11 +00:00
|
|
|
let mut child = Command::new("mini-journalreader")
|
2019-12-17 13:06:48 +00:00
|
|
|
.args(&args)
|
|
|
|
.stdout(Stdio::piped())
|
|
|
|
.spawn()?;
|
|
|
|
|
|
|
|
if let Some(ref mut stdout) = child.stdout {
|
|
|
|
for line in BufReader::new(stdout).lines() {
|
|
|
|
match line {
|
|
|
|
Ok(line) => {
|
|
|
|
lines.push(line);
|
|
|
|
}
|
|
|
|
Err(err) => {
|
|
|
|
log::error!("reading journal failed: {}", err);
|
|
|
|
let _ = child.kill();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let status = child.wait().unwrap();
|
|
|
|
if !status.success() {
|
|
|
|
log::error!("journalctl failed with {}", status);
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(json!(lines))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub const ROUTER: Router = Router::new()
|
|
|
|
.get(&API_METHOD_GET_JOURNAL);
|