@ -4,6 +4,7 @@ from dataclasses import dataclass, is_dataclass
from enum import Enum
from enum import Enum
from functools import lru_cache
from functools import lru_cache
from typing import (
from typing import (
TYPE_CHECKING ,
Any ,
Any ,
Callable ,
Callable ,
Deque ,
Deque ,
@ -23,7 +24,14 @@ from fastapi.types import IncEx, ModelNameMap, UnionType
from pydantic import BaseModel , create_model
from pydantic import BaseModel , create_model
from pydantic . version import VERSION as PYDANTIC_VERSION
from pydantic . version import VERSION as PYDANTIC_VERSION
from starlette . datastructures import UploadFile
from starlette . datastructures import UploadFile
from typing_extensions import Annotated , Literal , get_args , get_origin
from typing_extensions import (
Annotated ,
Literal ,
TypeAlias ,
assert_never ,
get_args ,
get_origin ,
)
PYDANTIC_VERSION_MINOR_TUPLE = tuple ( int ( x ) for x in PYDANTIC_VERSION . split ( " . " ) [ : 2 ] )
PYDANTIC_VERSION_MINOR_TUPLE = tuple ( int ( x ) for x in PYDANTIC_VERSION . split ( " . " ) [ : 2 ] )
PYDANTIC_V2 = PYDANTIC_VERSION_MINOR_TUPLE [ 0 ] == 2
PYDANTIC_V2 = PYDANTIC_VERSION_MINOR_TUPLE [ 0 ] == 2
@ -60,6 +68,7 @@ if PYDANTIC_V2:
from pydantic . json_schema import GenerateJsonSchema as GenerateJsonSchema
from pydantic . json_schema import GenerateJsonSchema as GenerateJsonSchema
from pydantic . json_schema import JsonSchemaValue as JsonSchemaValue
from pydantic . json_schema import JsonSchemaValue as JsonSchemaValue
from pydantic_core import CoreSchema as CoreSchema
from pydantic_core import CoreSchema as CoreSchema
from pydantic_core import ErrorDetails as ErrorDetails
from pydantic_core import PydanticUndefined , PydanticUndefinedType
from pydantic_core import PydanticUndefined , PydanticUndefinedType
from pydantic_core import Url as Url
from pydantic_core import Url as Url
@ -69,7 +78,7 @@ if PYDANTIC_V2:
)
)
except ImportError : # pragma: no cover
except ImportError : # pragma: no cover
from pydantic_core . core_schema import (
from pydantic_core . core_schema import (
general_plain_validator_function as with_info_plain_validator_function , # noqa: F401
general_plain_validator_function as with_info_plain_validator_function ,
)
)
RequiredParam = PydanticUndefined
RequiredParam = PydanticUndefined
@ -84,6 +93,9 @@ if PYDANTIC_V2:
class ErrorWrapper ( Exception ) :
class ErrorWrapper ( Exception ) :
pass
pass
# See https://github.com/pydantic/pydantic/blob/bb18ac5/pydantic/error_wrappers.py#L45-L47.
ErrorList : TypeAlias = Union [ Sequence [ " ErrorList " ] , ErrorWrapper ]
@dataclass
@dataclass
class ModelField :
class ModelField :
field_info : FieldInfo
field_info : FieldInfo
@ -117,22 +129,25 @@ if PYDANTIC_V2:
return Undefined
return Undefined
return self . field_info . get_default ( call_default_factory = True )
return self . field_info . get_default ( call_default_factory = True )
# See https://github.com/pydantic/pydantic/blob/bb18ac5/pydantic/fields.py#L850-L852 for the signature.
def validate (
def validate (
self ,
self ,
value : Any ,
value : Any ,
values : Dict [ str , Any ] = { } , # noqa: B006
values : Dict [ str , Any ] = { } , # noqa: B006
* ,
* ,
loc : Tuple [ Union [ int , str ] , . . . ] = ( ) ,
loc : Tuple [ Union [ int , str ] , . . . ] = ( ) ,
) - > Tuple [ Any , Union [ List [ Dict [ str , Any ] ] , None ] ] :
) - > Tuple [ Any , Union [ ErrorList , Sequence [ ErrorDetails ] , None ] ] :
try :
try :
return (
return (
self . _type_adapter . validate_python ( value , from_attributes = True ) ,
self . _type_adapter . validate_python ( value , from_attributes = True ) ,
None ,
None ,
)
)
except ValidationError as exc :
except ValidationError as exc :
return None , _regenerate_error_with_loc (
errors : List [ ErrorDetails ] = [
errors = exc . errors ( include_url = False ) , loc_prefix = loc
{ * * err , " loc " : loc + err [ " loc " ] }
)
for err in exc . errors ( include_url = False )
]
return None , errors
def serialize (
def serialize (
self ,
self ,
@ -169,7 +184,13 @@ if PYDANTIC_V2:
) - > Any :
) - > Any :
return annotation
return annotation
def _normalize_errors ( errors : Sequence [ Any ] ) - > List [ Dict [ str , Any ] ] :
def _normalize_errors (
errors : Union [ ErrorList , Sequence [ ErrorDetails ] ] ,
) - > List [ ErrorDetails ] :
assert isinstance ( errors , Sequence ) , type ( errors )
for error in errors :
assert not isinstance ( error , ErrorWrapper )
assert not isinstance ( error , Sequence )
return errors # type: ignore[return-value]
return errors # type: ignore[return-value]
def _model_rebuild ( model : Type [ BaseModel ] ) - > None :
def _model_rebuild ( model : Type [ BaseModel ] ) - > None :
@ -267,12 +288,12 @@ if PYDANTIC_V2:
assert issubclass ( origin_type , sequence_types ) # type: ignore[arg-type]
assert issubclass ( origin_type , sequence_types ) # type: ignore[arg-type]
return sequence_annotation_to_type [ origin_type ] ( value ) # type: ignore[no-any-return]
return sequence_annotation_to_type [ origin_type ] ( value ) # type: ignore[no-any-return]
def get_missing_field_error ( loc : Tuple [ str , . . . ] ) - > Dict [ str , Any ] :
def get_missing_field_error ( loc : Tuple [ str , . . . ] ) - > ErrorDetails :
error = ValidationError . from_exception_data (
[ error ] = ValidationError . from_exception_data (
" Field required " , [ { " type " : " missing " , " loc " : loc , " input " : { } } ]
" Field required " , [ { " type " : " missing " , " loc " : loc , " input " : { } } ]
) . errors ( include_url = False ) [ 0 ]
) . errors ( include_url = False , include_input = False )
error [ " input " ] = None
error [ " input " ] = None
return error # type: ignore[return-value]
return error
def create_body_model (
def create_body_model (
* , fields : Sequence [ ModelField ] , model_name : str
* , fields : Sequence [ ModelField ] , model_name : str
@ -291,14 +312,22 @@ else:
from fastapi . openapi . constants import REF_PREFIX as REF_PREFIX
from fastapi . openapi . constants import REF_PREFIX as REF_PREFIX
from pydantic import AnyUrl as Url # noqa: F401
from pydantic import AnyUrl as Url # noqa: F401
from pydantic import ( # type: ignore[assignment]
from pydantic import ( # type: ignore[assignment]
BaseConfig as BaseConfig , # noqa: F401
BaseConfig as BaseConfig ,
)
)
from pydantic import ValidationError as ValidationError # noqa: F401
from pydantic import ValidationError as ValidationError
from pydantic . class_validators import ( # type: ignore[no-redef]
from pydantic . class_validators import ( # type: ignore[no-redef]
Validator as Validator , # noqa: F401
Validator as Validator ,
)
if TYPE_CHECKING : # pragma: nocover
from pydantic . error_wrappers import ( # type: ignore[no-redef]
ErrorDict as ErrorDetails ,
)
from pydantic . error_wrappers import ( # type: ignore[no-redef]
ErrorList as ErrorList ,
)
)
from pydantic . error_wrappers import ( # type: ignore[no-redef]
from pydantic . error_wrappers import ( # type: ignore[no-redef]
ErrorWrapper as ErrorWrapper , # noqa: F401
ErrorWrapper as ErrorWrapper ,
)
)
from pydantic . errors import MissingError
from pydantic . errors import MissingError
from pydantic . fields import ( # type: ignore[attr-defined]
from pydantic . fields import ( # type: ignore[attr-defined]
@ -312,7 +341,7 @@ else:
)
)
from pydantic . fields import FieldInfo as FieldInfo
from pydantic . fields import FieldInfo as FieldInfo
from pydantic . fields import ( # type: ignore[no-redef,attr-defined]
from pydantic . fields import ( # type: ignore[no-redef,attr-defined]
ModelField as ModelField , # noqa: F401
ModelField as ModelField ,
)
)
# Keeping old "Required" functionality from Pydantic V1, without
# Keeping old "Required" functionality from Pydantic V1, without
@ -322,7 +351,7 @@ else:
Undefined as Undefined ,
Undefined as Undefined ,
)
)
from pydantic . fields import ( # type: ignore[no-redef, attr-defined]
from pydantic . fields import ( # type: ignore[no-redef, attr-defined]
UndefinedType as UndefinedType , # noqa: F401
UndefinedType as UndefinedType ,
)
)
from pydantic . schema import (
from pydantic . schema import (
field_schema ,
field_schema ,
@ -330,14 +359,14 @@ else:
get_model_name_map ,
get_model_name_map ,
model_process_schema ,
model_process_schema ,
)
)
from pydantic . schema import ( # type: ignore[no-redef] # noqa: F401
from pydantic . schema import ( # type: ignore[no-redef]
get_annotation_from_field_info as get_annotation_from_field_info ,
get_annotation_from_field_info as get_annotation_from_field_info ,
)
)
from pydantic . typing import ( # type: ignore[no-redef]
from pydantic . typing import ( # type: ignore[no-redef]
evaluate_forwardref as evaluate_forwardref , # noqa: F401
evaluate_forwardref as evaluate_forwardref ,
)
)
from pydantic . utils import ( # type: ignore[no-redef]
from pydantic . utils import ( # type: ignore[no-redef]
lenient_issubclass as lenient_issubclass , # noqa: F401
lenient_issubclass as lenient_issubclass ,
)
)
GetJsonSchemaHandler = Any # type: ignore[assignment,misc]
GetJsonSchemaHandler = Any # type: ignore[assignment,misc]
@ -427,18 +456,23 @@ else:
return True
return True
return False
return False
def _normalize_errors ( errors : Sequence [ Any ] ) - > List [ Dict [ str , Any ] ] :
def _normalize_errors (
use_errors : List [ Any ] = [ ]
errors : Union [ ErrorList , Sequence [ " ErrorDetails " ] ] ,
for error in errors :
) - > List [ " ErrorDetails " ] :
if isinstance ( error , ErrorWrapper ) :
use_errors : List [ ErrorDetails ] = [ ]
new_errors = ValidationError ( # type: ignore[call-arg]
if isinstance ( errors , ErrorWrapper ) :
errors = [ error ] , model = RequestErrorModel
use_errors . extend (
ValidationError ( # type: ignore[call-arg]
errors = [ errors ] , model = RequestErrorModel
) . errors ( )
) . errors ( )
use_errors . extend ( new_errors )
)
elif isinstance ( error , list ) :
elif isinstance ( errors , Sequence ) :
for error in errors :
assert not isinstance ( error , dict )
use_errors . extend ( _normalize_errors ( error ) )
use_errors . extend ( _normalize_errors ( error ) )
else :
return use_errors
use_errors . append ( error )
else :
assert_never ( errors ) # pragma: no cover
return use_errors
return use_errors
def _model_rebuild ( model : Type [ BaseModel ] ) - > None :
def _model_rebuild ( model : Type [ BaseModel ] ) - > None :
@ -509,10 +543,10 @@ else:
def serialize_sequence_value ( * , field : ModelField , value : Any ) - > Sequence [ Any ] :
def serialize_sequence_value ( * , field : ModelField , value : Any ) - > Sequence [ Any ] :
return sequence_shape_to_type [ field . shape ] ( value ) # type: ignore[no-any-return,attr-defined]
return sequence_shape_to_type [ field . shape ] ( value ) # type: ignore[no-any-return,attr-defined]
def get_missing_field_error ( loc : Tuple [ str , . . . ] ) - > Dict [ str , Any ] :
def get_missing_field_error ( loc : Tuple [ str , . . . ] ) - > " ErrorDetails " :
missing_field_error = ErrorWrapper ( MissingError ( ) , loc = loc ) # type: ignore[call-arg]
missing_field_error = ErrorWrapper ( MissingError ( ) , loc = loc ) # type: ignore[call-arg]
new_error = ValidationError ( [ missing_field_error ] , RequestErrorModel )
[ new_error ] = ValidationError ( [ missing_field_error ] , RequestErrorModel ) . errors ( )
return new_error . errors ( ) [ 0 ] # type: ignore[return-value]
return new_error
def create_body_model (
def create_body_model (
* , fields : Sequence [ ModelField ] , model_name : str
* , fields : Sequence [ ModelField ] , model_name : str
@ -526,17 +560,6 @@ else:
return list ( model . __fields__ . values ( ) ) # type: ignore[attr-defined]
return list ( model . __fields__ . values ( ) ) # type: ignore[attr-defined]
def _regenerate_error_with_loc (
* , errors : Sequence [ Any ] , loc_prefix : Tuple [ Union [ str , int ] , . . . ]
) - > List [ Dict [ str , Any ] ] :
updated_loc_errors : List [ Any ] = [
{ * * err , " loc " : loc_prefix + err . get ( " loc " , ( ) ) }
for err in _normalize_errors ( errors )
]
return updated_loc_errors
def _annotation_is_sequence ( annotation : Union [ Type [ Any ] , None ] ) - > bool :
def _annotation_is_sequence ( annotation : Union [ Type [ Any ] , None ] ) - > bool :
if lenient_issubclass ( annotation , ( str , bytes ) ) :
if lenient_issubclass ( annotation , ( str , bytes ) ) :
return False
return False