2022-11-02 08:03:19 +01:00
|
|
|
# Chronos Test Suite
|
|
|
|
# (c) Copyright 2022-Present
|
|
|
|
# Status Research & Development GmbH
|
|
|
|
#
|
|
|
|
# Licensed under either of
|
|
|
|
# Apache License, version 2.0, (LICENSE-APACHEv2)
|
|
|
|
# MIT license (LICENSE-MIT)
|
|
|
|
|
2023-03-31 07:35:04 +02:00
|
|
|
{.used.}
|
|
|
|
|
|
|
|
import unittest2
|
2022-11-02 08:03:19 +01:00
|
|
|
import ../chronos
|
|
|
|
import ../chronos/ratelimit
|
|
|
|
|
|
|
|
suite "Token Bucket":
|
|
|
|
test "Sync test":
|
|
|
|
var bucket = TokenBucket.new(1000, 1.milliseconds)
|
2023-06-23 10:11:14 +02:00
|
|
|
let
|
|
|
|
start = Moment.now()
|
|
|
|
fullTime = start + 1.milliseconds
|
2022-11-02 08:03:19 +01:00
|
|
|
check:
|
2023-06-23 10:11:14 +02:00
|
|
|
bucket.tryConsume(800, start) == true
|
|
|
|
bucket.tryConsume(200, start) == true
|
2022-11-02 08:03:19 +01:00
|
|
|
|
|
|
|
# Out of budget
|
2023-06-23 10:11:14 +02:00
|
|
|
bucket.tryConsume(100, start) == false
|
|
|
|
bucket.tryConsume(800, fullTime) == true
|
|
|
|
bucket.tryConsume(200, fullTime) == true
|
2022-11-02 08:03:19 +01:00
|
|
|
|
|
|
|
# Out of budget
|
2023-06-23 10:11:14 +02:00
|
|
|
bucket.tryConsume(100, fullTime) == false
|
2022-11-02 08:03:19 +01:00
|
|
|
|
|
|
|
test "Async test":
|
2023-06-23 10:11:14 +02:00
|
|
|
var bucket = TokenBucket.new(1000, 1000.milliseconds)
|
2022-11-02 08:03:19 +01:00
|
|
|
check: bucket.tryConsume(1000) == true
|
|
|
|
|
|
|
|
var toWait = newSeq[Future[void]]()
|
2023-01-18 16:02:00 +01:00
|
|
|
for _ in 0..<15:
|
|
|
|
toWait.add(bucket.consume(100))
|
2022-11-02 08:03:19 +01:00
|
|
|
|
|
|
|
let start = Moment.now()
|
|
|
|
waitFor(allFutures(toWait))
|
|
|
|
let duration = Moment.now() - start
|
|
|
|
|
2023-06-23 10:11:14 +02:00
|
|
|
check: duration in 1400.milliseconds .. 2200.milliseconds
|
2022-11-02 08:03:19 +01:00
|
|
|
|
|
|
|
test "Over budget async":
|
2023-06-23 10:11:14 +02:00
|
|
|
var bucket = TokenBucket.new(100, 100.milliseconds)
|
2022-11-02 08:03:19 +01:00
|
|
|
# Consume 10* the budget cap
|
|
|
|
let beforeStart = Moment.now()
|
2023-06-23 10:11:14 +02:00
|
|
|
waitFor(bucket.consume(1000).wait(5.seconds))
|
2023-09-15 19:38:39 +03:00
|
|
|
check Moment.now() - beforeStart in 900.milliseconds .. 2200.milliseconds
|
2022-11-02 08:03:19 +01:00
|
|
|
|
|
|
|
test "Sync manual replenish":
|
|
|
|
var bucket = TokenBucket.new(1000, 0.seconds)
|
2023-06-23 10:11:14 +02:00
|
|
|
let start = Moment.now()
|
2022-11-02 08:03:19 +01:00
|
|
|
check:
|
2023-06-23 10:11:14 +02:00
|
|
|
bucket.tryConsume(1000, start) == true
|
|
|
|
bucket.tryConsume(1000, start) == false
|
2022-11-02 08:03:19 +01:00
|
|
|
bucket.replenish(2000)
|
|
|
|
check:
|
2023-06-23 10:11:14 +02:00
|
|
|
bucket.tryConsume(1000, start) == true
|
2022-11-02 08:03:19 +01:00
|
|
|
# replenish is capped to the bucket max
|
2023-06-23 10:11:14 +02:00
|
|
|
bucket.tryConsume(1000, start) == false
|
2022-11-02 08:03:19 +01:00
|
|
|
|
|
|
|
test "Async manual replenish":
|
|
|
|
var bucket = TokenBucket.new(10 * 150, 0.seconds)
|
|
|
|
check:
|
|
|
|
bucket.tryConsume(10 * 150) == true
|
|
|
|
bucket.tryConsume(1000) == false
|
|
|
|
var toWait = newSeq[Future[void]]()
|
|
|
|
for _ in 0..<150:
|
|
|
|
toWait.add(bucket.consume(10))
|
|
|
|
|
|
|
|
let lastOne = bucket.consume(10)
|
|
|
|
|
|
|
|
# Test cap as well
|
|
|
|
bucket.replenish(1000000)
|
|
|
|
waitFor(allFutures(toWait).wait(10.milliseconds))
|
|
|
|
|
|
|
|
check: not lastOne.finished()
|
|
|
|
|
|
|
|
bucket.replenish(10)
|
|
|
|
waitFor(lastOne.wait(10.milliseconds))
|
|
|
|
|
|
|
|
test "Async cancellation":
|
|
|
|
var bucket = TokenBucket.new(100, 0.seconds)
|
|
|
|
let
|
|
|
|
fut1 = bucket.consume(20)
|
|
|
|
futBlocker = bucket.consume(1000)
|
|
|
|
fut2 = bucket.consume(50)
|
|
|
|
|
|
|
|
waitFor(fut1.wait(10.milliseconds))
|
|
|
|
waitFor(sleepAsync(10.milliseconds))
|
|
|
|
check:
|
|
|
|
futBlocker.finished == false
|
|
|
|
fut2.finished == false
|
|
|
|
|
2023-09-15 19:38:39 +03:00
|
|
|
futBlocker.cancelSoon()
|
2022-11-02 08:03:19 +01:00
|
|
|
waitFor(fut2.wait(10.milliseconds))
|
|
|
|
|
|
|
|
test "Very long replenish":
|
|
|
|
var bucket = TokenBucket.new(7000, 1.hours)
|
2023-06-23 10:11:14 +02:00
|
|
|
let start = Moment.now()
|
|
|
|
check bucket.tryConsume(7000, start)
|
|
|
|
check bucket.tryConsume(1, start) == false
|
2022-11-02 08:03:19 +01:00
|
|
|
|
|
|
|
# With this setting, it takes 514 milliseconds
|
|
|
|
# to tick one. Check that we can eventually
|
|
|
|
# consume, even if we update multiple time
|
|
|
|
# before that
|
2023-06-23 10:11:14 +02:00
|
|
|
var fakeNow = start
|
|
|
|
while fakeNow - start < 514.milliseconds:
|
|
|
|
check bucket.tryConsume(1, fakeNow) == false
|
|
|
|
fakeNow += 30.milliseconds
|
2023-01-18 16:02:00 +01:00
|
|
|
|
2023-06-23 10:11:14 +02:00
|
|
|
check bucket.tryConsume(1, fakeNow) == true
|
2022-11-02 08:03:19 +01:00
|
|
|
|
|
|
|
test "Short replenish":
|
2023-09-15 19:38:39 +03:00
|
|
|
skip()
|
|
|
|
# TODO (cheatfate): This test was disabled, because it continuosly fails in
|
|
|
|
# Github Actions Windows x64 CI when using Nim 1.6.14 version.
|
|
|
|
# Unable to reproduce failure locally.
|
|
|
|
|
|
|
|
# var bucket = TokenBucket.new(15000, 1.milliseconds)
|
|
|
|
# let start = Moment.now()
|
|
|
|
# check bucket.tryConsume(15000, start)
|
|
|
|
# check bucket.tryConsume(1, start) == false
|
2022-11-02 08:03:19 +01:00
|
|
|
|
2023-09-15 19:38:39 +03:00
|
|
|
# check bucket.tryConsume(15000, start + 1.milliseconds) == true
|