Skip to content

Commit

Permalink
Add a maximum over interval gauge
Browse files Browse the repository at this point in the history
Signed-off-by: Kushagra Udai <[email protected]>
  • Loading branch information
kushudai committed Oct 16, 2023
1 parent 6e81890 commit 3c5350d
Show file tree
Hide file tree
Showing 4 changed files with 263 additions and 14 deletions.
1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@ procfs = { version = "^0.14", optional = true, default-features = false }
criterion = "0.4"
getopts = "^0.2"
hyper = { version = "^0.14", features = ["server", "http1", "tcp"] }
mock_instant = { version = "0.3", features = ["sync"] }
tokio = { version = "^1.0", features = ["macros", "rt-multi-thread"] }

[build-dependencies]
Expand Down
39 changes: 25 additions & 14 deletions src/atomic64.rs
Original file line number Diff line number Diff line change
Expand Up @@ -107,24 +107,12 @@ impl Atomic for AtomicF64 {

#[inline]
fn inc_by(&self, delta: Self::T) {
loop {
let current = self.inner.load(Ordering::Acquire);
let new = u64_to_f64(current) + delta;
let result = self.inner.compare_exchange_weak(
current,
f64_to_u64(new),
Ordering::Release,
Ordering::Relaxed,
);
if result.is_ok() {
return;
}
}
self.fetch_add(delta);
}

#[inline]
fn dec_by(&self, delta: Self::T) {
self.inc_by(-delta);
self.fetch_add(-delta);
}
}

Expand All @@ -133,6 +121,29 @@ impl AtomicF64 {
pub fn swap(&self, val: f64, ordering: Ordering) -> f64 {
u64_to_f64(self.inner.swap(f64_to_u64(val), ordering))
}

/// Fetches the maximum of the current value and the provided value.
pub fn fetch_max(&self, val: f64, ordering: Ordering) -> f64 {
u64_to_f64(self.inner.fetch_max(f64_to_u64(val) ,ordering))
}

/// Fetches the old value after summing the provided value.
pub fn fetch_add(&self, val: f64) -> f64 {
let mut current: u64;
loop {
current = self.inner.load(Ordering::Acquire);
let new = u64_to_f64(current) + val;
let result = self.inner.compare_exchange_weak(
current,
f64_to_u64(new),
Ordering::Release,
Ordering::Relaxed,
);
if result.is_ok() {
return u64_to_f64(current);
}
}
}
}

/// A atomic signed integer.
Expand Down
1 change: 1 addition & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -157,6 +157,7 @@ mod encoder;
mod errors;
mod gauge;
mod histogram;
mod maximum_over_interval_gauge;
mod metrics;
mod pulling_gauge;
#[cfg(feature = "push")]
Expand Down
236 changes: 236 additions & 0 deletions src/maximum_over_interval_gauge.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,236 @@
use parking_lot::{RwLock, RwLockUpgradableReadGuard};
use std::{
sync::{
atomic::Ordering,
Arc,
},
time::Duration,
};

#[cfg(test)]
use mock_instant::Instant;

#[cfg(not(test))]
use std::time::Instant;

use crate::{core::{Collector, AtomicF64, Atomic}, Error, PullingGauge};

/// A prometheus gauge that exposes the maximum value of a gauge over an interval.
///
/// Used to expose instantaneous values that tend to move a lot within a small interval.
///
/// # Examples
/// ```
/// # use std::time::Duration;
/// # use prometheus::{Registry, MaximumOverIntervalGauge};
///
/// let registry = Registry::new();
/// let gauge = MaximumOverIntervalGauge::new(
/// "maximum_queue_size_30s",
/// "The high watermark queue size in the last 30 seconds.",
/// Duration::from_secs(30)
/// ).unwrap();
/// registry.register(Box::new(gauge.clone()));
///
/// gauge.inc_by(30);
/// gauge.dec_by(10);
///
/// // For the next 30 seconds, the metric will be 30 as that was the maximum value.
/// // Afterwards, it will drop to 10.
/// ```
#[derive(Clone, Debug)]
pub struct MaximumOverIntervalGauge {
// The current real-time value.
value: Arc<AtomicF64>,
// The maximum value in the current interval.
maximum_value: Arc<AtomicF64>,

// The length of a given interval.
interval_duration: Duration,
// The time at which the current interval will expose.
interval_expiry: Arc<RwLock<Instant>>,

gauge: PullingGauge,
}

