monitord/
pid1.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
//! # pid1 module
//!
//! `pid1` uses procfs to get some statistics on Linux's more important
//! process pid1. These metrics can help ensure newer systemds don't regress
//! or show stange behavior. E.g. more file descriptors without more units.

use std::sync::Arc;

#[cfg(target_os = "linux")]
use procfs::process::Process;
use tokio::sync::RwLock;
use tracing::error;

use crate::MachineStats;

#[derive(serde::Serialize, serde::Deserialize, Clone, Debug, Default, Eq, PartialEq)]
pub struct Pid1Stats {
    pub cpu_time_kernel: u64,
    pub cpu_time_user: u64,
    pub memory_usage_bytes: u64,
    pub fd_count: u64,
    pub tasks: u64,
}

/// Get procfs info on pid 1 - <https://manpages.debian.org/buster/manpages/procfs.5.en.html>
#[cfg(target_os = "linux")]
pub fn get_pid_stats(pid: i32) -> anyhow::Result<Pid1Stats> {
    let bytes_per_page = procfs::page_size();
    let ticks_per_second = procfs::ticks_per_second();

    let pid1_proc = Process::new(pid)?;
    let stat_file = pid1_proc.stat()?;

    // Living with integer rounding
    Ok(Pid1Stats {
        cpu_time_kernel: (stat_file.stime) / (ticks_per_second),
        cpu_time_user: (stat_file.utime) / (ticks_per_second),
        memory_usage_bytes: (stat_file.rss) * (bytes_per_page),
        fd_count: pid1_proc.fd_count()?.try_into()?,
        // Using 0 as impossible number of tasks
        tasks: pid1_proc
            .tasks()?
            .flatten()
            .collect::<Vec<_>>()
            .len()
            .try_into()?,
    })
}

#[cfg(not(target_os = "linux"))]
pub fn get_pid_stats(_pid: i32) -> anyhow::Result<Pid1Stats> {
    error!("pid1 stats not supported on this OS");
    Ok(Pid1Stats::default())
}

/// Async wrapper than can update PID1 stats when passed a locked struct
pub async fn update_pid1_stats(
    pid: i32,
    locked_machine_stats: Arc<RwLock<MachineStats>>,
) -> anyhow::Result<()> {
    let pid1_stats = match tokio::task::spawn_blocking(move || get_pid_stats(pid)).await {
        Ok(p1s) => p1s,
        Err(err) => return Err(err.into()),
    };

    let mut machine_stats = locked_machine_stats.write().await;
    machine_stats.pid1 = match pid1_stats {
        Ok(s) => Some(s),
        Err(err) => {
            error!("Unable to set pid1 stats: {:?}", err);
            None
        }
    };

    Ok(())
}

#[cfg(target_os = "linux")]
#[cfg(test)]
pub mod tests {
    use super::*;

    #[test]
    pub fn test_get_stats() -> anyhow::Result<()> {
        let pid1_stats = get_pid_stats(1)?;
        assert!(pid1_stats.tasks > 0);
        Ok(())
    }
}