Metadata-Version: 2.4
Name: pydantic-discriminated
Version: 0.1.3
Summary: Type-safe discriminated unions for Pydantic models
Project-URL: Homepage, https://github.com/talbotknighton/pydantic-discriminated
Project-URL: Bug Tracker, https://github.com/talbotknighton/pydantic-discriminated/issues
Project-URL: Documentation, https://github.com/talbotknighton/pydantic-discriminated/blob/main/README.md
Author-email: Talbot Knighton <talbotknighton@gmail.org>
Maintainer-email: Talbot Knighton <talbotknighton@gmail.org>
License: MIT
License-File: LICENSE
Keywords: discriminated,polymorphic,pydantic,serialization,unions,validation
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Classifier: Typing :: Typed
Requires-Python: >=3.8
Requires-Dist: pydantic>=2.0.0
Description-Content-Type: text/markdown

# pydantic-discriminated

[![PyPI Version](https://img.shields.io/pypi/v/pydantic-discriminated.svg)](https://pypi.org/project/pydantic-discriminated/)
[![Python Versions](https://img.shields.io/pypi/pyversions/pydantic-discriminated.svg)](https://pypi.org/project/pydantic-discriminated/)
[![License](https://img.shields.io/github/license/TalbotKnighton/pydantic-discriminated.svg)](https://github.com/TalbotKnighton/pydantic-discriminated/blob/main/LICENSE)
[![Documentation](https://img.shields.io/badge/docs-mkdocs-blue.svg)](https://talbotknighton.github.io/pydantic-discriminated/)

Type-safe discriminated unions for Pydantic models.

## What are Discriminated Unions?

Discriminated unions (also called tagged unions) let you work with polymorphic data in a type-safe way. A "discriminator" field tells you which concrete type you're dealing with.

```python
from pydantic_discriminated import discriminated_model, DiscriminatedBaseModel

@discriminated_model("shape_type", "circle")
class Circle(DiscriminatedBaseModel):
    radius: float
    
    def area(self) -> float:
        return 3.14159 * self.radius ** 2

@discriminated_model("shape_type", "rectangle")
class Rectangle(DiscriminatedBaseModel):
    width: float
    height: float
    
    def area(self) -> float:
        return self.width * self.height

# Parse data with the correct type
data = {"shape_type": "circle", "radius": 5}
circle = Circle.model_validate(data)  # Fully typed as Circle
print(f"Area: {circle.area()}")  # 78.53975
```

## Features

- **🔍 Type Safety**: Proper type hints for IDE autocomplete and static analysis
- **📦 Nested Models**: Works with models nested at any level
- **🔄 Seamless Integration**: Uses standard Pydantic methods (`model_validate`, `model_dump`)
- **🧩 Polymorphic Validation**: Automatically validates and dispatches to the correct model type
- **📚 OpenAPI Compatible**: Works great with FastAPI for generating correct schemas

## Installation

```bash
pip install pydantic-discriminated
```

## Quick Example

Define discriminated models for different event types:

```python
from enum import Enum
from typing import List, Union
from pydantic import BaseModel

from pydantic_discriminated import discriminated_model, DiscriminatedBaseModel

class EventType(str, Enum):
    USER_CREATED = "user_created"
    USER_UPDATED = "user_updated"
    LOGIN_ATTEMPT = "login_attempt"

@discriminated_model(EventType, EventType.USER_CREATED)
class UserCreatedEvent(DiscriminatedBaseModel):
    user_id: str
    username: str

@discriminated_model(EventType, EventType.USER_UPDATED)
class UserUpdatedEvent(DiscriminatedBaseModel):
    user_id: str
    fields_changed: List[str]

@discriminated_model(EventType, EventType.LOGIN_ATTEMPT)
class LoginAttemptEvent(DiscriminatedBaseModel):
    user_id: str
    success: bool
    ip_address: str

# Container that handles any event type
class EventProcessor(BaseModel):
    events: List[Union[UserCreatedEvent, UserUpdatedEvent, LoginAttemptEvent]]
    
    def process(self):
        for event in self.events:
            if isinstance(event, UserCreatedEvent):
                print(f"New user created: {event.username}")
            elif isinstance(event, UserUpdatedEvent):
                print(f"User {event.user_id} updated fields: {event.fields_changed}")
            elif isinstance(event, LoginAttemptEvent):
                result = "succeeded" if event.success else "failed"
                print(f"Login {result} for user {event.user_id} from {event.ip_address}")
```

## Documentation

- [Full Documentation](https://talbotknighton.github.io/pydantic-discriminated/)
- [API Reference](https://talbotknighton.github.io/pydantic-discriminated/api-reference/)
- [Examples](https://talbotknighton.github.io/pydantic-discriminated/examples/basic-usage/)

## When To Use

This library is perfect for:

- **API Responses**: When endpoints return different object types
- **Event Systems**: Handling different event types in a type-safe way
- **State Machines**: Representing different states with specific properties
- **Polymorphic Data**: Working with heterogeneous data structures

## Resources

- [GitHub Repository](https://github.com/TalbotKnighton/pydantic-discriminated)
- [PyPI Package](https://pypi.org/project/pydantic-discriminated/)
- [Issue Tracker](https://github.com/TalbotKnighton/pydantic-discriminated/issues)

## License

MIT

---

Built by [Talbot Knighton](https://github.com/TalbotKnighton)