Skip to main content

Fast local rate limiting for Python, powered by Rust.

Project description

rust-py-rate-limit

Fast local rate limiting for Python, powered by Rust.

A fast, thread-safe, in-process rate limiter for Python with a core written in Rust (via PyO3 + maturin). Use it to protect endpoints, functions, internal APIs, workers and backend scripts against bursts of traffic — with zero external services.

from rust_py_rate_limit import RateLimiter

limiter = RateLimiter(limit=10, window_seconds=60)

if limiter.allow("user:123"):
    print("allowed")
else:
    print("blocked")

Table of contents


What is this?

rust-py-rate-limit is a local (in-process) rate limiter. Every limiter instance keeps its counters in memory inside your Python process, guarded by a concurrent, sharded hash map on the Rust side. There is no Redis, no network hop, and no serialization on the hot path — just a couple of atomic operations per request.

It works anywhere Python runs:

  • Plain Python
  • FastAPI
  • Django
  • Flask (preview)
  • Background workers and scripts

Why Rust?

  • Speed — the counting logic is compiled native code; the hot path releases the GIL so multiple Python threads can check limits in parallel.
  • Safety — no data races by construction. State lives in a DashMap (a sharded concurrent map) and statistics use lock-free atomics, so there is no global lock on the critical path.
  • Simplicity — a tiny, predictable API surface that is hard to misuse.

Installation

pip install rust-py-rate-limit

Requires Python 3.10+. Wheels are published for Linux, macOS and Windows, so no Rust toolchain is needed to install.

Quick start

from rust_py_rate_limit import RateLimiter

limiter = RateLimiter(limit=3, window_seconds=60)

assert limiter.allow("ip:127.0.0.1") is True
assert limiter.allow("ip:127.0.0.1") is True
assert limiter.allow("ip:127.0.0.1") is True
assert limiter.allow("ip:127.0.0.1") is False   # limit reached

How Fixed Window works

The MVP implements the Fixed Window algorithm. Each key gets a counter and a window start time. Within a window of window_seconds, up to limit requests are admitted; once the window elapses, the counter resets.

limit = 3, window = 60s, key = "user:1"

request 1 -> allowed
request 2 -> allowed
request 3 -> allowed
request 4 -> blocked
... 60s later ...
request 5 -> allowed   (new window)

Fixed Window is simple and cheap. Its only caveat is that it can admit up to 2 * limit requests around a window boundary (a burst at the end of one window plus a burst at the start of the next). If you need stricter smoothing, the roadmap includes Sliding Window and Token Bucket.

API reference

RateLimiter(limit: int, window_seconds: int)

Both arguments must be positive integers. Passing 0 (or a negative value) raises ValueError.

Method Returns Description
allow(key: str) bool Consume one request. True if admitted, False if blocked.
check(key: str) dict Consume one request and return full detail (see below).
remaining(key: str) int Requests left in the current window without consuming one.
reset(key: str) bool Drop a key's state. True if it existed.
clear() None Drop all keys.
stats() dict Activity counters (see Statistics).
cleanup_expired() int Remove keys whose window has expired. Returns the count removed.

Read-only properties: limiter.max_requests and limiter.window_seconds. (The configured limit is max_requests, since .limit(...) is the decorator.)

check() return value

Allowed:

{
    "allowed": True,
    "limit": 100,
    "remaining": 99,
    "reset_after_seconds": 60,
    "retry_after_seconds": 0,
}

Blocked:

{
    "allowed": False,
    "limit": 100,
    "remaining": 0,
    "reset_after_seconds": 42,
    "retry_after_seconds": 42,
}

FastAPI

Manual check

from fastapi import FastAPI, Request, HTTPException
from rust_py_rate_limit import RateLimiter

app = FastAPI()
limiter = RateLimiter(limit=100, window_seconds=60)

@app.get("/api/users")
def list_users(request: Request):
    key = request.client.host
    if not limiter.allow(key):
        raise HTTPException(status_code=429, detail="Too many requests")
    return {"users": []}

Middleware

from rust_py_rate_limit.fastapi import RateLimitMiddleware

app.add_middleware(
    RateLimitMiddleware,
    limit=100,
    window_seconds=60,
    key_func=lambda request: request.client.host,
)

When a request is blocked the middleware responds with 429 and {"detail": "Too many requests"}. Every response carries the standard headers:

X-RateLimit-Limit
X-RateLimit-Remaining
X-RateLimit-Reset
Retry-After      (only when blocked)

Django

# settings.py
MIDDLEWARE = [
    # ...
    "rust_py_rate_limit.django.RateLimitMiddleware",
]

RUST_PY_RATE_LIMIT = {
    "LIMIT": 100,
    "WINDOW_SECONDS": 60,
    "KEY": "ip",  # "ip" or "user"
}

Or check manually in a view:

from django.http import JsonResponse
from rust_py_rate_limit import RateLimiter

limiter = RateLimiter(limit=100, window_seconds=60)

def my_view(request):
    key = request.META.get("REMOTE_ADDR")
    if not limiter.allow(key):
        return JsonResponse({"detail": "Too many requests"}, status=429)
    return JsonResponse({"ok": True})

Flask

from flask import Flask
from rust_py_rate_limit.flask import FlaskRateLimiter

app = Flask(__name__)
limiter = FlaskRateLimiter(app, limit=100, window_seconds=60)

@app.get("/api/users")
@limiter.limit()
def list_users():
    return {"users": []}

Decorator

from rust_py_rate_limit import RateLimiter, RateLimitExceeded

