LCOV - code coverage report
Current view: top level - libs/utils/src - rate_limit.rs (source / functions) Coverage Total Hit
Test: 2b0730d767f560e20b6748f57465922aa8bb805e.info Lines: 100.0 % 48 48
Test Date: 2024-09-25 14:04:07 Functions: 21.6 % 37 8

            Line data    Source code
       1              : //! A helper to rate limit operations.
       2              : 
       3              : use std::time::{Duration, Instant};
       4              : 
       5              : pub struct RateLimit {
       6              :     last: Option<Instant>,
       7              :     interval: Duration,
       8              :     dropped: u64,
       9              : }
      10              : 
      11              : pub struct RateLimitStats(u64);
      12              : 
      13              : impl std::fmt::Display for RateLimitStats {
      14            6 :     fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
      15            6 :         write!(f, "{} dropped calls", self.0)
      16            6 :     }
      17              : }
      18              : 
      19              : impl RateLimit {
      20          109 :     pub fn new(interval: Duration) -> Self {
      21          109 :         Self {
      22          109 :             last: None,
      23          109 :             interval,
      24          109 :             dropped: 0,
      25          109 :         }
      26          109 :     }
      27              : 
      28              :     /// Call `f` if the rate limit allows.
      29              :     /// Don't call it otherwise.
      30            6 :     pub fn call<F: FnOnce()>(&mut self, f: F) {
      31            6 :         self.call2(|_| f())
      32            6 :     }
      33              : 
      34           30 :     pub fn call2<F: FnOnce(RateLimitStats)>(&mut self, f: F) {
      35           30 :         let now = Instant::now();
      36           23 :         match self.last {
      37           23 :             Some(last) if now - last <= self.interval => {
      38           21 :                 // ratelimit
      39           21 :                 self.dropped += 1;
      40           21 :             }
      41            9 :             _ => {
      42            9 :                 self.last = Some(now);
      43            9 :                 f(RateLimitStats(self.dropped));
      44            9 :                 self.dropped = 0;
      45            9 :             }
      46              :         }
      47           30 :     }
      48              : }
      49              : 
      50              : #[cfg(test)]
      51              : mod tests {
      52              :     use std::sync::atomic::AtomicUsize;
      53              : 
      54              :     #[test]
      55            1 :     fn basics() {
      56              :         use super::RateLimit;
      57              :         use std::sync::atomic::Ordering::Relaxed;
      58              :         use std::time::Duration;
      59              : 
      60            1 :         let called = AtomicUsize::new(0);
      61            1 :         let mut f = RateLimit::new(Duration::from_millis(100));
      62            1 : 
      63            3 :         let cl = || {
      64            3 :             called.fetch_add(1, Relaxed);
      65            3 :         };
      66              : 
      67            1 :         f.call(cl);
      68            1 :         assert_eq!(called.load(Relaxed), 1);
      69            1 :         f.call(cl);
      70            1 :         assert_eq!(called.load(Relaxed), 1);
      71            1 :         f.call(cl);
      72            1 :         assert_eq!(called.load(Relaxed), 1);
      73            1 :         std::thread::sleep(Duration::from_millis(100));
      74            1 :         f.call(cl);
      75            1 :         assert_eq!(called.load(Relaxed), 2);
      76            1 :         f.call(cl);
      77            1 :         assert_eq!(called.load(Relaxed), 2);
      78            1 :         std::thread::sleep(Duration::from_millis(100));
      79            1 :         f.call(cl);
      80            1 :         assert_eq!(called.load(Relaxed), 3);
      81            1 :     }
      82              : }
        

Generated by: LCOV version 2.1-beta