x/time/rate: Rate limit can be exceeded 5000x under high lock contention due to bug in advance
#71360
Labels
BugReport
Issues describing a possible bug in the Go implementation.
Milestone
Go version
go version go1.23.2 X:nocoverageredesign linux/amd64
Output of
go env
in your module/workspace:What did you do?
When a call to AllowN(t, n) succeeds, it always sets lim.last to t, even if t is before lim.last. This allows the rate limit to be exceeded: https://go.dev/play/p/uR8KvkVJGN8
Clearly this is a bug in line 389, where t should be updated to lim.last.
In practice, since calls to
Allow
calltime.Now()
before acquiring the limiter's lock, high lock contention increases the likelihood ofAllowN
being called with a time in the past, thus resettinglim.last
to a past time and allowing more tokens to be acquired.What did you see happen?
When calling
Allow
with many concurrent goroutines, the rate limit can easily be exceeded by a factor 5000.For instance, 10k goroutines accessing
Allow
with a rate limit of 200 for 10 seconds produces something like:Ok: 1445521
Block: 15612127
(in my test, goroutines sleep for ~25ms after
Allow
succeeds, and exponentially back off with 5/15/45/135/...ms on failures)What did you expect to see?
After fixing the bug, I observed this:
Ok: 2992
Block: 16271193
I expected OK to be ~2000 (rate limit of 200 for 10 seconds), but this is much better then with the bug.
The text was updated successfully, but these errors were encountered: