looking at rate-limiting implementations
This commit is contained in:
commit
29656c8d15
5
rate_limiters/.gitignore
vendored
Normal file
5
rate_limiters/.gitignore
vendored
Normal file
@ -0,0 +1,5 @@
|
||||
# Python-generated files
|
||||
__pycache__/
|
||||
.venv
|
||||
.ruff_cache
|
||||
.pytest_cache
|
||||
1
rate_limiters/.python-version
Normal file
1
rate_limiters/.python-version
Normal file
@ -0,0 +1 @@
|
||||
3.11
|
||||
0
rate_limiters/README.md
Normal file
0
rate_limiters/README.md
Normal file
44
rate_limiters/fixed_window.py
Normal file
44
rate_limiters/fixed_window.py
Normal 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
6
rate_limiters/main.py
Normal file
@ -0,0 +1,6 @@
|
||||
def main():
|
||||
print("Hello from rate-limiters!")
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
9
rate_limiters/pyproject.toml
Normal file
9
rate_limiters/pyproject.toml
Normal 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",
|
||||
]
|
||||
20
rate_limiters/tests/test_fixed_window.py
Normal file
20
rate_limiters/tests/test_fixed_window.py
Normal 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)
|
||||
25
rate_limiters/tests/test_token_bucket.py
Normal file
25
rate_limiters/tests/test_token_bucket.py
Normal 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)
|
||||
|
||||
82
rate_limiters/token_bucket.py
Normal file
82
rate_limiters/token_bucket.py
Normal 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
75
rate_limiters/uv.lock
Normal 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" }]
|
||||
Loading…
Reference in New Issue
Block a user