-
Notifications
You must be signed in to change notification settings - Fork 4
/
rate_limiter.cpp
89 lines (70 loc) · 2.56 KB
/
rate_limiter.cpp
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
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
#include <algorithm>
#include <chrono>
#include <stdexcept>
#include <thread>
#include "rate_limiter.hpp"
RateLimiter::RateLimiter() : interval_(0), max_permits_(0), stored_permits_(0), next_free_(0) {
}
long RateLimiter::aquire() {
return aquire(1);
}
long RateLimiter::aquire(int permits) {
if (permits <= 0) {
std::runtime_error("RateLimiter: Must request positive amount of permits");
}
auto wait_time = claim_next(permits);
std::this_thread::sleep_for(wait_time);
return wait_time.count() / 1000.0;
}
bool RateLimiter::try_aquire(int permits) {
return try_aquire(permits, 0);
}
bool RateLimiter::try_aquire(int permits, int timeout) {
using namespace std::chrono;
unsigned long long now = duration_cast<microseconds>(system_clock::now().time_since_epoch()).count();
// Check to see if the next free aquire time is within the
// specified timeout. If it's not, return false and DO NOT BLOCK,
// otherwise, calculate time needed to claim, and block
if (next_free_ > now + timeout * 1000)
return false;
else {
aquire(permits);
}
return true;
}
void RateLimiter::sync(unsigned long long now) {
// If we're passed the next_free, then recalculate
// stored permits, and update next_free_
if (now > next_free_) {
stored_permits_ = std::min(max_permits_, stored_permits_ + (now - next_free_) / interval_);
next_free_ = now;
}
}
std::chrono::microseconds RateLimiter::claim_next(double permits) {
using namespace std::chrono;
std::lock_guard<std::mutex> lock(mut_);
unsigned long long now = duration_cast<microseconds>(system_clock::now().time_since_epoch()).count();
// Make sure we're synced
sync(now);
// Since we synced before hand, this will always be >= 0.
unsigned long long wait = next_free_ - now;
// Determine how many stored and fresh permits to consume
double stored = std::min(permits, stored_permits_);
double fresh = permits - stored;
// In the general RateLimiter, stored permits have no wait time,
// and thus we only have to wait for however many fresh permits we consume
long next_free = (long)(fresh * interval_);
next_free_ += next_free;
stored_permits_ -= stored;
return microseconds(wait);
}
double RateLimiter::get_rate() const {
return 1000000.0 / interval_;
}
void RateLimiter::set_rate(double rate) {
if (rate <= 0.0) {
throw std::runtime_error("RateLimiter: Rate must be greater than 0");
}
std::lock_guard<std::mutex> lock(mut_);
interval_ = 1000000.0 / rate;
}