1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
use super::tracer::{Tracer, TracerEvent};
use derive_more::{Deref, DerefMut};
use rill_protocol::provider::{Description, Path, RillData, RillEvent, StreamType, Timestamp};
use std::time::SystemTime;

#[derive(Debug)]
pub enum CounterDelta {
    Increment(f64),
}

#[derive(Debug, Default)]
pub struct CounterState {
    counter: f64,
    last_event: Option<RillEvent>,
}

impl TracerEvent for CounterDelta {
    type State = CounterState;

    fn aggregate(self, state: &mut Self::State, timestamp: Timestamp) -> Option<&RillEvent> {
        match self {
            Self::Increment(delta) => {
                state.counter += delta;
                let data = RillData::CounterRecord {
                    value: state.counter,
                };
                let last_event = RillEvent { timestamp, data };
                state.last_event = Some(last_event);
                state.last_event.as_ref()
            }
        }
    }

    fn make_snapshot(state: &Self::State) -> Vec<RillEvent> {
        state.last_event.clone().into_iter().collect()
    }
}

/// Tracers `Counter` metrics that can increments only.
#[derive(Debug, Deref, DerefMut)]
pub struct CounterTracer {
    #[deref]
    #[deref_mut]
    tracer: Tracer<CounterDelta>,
}

impl CounterTracer {
    /// Creates a new tracer instance.
    pub fn new(path: Path) -> Self {
        let info = format!("{} counter", path);
        let description = Description {
            path,
            info,
            stream_type: StreamType::CounterStream,
        };
        let tracer = Tracer::new(description);
        Self { tracer }
    }

    /// Increments value by the sepcific delta.
    pub fn inc(&self, delta: f64, timestamp: Option<SystemTime>) {
        let data = CounterDelta::Increment(delta);
        self.tracer.send(data, timestamp);
    }
}