looking at rate-limiting implementations

This commit is contained in:
V 2026-05-09 13:15:44 +01:00
commit 29656c8d15
10 changed files with 267 additions and 0 deletions

5
rate_limiters/.gitignore vendored Normal file
View File

@ -0,0 +1,5 @@
# Python-generated files
__pycache__/
.venv
.ruff_cache
.pytest_cache

View File

@ -0,0 +1 @@
3.11

0
rate_limiters/README.md Normal file
View File

View File

@ -0,0 +1,44 @@
import threading
from datetime import datetime, timedelta
from time import sleep
class FixedWindowLimiter:
def __init__(self, window_size: int, window_duration: int):
self.window_size = window_size
self.window_duration = window_duration
self.count = 0
self.last_reset_time = datetime.now()
self._lock = threading.Lock()
def __enter__(self):
with self._lock:
if datetime.now() - self.last_reset_time > timedelta(
seconds=self.window_duration
):
self.count = 0
self.last_reset_time = datetime.now()
if self.count < self.window_size:
self.count += 1
print("ALLOWED")
return self
else:
raise RuntimeError("Rate exceeded!")
def __exit__(self, exc_type, exc_value, exc_traceback):
return False
def make_requests(number: int, limiter: FixedWindowLimiter) -> None:
for i in range(1, number):
try:
with limiter as lim:
print(f"PRE-BUDGET: {lim.window_size - lim.count}")
print("REQUEST MADE!")
except RuntimeError:
print("Request limit reached!")
sleep(0.5)
# lim_1 = FixedWindowLimiter(window_duration=10, window_size=10)
# make_requests(100, lim_1)

6
rate_limiters/main.py Normal file
View File

@ -0,0 +1,6 @@
def main():
print("Hello from rate-limiters!")
if __name__ == "__main__":
main()

View File

@ -0,0 +1,9 @@
[project]
name = "rate-limiters"
version = "0.1.0"
description = "Add your description here"
readme = "README.md"
requires-python = ">=3.11"
dependencies = [
"pytest>=9.0.3",
]

View File

@ -0,0 +1,20 @@
import pytest
from fixed_window import FixedWindowLimiter
def test_limit_respected():
limiter = FixedWindowLimiter(10, 10)
try:
for i in range(9):
with limiter:
pass
except RuntimeError as e:
assert False, f"Limiter raised an exception -> {e}"
def test_limit_enforced():
limiter = FixedWindowLimiter(10, 10)
with pytest.raises(RuntimeError) as e:
for i in range(11):
with limiter:
pass
assert "Rate exceeded" in str(e.value)

View File

@ -0,0 +1,25 @@
import pytest
from time import sleep
from token_bucket import TokenBucketLimiter, RateLimitExceeded
def test_limit_respected() -> None:
limiter = TokenBucketLimiter(10, 10)
try:
for i in range(30):
with limiter:
sleep(0.1)
except RateLimitExceeded:
assert False, "Rate limit exceeded!"
def test_limit_exceeded() -> None:
limiter = TokenBucketLimiter(10, 10)
with pytest.raises(RateLimitExceeded) as e:
for i in range(20):
with limiter:
continue
assert "Rate exceeded" in str(e.value)

View File

@ -0,0 +1,82 @@
import threading
from time import sleep, time
class RateLimiterException(Exception):
pass
class RateLimitExceeded(RateLimiterException):
def __init__(self, retry_after: float, current_tokens: float):
self.retry_after = retry_after
self.current_tokens = current_tokens
super().__init__(f"Rate exceeded! Retry after {retry_after:.2f} seconds.")
class TokenBucketLimiter:
def __init__(self, capacity: int, refill_rate: int):
self.capacity = float(capacity)
self.refill_rate = float(refill_rate)
self.tokens = float(capacity)
self.last_refill_time = time()
self._lock = threading.Lock()
def __enter__(self):
with self._lock:
now = time()
elapsed = now - self.last_refill_time
new_tokens = elapsed * self.refill_rate
self.tokens = min(self.tokens + new_tokens, self.capacity)
self.last_refill_time = time()
if self.tokens < 1:
retry_after = (1 - self.tokens) / self.refill_rate
raise RateLimitExceeded(retry_after, self.tokens)
self.tokens -= 1
return self
def __exit__(self, exc_type, exc_value, exc_traceback) -> bool:
return False
def make_requests(number: int, limiter: TokenBucketLimiter) -> None:
start_time = time()
successful = 0
rate_limited = 0
failed = 0
print(f"Starting request cycle with {number} requests...")
for i in range(1, number):
try:
with limiter:
successful += 1
except RateLimitExceeded as e:
rate_limited += 1
sleep(e.retry_after)
try:
with limiter:
successful += 1
except RuntimeError:
failed += 1
print("Retry failed! Request limit reached!")
end_time = time()
execution_time = end_time - start_time
effective_rate = number / execution_time
print(
f"Results: \n\nSuccessfull: {successful}\nRate limited: {rate_limited}\nFailed: {failed}\n"
)
print(
f"Total execution time: {execution_time:.2f}\nEffective rate: {effective_rate:.2f}"
)
# limiter = TokenBucketLimiter(10, 10)
# make_requests(100, limiter)

75
rate_limiters/uv.lock Normal file
View File

