use prometheus::{
histogram_opts, register_histogram_vec, register_int_counter_vec, Error, HistogramVec,
IntCounterVec, Opts,
};
use std::time::Instant;
const LINERA_NAMESPACE: &str = "linera";
pub fn register_int_counter_vec(
name: &str,
description: &str,
label_names: &[&str],
) -> Result<IntCounterVec, Error> {
let counter_opts = Opts::new(name, description).namespace(LINERA_NAMESPACE);
register_int_counter_vec!(counter_opts, label_names)
}
pub fn register_histogram_vec(
name: &str,
description: &str,
label_names: &[&str],
buckets: Option<Vec<f64>>,
) -> Result<HistogramVec, Error> {
let histogram_opts = if let Some(buckets) = buckets {
histogram_opts!(name, description, buckets).namespace(LINERA_NAMESPACE)
} else {
histogram_opts!(name, description).namespace(LINERA_NAMESPACE)
};
register_histogram_vec!(histogram_opts, label_names)
}
pub struct ActiveMeasurementGuard<'metric, Metric>
where
Metric: MeasureLatency,
{
start: Instant,
metric: Option<&'metric Metric>,
}
impl<Metric> ActiveMeasurementGuard<'_, Metric>
where
Metric: MeasureLatency,
{
pub fn finish(mut self) -> f64 {
self.finish_by_ref()
}
fn finish_by_ref(&mut self) -> f64 {
match self.metric.take() {
Some(metric) => {
let latency = self.start.elapsed().as_secs_f64() * 1000.0;
metric.finish_measurement(latency);
latency
}
None => {
f64::NAN
}
}
}
}
impl<Metric> Drop for ActiveMeasurementGuard<'_, Metric>
where
Metric: MeasureLatency,
{
fn drop(&mut self) {
self.finish_by_ref();
}
}
pub trait MeasureLatency: Sized {
fn measure_latency(&self) -> ActiveMeasurementGuard<'_, Self>;
fn finish_measurement(&self, milliseconds: f64);
}
impl MeasureLatency for HistogramVec {
fn measure_latency(&self) -> ActiveMeasurementGuard<'_, Self> {
ActiveMeasurementGuard {
start: Instant::now(),
metric: Some(self),
}
}
fn finish_measurement(&self, milliseconds: f64) {
self.with_label_values(&[]).observe(milliseconds);
}
}