Async-first pydantic integration for Pico-IoC. Enables defining pydantic tasks as async methods on IoC-managed components, with automatic discovery, dependency injection, and container-scoped execution.
Project description
๐ฆ pico-pydantic
Pico-Pydantic
Pico-Pydantic integrates Pico-IoC with Pydantic, enabling declarative, aspect-oriented validation of method arguments within your service layer.
It uses Pico-IoC's MethodInterceptor system to perform validation based on Pydantic BaseModel type hints before your method's business logic runs. This is the ideal tool for ensuring arguments passed between IoC-managed services are structurally correct.
๐ Requires Python 3.10+ ๐งฉ Works with Pydantic 2.0+ ๐ Supports async and sync methods ๐งช Enables unit testing of validation separate from business logic
๐ฏ Why pico-pydantic
While web frameworks handle validation at the HTTP boundary, business services often need to guarantee input integrity internally, especially when components are called from CLI tools, workers, or other services.
Pico-Pydantic provides:
- Declarative
@validateboundaries for service methods. - Aspect-Oriented Programming (AOP) for argument validation.
- Clear error handling with
ValidationFailedError. - Centralized validation logic, decoupled from the core service code.
| Concern | Pico-IoC Default | pico-pydantic |
|---|---|---|
| Argument checking | Manual if/raise |
Declarative @validate |
| Schema definition | None | Pydantic BaseModel type hints |
| Handling errors | Raw ValidationError |
Wrapped ValidationFailedError |
๐งฑ Core Features
- Method validation via
@validatedecorator. ValidationInterceptorfor AOP execution.- Seamless compatibility with
BaseModeltype annotations. - Correct handling of positional, keyword, and default arguments.
- Zero coupling to web frameworks.
๐ฆ Installation
pip install pico-pydantic
Also install pico-ioc and pydantic:
pip install pico-ioc pydantic
๐ Quick Example
1. Define the Data Model and Service:
from pydantic import BaseModel, Field
from pico_ioc import component
from pico_pydantic import validate
class ItemData(BaseModel):
name: str = Field(min_length=3)
price: float = Field(gt=0)
@component
class InventoryService:
@validate
async def add_item(self, data: ItemData) -> dict:
# Validation happens BEFORE this line
print(f"Adding item: {data.name}")
return data.model_dump()
2. Full Example: Initialization, Success, and Failure Handling
import asyncio
from pico_ioc import init, DictSource, configuration
from pico_pydantic import ValidationFailedError
# Define the base configuration (optional)
config = configuration(DictSource({}))
# Initialize the container, registering pico_pydantic and your components
# 'components' is used here as the module containing InventoryService.
container = init(
modules=["pico_pydantic", "components"],
config=config,
)
async def main():
service = container.get(InventoryService)
# --- Success: Validation Passes ---
print("--- Testing Success ---")
result = await service.add_item({"name": "Hammer", "price": 10.50})
print(f"Result: {result}")
# --- Failure: ValidationFailedError is thrown by the Interceptor ---
print("\n--- Testing Failure ---")
try:
# Fails: 'price' is negative, violating Field(gt=0)
await service.add_item({"name": "A", "price": -5})
except ValidationFailedError as e:
print(f"Validation failed for method '{e.method_name}'.")
print(e.pydantic_error) # Shows the detailed Pydantic error
await container.cleanup_all_async()
container.shutdown()
if __name__ == "__main__":
asyncio.run(main())
โ๏ธ How It Works
- The
ValidationInterceptoris globally registered with Pico-IoC. - When a method decorated with
@validateis called: - The interceptor captures the call arguments.
- It inspects the method signature for arguments with the
BaseModeltype hint. - It calls
BaseModel.model_validate(value)on the argument value. - If validation fails, it wraps the error in
ValidationFailedErrorand stops execution. - If successful,
call_nextis executed, and the original method runs.
No manual checks inside the service method. Logic stays clean.
๐ก Architecture Overview
โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ
โ Your App โ
โ (Service Layer) โ
โโโโโโโโโโโโโโโโฌโโโโโโโโโโโโโโโ
โ
@validate called
โ
โโโโโโโโโโโโโโโโผโโโโโโโโโโโโโโโโ
โ Pico-IoC โ
โโโโโโโโโโโโโโโโฌโโโโโโโโโโโโโโโโ
โ
ValidationInterceptor (AOP)
โ
โโโโโโโโโโโโโโโโผโโโโโโโโโโโโโโโโ
โ pico-pydantic โ
โ Inspect & model_validate() โ
โโโโโโโโโโโโโโโโฌโโโโโโโโโโโโโโโโ
โ
Pydantic 2.0+
๐ 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 pico_pydantic-0.1.0.tar.gz.
File metadata
- Download URL: pico_pydantic-0.1.0.tar.gz
- Upload date:
- Size: 19.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
659e0b129f13d3755aefceedf718eeb6df71b251d75463949084a1148054e42a
|
|
| MD5 |
37d362583210202a56e0b165c94869da
|
|
| BLAKE2b-256 |
a3f703bdf4738b16bdc5aeba07a81190c50eb5ed809915d9235f90a9160b985e
|
File details
Details for the file pico_pydantic-0.1.0-py3-none-any.whl.
File metadata
- Download URL: pico_pydantic-0.1.0-py3-none-any.whl
- Upload date:
- Size: 7.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c737e5305963e38569da75e38358c9f6d627fc5a7315b3aa61f72c3319f38264
|
|
| MD5 |
4573da27f5fb8a2ae26938caafe0a47f
|
|
| BLAKE2b-256 |
b48291641a7f4903cc6b2e623b2483547a0bb3e1fb6231c7d3c11617c94900ac
|