use proxmox::list_subdirs_api_method; use anyhow::{Error}; use serde_json::{json, Value}; use proxmox::api::{ api, ApiMethod, Permission, Router, RpcEnvironment, SubdirMap, }; use crate::api2::types::{ DATASTORE_SCHEMA, RRDMode, RRDTimeFrameResolution, Authid, }; use crate::backup::{DataStore}; use crate::config::datastore; use crate::tools::statistics::{linear_regression}; use crate::config::cached_user_info::CachedUserInfo; use crate::config::acl::{ PRIV_DATASTORE_AUDIT, PRIV_DATASTORE_BACKUP, }; #[api( returns: { description: "Lists the Status of the Datastores.", type: Array, items: { description: "Status of a Datastore", type: Object, properties: { store: { schema: DATASTORE_SCHEMA, }, total: { type: Integer, description: "The Size of the underlying storage in bytes", }, used: { type: Integer, description: "The used bytes of the underlying storage", }, avail: { type: Integer, description: "The available bytes of the underlying storage", }, history: { type: Array, description: "A list of usages of the past (last Month).", items: { type: Number, description: "The usage of a time in the past. Either null or between 0.0 and 1.0.", } }, "estimated-full-date": { type: Integer, optional: true, description: "Estimation of the UNIX epoch when the storage will be full.\ This is calculated via a simple Linear Regression (Least Squares)\ of RRD data of the last Month. Missing if there are not enough data points yet.\ If the estimate lies in the past, the usage is decreasing.", }, }, }, }, access: { permission: &Permission::Anybody, }, )] /// List Datastore usages and estimates fn datastore_status( _param: Value, _info: &ApiMethod, rpcenv: &mut dyn RpcEnvironment, ) -> Result { let (config, _digest) = datastore::config()?; let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; let user_info = CachedUserInfo::new()?; let mut list = Vec::new(); for (store, (_, _)) in &config.sections { let user_privs = user_info.lookup_privs(&auth_id, &["datastore", &store]); let allowed = (user_privs & (PRIV_DATASTORE_AUDIT| PRIV_DATASTORE_BACKUP)) != 0; if !allowed { continue; } let datastore = DataStore::lookup_datastore(&store)?; let status = crate::tools::disks::disk_usage(&datastore.base_path())?; let mut entry = json!({ "store": store, "total": status.total, "used": status.used, "avail": status.avail, "gc-status": datastore.last_gc_status(), }); let rrd_dir = format!("datastore/{}", store); let now = proxmox::tools::time::epoch_f64(); let rrd_resolution = RRDTimeFrameResolution::Month; let rrd_mode = RRDMode::Average; let total_res = crate::rrd::extract_cached_data( &rrd_dir, "total", now, rrd_resolution, rrd_mode, ); let used_res = crate::rrd::extract_cached_data( &rrd_dir, "used", now, rrd_resolution, rrd_mode, ); match (total_res, used_res) { (Some((start, reso, total_list)), Some((_, _, used_list))) => { let mut usage_list: Vec = Vec::new(); let mut time_list: Vec = Vec::new(); let mut history = Vec::new(); for (idx, used) in used_list.iter().enumerate() { let total = if idx < total_list.len() { total_list[idx] } else { None }; match (total, used) { (Some(total), Some(used)) if total != 0.0 => { time_list.push(start + (idx as u64)*reso); let usage = used/total; usage_list.push(usage); history.push(json!(usage)); }, _ => { history.push(json!(null)) } } } entry["history-start"] = start.into(); entry["history-delta"] = reso.into(); entry["history"] = history.into(); // we skip the calculation for datastores with not enough data if usage_list.len() >= 7 { if let Some((a,b)) = linear_regression(&time_list, &usage_list) { if b != 0.0 { let estimate = (1.0 - a) / b; entry["estimated-full-date"] = Value::from(estimate.floor() as u64); } else { entry["estimated-full-date"] = Value::from(0); } } } }, _ => {}, } list.push(entry); } Ok(list.into()) } const SUBDIRS: SubdirMap = &[ ("datastore-usage", &Router::new().get(&API_METHOD_DATASTORE_STATUS)), ]; pub const ROUTER: Router = Router::new() .get(&list_subdirs_api_method!(SUBDIRS)) .subdirs(SUBDIRS);