@ -0,0 +1,75 @@
version = 1
revision = 3
requires-python = ">=3.11"
[[package]]
name = "colorama"
version = "0.4.6"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697, upload-time = "2022-10-25T02:36:22.414Z" }
wheels = [
{ url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335, upload-time = "2022-10-25T02:36:20.889Z" },
]
[[package]]
name = "iniconfig"
version = "2.3.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/72/34/14ca021ce8e5dfedc35312d08ba8bf51fdd999c576889fc2c24cb97f4f10/iniconfig-2.3.0.tar.gz", hash = "sha256:c76315c77db068650d49c5b56314774a7804df16fee4402c1f19d6d15d8c4730", size = 20503, upload-time = "2025-10-18T21:55:43.219Z" }
wheels = [
{ url = "https://files.pythonhosted.org/packages/cb/b1/3846dd7f199d53cb17f49cba7e651e9ce294d8497c8c150530ed11865bb8/iniconfig-2.3.0-py3-none-any.whl", hash = "sha256:f631c04d2c48c52b84d0d0549c99ff3859c98df65b3101406327ecc7d53fbf12", size = 7484, upload-time = "2025-10-18T21:55:41.639Z" },
]
[[package]]
name = "packaging"
version = "26.2"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/d7/f1/e7a6dd94a8d4a5626c03e4e99c87f241ba9e350cd9e6d75123f992427270/packaging-26.2.tar.gz", hash = "sha256:ff452ff5a3e828ce110190feff1178bb1f2ea2281fa2075aadb987c2fb221661", size = 228134, upload-time = "2026-04-24T20:15:23.917Z" }
wheels = [
{ url = "https://files.pythonhosted.org/packages/df/b2/87e62e8c3e2f4b32e5fe99e0b86d576da1312593b39f47d8ceef365e95ed/packaging-26.2-py3-none-any.whl", hash = "sha256:5fc45236b9446107ff2415ce77c807cee2862cb6fac22b8a73826d0693b0980e", size = 100195, upload-time = "2026-04-24T20:15:22.081Z" },
]
[[package]]
name = "pluggy"
version = "1.6.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/f9/e2/3e91f31a7d2b083fe6ef3fa267035b518369d9511ffab804f839851d2779/pluggy-1.6.0.tar.gz", hash = "sha256:7dcc130b76258d33b90f61b658791dede3486c3e6bfb003ee5c9bfb396dd22f3", size = 69412, upload-time = "2025-05-15T12:30:07.975Z" }
wheels = [
{ url = "https://files.pythonhosted.org/packages/54/20/4d324d65cc6d9205fabedc306948156824eb9f0ee1633355a8f7ec5c66bf/pluggy-1.6.0-py3-none-any.whl", hash = "sha256:e920276dd6813095e9377c0bc5566d94c932c33b27a3e3945d8389c374dd4746", size = 20538, upload-time = "2025-05-15T12:30:06.134Z" },
]
[[package]]
name = "pygments"
version = "2.20.0"
source = { registry = "https://pypi.org/simple" }
sdist = { url = "https://files.pythonhosted.org/packages/c3/b2/bc9c9196916376152d655522fdcebac55e66de6603a76a02bca1b6414f6c/pygments-2.20.0.tar.gz", hash = "sha256:6757cd03768053ff99f3039c1a36d6c0aa0b263438fcab17520b30a303a82b5f", size = 4955991, upload-time = "2026-03-29T13:29:33.898Z" }
wheels = [
{ url = "https://files.pythonhosted.org/packages/f4/7e/a72dd26f3b0f4f2bf1dd8923c85f7ceb43172af56d63c7383eb62b332364/pygments-2.20.0-py3-none-any.whl", hash = "sha256:81a9e26dd42fd28a23a2d169d86d7ac03b46e2f8b59ed4698fb4785f946d0176", size = 1231151, upload-time = "2026-03-29T13:29:30.038Z" },
]
[[package]]
name = "pytest"
version = "9.0.3"
source = { registry = "https://pypi.org/simple" }
dependencies = [
{ name = "colorama", marker = "sys_platform == 'win32'" },
{ name = "iniconfig" },
{ name = "packaging" },
{ name = "pluggy" },
{ name = "pygments" },
]
sdist = { url = "https://files.pythonhosted.org/packages/7d/0d/549bd94f1a0a402dc8cf64563a117c0f3765662e2e668477624baeec44d5/pytest-9.0.3.tar.gz", hash = "sha256:b86ada508af81d19edeb213c681b1d48246c1a91d304c6c81a427674c17eb91c", size = 1572165, upload-time = "2026-04-07T17:16:18.027Z" }
wheels = [
{ url = "https://files.pythonhosted.org/packages/d4/24/a372aaf5c9b7208e7112038812994107bc65a84cd00e0354a88c2c77a617/pytest-9.0.3-py3-none-any.whl", hash = "sha256:2c5efc453d45394fdd706ade797c0a81091eccd1d6e4bccfcd476e2b8e0ab5d9", size = 375249, upload-time = "2026-04-07T17:16:16.13Z" },
]
[[package]]
name = "rate-limiters"
version = "0.1.0"
source = { virtual = "." }
dependencies = [
{ name = "pytest" },
]
[package.metadata]
requires-dist = [{ name = "pytest", specifier = ">=9.0.3" }]