Skip to main content

Create partial models from your pydantic models. Partial models may allow None for certain or all fields.

Project description

pydantic-partial

Installation

Just use pip install pydantic-partial to install the library.

Note: pydantic-partial is compatible with pydantic version 2.x on Python 3.10, 3.11, 3.12 and 3.13. This is also ensured running all tests on all those versions using tox.

About

Create partial models from your normal pydantic models. Partial models will allow some or all fields to be optional and thus not be required when creating the model instance.

Partial models can be used to support PATCH HTTP requests where the user only wants to update some fields of the model and normal validation for required fields is not required. It may also be used to have partial response DTOs where you want to skip certain fields, this can be useful in combination with exclude_none. It is - like shown in these examples - intended to be used with API use cases, so when using pydantic with for example FastAPI.

Disclaimer: This is still an early release of pydantic-partial. Things might change in the future. PR welcome. ;-)

Usage example

pydantic-partial provides a mixin to generate partial model classes. The mixin can be used like this:

import pydantic
from pydantic_partial import PartialModelMixin

# Something model, then can be used as a partial, too:
class Something(PartialModelMixin, pydantic.BaseModel):
    name: str
    age: int


# Create a full partial model
FullSomethingPartial = Something.model_as_partial()
FullSomethingPartial()  # Same as FullSomethingPartial(name=None, age=None)

Without using the mixin

You also may create partial models without using the mixin:

import pydantic
from pydantic_partial import create_partial_model

# Something model, without the mixin:
class Something(pydantic.BaseModel):
    name: str
    age: int


# Create a full partial model
FullSomethingPartial = create_partial_model(Something)
FullSomethingPartial()  # Same as FullSomethingPartial(name=None, age=None)

Only changing some fields to being optional

pydantic-partial can be used to create partial models that only change some of the fields to being optional. Just pass the list of fields to be optional to the as_partial() or create_partial_model() function.

import pydantic
from pydantic_partial import create_partial_model

class Something(pydantic.BaseModel):
    name: str
    age: int

# Create a partial model only for the name attribute
FullSomethingPartial = create_partial_model(Something, 'name')
FullSomethingPartial(age=40)  # Same as FullSomethingPartial(name=None, age=40)
# This would still raise an error: FullSomethingPartial(age=None, ...)

Recursive partials

Partial models can be created changing the field of all nested models to being optional, too.

from typing import List

import pydantic
from pydantic_partial import PartialModelMixin, create_partial_model

class InnerSomething(PartialModelMixin, pydantic.BaseModel):
    name: str

class OuterSomething(pydantic.BaseModel):
    name: str
    things: List[InnerSomething]

# Create a full partial model
RecursiveOuterSomethingPartial = create_partial_model(OuterSomething, recursive=True)
RecursiveOuterSomethingPartial(things=[
    {},
])

Note: The inner model MUST extend the PartialModelMixin mixin. Otherwise pydantic-partial will not be able to detect which fields may allow to being converted to partial models.

Also note: My recommendation would be to always create such recursive partials by creating partials for all the required models and then override the fields on you outer partial model class. This is way more explicit.

Known limitations

pydantic-partial cannot generate new class types that actually are supported by the Python typing system rules. This means that the partial models will only be recognized as the same as their original model classes - type checkers will not know about the partial model changes and thus will think all those partial fields are still required.

This is due to the fact that Python itself has no concept of partials. pydantic-partial could (in theory) provide plugins for mypy for example to "patch" this in, but this would be a massive amount of work while being kind of a bad hack. The real solution would be to have a partial type in Python itself, but this is not planned for the near future as far as I know.

My recommendation is to use pydantic-partial only for API use cases where you do not need to work with the partial aspects of the models - they are just the DTOs (data transfer objects) you are using. If you need to use partial models in other cases you might get errors by your type checker - if you use one. Please be aware of this.

Note: Not having a good solution in Python itself for this is the reason pydantic does not support partial models in the first place. pydantic-partial is just a really good workaround for this issue.
See issue 2 in this project and issue 1673 in the pydantic project for reference.

Having that all said: If anyone wants to get a working plugin for mypy or others ready, I'm going to very much support this.

Contributing

If you want to contribute to this project, feel free to just fork the project, create a dev branch in your fork and then create a pull request (PR). If you are unsure about whether your changes really suit the project please create an issue first, to talk about this.

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

pydantic_partial-0.10.2.tar.gz (5.8 kB view details)

Uploaded Source

Built Distribution

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

pydantic_partial-0.10.2-py3-none-any.whl (7.3 kB view details)

Uploaded Python 3

File details

Details for the file pydantic_partial-0.10.2.tar.gz.

File metadata

  • Download URL: pydantic_partial-0.10.2.tar.gz
  • Upload date:
  • Size: 5.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.10.2 {"installer":{"name":"uv","version":"0.10.2","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for pydantic_partial-0.10.2.tar.gz
Algorithm Hash digest
SHA256 8dbfc9843c8ca854c56c368a276e2a61d9f65beefd1deee5c4dd8ed7d74435fd
MD5 12651b731763ba73dc2e835bd809ae20
BLAKE2b-256 4c996d9b16dd997ac6ae25d0c543325d5a444a6226b007efd7e9ddee41e2d62a

See more details on using hashes here.

File details

Details for the file pydantic_partial-0.10.2-py3-none-any.whl.

File metadata

  • Download URL: pydantic_partial-0.10.2-py3-none-any.whl
  • Upload date:
  • Size: 7.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.10.2 {"installer":{"name":"uv","version":"0.10.2","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for pydantic_partial-0.10.2-py3-none-any.whl
Algorithm Hash digest
SHA256 8fdb3f270740a88617e8e40dfbb3cce484502e48b1b5d3e8a6a311adfe33e016
MD5 ac6aaaab69de38b7ecf974dd0d66cc4c
BLAKE2b-256 369486baf7016637f254fe990c282d9d08a531e9fc5aa4b290c1b332fca4dd84

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