use async_channel::Receiver;
use std::collections::HashMap;
use std::fmt::{Display, Formatter};
use std::path::Path;
use sysinfo::{
Components, CpuRefreshKind, DiskRefreshKind, Disks, MemoryRefreshKind, RefreshKind, System,
};
static SYSTEM_REFRESH_KIND: std::sync::LazyLock<RefreshKind> = std::sync::LazyLock::new(|| {
RefreshKind::nothing()
.with_cpu(CpuRefreshKind::nothing().with_cpu_usage())
.with_memory(MemoryRefreshKind::nothing().with_ram())
});
static DISK_REFRESH_KIND: std::sync::LazyLock<DiskRefreshKind> =
std::sync::LazyLock::new(|| DiskRefreshKind::nothing().with_io_usage().with_storage());
#[derive(Default)]
pub struct SystemResources {
pub num_cpus: usize,
pub global_cpu_usage: f32,
pub cpu_usages: HashMap<String, f32>,
pub total_memory: u64,
pub used_memory: u64,
pub total_disk_space: u64,
pub available_disk_space: u64,
pub read_bytes: u64,
pub written_bytes: u64,
pub temperatures: HashMap<String, Option<f32>>,
}
#[must_use]
pub fn poll_system_resources(interval_msec: u64) -> Receiver<SystemResources> {
let (tx, rx) = async_channel::bounded(60);
std::thread::spawn(move || {
let mut sys = System::new_with_specifics(*SYSTEM_REFRESH_KIND);
let mut disks = Disks::new_with_refreshed_list_specifics(*DISK_REFRESH_KIND);
let mut components = Components::new_with_refreshed_list();
loop {
std::thread::sleep(std::time::Duration::from_millis(interval_msec));
sys.refresh_specifics(*SYSTEM_REFRESH_KIND);
disks.refresh_specifics(true, *DISK_REFRESH_KIND);
components.refresh(true);
let mut cpu_usages = HashMap::new();
for cpu in sys.cpus() {
let usage = cpu.cpu_usage();
cpu_usages.insert(cpu.name().to_string(), usage);
}
let mut total_disk_space = 0;
let mut available_disk_space = 0;
let mut read_bytes = 0;
let mut written_bytes = 0;
for disk in &disks {
if disk.mount_point() == Path::new("/") {
total_disk_space = disk.total_space();
available_disk_space = disk.available_space();
let disk_usage = disk.usage();
read_bytes = disk_usage.read_bytes;
written_bytes = disk_usage.written_bytes;
}
}
let mut temperatures = HashMap::new();
for component in &components {
let temperature = component.temperature();
temperatures.insert(component.label().to_string(), temperature);
}
let resources = SystemResources {
num_cpus: sys.cpus().len(),
global_cpu_usage: sys.global_cpu_usage(),
cpu_usages,
total_memory: sys.total_memory(),
used_memory: sys.used_memory(),
total_disk_space,
available_disk_space,
read_bytes,
written_bytes,
temperatures,
};
let Ok(()) = tx.send_blocking(resources) else {
return;
};
}
});
rx
}
impl Display for SystemResources {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
writeln!(
f,
"\n\nCPU -------------------------------------------------------------------------\n"
)?;
writeln!(f, "Number of CPUs: {}", self.num_cpus)?;
writeln!(f, "Global CPU usage: {:.2}%", self.global_cpu_usage)?;
let mut sorted_cpu_usages = self
.cpu_usages
.iter()
.map(|(k, v)| (k.clone(), *v))
.collect::<Vec<_>>();
sorted_cpu_usages.sort_by(|(l1, _), (l2, _)| l1.cmp(l2));
for (name, usage) in &sorted_cpu_usages {
writeln!(f, " - '{name}' usage: {usage:.2}%")?;
}
writeln!(
f,
"\n\nRAM -------------------------------------------------------------------------\n"
)?;
writeln!(f, "Total RAM space: {} bytes", self.total_memory)?;
writeln!(f, "Used RAM space: {} bytes", self.used_memory)?;
writeln!(
f,
"\n\nDISK ------------------------------------------------------------------------\n"
)?;
writeln!(f, "Total disk space : {} bytes", self.total_disk_space)?;
writeln!(
f,
"Available disk space: {} bytes",
self.available_disk_space
)?;
writeln!(
f,
"\n\nI/O -------------------------------------------------------------------------\n"
)?;
writeln!(f, "Read: {:?} bytes", self.read_bytes)?;
writeln!(f, "Written: {:?} bytes", self.written_bytes)?;
writeln!(
f,
"\n\nTEMPERATURE -----------------------------------------------------------------\n"
)?;
let mut sorted_temperatures = self
.temperatures
.iter()
.map(|(k, v)| (k.clone(), *v))
.collect::<Vec<_>>();
sorted_temperatures.sort_by(|(l1, _), (l2, _)| l1.cmp(l2));
for (label, temperature) in &sorted_temperatures {
let temperature_str = temperature
.map(|t| format!("{t:.2}°C"))
.unwrap_or("?".to_string());
writeln!(f, "{label}: {temperature_str}")?;
}
Ok(())
}
}