An API Rate Limiter is a mechanism that enables us to control the rate of requests made to our API (Application Programming Interface). This raises the question: why would we need such a thing? If we didn’t have a Rate Limiter in place, there would be nothing to prevent users of our API from making a large number of requests in quick succession. This could flood our API with internet traffic, overwhelming our services, and potentially causing a decrease in performance or even resulting in the service going down. This can also be referred to as a DDoS (Distributed Denial of Service) attack.
A Rate Limiter can also serve the purpose of cost optimization by curbing excessive requests. By imposing limits on both incoming (requests) and outgoing (responses) calls to our services, we can effectively minimize the load on our infrastructure required to handle these API requests. This strategic reduction in resource utilization not only enhances operational efficiency but also contributes to a reduction in overall expenditures.
An example of how other companies use Rate Limiters:
Now we know what Rate Limiters are good for let’s have a dive into how they are implemented.
There are a couple of ways we can implement Rate Limiters with each one having its own distinct pros and cons. Below is a list of common Rate Limiting algorithms
The token bucket is a widely used algorithms with companies such as Amazon and Stripe adopting this algorithm. The algorithm works as follows:
We specify a bucket which contains a set number of tokens (capacity). Tokens are then refilled periodically with any excess tokens which exceed the max capacity overflowing and not being added to the bucket.
Every time a request flows through our API we check to see if there are enough tokens in the bucket. If there are enough tokens in the bucket then we process the request.
If the bucket does not have enough tokens in it then we reject the request and return an error.
Usually a token bucket requires 2 parameters.
For example, we might have a bucket with a size of 5 and a refill rate of 2 tokens per second. Three requests hit our API at 10:00:00 AM whereby we deduct 3 tokens from our bucket leaving a total of 2 tokens in the bucket. We then proceed to process the requests.
At 10:00:01 we add 2 extra tokens to the bucket.
Just after adding the 2 extra tokens to the bucket, we get an influx of requests within the same second.
Pros: The token bucket is a fairly simple algorithm to implement and allows for short bursts of traffic.
Cons: It can be hard to fine tune this algorithm as we only have the bucket size and refill rate to go off.
The Leaking Bucket algorithm is very similar to that of the Token bucket except we swap out the bucket for a FIFO (First in First out) queue. We then consume the requests from the queue at a fixed rate.
If requests are added to the queue faster than the rate which we can process the requests from the queue then we drop the request and return an error.
The Leaking bucket algorithm usually takes two parameters.
The queue size is as expressed above where we can only fit 9 items in the queue. Any more than 9 items in the queue and we discard the request and return an error. The outflow rate is usually how many request we can process at a specified rate usually seconds.
Pros: Requests can be processed at a fixed rate which is ideal for throttling inbound requests and a stable outflow is needed.
Cons: A burst in traffic if not catered for by the process rate will cause requests to be dropped therefore making this algorithm hard to tune.
This approach takes a different direction to the two previous algorithms which had a single bucket or queue. In the case of the Fixed Window Counter algorithm we divide the time into windows and allocate a counter to each window. For example, we might want a window to be 1 second and have a max limit of 10 requests per second.
There is a major problem which comes along with using this algorithm. For example, if we allowed a max of 10 requests per 1-minute window, there is a potential that a burst in traffic at the edges of the window will actually cause more requests to be allowed through than the expected max requests per 1-minute window size.
Pros: Fairly simple algorithm to understand and implement.
Cons: A spike in traffic at the edges of the window can cause more requests through than our intended capacity, so this will have to be taken into account.
The Fixed Window Counter algorithm has a major flaw whereby it can let through more requests as intended if those requests are on the edges of the window. The Sliding Window Log algorithm helps to address this issue.
As requests flow through our API we allocate a timestamp to the request and keep track of it along with the other request timestamps. For example if we allow a maximum of 5 requests per 60 seconds then this might look like the below:
A new request flows in and there are no other timestamps that we are keeping track of, so we create a timestamp for the new request and keep track of it.
A few more requests flow in. We compare the timestamp of the current requests to the timestamp of the first request. We can see that they all lie within a time period of 60 seconds, so we process them.
Another request then flows in which is within the intended time period but exceeds the number of allowed requests, so we reject it.
A new request flows through our API, but we can see that this request exceeds the 60-second timeframe when compared to the first, second and third timestamp. We drop the first 3 timestamps therefore leaving us with a total of 4 timestamps including the current request which therefore leaves us with less than 5 requests within the valid timeframe. As a result we can process this request.
Pros: Very accurate as the window moves with the timestamps therefore fixing the issue that Fixed Window Counter algorithm had.
Cons: Can consume more memory as we still need to keep track of the timestamps for rejected requests.
The Sliding Window Counter algorithm seamlessly blends the strengths of both the Fixed Window Counter and Sliding Window Counter approaches. We can take on this algorithm from various angles, and one approach elegantly employs the formula outlined below:
C = Requests in current window
P = Requests in the previous window
O = Overlap percentage of the rolling window and previous window
C + P * O
An example of how this might work is lets say we allow 10 requests per minute.
Based on the above diagram and formula we can calculate the current rate limit:
7 + 4 * 0.36 = 8.44
We can then decide to either round down to 8 or round up to 9. Either way we are within our rate limit threshold of 10 requests. For the sake of this example lets assume that we are rounding down.
If we decide to add a few more requests to the previous window and then recalculate the rate limit.
7 + 8 * 0.53 = 11.24
You can see that we now have 11 requests which has exceeded our rate limit, so we reject the request.
Pros: Good for smoothing out spikes in traffic as the calculation is based on the average rate of the previous window.
Cons: Is not accurate and is just an approximation because it assumes requests in the previous window are evenly distributed. Based on the cloudflare article: How we built rate limiting capable of scaling to millions of domains the Sliding Window Counter algorithm is still very accurate, as an analysis on 400 million requests from 270,000 distinct sources show:
There you have it folks API rate limiting algorithms play a pivotal role in maintaining the equilibrium between efficient service delivery and safeguarding system integrity. By intelligently controlling the pace at which requests are processed, these algorithms prevent undue strain on resources, optimize operational costs, and ensure a fair and reliable user experience. Whether employing straightforward token buckets, leaky buckets, or more advanced techniques, the careful implementation of rate limiting strategies empowers organizations to strike an optimal balance between accommodating user demands and preserving the stability and responsiveness of their API-driven services. As digital interactions continue to proliferate, the judicious use of API rate limiting algorithms emerges as an essential tool in sustaining the resilience and responsiveness of modern interconnected systems.
System Design Interview – An insider’s guide
4 Rate Limit Algorithms Every Developer Should Know
System Design — Rate limiter and Data modelling
How we built rate limiting capable of scaling to millions of domains