#![allow(clippy::type_complexity)]
use crate::engine::EngineSignal;
use crate::state::{StateView, UserState};
use std::sync::{Arc, Mutex};
#[cfg(feature = "writing")]
mod csv_file;
mod failure;
mod metrics;
#[cfg(feature = "plotting")]
mod plot;
mod sampler;
mod tracing;
#[cfg(feature = "writing")]
pub use csv_file::CsvProgressWriter;
#[cfg(feature = "plotting")]
pub use plot::PlotObserver;
pub use tracing::Tracer;
pub trait Observe<S: UserState>: Send + Sync {
fn observe(&self, name: &'static str, state: StateView<'_, S>, event: &EngineSignal<S::Float>);
}
impl<S, T> Observe<S> for Arc<T>
where
S: UserState,
T: Observe<S> + ?Sized,
{
fn observe(&self, name: &'static str, state: StateView<'_, S>, event: &EngineSignal<S::Float>) {
(**self).observe(name, state, event)
}
}
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
pub enum Frequency {
Always,
Every(usize),
OnExit,
Never,
}
impl Frequency {
fn should_run<F>(&self, event: &EngineSignal<F>, iteration: usize) -> bool {
match self {
Self::Never => false,
Self::Always => true,
Self::OnExit => matches!(event, EngineSignal::Termination(_)),
Self::Every(n) => *n != 0 && iteration.is_multiple_of(*n),
}
}
}
pub struct Observers<S> {
inner: Vec<(Arc<Mutex<dyn Observe<S>>>, Frequency)>,
}
impl<S: UserState> Observers<S> {
pub fn new() -> Self {
Self { inner: Vec::new() }
}
pub fn attach(&mut self, observer: Arc<Mutex<dyn Observe<S>>>, frequency: Frequency) {
self.inner.push((observer, frequency));
}
pub fn dispatch(
&self,
ident: &'static str,
state: StateView<'_, S>,
event: &EngineSignal<S::Float>,
) {
let iter = state.iteration();
for (obs, freq) in &self.inner {
if !freq.should_run(event, iter) {
continue;
}
let obs = obs.lock().unwrap();
obs.observe(ident, state, event);
}
}
}