use std::time::Instant;
use crate::instruments::{
AcceptAllLabels, Instrument, InstrumentAdapter, LabelFilter, LabelPredicate, Update, Updates,
};
use crate::snapshot::Snapshot;
use crate::util;
use crate::{Descriptive, PutsSnapshot};
pub struct LastOccurrenceTracker {
name: String,
title: Option<String>,
description: Option<String>,
happened_last: Option<Instant>,
invert: bool,
make_none_zero: bool,
}
impl LastOccurrenceTracker {
pub fn new<T: Into<String>>(name: T) -> LastOccurrenceTracker {
LastOccurrenceTracker {
name: name.into(),
title: None,
description: None,
happened_last: None,
invert: false,
make_none_zero: false,
}
}
pub fn new_with_defaults<T: Into<String>>(name: T) -> LastOccurrenceTracker {
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 set_invert_enabled(&mut self, invert: bool) {
self.invert = invert
}
pub fn invert_enabled(mut self, invert: bool) -> Self {
self.set_invert_enabled(invert);
self
}
pub fn inverted(mut self) -> Self {
self.set_invert_enabled(true);
self
}
pub fn is_inverted(&self) -> bool {
self.invert
}
pub fn set_make_none_zero(&mut self, make_zero: bool) {
self.make_none_zero = make_zero
}
pub fn make_none_zero(mut self, make_zero: bool) -> Self {
self.set_make_none_zero(make_zero);
self
}
pub fn get_make_none_zero(&self) -> bool {
self.make_none_zero
}
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)
}
fn elapsed_since_last_occurrence(&self) -> Option<u64> {
self.happened_last
.map(|last| (Instant::now() - last).as_secs())
}
}
impl Instrument for LastOccurrenceTracker {}
impl PutsSnapshot for LastOccurrenceTracker {
fn put_snapshot(&self, into: &mut Snapshot, descriptive: bool) {
util::put_postfixed_descriptives(self, &self.name, into, descriptive);
if let Some(v) = self.elapsed_since_last_occurrence() {
into.push(self.name.clone(), v);
} else if self.get_make_none_zero() {
into.push(self.name.clone(), 0);
}
}
}
impl Updates for LastOccurrenceTracker {
fn update(&mut self, _: &Update) -> usize {
self.happened_last = Some(Instant::now());
1
}
}
impl Descriptive for LastOccurrenceTracker {
fn title(&self) -> Option<&str> {
self.title.as_ref().map(|n| &**n)
}
fn description(&self) -> Option<&str> {
self.description.as_ref().map(|n| &**n)
}
}