use inspect::Inspect;
use inspect::Value;
use std::sync::atomic::AtomicU64;
use std::sync::atomic::Ordering;
#[derive(Debug, Default, Clone)]
pub struct Counter(u64);
impl Counter {
pub fn new() -> Self {
Self::default()
}
pub fn increment(&mut self) {
self.add(1);
}
pub fn add(&mut self, n: u64) {
self.0 = self.0.wrapping_add(n);
}
pub fn get(&self) -> u64 {
self.0
}
}
impl Inspect for Counter {
fn inspect(&self, req: inspect::Request<'_>) {
req.value(Value::counter(self.0))
}
}
#[derive(Debug, Default)]
pub struct SharedCounter(AtomicU64);
impl SharedCounter {
pub fn new() -> Self {
Self::default()
}
pub fn increment(&self) {
self.add(1);
}
pub fn add(&self, n: u64) {
self.0.fetch_add(n, Ordering::Relaxed);
}
pub fn get(&self) -> u64 {
self.0.load(Ordering::Relaxed)
}
}
impl Inspect for SharedCounter {
fn inspect(&self, req: inspect::Request<'_>) {
req.value(Value::counter(self.0.load(Ordering::Relaxed)))
}
}
#[derive(Clone, Debug)]
pub struct Histogram<const N: usize>([u64; N]);
impl<const N: usize> Default for Histogram<N> {
fn default() -> Self {
Self::new()
}
}
impl<const N: usize> Histogram<N> {
pub fn new() -> Self {
assert!(N > 2);
assert!(N < BUCKETS.len());
Self([0; N])
}
pub fn add_sample(&mut self, n: impl Into<u64>) {
self.0[(64 - n.into().leading_zeros() as usize).min(N - 1)] += 1;
}
}
static BUCKETS: &[&str] = &[
"0",
"1",
"2-3",
"4-7",
"8-15",
"16-31",
"32-63",
"64-127",
"128-255",
"256-511",
"512-1023",
"1024-2047",
"2048-4195",
"4196-8191",
"8192-16383",
"16384-32767",
"32768-65535",
];
static WIDTH: &[usize] = &[1, 1, 1, 1, 1, 2, 2, 2, 3, 3, 3, 4, 4, 4, 4, 5, 5];
impl<const N: usize> Inspect for Histogram<N> {
fn inspect(&self, req: inspect::Request<'_>) {
let mut resp = req.respond();
for (i, &n) in self.0[..N - 1].iter().enumerate() {
resp.counter(BUCKETS[i], n);
}
resp.counter(&BUCKETS[N - 1][..WIDTH[N - 1] + 1], self.0[N - 1]);
}
}