- Added new dependencies: `adler2`, `crc32fast`, `flate2`, `miniz_oxide`, and `libredox`. - Updated existing dependencies: `tokio-rustls` to version 0.26.4 and `filetime` to version 0.2.27. - Removed the `backup.rs` file as it is no longer needed. - Introduced tests for configuration and credential management. - Enhanced the `identity` module to generate W3C compliant DID documents. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
306 lines
11 KiB
Rust
306 lines
11 KiB
Rust
// Disk Space Monitor
|
|
// Periodically checks disk usage and triggers automatic cleanup at 90%.
|
|
|
|
use anyhow::{Context, Result};
|
|
use tracing::{info, warn};
|
|
|
|
/// Parse df output into (used_bytes, total_bytes, used_percent).
|
|
/// Expects output from `df --block-size=1 --output=used,size /` which has a header line
|
|
/// followed by a data line with two whitespace-separated numbers.
|
|
fn parse_df_output(stdout: &str) -> Result<(u64, u64, f64)> {
|
|
let data_line = stdout
|
|
.lines()
|
|
.nth(1)
|
|
.ok_or_else(|| anyhow::anyhow!("No data line from df"))?;
|
|
let mut parts = data_line.split_whitespace();
|
|
let used: u64 = parts
|
|
.next()
|
|
.ok_or_else(|| anyhow::anyhow!("Missing used"))?
|
|
.parse()
|
|
.context("parse df used")?;
|
|
let total: u64 = parts
|
|
.next()
|
|
.ok_or_else(|| anyhow::anyhow!("Missing total"))?
|
|
.parse()
|
|
.context("parse df total")?;
|
|
|
|
let percent = if total > 0 {
|
|
(used as f64 / total as f64) * 100.0
|
|
} else {
|
|
0.0
|
|
};
|
|
|
|
Ok((used, total, percent))
|
|
}
|
|
|
|
/// Check disk usage percentage for the root filesystem.
|
|
/// Returns (used_bytes, total_bytes, used_percent).
|
|
pub async fn check_disk_usage() -> Result<(u64, u64, f64)> {
|
|
let output = tokio::process::Command::new("df")
|
|
.args(["--block-size=1", "--output=used,size", "/"])
|
|
.output()
|
|
.await
|
|
.context("Failed to run df")?;
|
|
|
|
if !output.status.success() {
|
|
anyhow::bail!("df failed: {}", String::from_utf8_lossy(&output.stderr));
|
|
}
|
|
|
|
let stdout = String::from_utf8(output.stdout).context("df output not utf8")?;
|
|
parse_df_output(&stdout)
|
|
}
|
|
|
|
/// Run automatic cleanup when disk usage exceeds 90%.
|
|
async fn auto_cleanup() -> Result<u64> {
|
|
let mut freed: u64 = 0;
|
|
|
|
// Prune dangling images
|
|
let output = tokio::process::Command::new("sudo")
|
|
.args(["podman", "image", "prune", "-f"])
|
|
.output()
|
|
.await;
|
|
if let Ok(out) = output {
|
|
if out.status.success() {
|
|
let count = String::from_utf8_lossy(&out.stdout)
|
|
.lines()
|
|
.filter(|l| !l.trim().is_empty())
|
|
.count();
|
|
freed += count as u64 * 100_000_000;
|
|
}
|
|
}
|
|
|
|
// Clean old rotated logs (> 14 days for auto-cleanup, more aggressive)
|
|
let _ = tokio::process::Command::new("sudo")
|
|
.args([
|
|
"find", "/var/log", "-type", "f", "-name", "*.log.*",
|
|
"-mtime", "+14", "-delete",
|
|
])
|
|
.output()
|
|
.await;
|
|
|
|
let _ = tokio::process::Command::new("sudo")
|
|
.args([
|
|
"find", "/var/log", "-type", "f", "-name", "*.gz",
|
|
"-mtime", "+14", "-delete",
|
|
])
|
|
.output()
|
|
.await;
|
|
|
|
// Truncate large journal logs
|
|
let _ = tokio::process::Command::new("sudo")
|
|
.args(["journalctl", "--vacuum-size=100M"])
|
|
.output()
|
|
.await;
|
|
|
|
Ok(freed)
|
|
}
|
|
|
|
/// Spawn a background task that monitors disk usage every 5 minutes.
|
|
/// Triggers automatic cleanup at 90% and logs warnings at 85%.
|
|
pub fn spawn_disk_monitor(data_dir: std::path::PathBuf) {
|
|
tokio::spawn(async move {
|
|
// Initial delay to let system stabilize
|
|
tokio::time::sleep(std::time::Duration::from_secs(60)).await;
|
|
|
|
let mut interval = tokio::time::interval(std::time::Duration::from_secs(300));
|
|
let mut last_warning_level: Option<&str> = None;
|
|
|
|
loop {
|
|
interval.tick().await;
|
|
|
|
match check_disk_usage().await {
|
|
Ok((_used, _total, percent)) => {
|
|
if percent >= 90.0 {
|
|
if last_warning_level != Some("critical") {
|
|
warn!("Disk usage critical: {:.1}% — triggering automatic cleanup", percent);
|
|
last_warning_level = Some("critical");
|
|
}
|
|
match auto_cleanup().await {
|
|
Ok(freed) => {
|
|
if freed > 0 {
|
|
info!("Auto-cleanup freed approximately {} bytes", freed);
|
|
}
|
|
}
|
|
Err(e) => warn!("Auto-cleanup failed: {}", e),
|
|
}
|
|
// Write disk warning file for the frontend to poll
|
|
let warning_path = data_dir.join("disk-warning.json");
|
|
let _ = tokio::fs::write(
|
|
&warning_path,
|
|
serde_json::json!({
|
|
"level": "critical",
|
|
"percent": (percent * 10.0).round() / 10.0,
|
|
"timestamp": chrono::Utc::now().to_rfc3339(),
|
|
})
|
|
.to_string(),
|
|
)
|
|
.await;
|
|
} else if percent >= 85.0 {
|
|
if last_warning_level != Some("warning") {
|
|
warn!("Disk usage warning: {:.1}% — approaching critical threshold", percent);
|
|
last_warning_level = Some("warning");
|
|
}
|
|
let warning_path = data_dir.join("disk-warning.json");
|
|
let _ = tokio::fs::write(
|
|
&warning_path,
|
|
serde_json::json!({
|
|
"level": "warning",
|
|
"percent": (percent * 10.0).round() / 10.0,
|
|
"timestamp": chrono::Utc::now().to_rfc3339(),
|
|
})
|
|
.to_string(),
|
|
)
|
|
.await;
|
|
} else {
|
|
// Clear warning file if disk is healthy
|
|
if last_warning_level.is_some() {
|
|
let warning_path = data_dir.join("disk-warning.json");
|
|
let _ = tokio::fs::remove_file(&warning_path).await;
|
|
last_warning_level = None;
|
|
info!("Disk usage back to normal: {:.1}%", percent);
|
|
}
|
|
}
|
|
}
|
|
Err(e) => {
|
|
tracing::debug!("Disk usage check failed (non-fatal): {}", e);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn test_parse_df_output_normal() {
|
|
// Simulates typical df --block-size=1 --output=used,size / output
|
|
let output = " Used Size\n 500000000000 1000000000000\n";
|
|
let (used, total, percent) = parse_df_output(output).unwrap();
|
|
assert_eq!(used, 500_000_000_000);
|
|
assert_eq!(total, 1_000_000_000_000);
|
|
assert!((percent - 50.0).abs() < 0.01);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_high_usage() {
|
|
let output = " Used Size\n 900000000000 1000000000000\n";
|
|
let (used, total, percent) = parse_df_output(output).unwrap();
|
|
assert_eq!(used, 900_000_000_000);
|
|
assert_eq!(total, 1_000_000_000_000);
|
|
assert!((percent - 90.0).abs() < 0.01);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_almost_full() {
|
|
let output = "Used Size\n999 1000\n";
|
|
let (used, total, percent) = parse_df_output(output).unwrap();
|
|
assert_eq!(used, 999);
|
|
assert_eq!(total, 1000);
|
|
assert!((percent - 99.9).abs() < 0.01);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_empty_disk() {
|
|
let output = "Used Size\n0 1000000000000\n";
|
|
let (used, total, percent) = parse_df_output(output).unwrap();
|
|
assert_eq!(used, 0);
|
|
assert_eq!(total, 1_000_000_000_000);
|
|
assert!((percent - 0.0).abs() < 0.01);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_zero_total() {
|
|
// Edge case: total is 0 (should not happen but should not panic/divide-by-zero)
|
|
let output = "Used Size\n0 0\n";
|
|
let (used, total, percent) = parse_df_output(output).unwrap();
|
|
assert_eq!(used, 0);
|
|
assert_eq!(total, 0);
|
|
assert!((percent - 0.0).abs() < 0.01);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_no_data_line() {
|
|
let output = "Used Size\n";
|
|
let result = parse_df_output(output);
|
|
assert!(result.is_err());
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_empty_string() {
|
|
let result = parse_df_output("");
|
|
assert!(result.is_err());
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_single_header_only() {
|
|
let output = "Header Only";
|
|
let result = parse_df_output(output);
|
|
assert!(result.is_err());
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_non_numeric() {
|
|
let output = "Used Size\nabc def\n";
|
|
let result = parse_df_output(output);
|
|
assert!(result.is_err());
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_missing_second_field() {
|
|
let output = "Used Size\n12345\n";
|
|
let result = parse_df_output(output);
|
|
assert!(result.is_err());
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_extra_whitespace() {
|
|
let output = " Used Size \n 123456 7890000 \n";
|
|
let (used, total, _) = parse_df_output(output).unwrap();
|
|
assert_eq!(used, 123456);
|
|
assert_eq!(total, 7890000);
|
|
}
|
|
|
|
#[test]
|
|
fn test_parse_df_output_real_world_format() {
|
|
// Closer to real df output with header padding
|
|
let output = " Used Size\n 328000000000 1800000000000\n";
|
|
let (used, total, percent) = parse_df_output(output).unwrap();
|
|
assert_eq!(used, 328_000_000_000);
|
|
assert_eq!(total, 1_800_000_000_000);
|
|
// ~18.2%
|
|
assert!(percent > 18.0 && percent < 19.0);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_disk_warning_json_format() {
|
|
// Verify that the JSON structure we write for disk warnings is valid
|
|
let percent: f64 = 92.3;
|
|
let json = serde_json::json!({
|
|
"level": "critical",
|
|
"percent": (percent * 10.0).round() / 10.0,
|
|
"timestamp": chrono::Utc::now().to_rfc3339(),
|
|
});
|
|
let s = json.to_string();
|
|
let parsed: serde_json::Value = serde_json::from_str(&s).unwrap();
|
|
assert_eq!(parsed["level"], "critical");
|
|
assert_eq!(parsed["percent"], 92.3);
|
|
assert!(parsed["timestamp"].is_string());
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_disk_warning_json_warning_level() {
|
|
let percent: f64 = 87.5;
|
|
let json = serde_json::json!({
|
|
"level": "warning",
|
|
"percent": (percent * 10.0).round() / 10.0,
|
|
"timestamp": chrono::Utc::now().to_rfc3339(),
|
|
});
|
|
let parsed: serde_json::Value = serde_json::from_str(&json.to_string()).unwrap();
|
|
assert_eq!(parsed["level"], "warning");
|
|
// 87.5 rounded to 1 decimal = 87.5
|
|
assert_eq!(parsed["percent"], 87.5);
|
|
}
|
|
}
|