limiter = RateLimiter(limit=5, window_seconds=60)

@limiter.limit("login")
def login():
    return "ok"

When the limit is exceeded the decorated function raises RateLimitExceeded (which carries .key, .limit and .retry_after). The key may also be a callable that derives the key from the function's arguments:

@limiter.limit(lambda user_id: f"user:{user_id}")
def fetch(user_id):
    ...

Statistics

limiter.stats()
# {
#     "allowed": 1200,
#     "blocked": 35,
#     "total_checks": 1235,
#     "active_keys": 20,
# }

Limitations

Be honest with yourself about what an in-process limiter can and cannot do:

  • The rate-limit state is local to the process.
  • Under Gunicorn/Uvicorn with multiple workers, each worker keeps its own counters, so the effective global limit is roughly limit × workers.
  • It is not a replacement for Redis when you need distributed rate limiting.
  • Fixed Window can allow short bursts at the boundary between two windows.
  • For distributed production setups, a Redis/Postgres backend is planned (see the roadmap).

Roadmap

Version Highlights
v0.1.0 Fixed Window · allow/check/remaining/reset/clear/stats/cleanup_expired · pytest · README
v0.2.0 Decorator · FastAPI/Django middleware · HTTP headers
v0.3.0 Sliding Window · Token Bucket · background cleanup
v0.4.0 Redis backend · distributed rate limiting
v0.5.0 Prometheus metrics · ImmutableLog integration

Development

# Rust unit tests
cargo test

# Build the extension into a virtualenv and run the Python tests
python -m venv .venv && source .venv/bin/activate
pip install -e ".[dev]"      # or: pip install maturin && maturin develop
maturin develop
pytest

License

MIT © Roberto Lima

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

rust_py_rate_limit-0.1.4.tar.gz (27.7 kB view details)

Uploaded Source

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

rust_py_rate_limit-0.1.4-cp312-cp312-win_amd64.whl (151.1 kB view details)

Uploaded CPython 3.12Windows x86-64

rust_py_rate_limit-0.1.4-cp312-cp312-manylinux_2_34_x86_64.whl (272.3 kB view details)

Uploaded CPython 3.12manylinux: glibc 2.34+ x86-64

rust_py_rate_limit-0.1.4-cp312-cp312-macosx_11_0_arm64.whl (244.1 kB view details)

Uploaded CPython 3.12macOS 11.0+ ARM64

File details

Details for the file rust_py_rate_limit-0.1.4.tar.gz.

File metadata

  • Download URL: rust_py_rate_limit-0.1.4.tar.gz
  • Upload date:
  • Size: 27.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for rust_py_rate_limit-0.1.4.tar.gz
Algorithm Hash digest
SHA256 bc5fe7ca344634b2c6124cd47415ec9552abfd69245fc6aa5e3620979578fbf9
MD5 a6a4ae6661ffe82c7033137e87c75c51
BLAKE2b-256 7d0f97514c13656aa391265a138f1db7ef17b61779c35298d9cc50dfaaf0bb19

See more details on using hashes here.

Provenance

The following attestation bundles were made for rust_py_rate_limit-0.1.4.tar.gz:

Publisher: release.yml on robertolima-dev/rust-py-rate-limit

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file rust_py_rate_limit-0.1.4-cp312-cp312-win_amd64.whl.

File metadata

File hashes

Hashes for rust_py_rate_limit-0.1.4-cp312-cp312-win_amd64.whl
Algorithm Hash digest
SHA256 ee30a2bfd8f665ef2de87e38aecf263895d9ce773bc17f93d8bcb5adb438b58e
MD5 584a451502036b7ed7f6f5efbdcea46a
BLAKE2b-256 78abe16641ea738778245e2472da9e329c46ea76727fa2265eb51ac8d98527a0

See more details on using hashes here.

Provenance

The following attestation bundles were made for rust_py_rate_limit-0.1.4-cp312-cp312-win_amd64.whl:

Publisher: release.yml on robertolima-dev/rust-py-rate-limit

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file rust_py_rate_limit-0.1.4-cp312-cp312-manylinux_2_34_x86_64.whl.

File metadata

File hashes

Hashes for rust_py_rate_limit-0.1.4-cp312-cp312-manylinux_2_34_x86_64.whl
Algorithm Hash digest
SHA256 c4c3597229d451f667ba79bf86d8a0ad4d6508b0e84798d3260bd7bf29627d79
MD5 b2145fe06d195770f5669476c7d309dd
BLAKE2b-256 60b2039c45f2d2c06e288c08fdfc2510bc03091b3a120fe3bbd36ef681a7c086

See more details on using hashes here.

Provenance

The following attestation bundles were made for rust_py_rate_limit-0.1.4-cp312-cp312-manylinux_2_34_x86_64.whl:

Publisher: release.yml on robertolima-dev/rust-py-rate-limit

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file rust_py_rate_limit-0.1.4-cp312-cp312-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for rust_py_rate_limit-0.1.4-cp312-cp312-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 b6d2af36acfce38d09d03a88a38f42617e85db7ad7d02b588e297dbdd7dbd13e
MD5 e8adefac3e3c9ba6b1c72ff70feed8e4
BLAKE2b-256 51156bb330959bb82475c014a29506935d24893ee2b3aadde6c2f9db82b677a0

See more details on using hashes here.

Provenance

The following attestation bundles were made for rust_py_rate_limit-0.1.4-cp312-cp312-macosx_11_0_arm64.whl:

Publisher: release.yml on robertolima-dev/rust-py-rate-limit

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page