Skip to main content

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

PyPI Ask DeepWiki License: MIT CI (tox matrix) codecov Quality Gate Status Duplicated Lines (%) Maintainability Rating Docs

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 @validate boundaries 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 @validate decorator.
  • ValidationInterceptor for AOP execution.
  • Seamless compatibility with BaseModel type 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 ValidationInterceptor is globally registered with Pico-IoC.
  • When a method decorated with @validate is called:
  • The interceptor captures the call arguments.
  • It inspects the method signature for arguments with the BaseModel type hint.
  • It calls BaseModel.model_validate(value) on the argument value.
  • If validation fails, it wraps the error in ValidationFailedError and stops execution.
  • If successful, call_next is 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


Download files

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

Source Distribution

pico_pydantic-0.1.0.tar.gz (19.4 kB view details)

Uploaded Source

Built Distribution

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

pico_pydantic-0.1.0-py3-none-any.whl (7.6 kB view details)

Uploaded Python 3

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

Hashes for pico_pydantic-0.1.0.tar.gz
Algorithm Hash digest
SHA256 659e0b129f13d3755aefceedf718eeb6df71b251d75463949084a1148054e42a
MD5 37d362583210202a56e0b165c94869da
BLAKE2b-256 a3f703bdf4738b16bdc5aeba07a81190c50eb5ed809915d9235f90a9160b985e

See more details on using hashes here.

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

Hashes for pico_pydantic-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 c737e5305963e38569da75e38358c9f6d627fc5a7315b3aa61f72c3319f38264
MD5 4573da27f5fb8a2ae26938caafe0a47f
BLAKE2b-256 b48291641a7f4903cc6b2e623b2483547a0bb3e1fb6231c7d3c11617c94900ac

See more details on using hashes here.

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