An httpx client with persistent caching and rate limiting for data pipelines
Project description
httpx-frugal
An httpx client with persistent caching (hishel) and per-domain rate limiting (pyrate-limiter), designed for data pipelines that need to conserve API quota across process restarts.
Cache is checked before rate limiting, so cached responses do not consume rate-limit tokens.
Install
pip install httpx-frugal
Optional extras:
pip install httpx-frugal[http2] # HTTP/2 inner transport
Quickstart (sync)
import pathlib
from pyrate_limiter import Duration, Rate
from httpx_frugal import RateLimitedCacheClient
rates = [Rate(5, Duration.MINUTE)]
cache_db = pathlib.Path("~/.cache/myapp/http-cache.sqlite").expanduser()
rate_db = pathlib.Path("~/.cache/myapp/rate-limiter.sqlite").expanduser()
cache_db.parent.mkdir(parents=True, exist_ok=True)
rate_db.parent.mkdir(parents=True, exist_ok=True)
client_wrapper = RateLimitedCacheClient(
rates=rates,
cache_db_path=cache_db,
rate_limiter_db_path=rate_db,
)
if client_wrapper.would_hit_cache("https://api.example.com/data"):
print("served from cache on next request")
with client_wrapper as client:
response = client.get("https://api.example.com/data")
print(response.extensions.get("hishel_from_cache"))
Async
from httpx_frugal import AsyncRateLimitedCacheClient
async with AsyncRateLimitedCacheClient(
rates=rates,
cache_db_path=cache_db,
rate_limiter_db_path=rate_db,
) as client:
response = await client.get("https://api.example.com/data")
Blocking mode
Wait for a rate-limit token instead of raising immediately:
client_wrapper = RateLimitedCacheClient(
rates=rates,
cache_db_path=cache_db,
rate_limiter_db_path=rate_db,
blocking=True,
rate_limit_timeout_seconds=30.0,
)
Per-request cache TTL
Override TTL for a single request via httpx extensions:
with client_wrapper as client:
client.get("https://api.example.com/short-lived", extensions={"hishel_ttl": 60})
Or use the helper:
from httpx_frugal import request_with_ttl
with client_wrapper as client:
req = request_with_ttl(client, "GET", "https://api.example.com/x", ttl=120)
client.send(req)
Rate limit introspection
remaining = client_wrapper.tokens_available("api.example.com")
# e.g. {"5/60000s": 3}
Cache invalidation
client_wrapper.clear_cache() # all entries
client_wrapper.clear_cache_for("https://...") # one URL
client_wrapper.clear_rate_limiter() # reset tokens
Multi-process pipelines
Enable SQLite file locking on the rate limiter bucket:
RateLimitedCacheClient(..., use_file_lock=True)
would_hit_cache caveat
would_hit_cache() uses hishel internal APIs and may break if hishel changes them. Prefer checking response.extensions["hishel_from_cache"] after requests when possible. httpx-frugal pins hishel>=1.2,<2.
Development
uv sync
uv run ruff check .
uv run pytest
License
MIT
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file httpx_frugal-0.2.0.tar.gz.
File metadata
- Download URL: httpx_frugal-0.2.0.tar.gz
- Upload date:
- Size: 49.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
34e7064259ed28d03865f6ac9125734dff3aa1ab5f147e56bc3f35bcd2e92a40
|
|
| MD5 |
7942d4ea51257a50766b3f1c9d3b7f2a
|
|
| BLAKE2b-256 |
bf32914baf71c9f05e3e405c73e42ea6a8cbe51a40191019afceaaa1e0185691
|
File details
Details for the file httpx_frugal-0.2.0-py3-none-any.whl.
File metadata
- Download URL: httpx_frugal-0.2.0-py3-none-any.whl
- Upload date:
- Size: 11.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6f097c6c7740f830efdc4aef117b307e8603e3d65ef38bd846ecba667de3c78b
|
|
| MD5 |
f0008a689c07ee3aa9714006ef077b25
|
|
| BLAKE2b-256 |
30d9fbf7c6233ea0ac66c444805633539877a219f1cdc4591a142b3f1e9c4000
|