use std::time::{Duration, Instant};
use crate::instruments::{
AcceptAllLabels, Instrument, InstrumentAdapter, LabelFilter, LabelPredicate, Update, Updates,
};
use crate::snapshot::{ItemKind, Snapshot};
use crate::util;
use crate::{Descriptive, PutsSnapshot};
pub struct DataDisplay {
name: String,
title: Option<String>,
description: Option<String>,
value: Option<ItemKind>,
reset_after: Option<Duration>,
stay_on_until: Option<Instant>,
reset_value: Option<ItemKind>,
}
impl DataDisplay {
pub fn new<T: Into<String>>(name: T) -> DataDisplay {
DataDisplay {
name: name.into(),
title: None,
description: None,
value: None,
reset_after: None,
stay_on_until: None,
reset_value: None,
}
}
pub fn new_with_defaults<T: Into<String>>(name: T) -> DataDisplay {
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_reset_after(&mut self, d: Duration) {
self.reset_after = Some(d);
}
pub fn reset_after(mut self, d: Duration) -> Self {
self.set_reset_after(d);
self
}
pub fn get_reset_after(&self) -> Option<Duration> {
self.reset_after
}
pub fn set_reset_value<T: Into<ItemKind>>(&mut self, v: T) {
self.reset_value = Some(v.into());
self.value = self.reset_value.clone();
}
pub fn reset_value<T: Into<ItemKind>>(mut self, v: T) -> Self {
self.set_reset_value(v);
self
}
pub fn get_reset_value(&self) -> Option<&ItemKind> {
self.reset_value.as_ref()
}
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)
}
pub fn value(&self) -> Option<ItemKind> {
if let Some(stay_on_until) = self.stay_on_until {
if stay_on_until >= Instant::now() {
self.value.clone()
} else {
self.reset_value.clone()
}
} else {
self.value.clone()
}
}
}
impl Instrument for DataDisplay {}
impl PutsSnapshot for DataDisplay {
fn put_snapshot(&self, into: &mut Snapshot, descriptive: bool) {
util::put_postfixed_descriptives(self, &self.name, into, descriptive);
if let Some(value) = self.value() {
into.push(self.name.clone(), value);
}
}
}
impl Updates for DataDisplay {
fn update(&mut self, update: &Update) -> usize {
if let Some(value) = update.observed_value().and_then(|v| v.to_item_kind()) {
if let Some(reset_after) = self.reset_after {
self.stay_on_until = Some(Instant::now() + reset_after);
}
self.value = Some(value);
1
} else {
0
}
}
}
impl Descriptive for DataDisplay {
fn title(&self) -> Option<&str> {
self.title.as_ref().map(|n| &**n)
}
fn description(&self) -> Option<&str> {
self.description.as_ref().map(|n| &**n)
}
}