Testing
This commit is contained in:
parent
0b1d3ae0ad
commit
fb91d8346f
@ -6,6 +6,7 @@ use std::process::Stdio;
|
|||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use tokio::process::Command;
|
use tokio::process::Command;
|
||||||
use tokio::time::timeout;
|
use tokio::time::timeout;
|
||||||
|
use tokio::fs;
|
||||||
|
|
||||||
use super::{AgentType, Collector, CollectorError, CollectorOutput};
|
use super::{AgentType, Collector, CollectorError, CollectorOutput};
|
||||||
|
|
||||||
@ -34,6 +35,24 @@ impl BackupCollector {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn get_borgbackup_metrics(&self) -> Result<BorgbackupMetrics, CollectorError> {
|
||||||
|
// Read metrics from the borgbackup JSON file
|
||||||
|
let metrics_path = "/var/lib/backup/backup-metrics.json";
|
||||||
|
|
||||||
|
let content = fs::read_to_string(metrics_path)
|
||||||
|
.await
|
||||||
|
.map_err(|e| CollectorError::IoError {
|
||||||
|
message: format!("Failed to read backup metrics file: {}", e),
|
||||||
|
})?;
|
||||||
|
|
||||||
|
let metrics: BorgbackupMetrics = serde_json::from_str(&content)
|
||||||
|
.map_err(|e| CollectorError::ParseError {
|
||||||
|
message: format!("Failed to parse backup metrics JSON: {}", e),
|
||||||
|
})?;
|
||||||
|
|
||||||
|
Ok(metrics)
|
||||||
|
}
|
||||||
|
|
||||||
async fn get_restic_snapshots(&self) -> Result<ResticStats, CollectorError> {
|
async fn get_restic_snapshots(&self) -> Result<ResticStats, CollectorError> {
|
||||||
let repo = self
|
let repo = self
|
||||||
.restic_repo
|
.restic_repo
|
||||||
@ -290,8 +309,66 @@ impl Collector for BackupCollector {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async fn collect(&self) -> Result<CollectorOutput, CollectorError> {
|
async fn collect(&self) -> Result<CollectorOutput, CollectorError> {
|
||||||
// Get restic repository stats
|
// Try to get borgbackup metrics first, fall back to restic if not available
|
||||||
let restic_stats = self.get_restic_snapshots().await;
|
let borgbackup_result = self.get_borgbackup_metrics().await;
|
||||||
|
|
||||||
|
let (backup_info, overall_status) = match borgbackup_result {
|
||||||
|
Ok(borg_metrics) => {
|
||||||
|
// Parse borgbackup timestamp to DateTime
|
||||||
|
let last_success = chrono::DateTime::from_timestamp(borg_metrics.timestamp, 0);
|
||||||
|
|
||||||
|
// Determine status from borgbackup data
|
||||||
|
let status = match borg_metrics.status.as_str() {
|
||||||
|
"success" => BackupStatus::Healthy,
|
||||||
|
"warning" => BackupStatus::Warning,
|
||||||
|
"failed" => BackupStatus::Failed,
|
||||||
|
_ => BackupStatus::Unknown,
|
||||||
|
};
|
||||||
|
|
||||||
|
let backup_info = BackupInfo {
|
||||||
|
last_success,
|
||||||
|
last_failure: None, // borgbackup metrics don't include failure info
|
||||||
|
size_gb: borg_metrics.repository.total_repository_size_bytes as f32 / (1024.0 * 1024.0 * 1024.0),
|
||||||
|
snapshot_count: borg_metrics.repository.total_archives as u32,
|
||||||
|
};
|
||||||
|
|
||||||
|
(backup_info, status)
|
||||||
|
},
|
||||||
|
Err(_) => {
|
||||||
|
// Fall back to restic if borgbackup metrics not available
|
||||||
|
let restic_stats = self.get_restic_snapshots().await;
|
||||||
|
let last_failure = self.get_backup_logs_for_failures().await.unwrap_or(None);
|
||||||
|
|
||||||
|
// Get backup service status for fallback determination
|
||||||
|
let service_data = self
|
||||||
|
.get_backup_service_status()
|
||||||
|
.await
|
||||||
|
.unwrap_or(BackupServiceData {
|
||||||
|
enabled: false,
|
||||||
|
pending_jobs: 0,
|
||||||
|
last_message: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
let overall_status = self.determine_backup_status(&restic_stats, &service_data, last_failure);
|
||||||
|
|
||||||
|
let backup_info = match &restic_stats {
|
||||||
|
Ok(stats) => BackupInfo {
|
||||||
|
last_success: stats.last_success,
|
||||||
|
last_failure,
|
||||||
|
size_gb: stats.total_size as f32 / (1024.0 * 1024.0 * 1024.0),
|
||||||
|
snapshot_count: stats.snapshot_count,
|
||||||
|
},
|
||||||
|
Err(_) => BackupInfo {
|
||||||
|
last_success: None,
|
||||||
|
last_failure,
|
||||||
|
size_gb: 0.0,
|
||||||
|
snapshot_count: 0,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
(backup_info, overall_status)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// Get backup service status
|
// Get backup service status
|
||||||
let service_data = self
|
let service_data = self
|
||||||
@ -303,34 +380,6 @@ impl Collector for BackupCollector {
|
|||||||
last_message: None,
|
last_message: None,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Check for recent failures
|
|
||||||
let last_failure = self.get_backup_logs_for_failures().await.unwrap_or(None);
|
|
||||||
|
|
||||||
// Determine overall backup status
|
|
||||||
let overall_status =
|
|
||||||
self.determine_backup_status(&restic_stats, &service_data, last_failure);
|
|
||||||
|
|
||||||
let (backup_info, _size_gb) = match &restic_stats {
|
|
||||||
Ok(stats) => (
|
|
||||||
BackupInfo {
|
|
||||||
last_success: stats.last_success,
|
|
||||||
last_failure,
|
|
||||||
size_gb: stats.total_size as f32 / (1024.0 * 1024.0 * 1024.0),
|
|
||||||
snapshot_count: stats.snapshot_count,
|
|
||||||
},
|
|
||||||
stats.total_size as f32 / (1024.0 * 1024.0 * 1024.0),
|
|
||||||
),
|
|
||||||
Err(_) => (
|
|
||||||
BackupInfo {
|
|
||||||
last_success: None,
|
|
||||||
last_failure,
|
|
||||||
size_gb: 0.0,
|
|
||||||
snapshot_count: 0,
|
|
||||||
},
|
|
||||||
0.0,
|
|
||||||
),
|
|
||||||
};
|
|
||||||
|
|
||||||
let backup_metrics = json!({
|
let backup_metrics = json!({
|
||||||
"overall_status": overall_status,
|
"overall_status": overall_status,
|
||||||
"backup": backup_info,
|
"backup": backup_info,
|
||||||
@ -386,3 +435,43 @@ struct JournalEntry {
|
|||||||
#[serde(rename = "__REALTIME_TIMESTAMP")]
|
#[serde(rename = "__REALTIME_TIMESTAMP")]
|
||||||
realtime_timestamp: String,
|
realtime_timestamp: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Borgbackup metrics structure from backup script
|
||||||
|
#[derive(Debug, Deserialize)]
|
||||||
|
struct BorgbackupMetrics {
|
||||||
|
backup_name: String,
|
||||||
|
start_time: String,
|
||||||
|
end_time: String,
|
||||||
|
duration_seconds: i64,
|
||||||
|
status: String,
|
||||||
|
exit_codes: ExitCodes,
|
||||||
|
repository: Repository,
|
||||||
|
backup_disk: BackupDisk,
|
||||||
|
timestamp: i64,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Deserialize)]
|
||||||
|
struct ExitCodes {
|
||||||
|
global: i32,
|
||||||
|
backup: i32,
|
||||||
|
prune: i32,
|
||||||
|
compact: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Deserialize)]
|
||||||
|
struct Repository {
|
||||||
|
total_archives: i32,
|
||||||
|
latest_archive_size_bytes: i64,
|
||||||
|
total_repository_size_bytes: i64,
|
||||||
|
path: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Deserialize)]
|
||||||
|
struct BackupDisk {
|
||||||
|
device: String,
|
||||||
|
health: String,
|
||||||
|
total_bytes: i64,
|
||||||
|
used_bytes: i64,
|
||||||
|
available_bytes: i64,
|
||||||
|
usage_percent: f32,
|
||||||
|
}
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user