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
90
use crate::blocks::{Block, Configure, Message, Sender};
use crate::{ema, utils};
use regex::Regex;
use serde::Deserialize;
use std::thread;
const PATTERN: &str = r"cpu\s+(\d+)\s+(\d+)\s+(\d+)\s+(\d+)\s+(\d+)\s+(\d+)\s+(\d+)";
const PATH: &str = "/proc/stat";
#[derive(Configure, Deserialize)]
pub struct Cpu {
#[serde(default = "default_name")]
name: String,
#[serde(default = "default_period")]
period: f32,
#[serde(default = "default_alpha")]
alpha: f32,
}
fn default_name() -> String {
"cpu".to_string()
}
fn default_period() -> f32 {
1.0
}
fn default_alpha() -> f32 {
0.7
}
impl Sender for Cpu {
fn add_sender(&self, channel: crossbeam_channel::Sender<Message>) {
let name = self.get_name();
let monitor = utils::monitor_file(PATH.to_string(), self.period);
let mut perc = ema::Ema::new(self.alpha);
let mut cpu = Usage {
idle: 0.0,
total: 0.0,
};
let mut block = Block::new(name.clone(), true);
thread::spawn(move || {
for c in monitor {
let current_cpu = calc_cpu(match_proc(&c));
block.full_text = Some(format!(
" {:.1}%",
perc.push(calc_dcpu(¤t_cpu, &cpu))
));
channel.send((name.clone(), block.to_string())).unwrap();
cpu = current_cpu;
}
});
}
}
struct Usage {
idle: f32,
total: f32,
}
fn calc_cpu(stat: regex::Captures) -> Usage {
let stats: Vec<f32> = stat
.iter()
.skip(1)
.map(|x| x.unwrap().as_str().parse().unwrap())
.collect();
Usage {
idle: (stats[3] + stats[4]),
total: stats.iter().sum(),
}
}
fn calc_dcpu(cpu: &Usage, prevcpu: &Usage) -> f32 {
(1.0 - (cpu.idle - prevcpu.idle) / (cpu.total - prevcpu.total)) * 100.0
}
fn match_proc(s: &str) -> regex::Captures {
lazy_static! {
static ref RE: Regex = Regex::new(PATTERN).unwrap();
}
RE.captures(s).unwrap()
}