Convert Pydantic from V1 to V2 ♻
Project description
Bump Pydantic ♻️
Bump Pydantic is a tool to help you migrate your code from Pydantic V1 to V2.
[!NOTE]
If you find bugs, please report them on the issue tracker.
Table of contents
- Bump Pydantic ♻️
-
- BP001: Add default
None
toOptional[T]
,Union[T, None]
andAny
fields - BP002: Replace
Config
class bymodel_config
attribute - BP003: Replace
Field
old parameters to new ones - BP004: Replace imports
- BP005: Replace
GenericModel
byBaseModel
- BP006: Replace
__root__
byRootModel
- BP007: Replace decorators
- BP008: Replace
con*
functions byAnnotated
versions - BP009: Mark pydantic "protocol" functions in custom types with proper TODOs
- BP001: Add default
Installation
The installation is as simple as:
pip install bump-pydantic
Usage
bump-pydantic
is a CLI tool, hence you can use it from your terminal.
It's easy to use. If your project structure is:
repository/
└── my_package/
└── <python source files>
Then you'll want to do:
cd /path/to/repository
bump-pydantic my_package
Check diff before applying changes
To check the diff before applying the changes, you can run:
bump-pydantic --diff <path>
Apply changes
To apply the changes, you can run:
bump-pydantic <path>
Rules
You can find below the list of rules that are applied by bump-pydantic
.
It's also possible to disable rules by using the --disable
option.
BP001: Add default None
to Optional[T]
, Union[T, None]
and Any
fields
- ✅ Add default
None
toOptional[T]
fields.
The following code will be transformed:
class User(BaseModel):
name: Optional[str]
Into:
class User(BaseModel):
name: Optional[str] = None
BP002: Replace Config
class by model_config
attribute
- ✅ Replace
Config
class bymodel_config = ConfigDict()
. - ✅ Rename old
Config
attributes to newmodel_config
attributes. - ✅ Add a TODO comment in case the transformation can't be done automatically.
- ✅ Replace
Extra
enum by string values.
The following code will be transformed:
from pydantic import BaseModel, Extra
class User(BaseModel):
name: str
class Config:
extra = Extra.forbid
Into:
from pydantic import ConfigDict, BaseModel
class User(BaseModel):
name: str
model_config = ConfigDict(extra="forbid")
BP003: Replace Field
old parameters to new ones
- ✅ Replace
Field
old parameters to new ones. - ✅ Replace
field: Enum = Field(Enum.VALUE, const=True)
byfield: Literal[Enum.VALUE] = Enum.VALUE
.
The following code will be transformed:
from typing import List
from pydantic import BaseModel, Field
class User(BaseModel):
name: List[str] = Field(..., min_items=1)
Into:
from typing import List
from pydantic import BaseModel, Field
class User(BaseModel):
name: List[str] = Field(..., min_length=1)
BP004: Replace imports
- ✅ Replace
BaseSettings
frompydantic
topydantic_settings
. - ✅ Replace
Color
andPaymentCardNumber
frompydantic
topydantic_extra_types
.
BP005: Replace GenericModel
by BaseModel
- ✅ Replace
GenericModel
byBaseModel
.
The following code will be transformed:
from typing import Generic, TypeVar
from pydantic.generics import GenericModel
T = TypeVar('T')
class User(GenericModel, Generic[T]):
name: str
Into:
from typing import Generic, TypeVar
from pydantic import BaseModel
T = TypeVar('T')
class User(BaseModel, Generic[T]):
name: str
BP006: Replace __root__
by RootModel
- ✅ Replace
__root__
byRootModel
.
The following code will be transformed:
from typing import List
from pydantic import BaseModel
class User(BaseModel):
age: int
name: str
class Users(BaseModel):
__root__ = List[User]
Into:
from typing import List
from pydantic import RootModel, BaseModel
class User(BaseModel):
age: int
name: str
class Users(RootModel[List[User]]):
pass
BP007: Replace decorators
- ✅ Replace
@validator
by@field_validator
. - ✅ Replace
@root_validator
by@model_validator
.
The following code will be transformed:
from pydantic import BaseModel, validator, root_validator
class User(BaseModel):
name: str
@validator('name', pre=True)
def validate_name(cls, v):
return v
@root_validator(pre=True)
def validate_root(cls, values):
return values
Into:
from pydantic import BaseModel, field_validator, model_validator
class User(BaseModel):
name: str
@field_validator('name', mode='before')
def validate_name(cls, v):
return v
@model_validator(mode='before')
def validate_root(cls, values):
return values
BP008: Replace con*
functions by Annotated
versions
- ✅ Replace
constr(*args)
byAnnotated[str, StringConstraints(*args)]
. - ✅ Replace
conint(*args)
byAnnotated[int, Field(*args)]
. - ✅ Replace
confloat(*args)
byAnnotated[float, Field(*args)]
. - ✅ Replace
conbytes(*args)
byAnnotated[bytes, Field(*args)]
. - ✅ Replace
condecimal(*args)
byAnnotated[Decimal, Field(*args)]
. - ✅ Replace
conset(T, *args)
byAnnotated[Set[T], Field(*args)]
. - ✅ Replace
confrozenset(T, *args)
byAnnotated[Set[T], Field(*args)]
. - ✅ Replace
conlist(T, *args)
byAnnotated[List[T], Field(*args)]
.
The following code will be transformed:
from pydantic import BaseModel, constr
class User(BaseModel):
name: constr(min_length=1)
Into:
from pydantic import BaseModel, StringConstraints
from typing_extensions import Annotated
class User(BaseModel):
name: Annotated[str, StringConstraints(min_length=1)]
BP009: Mark Pydantic "protocol" functions in custom types with proper TODOs
- ✅ Mark
__get_validators__
as to be replaced by__get_pydantic_core_schema__
. - ✅ Mark
__modify_schema__
as to be replaced by__get_pydantic_json_schema__
.
The following code will be transformed:
class SomeThing:
@classmethod
def __get_validators__(cls):
yield from []
@classmethod
def __modify_schema__(cls, field_schema, field):
if field:
field_schema['example'] = "Weird example"
Into:
class SomeThing:
@classmethod
# TODO[pydantic]: We couldn't refactor `__get_validators__`, please create the `__get_pydantic_core_schema__` manually.
# Check https://docs.pydantic.dev/latest/migration/#defining-custom-types for more information.
def __get_validators__(cls):
yield from []
@classmethod
# TODO[pydantic]: We couldn't refactor `__modify_schema__`, please create the `__get_pydantic_json_schema__` manually.
# Check https://docs.pydantic.dev/latest/migration/#defining-custom-types for more information.
def __modify_schema__(cls, field_schema, field):
if field:
field_schema['example'] = "Weird example"
License
This project is licensed under the terms of the MIT license.
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
File details
Details for the file bump_pydantic-0.7.0.tar.gz
.
File metadata
- Download URL: bump_pydantic-0.7.0.tar.gz
- Upload date:
- Size: 28.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/4.0.2 CPython/3.11.5
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | c74d88550225b6c413b3b87e750d04054a2234724b0cc5592011502ac994ff7e |
|
MD5 | 9d935fcca90b823b22a133926dfb1468 |
|
BLAKE2b-256 | 13baf9ba7d28c5914c53232681c7bf0dda747cf411a0955e05e3de9ff85ca883 |
File details
Details for the file bump_pydantic-0.7.0-py3-none-any.whl
.
File metadata
- Download URL: bump_pydantic-0.7.0-py3-none-any.whl
- Upload date:
- Size: 28.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/4.0.2 CPython/3.11.5
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | f8920da00302b4883eb255b15a12ccc536859ace1437099815f525573b5b3840 |
|
MD5 | f72fc993bb8941e6cd9fa16c4e9f70ed |
|
BLAKE2b-256 | 1fecb6a311b409f1529500c34b6f901392d2595802dd39309951348d3a2f7dab |