hyperlight_host/metrics/
histogram_vec.rsuse prometheus::{register_histogram_vec_with_registry, HistogramVec as PHistogramVec};
use tracing::{instrument, Span};
use super::{
get_histogram_opts, get_metrics_registry, GetHyperlightMetric, HyperlightMetric,
HyperlightMetricOps,
};
use crate::{new_error, HyperlightError, Result};
#[derive(Debug)]
pub struct HistogramVec {
histogram: PHistogramVec,
pub name: &'static str,
}
impl HistogramVec {
#[instrument(err(Debug), skip_all, parent = Span::current(), level= "Trace")]
pub fn new(name: &'static str, help: &str, labels: &[&str], buckets: Vec<f64>) -> Result<Self> {
let registry = get_metrics_registry();
let opts = get_histogram_opts(name, help, buckets);
let histogram = register_histogram_vec_with_registry!(opts, labels, registry)?;
Ok(Self { histogram, name })
}
#[instrument(err(Debug), skip_all, parent = Span::current(), level= "Trace")]
pub fn observe(&self, label_vals: &[&str], val: f64) -> Result<()> {
self.histogram
.get_metric_with_label_values(label_vals)?
.observe(val);
Ok(())
}
#[instrument(err(Debug), skip_all, parent = Span::current(), level= "Trace")]
pub fn get_sample_sum(&self, label_vals: &[&str]) -> Result<f64> {
Ok(self
.histogram
.get_metric_with_label_values(label_vals)?
.get_sample_sum())
}
#[instrument(err(Debug), skip_all, parent = Span::current(), level= "Trace")]
pub fn get_sample_count(&self, label_vals: &[&str]) -> Result<u64> {
Ok(self
.histogram
.get_metric_with_label_values(label_vals)?
.get_sample_count())
}
}
impl<S: HyperlightMetricOps> GetHyperlightMetric<HistogramVec> for S {
#[instrument(err(Debug), skip_all, parent = Span::current(), level= "Trace")]
fn metric(&self) -> Result<&HistogramVec> {
let metric = self.get_metric()?;
<&HyperlightMetric as TryInto<&HistogramVec>>::try_into(metric)
}
}
impl<'a> TryFrom<&'a HyperlightMetric> for &'a HistogramVec {
type Error = HyperlightError;
#[instrument(err(Debug), skip_all, parent = Span::current(), level= "Trace")]
fn try_from(metric: &'a HyperlightMetric) -> Result<Self> {
match metric {
HyperlightMetric::HistogramVec(histogram) => Ok(histogram),
_ => Err(new_error!("metric is not a HistogramVec")),
}
}
}
impl From<HistogramVec> for HyperlightMetric {
#[instrument(skip_all, parent = Span::current(), level= "Trace")]
fn from(histogram_vec: HistogramVec) -> Self {
HyperlightMetric::HistogramVec(histogram_vec)
}
}
#[macro_export]
macro_rules! histogram_vec_observe {
($metric:expr, $label_vals:expr, $val:expr) => {{
match $crate::metrics::GetHyperlightMetric::<$crate::metrics::HistogramVec>::metric($metric)
{
Ok(val) => {
if let Err(e) = val.observe($label_vals, $val) {
log::error!(
"error calling observe with {} on metric with labels: {} {:?}",
$val,
e,
$label_vals
)
}
}
Err(e) => log::error!("error getting metric: {}", e),
};
}};
}
#[macro_export]
macro_rules! histogram_vec_sample_sum {
($metric:expr, $label_vals:expr) => {{
match $crate::metrics::GetHyperlightMetric::<$crate::metrics::HistogramVec>::metric($metric)
{
Ok(val) => match val.get_sample_sum($label_vals) {
Ok(val) => val,
Err(e) => {
log::error!(
"error getting samples sum of metric with labels: {} {:?}",
e,
$label_vals
);
0.0
}
},
Err(e) => {
log::error!("error getting metric: {}", e);
0.0
}
}
}};
}
#[macro_export]
macro_rules! histogram_vec_sample_count {
($metric:expr, $label_vals:expr) => {{
match $crate::metrics::GetHyperlightMetric::<$crate::metrics::HistogramVec>::metric($metric)
{
Ok(val) => match val.get_sample_count($label_vals) {
Ok(val) => val,
Err(e) => {
log::error!(
"error getting samples count of metric with labels: {} {:?}",
e,
$label_vals
);
0
}
},
Err(e) => {
log::error!("error getting metric: {}", e);
0
}
}
}};
}
#[macro_export]
macro_rules! histogram_vec_time_micros {
($metric:expr, $label_vals:expr, $expr:expr) => {{
let start = std::time::Instant::now();
let result = $expr;
$crate::histogram_vec_observe!($metric, $label_vals, start.elapsed().as_micros() as f64);
result
}};
}