impl MaximumOverIntervalGauge {
/// Create a new [`MaximumOverIntervalGauge`].
pub fn new<S1: Into<String>, S2: Into<String>>(
name: S1,
help: S2,
interval: Duration,
) -> Result<Self, Error> {
let maximum_value = Arc::new(AtomicF64::new(0.0));

Ok(Self {
value: Arc::new(AtomicF64::new(0.0)),
maximum_value: maximum_value.clone(),

interval_expiry: Arc::new(RwLock::new(Instant::now() + interval)),
interval_duration: interval,
gauge: PullingGauge::new(
name,
help,
Box::new(move || maximum_value.get()),
)?,
})
}

/// Increments the gauge by 1.
pub fn inc(&self) {
self.apply_delta(1.0);
}

/// Decrements the gauge by 1.
pub fn dec(&self) {
self.apply_delta(-1.0);
}

/// Add the given value to the gauge.
///
/// (The value can be negative, resulting in a decrement of the gauge.)
pub fn inc_by(&self, v: f64) {
self.apply_delta(v);
}

/// Subtract the given value from the gauge.
///
/// (The value can be negative, resulting in an increment of the gauge.)
pub fn dec_by(&self, v: f64) {
self.apply_delta(-v);
}

pub fn observe(&self, v: f64) {
let previous_value = self.value.swap(v, Ordering::AcqRel);
if self.maximum_value.get() < previous_value {
self.maximum_value.set(previous_value);
}
}

fn apply_delta(&self, delta: f64) {
let previous_value = self.value.fetch_add(delta);
let new_value = previous_value + delta;

let now = Instant::now();
let interval_expiry = self.interval_expiry.upgradable_read();
let loaded_interval_expiry = *interval_expiry;

// Check whether we've crossed into the new interval.
if loaded_interval_expiry < now {
// There's a possible optimization here of using try_upgrade in a loop. Need to write
// benchmarks to verify.
let mut interval_expiry = RwLockUpgradableReadGuard::upgrade(interval_expiry);

// Did we get to be the thread that actually started the new interval? Other threads
// could have updated the value before we got the exclusive lock.
if *interval_expiry == loaded_interval_expiry {
*interval_expiry = now + self.interval_duration;
self.maximum_value.set(new_value);

return;
}
}

// Set the maximum_value to the max of the current value & previous max.
self.maximum_value.fetch_max(new_value, Ordering::Relaxed);
}
}

impl Collector for MaximumOverIntervalGauge {
fn desc(&self) -> Vec<&crate::core::Desc> {
self.gauge.desc()
}

fn collect(&self) -> Vec<crate::proto::MetricFamily> {
// Apply a delta of '0' to ensure that the reset-value-if-interval-expired-logic kicks in.
self.apply_delta(0.0);

self.gauge.collect()
}
}

#[cfg(test)]
mod test {
use mock_instant::MockClock;

use super::*;

static INTERVAL: Duration = Duration::from_secs(30);

#[test]
fn test_correct_behaviour() {
let gauge = MaximumOverIntervalGauge::new(
"test_counter".to_string(),
"This won't help you".to_string(),
INTERVAL,
)
.unwrap();

assert_metric_value(&gauge, 0.0);

gauge.inc_by(5.0);

assert_metric_value(&gauge, 5.0);

gauge.dec();

// The value should still be five after we decreased it as the max within the interval was 5.
assert_metric_value(&gauge, 5.0);

MockClock::advance(INTERVAL + Duration::from_secs(1));

// The value should be 4 now as the next interval has started.
assert_metric_value(&gauge, 4.0);

gauge.observe(3.0);

// The value should still be five after we decreased it as the max within the interval was 5.
assert_metric_value(&gauge, 4.0);

gauge.observe(6.0);

// The value should be six after we inreased it as the max within the interval was 6.
assert_metric_value(&gauge, 6.0);

gauge.observe(2.0);

MockClock::advance(INTERVAL + Duration::from_secs(1));

// The value should be 2 now as the next interval has started.
assert_metric_value(&gauge, 2.0);
}

#[test]
fn test_cloning() {
let gauge = MaximumOverIntervalGauge::new(
"test_counter".to_string(),
"This won't help you".to_string(),
INTERVAL,
)
.unwrap();

let same_gauge = gauge.clone();

assert_metric_value(&gauge, 0.0);

gauge.inc_by(5.0);

// Read from the cloned gauge to veriy that they share data.
assert_metric_value(&same_gauge, 5.0);
}

fn assert_metric_value(gauge: &MaximumOverIntervalGauge, val: f64) {
let result = gauge.collect();

let metric_family = result
.first()
.expect("expected one MetricFamily to be returned");

let metric = metric_family
.get_metric()
.first()
.expect("expected one Metric to be returned");

assert_eq!(val, metric.get_gauge().get_value());
}
}

0 comments on commit 3c5350d

Please sign in to comment.