use std::time::Duration;
use mz_ore::cast::CastFrom;
use mz_ore::metrics::MetricsRegistry;
use prometheus::{Gauge, IntGauge};
macro_rules! metrics {
($namespace:ident $(($name:ident, $desc:expr, $suffix:expr, $type:ident)),*) => {
metrics! { @define $namespace $(($name, $desc, $suffix, $type)),*}
};
(@define $namespace:ident $(($name:ident, $desc:expr, $suffix:expr, $type:ident)),*) => {
struct RuMetrics {
$($name: <$type as Unit>::Gauge,)*
}
impl RuMetrics {
fn new(registry: &MetricsRegistry) -> Self {
Self {
$($name: registry.register(mz_ore::metric!(
name: concat!(stringify!($namespace), "_", stringify!($name), $suffix),
help: $desc,
)),)*
}
}
fn update(&self) {
let rusage = unsafe {
let mut rusage = std::mem::zeroed();
let ret = libc::getrusage(libc::RUSAGE_SELF, &mut rusage);
if ret < 0 {
return;
}
rusage
};
$(self.$name.set(<$type as Unit>::from(rusage.$name));)*
}
}
};
}
trait Unit {
type Gauge;
type From;
type To;
fn from(value: Self::From) -> Self::To;
}
struct Timeval;
impl Unit for Timeval {
type Gauge = Gauge;
type From = libc::timeval;
type To = f64;
fn from(Self::From { tv_sec, tv_usec }: Self::From) -> Self::To {
(Duration::from_secs(u64::cast_from(tv_sec.abs_diff(0)))
+ Duration::from_micros(u64::cast_from(tv_usec.abs_diff(0))))
.as_secs_f64()
}
}
struct Unitless;
impl Unit for Unitless {
type Gauge = IntGauge;
type From = libc::c_long;
type To = i64;
fn from(value: Self::From) -> Self::To {
value
}
}
struct MaxrssToBytes;
impl Unit for MaxrssToBytes {
type Gauge = IntGauge;
type From = libc::c_long;
type To = i64;
#[cfg(not(target_os = "macos"))]
fn from(value: Self::From) -> Self::To {
value.saturating_mul(1024)
}
#[cfg(target_os = "macos")]
fn from(value: Self::From) -> Self::To {
value
}
}
metrics! {
mz_metrics_libc
(ru_utime, "user CPU time used", "_seconds_total", Timeval),
(ru_stime, "system CPU time used", "_seconds_total", Timeval),
(ru_maxrss, "maximum resident set size", "_bytes", MaxrssToBytes),
(ru_minflt, "page reclaims (soft page faults)", "_total", Unitless),
(ru_majflt, "page faults (hard page faults)", "_total", Unitless),
(ru_inblock, "block input operations", "_total", Unitless),
(ru_oublock, "block output operations", "_total", Unitless),
(ru_nvcsw, "voluntary context switches", "_total", Unitless),
(ru_nivcsw, "involuntary context switches", "_total", Unitless)
}
#[allow(clippy::unused_async)]
pub async fn register_metrics_into(metrics_registry: &MetricsRegistry) {
let rusage = RuMetrics::new(metrics_registry);
mz_ore::task::spawn(|| "rusage_stats_update", async move {
let mut interval = tokio::time::interval(Duration::from_secs(30));
interval.set_missed_tick_behavior(tokio::time::MissedTickBehavior::Skip);
loop {
interval.tick().await;
rusage.update();
}
});
}