use crate::instruments::{
AcceptAllLabels, Instrument, InstrumentAdapter, LabelFilter, LabelPredicate, Update, Updates,
};
use crate::snapshot::Snapshot;
use crate::util;
use crate::{Descriptive, PutsSnapshot};
pub struct Counter {
name: String,
title: Option<String>,
description: Option<String>,
count: u64,
}
impl Counter {
pub fn new<T: Into<String>>(name: T) -> Counter {
Counter {
name: name.into(),
title: None,
description: None,
count: 0,
}
}
pub fn new_with_defaults<T: Into<String>>(name: T) -> Counter {
Self::new(name)
}
pub fn get_name(&self) -> &str {
&self.name
}
pub fn set_name<T: Into<String>>(&mut self, name: T) {
self.name = name.into();
}
pub fn name<T: Into<String>>(mut self, name: T) -> Self {
self.set_name(name);
self
}
pub fn set_title<T: Into<String>>(&mut self, title: T) {
self.title = Some(title.into())
}
pub fn title<T: Into<String>>(mut self, title: T) -> Self {
self.set_title(title);
self
}
pub fn set_description<T: Into<String>>(&mut self, description: T) {
self.description = Some(description.into())
}
pub fn description<T: Into<String>>(mut self, description: T) -> Self {
self.set_description(description);
self
}
pub fn inc(&mut self) {
self.count += 1;
}
pub fn inc_by(&mut self, n: u64) {
self.count += n;
}
pub fn get(&self) -> u64 {
self.count
}
pub fn accept<L: Eq + Send + 'static, F: Into<LabelFilter<L>>>(
self,
accept: F,
) -> InstrumentAdapter<L, Self> {
InstrumentAdapter::accept(accept, self)
}
pub fn for_label<L: Eq + Send + 'static>(self, label: L) -> InstrumentAdapter<L, Self> {
self.accept(label)
}
pub fn for_labels<L: Eq + Send + 'static>(self, labels: Vec<L>) -> InstrumentAdapter<L, Self> {
self.accept(labels)
}
pub fn for_all_labels<L: Eq + Send + 'static>(self) -> InstrumentAdapter<L, Self> {
self.accept(AcceptAllLabels)
}
pub fn for_labels_by_predicate<L, P>(self, label_predicate: P) -> InstrumentAdapter<L, Self>
where
L: Eq + Send + 'static,
P: Fn(&L) -> bool + Send + 'static,
{
self.accept(LabelPredicate(label_predicate))
}
pub fn adapter<L: Eq + Send + 'static>(self) -> InstrumentAdapter<L, Self> {
InstrumentAdapter::deaf(self)
}
}
impl Instrument for Counter {}
impl PutsSnapshot for Counter {
fn put_snapshot(&self, into: &mut Snapshot, descriptive: bool) {
util::put_postfixed_descriptives(self, &self.name, into, descriptive);
into.items.push((self.name.clone(), self.count.into()));
}
}
impl Updates for Counter {
fn update(&mut self, with: &Update) -> usize {
match *with {
Update::Observation(_) => {
self.inc();
1
}
Update::Observations(n, _) => {
self.inc_by(n);
1
}
Update::ObservationWithValue(_, _) => {
self.inc();
1
}
}
}
}
impl Descriptive for Counter {
fn title(&self) -> Option<&str> {
self.title.as_ref().map(|n| &**n)
}
fn description(&self) -> Option<&str> {
self.description.as_ref().map(|n| &**n)
}
}
#[cfg(test)]
mod test {
use std::time::Instant;
use super::*;
#[test]
fn updates() {
let mut counter = Counter::new("");
assert_eq!(counter.get(), 0);
counter.update(&Update::Observation(Instant::now()));
assert_eq!(counter.get(), 1);
counter.update(&Update::Observations(1, Instant::now()));
assert_eq!(counter.get(), 2);
counter.update(&Update::Observations(3, Instant::now()));
assert_eq!(counter.get(), 5);
counter.update(&Update::ObservationWithValue(33.into(), Instant::now()));
assert_eq!(counter.get(), 6);
}
}