1use std::sync::Arc;
23use crate::IntoF64;
45/// A counter handler.
6pub trait CounterFn {
7/// Increments the counter by the given amount.
8fn increment(&self, value: u64);
910/// Sets the counter to at least the given amount.
11 ///
12 /// This is intended to support use cases where multiple callers are attempting to synchronize
13 /// this counter with an external counter that they have no control over. As multiple callers
14 /// may read that external counter, and attempt to set it here, there could be reordering issues
15 /// where a caller attempts to set an older (smaller) value after the counter has been updated to
16 /// the latest (larger) value.
17 ///
18 /// This method must cope with those cases. An example of doing so atomically can be found in
19 /// `AtomicCounter`.
20fn absolute(&self, value: u64);
21}
2223/// A gauge handler.
24pub trait GaugeFn {
25/// Increments the gauge by the given amount.
26fn increment(&self, value: f64);
2728/// Decrements the gauge by the given amount.
29fn decrement(&self, value: f64);
3031/// Sets the gauge to the given amount.
32fn set(&self, value: f64);
33}
3435/// A histogram handler.
36pub trait HistogramFn {
37/// Records a value into the histogram.
38fn record(&self, value: f64);
39}
4041/// A counter.
42#[derive(Clone)]
43#[must_use = "counters do nothing unless you use them"]
44pub struct Counter {
45 inner: Option<Arc<dyn CounterFn + Send + Sync>>,
46}
4748/// A gauge.
49#[derive(Clone)]
50#[must_use = "gauges do nothing unless you use them"]
51pub struct Gauge {
52 inner: Option<Arc<dyn GaugeFn + Send + Sync>>,
53}
5455/// A histogram.
56#[derive(Clone)]
57#[must_use = "histograms do nothing unless you use them"]
58pub struct Histogram {
59 inner: Option<Arc<dyn HistogramFn + Send + Sync>>,
60}
6162impl Counter {
63/// Creates a no-op `Counter` which does nothing.
64 ///
65 /// Suitable when a handle must be provided that does nothing i.e. a no-op recorder or a layer
66 /// that disables specific metrics, and so on.
67pub fn noop() -> Self {
68Self { inner: None }
69 }
7071/// Creates a `Counter` based on a shared handler.
72pub fn from_arc<F: CounterFn + Send + Sync + 'static>(a: Arc<F>) -> Self {
73Self { inner: Some(a) }
74 }
7576/// Increments the counter.
77pub fn increment(&self, value: u64) {
78if let Some(c) = &self.inner {
79 c.increment(value)
80 }
81 }
8283/// Sets the counter to an absolute value.
84pub fn absolute(&self, value: u64) {
85if let Some(c) = &self.inner {
86 c.absolute(value)
87 }
88 }
89}
9091impl Gauge {
92/// Creates a no-op `Gauge` which does nothing.
93 ///
94 /// Suitable when a handle must be provided that does nothing i.e. a no-op recorder or a layer
95 /// that disables specific metrics, and so on.
96pub fn noop() -> Self {
97Self { inner: None }
98 }
99100/// Creates a `Gauge` based on a shared handler.
101pub fn from_arc<F: GaugeFn + Send + Sync + 'static>(a: Arc<F>) -> Self {
102Self { inner: Some(a) }
103 }
104105/// Increments the gauge.
106pub fn increment<T: IntoF64>(&self, value: T) {
107if let Some(g) = &self.inner {
108 g.increment(value.into_f64())
109 }
110 }
111112/// Decrements the gauge.
113pub fn decrement<T: IntoF64>(&self, value: T) {
114if let Some(g) = &self.inner {
115 g.decrement(value.into_f64())
116 }
117 }
118119/// Sets the gauge.
120pub fn set<T: IntoF64>(&self, value: T) {
121if let Some(g) = &self.inner {
122 g.set(value.into_f64())
123 }
124 }
125}
126127impl Histogram {
128/// Creates a no-op `Histogram` which does nothing.
129 ///
130 /// Suitable when a handle must be provided that does nothing i.e. a no-op recorder or a layer
131 /// that disables specific metrics, and so on.
132pub fn noop() -> Self {
133Self { inner: None }
134 }
135136/// Creates a `Histogram` based on a shared handler.
137pub fn from_arc<F: HistogramFn + Send + Sync + 'static>(a: Arc<F>) -> Self {
138Self { inner: Some(a) }
139 }
140141/// Records a value in the histogram.
142pub fn record<T: IntoF64>(&self, value: T) {
143if let Some(ref inner) = self.inner {
144 inner.record(value.into_f64())
145 }
146 }
147}
148149impl<T> CounterFn for Arc<T>
150where
151T: CounterFn,
152{
153fn increment(&self, value: u64) {
154 (**self).increment(value)
155 }
156157fn absolute(&self, value: u64) {
158 (**self).absolute(value)
159 }
160}
161impl<T> GaugeFn for Arc<T>
162where
163T: GaugeFn,
164{
165fn increment(&self, value: f64) {
166 (**self).increment(value)
167 }
168169fn decrement(&self, value: f64) {
170 (**self).decrement(value)
171 }
172173fn set(&self, value: f64) {
174 (**self).set(value)
175 }
176}
177178impl<T> HistogramFn for Arc<T>
179where
180T: HistogramFn,
181{
182fn record(&self, value: f64) {
183 (**self).record(value);
184 }
185}
186187impl<T> From<Arc<T>> for Counter
188where
189T: CounterFn + Send + Sync + 'static,
190{
191fn from(inner: Arc<T>) -> Self {
192 Counter::from_arc(inner)
193 }
194}
195196impl<T> From<Arc<T>> for Gauge
197where
198T: GaugeFn + Send + Sync + 'static,
199{
200fn from(inner: Arc<T>) -> Self {
201 Gauge::from_arc(inner)
202 }
203}
204205impl<T> From<Arc<T>> for Histogram
206where
207T: HistogramFn + Send + Sync + 'static,
208{
209fn from(inner: Arc<T>) -> Self {
210 Histogram::from_arc(inner)
211 }
212}