mirror of
https://codeberg.org/icewind/palantir.git
synced 2026-06-03 18:24:08 +02:00
power
This commit is contained in:
parent
c1b77cd2ab
commit
825e7cace8
4 changed files with 105 additions and 1 deletions
|
|
@ -1,4 +1,5 @@
|
|||
pub mod docker;
|
||||
pub mod power;
|
||||
pub mod sensors;
|
||||
pub mod zfs;
|
||||
|
||||
|
|
|
|||
|
|
@ -4,6 +4,7 @@ use futures_util::pin_mut;
|
|||
use futures_util::StreamExt;
|
||||
use palantir::docker::{get_docker, stat, Container};
|
||||
use palantir::get_metrics;
|
||||
use palantir::power::power_usage;
|
||||
use warp::reject::Reject;
|
||||
use warp::{Filter, Rejection};
|
||||
|
||||
|
|
@ -21,8 +22,8 @@ impl Reject for ReportRejection {}
|
|||
|
||||
async fn serve_inner(docker: Option<Docker>) -> Result<String> {
|
||||
let mut metrics = get_metrics()?;
|
||||
let hostname = palantir::sensors::hostname()?;
|
||||
if let Some(docker) = docker {
|
||||
let hostname = palantir::sensors::hostname()?;
|
||||
let containers = stat(docker).await?;
|
||||
pin_mut!(containers);
|
||||
while let Some(container) = containers.next().await {
|
||||
|
|
@ -30,6 +31,9 @@ async fn serve_inner(docker: Option<Docker>) -> Result<String> {
|
|||
container.write(&mut metrics, &hostname);
|
||||
}
|
||||
}
|
||||
if let Some(power) = power_usage()? {
|
||||
power.write(&mut metrics, &hostname);
|
||||
}
|
||||
Ok(metrics)
|
||||
}
|
||||
|
||||
|
|
|
|||
65
src/power.rs
Normal file
65
src/power.rs
Normal file
|
|
@ -0,0 +1,65 @@
|
|||
use color_eyre::{Report, Result};
|
||||
use std::fmt::Write;
|
||||
use std::fs::{read_dir, read_to_string};
|
||||
use std::sync::atomic::{AtomicBool, Ordering};
|
||||
|
||||
static CAN_READ: AtomicBool = AtomicBool::new(true);
|
||||
|
||||
#[derive(Debug, Default)]
|
||||
pub struct PowerUsage {
|
||||
total_uj: u64,
|
||||
packages_uj: Vec<u64>,
|
||||
}
|
||||
|
||||
impl PowerUsage {
|
||||
pub fn write<W: Write>(&self, mut w: W, hostname: &str) {
|
||||
writeln!(
|
||||
&mut w,
|
||||
"total_power{{host=\"{}\"}} {:.3}",
|
||||
hostname,
|
||||
self.total_uj as f64 / 1_000_000.0
|
||||
)
|
||||
.ok();
|
||||
for (i, package) in self.packages_uj.iter().enumerate() {
|
||||
writeln!(
|
||||
&mut w,
|
||||
"total_power{{host=\"{}\", package=\"{}\"}} {:.3}",
|
||||
hostname,
|
||||
i,
|
||||
*package as f64 / 1_000_000.0
|
||||
)
|
||||
.ok();
|
||||
}
|
||||
}
|
||||
}
|
||||
pub fn power_usage() -> Result<Option<PowerUsage>> {
|
||||
if !CAN_READ.load(Ordering::Relaxed) {
|
||||
return Ok(None);
|
||||
}
|
||||
|
||||
let dir = read_dir("/sys/devices/virtual/powercap/intel-rapl")?;
|
||||
let mut usage = PowerUsage::default();
|
||||
for package in dir {
|
||||
let package = package?;
|
||||
if package
|
||||
.file_name()
|
||||
.to_str()
|
||||
.ok_or_else(|| Report::msg("Invalid name"))?
|
||||
.starts_with("intel-rapl")
|
||||
{
|
||||
let mut package_path = package.path();
|
||||
package_path.push("energy_uj");
|
||||
let package_usage = match read_to_string(&package_path) {
|
||||
Err(e) if e.raw_os_error() == Some(13) => {
|
||||
CAN_READ.store(false, Ordering::Relaxed);
|
||||
return Ok(None);
|
||||
}
|
||||
result => result,
|
||||
}?;
|
||||
let package_usage = dbg!(package_usage.trim().parse::<u64>()?);
|
||||
usage.total_uj += package_usage;
|
||||
usage.packages_uj.push(package_usage);
|
||||
}
|
||||
}
|
||||
Ok(Some(usage))
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue