nullnet_libresmon/
lib.rs

1use async_channel::Receiver;
2use std::collections::HashMap;
3use std::fmt::{Display, Formatter};
4use std::path::Path;
5use sysinfo::{
6    Components, CpuRefreshKind, DiskRefreshKind, Disks, MemoryRefreshKind, RefreshKind, System,
7};
8
9static SYSTEM_REFRESH_KIND: std::sync::LazyLock<RefreshKind> = std::sync::LazyLock::new(|| {
10    RefreshKind::nothing()
11        .with_cpu(CpuRefreshKind::nothing().with_cpu_usage())
12        .with_memory(MemoryRefreshKind::nothing().with_ram())
13});
14
15static DISK_REFRESH_KIND: std::sync::LazyLock<DiskRefreshKind> =
16    std::sync::LazyLock::new(|| DiskRefreshKind::nothing().with_io_usage().with_storage());
17
18#[derive(Default)]
19pub struct SystemResources {
20    pub num_cpus: usize,
21    pub global_cpu_usage: f32,
22    pub cpu_usages: HashMap<String, f32>,
23    pub total_memory: u64,
24    pub used_memory: u64,
25    pub total_disk_space: u64,
26    pub available_disk_space: u64,
27    pub read_bytes: u64,
28    pub written_bytes: u64,
29    pub temperatures: HashMap<String, Option<f32>>,
30}
31
32#[must_use]
33pub fn poll_system_resources(interval_msec: u64) -> Receiver<SystemResources> {
34    let (tx, rx) = async_channel::unbounded();
35
36    std::thread::spawn(move || {
37        let mut sys = System::new_with_specifics(*SYSTEM_REFRESH_KIND);
38        let mut disks = Disks::new_with_refreshed_list_specifics(*DISK_REFRESH_KIND);
39        let mut components = Components::new_with_refreshed_list();
40        loop {
41            // sleep for 1 second
42            std::thread::sleep(std::time::Duration::from_millis(interval_msec));
43
44            // refresh system resources
45            sys.refresh_specifics(*SYSTEM_REFRESH_KIND);
46            disks.refresh_specifics(true, *DISK_REFRESH_KIND);
47            components.refresh(true);
48
49            // update resources
50
51            let mut cpu_usages = HashMap::new();
52            for cpu in sys.cpus() {
53                let usage = cpu.cpu_usage();
54                cpu_usages.insert(cpu.name().to_string(), usage);
55            }
56
57            let mut total_disk_space = 0;
58            let mut available_disk_space = 0;
59            let mut read_bytes = 0;
60            let mut written_bytes = 0;
61            for disk in &disks {
62                if disk.mount_point() == Path::new("/") {
63                    total_disk_space = disk.total_space();
64                    available_disk_space = disk.available_space();
65                    let disk_usage = disk.usage();
66                    read_bytes = disk_usage.read_bytes;
67                    written_bytes = disk_usage.written_bytes;
68                }
69            }
70
71            let mut temperatures = HashMap::new();
72            for component in &components {
73                let temperature = component.temperature();
74                temperatures.insert(component.label().to_string(), temperature);
75            }
76
77            let resources = SystemResources {
78                num_cpus: sys.cpus().len(),
79                global_cpu_usage: sys.global_cpu_usage(),
80                cpu_usages,
81                total_memory: sys.total_memory(),
82                used_memory: sys.used_memory(),
83                total_disk_space,
84                available_disk_space,
85                read_bytes,
86                written_bytes,
87                temperatures,
88            };
89
90            // send resources to caller
91            let _ = tx.send_blocking(resources);
92        }
93    });
94
95    rx
96}
97
98impl Display for SystemResources {
99    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
100        writeln!(
101            f,
102            "\n\nCPU -------------------------------------------------------------------------\n"
103        )?;
104        writeln!(f, "Number of CPUs: {}", self.num_cpus)?;
105        writeln!(f, "Global CPU usage: {:.2}%", self.global_cpu_usage)?;
106        let mut sorted_cpu_usages = self
107            .cpu_usages
108            .iter()
109            .map(|(k, v)| (k.clone(), *v))
110            .collect::<Vec<_>>();
111        sorted_cpu_usages.sort_by(|(l1, _), (l2, _)| l1.cmp(l2));
112        for (name, usage) in &sorted_cpu_usages {
113            writeln!(f, "  - '{name}' usage: {usage:.2}%")?;
114        }
115
116        writeln!(
117            f,
118            "\n\nRAM -------------------------------------------------------------------------\n"
119        )?;
120        writeln!(f, "Total RAM space: {} bytes", self.total_memory)?;
121        writeln!(f, "Used RAM space: {} bytes", self.used_memory)?;
122
123        writeln!(
124            f,
125            "\n\nDISK ------------------------------------------------------------------------\n"
126        )?;
127        writeln!(f, "Total disk space : {} bytes", self.total_disk_space)?;
128        writeln!(
129            f,
130            "Available disk space: {} bytes",
131            self.available_disk_space
132        )?;
133
134        writeln!(
135            f,
136            "\n\nI/O -------------------------------------------------------------------------\n"
137        )?;
138        writeln!(f, "Read: {:?} bytes", self.read_bytes)?;
139        writeln!(f, "Written: {:?} bytes", self.written_bytes)?;
140
141        writeln!(
142            f,
143            "\n\nTEMPERATURE -----------------------------------------------------------------\n"
144        )?;
145        let mut sorted_temperatures = self
146            .temperatures
147            .iter()
148            .map(|(k, v)| (k.clone(), *v))
149            .collect::<Vec<_>>();
150        sorted_temperatures.sort_by(|(l1, _), (l2, _)| l1.cmp(l2));
151        for (label, temperature) in &sorted_temperatures {
152            let temperature_str = temperature
153                .map(|t| format!("{t:.2}°C"))
154                .unwrap_or("?".to_string());
155            writeln!(f, "{label}: {temperature_str}")?;
156        }
157
158        Ok(())
159    }
160}