1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
//! A module providing the `InFlight` metric.

use crate::atomic::AtomicInt;
use crate::clear::Clear;
use crate::metric::{Gauge, Metric};
use aspect::{Advice, Enter, OnResult};
use serde::Serialize;

/// A metric providing an in-flight gauge, showing how many calls are currently active for an expression.
///
/// This is a light-weight metric.
///
/// This makes sense mostly in a multi-threaded situation where several threads may call the same method constantly, and we want to monitor how many are active at a given time.
///
/// The [`Throughput`](struct.Throughput.html) metric shows an alternative view of the same picture, by reporting how many transactions per seconds are processed by an expression.
///
/// By default, `InFlight` uses a lock-free `u64` `Gauge`, which makes sense in multithread scenarios. Non-threaded applications can gain performance by using a `std::cell:Cell<u64>` instead.

#[derive(Clone, Default, Debug, Serialize)]
pub struct InFlight<G: Gauge = AtomicInt<u64>>(G);

impl<G: Gauge, R> Metric<R> for InFlight<G> {}

impl<G: Gauge> Enter for InFlight<G> {
    type E = ();
    fn enter(&self) {
        self.0.incr();
    }
}

impl<G: Gauge, R> OnResult<R> for InFlight<G> {
    fn on_result(&self, _: (), _: &R) -> Advice {
        self.0.decr();
        Advice::Return
    }
}

impl<G: Gauge> Clear for InFlight<G> {
    fn clear(&self) {
        // Do nothing: an InFlight metric
        // would get in an inconsistent state if cleared
    }
}