Re-usable code for Python 3 projects.
uv add python3-commonsSome features require extra dependencies. You can install them individually or all at once:
api-client: Forpython3_commons.api_clientaudit: Forpython3_commons.auditauthn: Forpython3_commons.authauthz: Forpython3_commons.permissionscache: Forpython3_commons.cachedatabase: Forpython3_commons.dbobject-storage: Forpython3_commons.object_storagesoap-client: Forpython3_commons.soap_clientall: Install all optional dependencies
uv add "python3-commons[all]"An LRU cache decorator for asynchronous functions that handles await and prevents the dogpile effect (mitigating multiple concurrent calls for the same key by waiting for the first one to finish).
from python3_commons.async_functools import async_lru_cache
@async_lru_cache(maxsize=128)
async def fetch_expensive_data(item_id: int):
# This function will only be called once for a given item_id
# even if multiple tasks await it simultaneously.
return await db.get(item_id)
# Usage
result = await fetch_expensive_data(42)A context manager for aiohttp requests with built-in audit logging to S3 and standardized error mapping to Python exceptions.
from python3_commons.api_client import request
from aiohttp import ClientSession
async with ClientSession() as session:
async with request(
session,
base_url="https://api.example.com",
uri="/data",
method="get",
audit_name="my_service_audit"
) as response:
data = await response.json()Async SOAP client support for zeep with S3 auditing capabilities.
from python3_commons.soap_client import soap_client
async with soap_client("https://example.com/service?wsdl") as client:
result = await client.service.GetData(id=42)SQLAlchemy async engine and session management with pool tuning, health checks, and dynamic query builders.
from python3_commons.db import AsyncSessionManager
from python3_commons.conf import DBSettings
# Configuration
configs = {"default": DBSettings(dsn="postgresql+asyncpg://user:pass@localhost/db")}
manager = AsyncSessionManager(configs)
# Usage
async with manager.get_session_context("default") as session:
result = await session.execute(...)
# Health check
from python3_commons.db import is_healthy
await is_healthy(manager.get_engine("default"))Utilities for async S3 operations using aiobotocore.
from python3_commons import object_storage
import io
# Upload an object
await object_storage.put_object(
bucket_name="my-bucket",
path="uploads/file.txt",
data=io.BytesIO(b"Hello World"),
length=11
)
# Download an object
content = await object_storage.get_object("my-bucket", "uploads/file.txt")
# List objects
async for obj in object_storage.list_objects("my-bucket", "uploads/"):
print(obj['Key'])Client for OpenID Connect authentication, supporting configuration fetching, JWKS, and token acquisition.
from python3_commons.auth import OIDCClient
from pydantic import HttpUrl
client = OIDCClient(
authority_url=HttpUrl("https://auth.example.com/realms/myrealm"),
client_id="my-app-client",
client_secret="secret"
)
async with client:
token_response = await client.fetch_token(username="user", password="password")
print(token_response.access_token)Async caching using Valkey (Redis-compatible) with automatic Msgpack serialization for complex types.
from python3_commons import cache
# Store a dictionary
await cache.store("user:123", {"name": "Alice", "role": "admin"}, ttl=3600)
# Retrieve it
user_data = await cache.get("user:123")
# Set operations
await cache.add_set_item("active_users", "user:123")
is_active = await cache.has_set_item("active_users", "user:123")A JSONFormatter for structured logging, compatible with standard Python logging.
import logging
from python3_commons.log.formatters import JSONFormatter
handler = logging.StreamHandler()
handler.setFormatter(JSONFormatter())
logging.getLogger().addHandler(handler)
logger = logging.getLogger("app")
logger.info("User logged in", extra={"user_id": "abc-123", "ip": "1.2.3.4"})Enhanced JSON and Msgpack serialization for types not handled by default (Decimal, datetime, date, dataclasses, Pydantic models).
from python3_commons.serializers.msgspec import serialize_msgpack, deserialize_msgpack
from decimal import Decimal
from datetime import datetime
data = {
"amount": Decimal("150.75"),
"timestamp": datetime.now(),
"tags": {"finance", "internal"}
}
# Serialize to Msgpack
binary = serialize_msgpack(data)
# Deserialize back
restored = deserialize_msgpack(binary)Database-backed Role-Based Access Control (RBAC) permission checking.
from python3_commons.permissions import has_user_permission
from uuid import UUID
user_uuid = UUID("...")
allowed = await has_user_permission(session, user_uuid, "reports.view")A collection of useful utility functions:
to_snake_case(text): Converts strings to snake_case.round_decimal(value, places): RoundsDecimalvalues.tries(n): An async retry decorator.log_execution_time: An async decorator to log how long a function takes.date_from_string/datetime_from_string: Flexible date/time parsing.request_to_curl: Converts request parameters to acurlcommand string.
from python3_commons.helpers import tries, log_execution_time
@tries(3)
@log_execution_time
async def flaky_network_call():
...Efficiently generate CSV data as a byte stream from an async generator of tuples.
from python3_commons.generators import tuple_csv_stream
async def generate_rows():
for i in range(1000):
yield (i, f"Name {i}", 10.5 * i)
async for chunk in tuple_csv_stream(generate_rows(), header=("ID", "Name", "Value")):
# Send chunk to HTTP response or write to file
pass