pub mod diagnostics; mod diagnostic_plugin; pub use diagnostic_plugin::*; use std::{ collections::{HashMap, VecDeque}, time::{Duration, SystemTime}, }; use uuid::Uuid; #[derive(Debug, Copy, Clone, Hash, Eq, PartialEq)] pub struct DiagnosticId(Uuid); #[derive(Debug)] pub struct DiagnosticMeasurement { pub time: SystemTime, pub value: f64, } #[derive(Debug)] pub struct Diagnostic { pub id: DiagnosticId, pub name: String, history: VecDeque, sum: f64, max_history_length: usize, } impl Diagnostic { pub fn add_measurement(&mut self, value: f64) { let time = SystemTime::now(); if self.history.len() == self.max_history_length { if let Some(removed_diagnostic) = self.history.pop_back() { self.sum -= removed_diagnostic.value; } } self.sum += value; self.history.push_front(DiagnosticMeasurement { time, value, }); } pub fn new(id: DiagnosticId, name: &str, max_history_length: usize) -> Diagnostic { Diagnostic { id, name: name.to_string(), history: VecDeque::with_capacity(max_history_length), max_history_length, sum: 0.0, } } pub fn value(&self) -> Option { self.history.back().map(|measurement| measurement.value) } pub fn sum(&self) -> f64 { self.sum } pub fn average(&self) -> Option { if self.history.len() > 0 { Some(self.sum / self.history.len() as f64) } else { None } } pub fn history_len(&self) -> usize { self.history.len() } pub fn duration(&self) -> Option { if self.history.len() < 2 { return None } if let Some(oldest) = self.history.back() { if let Some(newest) = self.history.front() { return newest.time.duration_since(oldest.time).ok() } } return None } pub fn get_max_history_length(&self) -> usize { self.max_history_length } } #[derive(Default)] pub struct Diagnostics { diagnostics: HashMap, } impl Diagnostics { pub fn add(&mut self, diagnostic: Diagnostic) { self.diagnostics.insert(diagnostic.id, diagnostic); } pub fn get(&self, id: DiagnosticId) -> Option<&Diagnostic> { self.diagnostics.get(&id) } pub fn get_mut(&mut self, id: DiagnosticId) -> Option<&mut Diagnostic> { self.diagnostics.get_mut(&id) } pub fn get_measurement(&self, id: DiagnosticId) -> Option<&DiagnosticMeasurement> { self.diagnostics.get(&id).and_then(|diagnostic| diagnostic.history.front()) } pub fn add_measurement(&mut self, id: DiagnosticId, value: f64) { if let Some(diagnostic) = self.diagnostics.get_mut(&id) { diagnostic.add_measurement(value); } } pub fn iter(&self) -> impl Iterator { self.diagnostics.values() } }