mirror of
https://github.com/waku-org/nwaku.git
synced 2025-01-16 09:55:07 +00:00
65 lines
2.0 KiB
Nim
65 lines
2.0 KiB
Nim
|
when (NimMajor, NimMinor) < (1, 4):
|
||
|
{.push raises: [Defect].}
|
||
|
else:
|
||
|
{.push raises: [].}
|
||
|
|
||
|
import chronos
|
||
|
|
||
|
## This is an extract from chronos/ratelimit.nim due to the found bug in the original implementation.
|
||
|
## Unfortunately that bug cannot be solved without harm the original features of TokenBucket class.
|
||
|
## So, this current shortcut is used to enable move ahead with nwaku rate limiter implementation.
|
||
|
type TokenBucket* = ref object
|
||
|
budget*: int ## Current number of tokens in the bucket
|
||
|
budgetCap: int ## Bucket capacity
|
||
|
lastTimeFull: Moment
|
||
|
## This timer measures the proper periodizaiton of the bucket refilling
|
||
|
fillDuration: Duration ## Refill period
|
||
|
|
||
|
## Update will take place if bucket is empty and trying to consume tokens.
|
||
|
## It checks if the bucket can be replenished as refill duration is passed or not.
|
||
|
proc update(bucket: TokenBucket, currentTime: Moment) =
|
||
|
if bucket.fillDuration == default(Duration):
|
||
|
bucket.budget = min(bucket.budgetCap, bucket.budget)
|
||
|
return
|
||
|
|
||
|
let timeDeltaFromLastFull = currentTime - bucket.lastTimeFull
|
||
|
|
||
|
if timeDeltaFromLastFull.milliseconds < bucket.fillDuration.milliseconds:
|
||
|
return
|
||
|
|
||
|
bucket.budget = bucket.budgetCap
|
||
|
bucket.lastTimeFull = currentTime
|
||
|
|
||
|
proc tryConsume*(bucket: TokenBucket, tokens: int, now = Moment.now()): bool =
|
||
|
## If `tokens` are available, consume them,
|
||
|
## Otherwhise, return false.
|
||
|
|
||
|
if bucket.budget == bucket.budgetCap:
|
||
|
bucket.lastTimeFull = now
|
||
|
|
||
|
if bucket.budget >= tokens:
|
||
|
bucket.budget -= tokens
|
||
|
return true
|
||
|
|
||
|
bucket.update(now)
|
||
|
|
||
|
if bucket.budget >= tokens:
|
||
|
bucket.budget -= tokens
|
||
|
return true
|
||
|
else:
|
||
|
return false
|
||
|
|
||
|
proc replenish*(bucket: TokenBucket, tokens: int, now = Moment.now()) =
|
||
|
## Add `tokens` to the budget (capped to the bucket capacity)
|
||
|
bucket.budget += tokens
|
||
|
bucket.update(now)
|
||
|
|
||
|
proc new*(T: type[TokenBucket], budgetCap: int, fillDuration: Duration = 1.seconds): T =
|
||
|
## Create a TokenBucket
|
||
|
T(
|
||
|
budget: budgetCap,
|
||
|
budgetCap: budgetCap,
|
||
|
fillDuration: fillDuration,
|
||
|
lastTimeFull: Moment.now(),
|
||
|
)
|