foundations 5.6.1

A Rust service foundations library.
Documentation
use crate::ratelimit::{DirectRateLimiter, Quota, StaticQuantaClock};
use crate::telemetry::settings::RateLimitingSettings;
use slog::{Drain, OwnedKVList, Record};
use std::num::NonZeroU32;

pub(crate) struct RateLimitingDrain<D> {
    inner: D,
    rate_limiter: Option<DirectRateLimiter>,
}

impl<D: Drain> RateLimitingDrain<D> {
    pub(crate) fn new(inner: D, settings: &RateLimitingSettings) -> Self {
        let rate_limiter = if settings.enabled
            && let Some(rate) = NonZeroU32::new(settings.max_events_per_second)
        {
            Some(DirectRateLimiter::direct_with_clock(
                Quota::per_second(rate),
                StaticQuantaClock::default(),
            ))
        } else {
            None
        };

        Self {
            inner,
            rate_limiter,
        }
    }
}

impl<D: Drain> Drain for RateLimitingDrain<D> {
    type Ok = ();
    type Err = D::Err;

    fn log(&self, record: &Record, values: &OwnedKVList) -> Result<Self::Ok, Self::Err> {
        if let Some(limiter) = &self.rate_limiter
            && limiter.check().is_err()
        {
            return Ok(());
        }

        self.inner.log(record, values).map(|_| ())
    }

    #[inline]
    fn is_enabled(&self, level: slog::Level) -> bool {
        Drain::is_enabled(&self.inner, level)
    }

    #[inline]
    fn flush(&self) -> Result<(), slog::FlushError> {
        Drain::flush(&self.inner)
    }
}