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
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
use std::{collections::HashMap, fmt, sync::Arc};

use crate::{
    core::Collector,
    proto::{Gauge, Metric, MetricFamily, MetricType},
};

/// A [Gauge] that returns the value from a provided function on every collect run.
///
/// This metric is the equivalant of Go's
/// <https://pkg.go.dev/github.com/prometheus/client_golang@v1.11.0/prometheus#GaugeFunc>
///
/// # Examples
/// ```
/// # use prometheus::{Registry, PullingGauge};
/// # // We are stubbing out std::thread::available_parallelism since it's not available in the
/// # // oldest Rust version that we support.
/// # fn available_parallelism() -> f64 { 0.0 }
///
/// let registry = Registry::new();
/// let gauge = PullingGauge::new(
///     "available_parallelism",
///     "The available parallelism, usually the numbers of logical cores.",
///     Box::new(|| available_parallelism())
/// ).unwrap();
/// registry.register(Box::new(gauge));
/// ```
#[derive(Clone)]
pub struct PullingGauge {
    desc: crate::core::Desc,
    value: Arc<Box<dyn Fn() -> f64 + Send + Sync>>,
}

impl fmt::Debug for PullingGauge {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        f.debug_struct("PullingGauge")
            .field("desc", &self.desc)
            .field("value", &"<opaque>")
            .finish()
    }
}

impl PullingGauge {
    /// Create a new [`PullingGauge`].
    pub fn new<S1: Into<String>, S2: Into<String>>(
        name: S1,
        help: S2,
        value: Box<dyn Fn() -> f64 + Send + Sync>,
    ) -> crate::Result<Self> {
        Ok(PullingGauge {
            value: Arc::new(value),
            desc: crate::core::Desc::new(name.into(), help.into(), Vec::new(), HashMap::new())?,
        })
    }

    fn metric(&self) -> Metric {
        let mut gauge = Gauge::default();
        let getter = &self.value;
        gauge.set_value(getter());

        let mut metric = Metric::default();
        metric.set_gauge(gauge);

        metric
    }
}

impl Collector for PullingGauge {
    fn desc(&self) -> Vec<&crate::core::Desc> {
        vec![&self.desc]
    }

    fn collect(&self) -> Vec<crate::proto::MetricFamily> {
        let mut m = MetricFamily::default();
        m.set_name(self.desc.fq_name.clone());
        m.set_help(self.desc.help.clone());
        m.set_field_type(MetricType::GAUGE);
        m.set_metric(from_vec!(vec![self.metric()]));
        vec![m]
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use crate::metrics::Collector;

    #[test]
    fn test_pulling_gauge() {
        const VALUE: f64 = 10.0;

        let gauge =
            PullingGauge::new("test_gauge", "Purely for testing", Box::new(|| VALUE)).unwrap();

        let metrics = gauge.collect();
        assert_eq!(metrics.len(), 1);

        assert_eq!(VALUE, metrics[0].get_metric()[0].get_gauge().get_value());
    }
}