mirror of
https://github.com/fastapi/fastapi.git
synced 2026-01-11 15:38:46 -05:00
* ✨ Pydantic v2 migration, initial implementation (#9500) * ✨ Add compat layer, for Pydantic v1 and v2 * ✨ Re-export Pydantic needed internals from compat, to later patch them for v1 * ♻️ Refactor internals to use new compatibility layers and run with Pydantic v2 * 📝 Update examples to run with Pydantic v2 * ✅ Update tests to use Pydantic v2 * 🎨 [pre-commit.ci] Auto format from pre-commit.com hooks * ✅ Temporarily disable Peewee tests, afterwards I'll enable them only for Pydantic v1 * 🐛 Fix JSON Schema generation and OpenAPI ref template * 🐛 Fix model field creation with defaults from Pydantic v2 * 🐛 Fix body field creation, with new FieldInfo * ✨ Use and check new ResponseValidationError for server validation errors * ✅ Fix test_schema_extra_examples tests with ResponseValidationError * ✅ Add dirty-equals to tests for compatibility with Pydantic v1 and v2 * ✨ Add util to regenerate errors with custom loc * ✨ Generate validation errors with loc * ✅ Update tests for compatibility with Pydantic v1 and v2 * ✅ Update tests for Pydantic v2 in tests/test_filter_pydantic_sub_model.py * ✅ Refactor tests in tests/test_dependency_overrides.py for Pydantic v2, separate parameterized into independent tests to use insert_assert * ✅ Refactor OpenAPI test for tests/test_infer_param_optionality.py for consistency, and make it compatible with Pydantic v1 and v2 * ✅ Update tests for tests/test_multi_query_errors.py for Pydantic v1 and v2 * ✅ Update tests for tests/test_multi_body_errors.py for Pydantic v1 and v2 * ✅ Update tests for tests/test_multi_body_errors.py for Pydantic v1 and v2 * 🎨 [pre-commit.ci] Auto format from pre-commit.com hooks * ♻️ Refactor tests for tests/test_path.py to inline pytest parameters, to make it easier to make them compatible with Pydantic v2 * ✅ Refactor and udpate tests for tests/test_path.py for Pydantic v1 and v2 * ♻️ Refactor and update tests for tests/test_query.py with compatibility for Pydantic v1 and v2 * ✅ Fix test with optional field without default None * ✅ Update tests for compatibility with Pydantic v2 * ✅ Update tutorial tests for Pydantic v2 * ♻️ Update OAuth2 dependencies for Pydantic v2 * ♻️ Refactor str check when checking for sequence types * ♻️ Rename regex to pattern to keep in sync with Pydantic v2 * ♻️ Refactor _compat.py, start moving conditional imports and declarations to specifics of Pydantic v1 or v2 * ✅ Update tests for OAuth2 security optional * ✅ Refactor tests for OAuth2 optional for Pydantic v2 * ✅ Refactor tests for OAuth2 security for compatibility with Pydantic v2 * 🐛 Fix location in compat layer for Pydantic v2 ModelField * ✅ Refactor tests for Pydantic v2 in tests/test_tutorial/test_bigger_applications/test_main_an_py39.py * 🐛 Add missing markers in Python 3.9 tests * ✅ Refactor tests for bigger apps for consistency with annotated ones and with support for Pydantic v2 * 🐛 Fix jsonable_encoder with new Pydantic v2 data types and Url * 🐛 Fix invalid JSON error for compatibility with Pydantic v2 * ✅ Update tests for behind_a_proxy for Pydantic v2 * ✅ Update tests for tests/test_tutorial/test_body/test_tutorial001_py310.py for Pydantic v2 * ✅ Update tests for tests/test_tutorial/test_body/test_tutorial001.py with Pydantic v2 and consistency with Python 3.10 tests * ✅ Fix tests for tutorial/body_fields for Pydantic v2 * ✅ Refactor tests for tutorial/body_multiple_params with Pydantic v2 * ✅ Update tests for tutorial/body_nested_models for Pydantic v2 * ✅ Update tests for tutorial/body_updates for Pydantic v2 * ✅ Update test for tutorial/cookie_params for Pydantic v2 * ✅ Fix tests for tests/test_tutorial/test_custom_request_and_route/test_tutorial002.py for Pydantic v2 * ✅ Update tests for tutorial/dataclasses for Pydantic v2 * ✅ Update tests for tutorial/dependencies for Pydantic v2 * ✅ Update tests for tutorial/extra_data_types for Pydantic v2 * ✅ Update tests for tutorial/handling_errors for Pydantic v2 * ✅ Fix test markers for Python 3.9 * ✅ Update tests for tutorial/header_params for Pydantic v2 * ✅ Update tests for Pydantic v2 in tests/test_tutorial/test_openapi_callbacks/test_tutorial001.py * ✅ Fix extra tests for Pydantic v2 * ✅ Refactor test for parameters, to later fix Pydantic v2 * ✅ Update tests for tutorial/query_params for Pydantic v2 * ♻️ Update examples in docs to use new pattern instead of the old regex * ✅ Fix several tests for Pydantic v2 * ✅ Update and fix test for ResponseValidationError * 🐛 Fix check for sequences vs scalars, include bytes as scalar * 🐛 Fix check for complex data types, include UploadFile * 🐛 Add list to sequence annotation types * 🐛 Fix checks for uploads and add utils to find if an annotation is an upload (or bytes) * ✨ Add UnionType and NoneType to compat layer * ✅ Update tests for request_files for compatibility with Pydantic v2 and consistency with other tests * ✅ Fix testsw for request_forms for Pydantic v2 * ✅ Fix tests for request_forms_and_files for Pydantic v2 * ✅ Fix tests in tutorial/security for compatibility with Pydantic v2 * ⬆️ Upgrade required version of email_validator * ✅ Fix tests for params repr * ✅ Add Pydantic v2 pytest markers * Use match_pydantic_error_url * 🎨 [pre-commit.ci] Auto format from pre-commit.com hooks * Use field_serializer instead of encoders in some tests * Show Undefined as ... in repr * Mark custom encoders test with xfail * Update test to reflect new serialization of Decimal as str * Use `model_validate` instead of `from_orm` * Update JSON schema to reflect required nullable * Add dirty-equals to pyproject.toml * Fix locs and error creation for use with pydantic 2.0a4 * Use the type adapter for serialization. This is hacky. * 🎨 [pre-commit.ci] Auto format from pre-commit.com hooks * ✅ Refactor test_multi_body_errors for compatibility with Pydantic v1 and v2 * ✅ Refactor test_custom_encoder for Pydantic v1 and v2 * ✅ Set input to None for now, for compatibility with current tests * 🐛 Fix passing serialization params to model field when handling the response * ♻️ Refactor exceptions to not depend on Pydantic ValidationError class * ♻️ Revert/refactor params to simplify repr * ✅ Tweak tests for custom class encoders for Pydantic v1 and v2 * ✅ Tweak tests for jsonable_encoder for Pydantic v1 and v2 * ✅ Tweak test for compatibility with Pydantic v1 and v2 * 🐛 Fix filtering data with subclasses * 🐛 Workaround examples in OpenAPI schema * ✅ Add skip marker for SQL tutorial, needs to be updated either way * ✅ Update test for broken JSON * ✅ Fix test for broken JSON * ✅ Update tests for timedeltas * ✅ Fix test for plain text validation errors * ✅ Add markers for Pydantic v1 exclusive tests (for now) * ✅ Update test for path_params with enums for compatibility with Pydantic v1 and v2 * ✅ Update tests for extra examples in OpenAPI * ✅ Fix tests for response_model with compatibility with Pydantic v1 and v2 * 🐛 Fix required double serialization for different types of models * ✅ Fix tests for response model with compatibility with new Pydantic v2 * 🐛 Import Undefined from compat layer * ✅ Fix tests for response_model for Pydantic v2 * ✅ Fix tests for schema_extra for Pydantic v2 * ✅ Add markers and update tests for Pydantic v2 * 💡 Comment out logic for double encoding that breaks other usecases * ✅ Update errors for int parsing * ♻️ Refactor re-enabling compatibility for Pydantic v1 * ♻️ Refactor OpenAPI utils to re-enable support for Pydantic v1 * ♻️ Refactor dependencies/utils and _compat for compatibility with Pydantic v1 * 🐛 Fix and tweak compatibility with Pydantic v1 and v2 in dependencies/utils * ✅ Tweak tests and examples for Pydantic v1 * ♻️ Tweak call to ModelField.validate for compatibility with Pydantic v1 * ✨ Use new global override TypeAdapter from_attributes * ✅ Update tests after updating from_attributes * 🔧 Update pytest config to avoid collecting tests from docs, useful for editor-integrated tests * ✅ Add test for data filtering, including inheritance and models in fields or lists of models * ♻️ Make OpenAPI models compatible with both Pydantic v1 and v2 * ♻️ Fix compatibility for Pydantic v1 and v2 in jsonable_encoder * ♻️ Fix compatibility in params with Pydantic v1 and v2 * ♻️ Fix compatibility when creating a FieldInfo in Pydantic v1 and v2 in utils.py * ♻️ Fix generation of flat_models and JSON Schema definitions in _compat.py for Pydantic v1 and v2 * ♻️ Update handling of ErrorWrappers for Pydantic v1 * ♻️ Refactor checks and handling of types an sequences * ♻️ Refactor and cleanup comments with compatibility for Pydantic v1 and v2 * ♻️ Update UploadFile for compatibility with both Pydantic v1 and v2 * 🔥 Remove commented out unneeded code * 🐛 Fix mock of get_annotation_from_field_info for Pydantic v1 and v2 * 🐛 Fix params with compatibility for Pydantic v1 and v2, with schemas and new pattern vs regex * 🐛 Fix check if field is sequence for Pydantic v1 * ✅ Fix tests for custom_schema_fields, for compatibility with Pydantic v1 and v2 * ✅ Simplify and fix tests for jsonable_encoder with compatibility for Pydantic v1 and v2 * ✅ Fix tests for orm_mode with Pydantic v1 and compatibility with Pydantic v2 * ♻️ Refactor logic for normalizing Pydantic v1 ErrorWrappers * ♻️ Workaround for params with examples, before defining what to deprecate in Pydantic v1 and v2 for examples with JSON Schema vs OpenAPI * ✅ Fix tests for Pydantic v1 and v2 for response_by_alias * ✅ Fix test for schema_extra with compatibility with Pydantic v1 and v2 * ♻️ Tweak error regeneration with loc * ♻️ Update error handling and serializationwith compatibility for Pydantic v1 and v2 * ♻️ Re-enable custom encoders for Pydantic v1 * ♻️ Update ErrorWrapper reserialization in Pydantic v1, do it outside of FastAPI ValidationExceptions * ✅ Update test for filter_submodel, re-structure to simplify testing while keeping division of Pydantic v1 and v2 * ✅ Refactor Pydantic v1 only test that requires modifying environment variables * 🔥 Update test for plaintext error responses, for Pydantic v1 and v2 * ⏪️ Revert changes in DB tutorial to use Pydantic v1 (the new guide will have SQLModel) * ✅ Mark current SQL DB tutorial tests as Pydantic only * ♻️ Update datastructures for compatibility with Pydantic v1, not requiring pydantic-core * ♻️ Update encoders.py for compatibility with Pydantic v1 * ⏪️ Revert changes to Peewee, the docs for that are gonna live in a new HowTo section, not in the main tutorials * ♻️ Simplify response body kwargs generation * 🔥 Clean up comments * 🔥 Clean some tests and comments * ✅ Refactor tests to match new Pydantic error string URLs * ✅ Refactor tests for recursive models for Pydantic v1 and v2 * ✅ Update tests for Peewee, re-enable, Pydantic-v1-only * ♻️ Update FastAPI params to take regex and pattern arguments * ⏪️ Revert tutorial examples for pattern, it will be done in a subsequent PR * ⏪️ Revert changes in schema extra examples, it will be added later in a docs-specific PR * 💡 Add TODO comment to document str validations with pattern * 🔥 Remove unneeded comment * 📌 Upgrade Pydantic pin dependency * ⬆️ Upgrade email_validator dependency * 🐛 Tweak type annotations in _compat.py * 🔇 Tweak mypy errors for compat, for Pydantic v1 re-imports * 🐛 Tweak and fix type annotations * ➕ Update requirements-test.txt, re-add dirty-equals * 🔥 Remove unnecessary config * 🐛 Tweak type annotations * 🔥 Remove unnecessary type in dependencies/utils.py * 💡 Update comment in routing.py --------- Co-authored-by: David Montague <35119617+dmontagu@users.noreply.github.com> Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> * 👷 Add CI for both Pydantic v1 and v2 (#9688) * 👷 Test and install Pydantic v1 and v2 in CI * 💚 Tweak CI config for Pydantic v1 and v2 * 💚 Fix Pydantic v2 specification in CI * 🐛 Fix type annotations for compatibility with Python 3.7 * 💚 Install Pydantic v2 for lints * 🐛 Fix type annotations for Pydantic v2 * 💚 Re-use test cache for lint * ♻️ Refactor internals for test coverage and performance (#9691) * ♻️ Tweak import of Annotated from typing_extensions, they are installed anyway * ♻️ Refactor _compat to define functions for Pydantic v1 or v2 once instead of checking inside * ✅ Add test for UploadFile for Pydantic v2 * ♻️ Refactor types and remove logic for impossible cases * ✅ Add missing tests from test refactor for path params * ✅ Add tests for new decimal encoder * 💡 Add TODO comment for decimals in encoders * 🔥 Remove unneeded dummy function * 🔥 Remove section of code in field_annotation_is_scalar covered by sub-call to field_annotation_is_complex * ♻️ Refactor and tweak variables and types in _compat * ✅ Add tests for corner cases and compat with Pydantic v1 and v2 * ♻️ Refactor type annotations * 🔖 Release version 0.100.0-beta1 * ♻️ Refactor parts that use optional requirements to make them compatible with installations without them (#9707) * ♻️ Refactor parts that use optional requirements to make them compatible with installations without them * ♻️ Update JSON Schema for email field without email-validator installed * 🐛 Fix support for Pydantic v2.0, small changes in their final release (#9771) Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> Co-authored-by: Sebastián Ramírez <tiangolo@gmail.com> * 🔖 Release version 0.100.0-beta2 * ✨ OpenAPI 3.1.0 with Pydantic v2, merge `master` (#9773) * ➕ Add dirty-equals as a testing dependency (#9778) ➕ Add dirty-equals as a testing dependency, it seems it got lsot at some point * 🔀 Merge master, fix valid JSON Schema accepting bools (#9782) * ⏪️ Revert usage of custom logic for TypeAdapter JSON Schema, solved on the Pydantic side (#9787) ⏪️ Revert usage of custom logic for TypeAdapter JSON Schema, solved on Pydantic side * ♻️ Deprecate parameter `regex`, use `pattern` instead (#9786) * 📝 Update docs to deprecate regex, recommend pattern * ♻️ Update examples to use new pattern instead of regex * 📝 Add new example with deprecated regex * ♻️ Add deprecation notes and warnings for regex * ✅ Add tests for regex deprecation * ✅ Update tests for compatibility with Pydantic v1 * ✨ Update docs to use Pydantic v2 settings and add note and example about v1 (#9788) * ➕ Add pydantic-settings to all extras * 📝 Update docs for Pydantic settings * 📝 Update Settings source examples to use Pydantic v2, and add a Pydantic v1 version * ✅ Add tests for settings with Pydantic v1 and v2 * 🔥 Remove solved TODO comment * ♻️ Update conditional OpenAPI to use new Pydantic v2 settings * ✅ Update tests to import Annotated from typing_extensions for Python < 3.9 (#9795) * ➕ Add pydantic-extra-types to fastapi[extra] * ➕ temp: Install Pydantic from source to test JSON Schema metadata fixes (#9777) * ➕ Install Pydantic from source, from branch for JSON Schema with metadata * ➕ Update dependencies, install Pydantic main * ➕ Fix dependency URL for Pydantic from source * ➕ Add pydantic-settings for test requirements * 💡 Add TODO comments to re-enable Pydantic main (not from source) (#9796) * ✨ Add new Pydantic Field param options to Query, Cookie, Body, etc. (#9797) * 📝 Add docs for Pydantic v2 for `docs/en/docs/advanced/path-operation-advanced-configuration.md` (#9798) * 📝 Update docs in examples for settings with Pydantic v2 (#9799) * 📝 Update JSON Schema `examples` docs with Pydantic v2 (#9800) * ♻️ Use new Pydantic v2 JSON Schema generator (#9813) Co-authored-by: David Montague <35119617+dmontagu@users.noreply.github.com> * ♻️ Tweak type annotations and Pydantic version range (#9801) * 📌 Re-enable GA Pydantic, for v2, require minimum 2.0.2 (#9814) * 🔖 Release version 0.100.0-beta3 * 🔥 Remove duplicate type declaration from merge conflicts (#9832) * 👷♂️ Run tests with Pydantic v2 GA (#9830) 👷 Run tests for Pydantic v2 GA * 📝 Add notes to docs expecting Pydantic v2 and future updates (#9833) * 📝 Update index with new extras * 📝 Update release notes --------- Co-authored-by: David Montague <35119617+dmontagu@users.noreply.github.com> Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> Co-authored-by: Pastukhov Nikita <diementros@yandex.ru>
612 lines
17 KiB
Python
612 lines
17 KiB
Python
from enum import Enum
|
|
from typing import Any, Callable, Dict, Iterable, List, Optional, Set, Type, Union
|
|
|
|
from fastapi._compat import (
|
|
PYDANTIC_V2,
|
|
CoreSchema,
|
|
GetJsonSchemaHandler,
|
|
JsonSchemaValue,
|
|
_model_rebuild,
|
|
general_plain_validator_function,
|
|
)
|
|
from fastapi.logger import logger
|
|
from pydantic import AnyUrl, BaseModel, Field
|
|
from typing_extensions import Annotated, Literal
|
|
from typing_extensions import deprecated as typing_deprecated
|
|
|
|
try:
|
|
import email_validator
|
|
|
|
assert email_validator # make autoflake ignore the unused import
|
|
from pydantic import EmailStr
|
|
except ImportError: # pragma: no cover
|
|
|
|
class EmailStr(str): # type: ignore
|
|
@classmethod
|
|
def __get_validators__(cls) -> Iterable[Callable[..., Any]]:
|
|
yield cls.validate
|
|
|
|
@classmethod
|
|
def validate(cls, v: Any) -> str:
|
|
logger.warning(
|
|
"email-validator not installed, email fields will be treated as str.\n"
|
|
"To install, run: pip install email-validator"
|
|
)
|
|
return str(v)
|
|
|
|
@classmethod
|
|
def _validate(cls, __input_value: Any, _: Any) -> str:
|
|
logger.warning(
|
|
"email-validator not installed, email fields will be treated as str.\n"
|
|
"To install, run: pip install email-validator"
|
|
)
|
|
return str(__input_value)
|
|
|
|
@classmethod
|
|
def __get_pydantic_json_schema__(
|
|
cls, core_schema: CoreSchema, handler: GetJsonSchemaHandler
|
|
) -> JsonSchemaValue:
|
|
return {"type": "string", "format": "email"}
|
|
|
|
@classmethod
|
|
def __get_pydantic_core_schema__(
|
|
cls, source: Type[Any], handler: Callable[[Any], CoreSchema]
|
|
) -> CoreSchema:
|
|
return general_plain_validator_function(cls._validate)
|
|
|
|
|
|
class Contact(BaseModel):
|
|
name: Optional[str] = None
|
|
url: Optional[AnyUrl] = None
|
|
email: Optional[EmailStr] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class License(BaseModel):
|
|
name: str
|
|
identifier: Optional[str] = None
|
|
url: Optional[AnyUrl] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class Info(BaseModel):
|
|
title: str
|
|
summary: Optional[str] = None
|
|
description: Optional[str] = None
|
|
termsOfService: Optional[str] = None
|
|
contact: Optional[Contact] = None
|
|
license: Optional[License] = None
|
|
version: str
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class ServerVariable(BaseModel):
|
|
enum: Annotated[Optional[List[str]], Field(min_length=1)] = None
|
|
default: str
|
|
description: Optional[str] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class Server(BaseModel):
|
|
url: Union[AnyUrl, str]
|
|
description: Optional[str] = None
|
|
variables: Optional[Dict[str, ServerVariable]] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class Reference(BaseModel):
|
|
ref: str = Field(alias="$ref")
|
|
|
|
|
|
class Discriminator(BaseModel):
|
|
propertyName: str
|
|
mapping: Optional[Dict[str, str]] = None
|
|
|
|
|
|
class XML(BaseModel):
|
|
name: Optional[str] = None
|
|
namespace: Optional[str] = None
|
|
prefix: Optional[str] = None
|
|
attribute: Optional[bool] = None
|
|
wrapped: Optional[bool] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class ExternalDocumentation(BaseModel):
|
|
description: Optional[str] = None
|
|
url: AnyUrl
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class Schema(BaseModel):
|
|
# Ref: JSON Schema 2020-12: https://json-schema.org/draft/2020-12/json-schema-core.html#name-the-json-schema-core-vocabu
|
|
# Core Vocabulary
|
|
schema_: Optional[str] = Field(default=None, alias="$schema")
|
|
vocabulary: Optional[str] = Field(default=None, alias="$vocabulary")
|
|
id: Optional[str] = Field(default=None, alias="$id")
|
|
anchor: Optional[str] = Field(default=None, alias="$anchor")
|
|
dynamicAnchor: Optional[str] = Field(default=None, alias="$dynamicAnchor")
|
|
ref: Optional[str] = Field(default=None, alias="$ref")
|
|
dynamicRef: Optional[str] = Field(default=None, alias="$dynamicRef")
|
|
defs: Optional[Dict[str, "SchemaOrBool"]] = Field(default=None, alias="$defs")
|
|
comment: Optional[str] = Field(default=None, alias="$comment")
|
|
# Ref: JSON Schema 2020-12: https://json-schema.org/draft/2020-12/json-schema-core.html#name-a-vocabulary-for-applying-s
|
|
# A Vocabulary for Applying Subschemas
|
|
allOf: Optional[List["SchemaOrBool"]] = None
|
|
anyOf: Optional[List["SchemaOrBool"]] = None
|
|
oneOf: Optional[List["SchemaOrBool"]] = None
|
|
not_: Optional["SchemaOrBool"] = Field(default=None, alias="not")
|
|
if_: Optional["SchemaOrBool"] = Field(default=None, alias="if")
|
|
then: Optional["SchemaOrBool"] = None
|
|
else_: Optional["SchemaOrBool"] = Field(default=None, alias="else")
|
|
dependentSchemas: Optional[Dict[str, "SchemaOrBool"]] = None
|
|
prefixItems: Optional[List["SchemaOrBool"]] = None
|
|
# TODO: uncomment and remove below when deprecating Pydantic v1
|
|
# It generales a list of schemas for tuples, before prefixItems was available
|
|
# items: Optional["SchemaOrBool"] = None
|
|
items: Optional[Union["SchemaOrBool", List["SchemaOrBool"]]] = None
|
|
contains: Optional["SchemaOrBool"] = None
|
|
properties: Optional[Dict[str, "SchemaOrBool"]] = None
|
|
patternProperties: Optional[Dict[str, "SchemaOrBool"]] = None
|
|
additionalProperties: Optional["SchemaOrBool"] = None
|
|
propertyNames: Optional["SchemaOrBool"] = None
|
|
unevaluatedItems: Optional["SchemaOrBool"] = None
|
|
unevaluatedProperties: Optional["SchemaOrBool"] = None
|
|
# Ref: JSON Schema Validation 2020-12: https://json-schema.org/draft/2020-12/json-schema-validation.html#name-a-vocabulary-for-structural
|
|
# A Vocabulary for Structural Validation
|
|
type: Optional[str] = None
|
|
enum: Optional[List[Any]] = None
|
|
const: Optional[Any] = None
|
|
multipleOf: Optional[float] = Field(default=None, gt=0)
|
|
maximum: Optional[float] = None
|
|
exclusiveMaximum: Optional[float] = None
|
|
minimum: Optional[float] = None
|
|
exclusiveMinimum: Optional[float] = None
|
|
maxLength: Optional[int] = Field(default=None, ge=0)
|
|
minLength: Optional[int] = Field(default=None, ge=0)
|
|
pattern: Optional[str] = None
|
|
maxItems: Optional[int] = Field(default=None, ge=0)
|
|
minItems: Optional[int] = Field(default=None, ge=0)
|
|
uniqueItems: Optional[bool] = None
|
|
maxContains: Optional[int] = Field(default=None, ge=0)
|
|
minContains: Optional[int] = Field(default=None, ge=0)
|
|
maxProperties: Optional[int] = Field(default=None, ge=0)
|
|
minProperties: Optional[int] = Field(default=None, ge=0)
|
|
required: Optional[List[str]] = None
|
|
dependentRequired: Optional[Dict[str, Set[str]]] = None
|
|
# Ref: JSON Schema Validation 2020-12: https://json-schema.org/draft/2020-12/json-schema-validation.html#name-vocabularies-for-semantic-c
|
|
# Vocabularies for Semantic Content With "format"
|
|
format: Optional[str] = None
|
|
# Ref: JSON Schema Validation 2020-12: https://json-schema.org/draft/2020-12/json-schema-validation.html#name-a-vocabulary-for-the-conten
|
|
# A Vocabulary for the Contents of String-Encoded Data
|
|
contentEncoding: Optional[str] = None
|
|
contentMediaType: Optional[str] = None
|
|
contentSchema: Optional["SchemaOrBool"] = None
|
|
# Ref: JSON Schema Validation 2020-12: https://json-schema.org/draft/2020-12/json-schema-validation.html#name-a-vocabulary-for-basic-meta
|
|
# A Vocabulary for Basic Meta-Data Annotations
|
|
title: Optional[str] = None
|
|
description: Optional[str] = None
|
|
default: Optional[Any] = None
|
|
deprecated: Optional[bool] = None
|
|
readOnly: Optional[bool] = None
|
|
writeOnly: Optional[bool] = None
|
|
examples: Optional[List[Any]] = None
|
|
# Ref: OpenAPI 3.1.0: https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#schema-object
|
|
# Schema Object
|
|
discriminator: Optional[Discriminator] = None
|
|
xml: Optional[XML] = None
|
|
externalDocs: Optional[ExternalDocumentation] = None
|
|
example: Annotated[
|
|
Optional[Any],
|
|
typing_deprecated(
|
|
"Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, "
|
|
"although still supported. Use examples instead."
|
|
),
|
|
] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
# Ref: https://json-schema.org/draft/2020-12/json-schema-core.html#name-json-schema-documents
|
|
# A JSON Schema MUST be an object or a boolean.
|
|
SchemaOrBool = Union[Schema, bool]
|
|
|
|
|
|
class Example(BaseModel):
|
|
summary: Optional[str] = None
|
|
description: Optional[str] = None
|
|
value: Optional[Any] = None
|
|
externalValue: Optional[AnyUrl] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class ParameterInType(Enum):
|
|
query = "query"
|
|
header = "header"
|
|
path = "path"
|
|
cookie = "cookie"
|
|
|
|
|
|
class Encoding(BaseModel):
|
|
contentType: Optional[str] = None
|
|
headers: Optional[Dict[str, Union["Header", Reference]]] = None
|
|
style: Optional[str] = None
|
|
explode: Optional[bool] = None
|
|
allowReserved: Optional[bool] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class MediaType(BaseModel):
|
|
schema_: Optional[Union[Schema, Reference]] = Field(default=None, alias="schema")
|
|
example: Optional[Any] = None
|
|
examples: Optional[Dict[str, Union[Example, Reference]]] = None
|
|
encoding: Optional[Dict[str, Encoding]] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class ParameterBase(BaseModel):
|
|
description: Optional[str] = None
|
|
required: Optional[bool] = None
|
|
deprecated: Optional[bool] = None
|
|
# Serialization rules for simple scenarios
|
|
style: Optional[str] = None
|
|
explode: Optional[bool] = None
|
|
allowReserved: Optional[bool] = None
|
|
schema_: Optional[Union[Schema, Reference]] = Field(default=None, alias="schema")
|
|
example: Optional[Any] = None
|
|
examples: Optional[Dict[str, Union[Example, Reference]]] = None
|
|
# Serialization rules for more complex scenarios
|
|
content: Optional[Dict[str, MediaType]] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class Parameter(ParameterBase):
|
|
name: str
|
|
in_: ParameterInType = Field(alias="in")
|
|
|
|
|
|
class Header(ParameterBase):
|
|
pass
|
|
|
|
|
|
class RequestBody(BaseModel):
|
|
description: Optional[str] = None
|
|
content: Dict[str, MediaType]
|
|
required: Optional[bool] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class Link(BaseModel):
|
|
operationRef: Optional[str] = None
|
|
operationId: Optional[str] = None
|
|
parameters: Optional[Dict[str, Union[Any, str]]] = None
|
|
requestBody: Optional[Union[Any, str]] = None
|
|
description: Optional[str] = None
|
|
server: Optional[Server] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class Response(BaseModel):
|
|
description: str
|
|
headers: Optional[Dict[str, Union[Header, Reference]]] = None
|
|
content: Optional[Dict[str, MediaType]] = None
|
|
links: Optional[Dict[str, Union[Link, Reference]]] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class Operation(BaseModel):
|
|
tags: Optional[List[str]] = None
|
|
summary: Optional[str] = None
|
|
description: Optional[str] = None
|
|
externalDocs: Optional[ExternalDocumentation] = None
|
|
operationId: Optional[str] = None
|
|
parameters: Optional[List[Union[Parameter, Reference]]] = None
|
|
requestBody: Optional[Union[RequestBody, Reference]] = None
|
|
# Using Any for Specification Extensions
|
|
responses: Optional[Dict[str, Union[Response, Any]]] = None
|
|
callbacks: Optional[Dict[str, Union[Dict[str, "PathItem"], Reference]]] = None
|
|
deprecated: Optional[bool] = None
|
|
security: Optional[List[Dict[str, List[str]]]] = None
|
|
servers: Optional[List[Server]] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class PathItem(BaseModel):
|
|
ref: Optional[str] = Field(default=None, alias="$ref")
|
|
summary: Optional[str] = None
|
|
description: Optional[str] = None
|
|
get: Optional[Operation] = None
|
|
put: Optional[Operation] = None
|
|
post: Optional[Operation] = None
|
|
delete: Optional[Operation] = None
|
|
options: Optional[Operation] = None
|
|
head: Optional[Operation] = None
|
|
patch: Optional[Operation] = None
|
|
trace: Optional[Operation] = None
|
|
servers: Optional[List[Server]] = None
|
|
parameters: Optional[List[Union[Parameter, Reference]]] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class SecuritySchemeType(Enum):
|
|
apiKey = "apiKey"
|
|
http = "http"
|
|
oauth2 = "oauth2"
|
|
openIdConnect = "openIdConnect"
|
|
|
|
|
|
class SecurityBase(BaseModel):
|
|
type_: SecuritySchemeType = Field(alias="type")
|
|
description: Optional[str] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class APIKeyIn(Enum):
|
|
query = "query"
|
|
header = "header"
|
|
cookie = "cookie"
|
|
|
|
|
|
class APIKey(SecurityBase):
|
|
type_: SecuritySchemeType = Field(default=SecuritySchemeType.apiKey, alias="type")
|
|
in_: APIKeyIn = Field(alias="in")
|
|
name: str
|
|
|
|
|
|
class HTTPBase(SecurityBase):
|
|
type_: SecuritySchemeType = Field(default=SecuritySchemeType.http, alias="type")
|
|
scheme: str
|
|
|
|
|
|
class HTTPBearer(HTTPBase):
|
|
scheme: Literal["bearer"] = "bearer"
|
|
bearerFormat: Optional[str] = None
|
|
|
|
|
|
class OAuthFlow(BaseModel):
|
|
refreshUrl: Optional[str] = None
|
|
scopes: Dict[str, str] = {}
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class OAuthFlowImplicit(OAuthFlow):
|
|
authorizationUrl: str
|
|
|
|
|
|
class OAuthFlowPassword(OAuthFlow):
|
|
tokenUrl: str
|
|
|
|
|
|
class OAuthFlowClientCredentials(OAuthFlow):
|
|
tokenUrl: str
|
|
|
|
|
|
class OAuthFlowAuthorizationCode(OAuthFlow):
|
|
authorizationUrl: str
|
|
tokenUrl: str
|
|
|
|
|
|
class OAuthFlows(BaseModel):
|
|
implicit: Optional[OAuthFlowImplicit] = None
|
|
password: Optional[OAuthFlowPassword] = None
|
|
clientCredentials: Optional[OAuthFlowClientCredentials] = None
|
|
authorizationCode: Optional[OAuthFlowAuthorizationCode] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class OAuth2(SecurityBase):
|
|
type_: SecuritySchemeType = Field(default=SecuritySchemeType.oauth2, alias="type")
|
|
flows: OAuthFlows
|
|
|
|
|
|
class OpenIdConnect(SecurityBase):
|
|
type_: SecuritySchemeType = Field(
|
|
default=SecuritySchemeType.openIdConnect, alias="type"
|
|
)
|
|
openIdConnectUrl: str
|
|
|
|
|
|
SecurityScheme = Union[APIKey, HTTPBase, OAuth2, OpenIdConnect, HTTPBearer]
|
|
|
|
|
|
class Components(BaseModel):
|
|
schemas: Optional[Dict[str, Union[Schema, Reference]]] = None
|
|
responses: Optional[Dict[str, Union[Response, Reference]]] = None
|
|
parameters: Optional[Dict[str, Union[Parameter, Reference]]] = None
|
|
examples: Optional[Dict[str, Union[Example, Reference]]] = None
|
|
requestBodies: Optional[Dict[str, Union[RequestBody, Reference]]] = None
|
|
headers: Optional[Dict[str, Union[Header, Reference]]] = None
|
|
securitySchemes: Optional[Dict[str, Union[SecurityScheme, Reference]]] = None
|
|
links: Optional[Dict[str, Union[Link, Reference]]] = None
|
|
# Using Any for Specification Extensions
|
|
callbacks: Optional[Dict[str, Union[Dict[str, PathItem], Reference, Any]]] = None
|
|
pathItems: Optional[Dict[str, Union[PathItem, Reference]]] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class Tag(BaseModel):
|
|
name: str
|
|
description: Optional[str] = None
|
|
externalDocs: Optional[ExternalDocumentation] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
class OpenAPI(BaseModel):
|
|
openapi: str
|
|
info: Info
|
|
jsonSchemaDialect: Optional[str] = None
|
|
servers: Optional[List[Server]] = None
|
|
# Using Any for Specification Extensions
|
|
paths: Optional[Dict[str, Union[PathItem, Any]]] = None
|
|
webhooks: Optional[Dict[str, Union[PathItem, Reference]]] = None
|
|
components: Optional[Components] = None
|
|
security: Optional[List[Dict[str, List[str]]]] = None
|
|
tags: Optional[List[Tag]] = None
|
|
externalDocs: Optional[ExternalDocumentation] = None
|
|
|
|
if PYDANTIC_V2:
|
|
model_config = {"extra": "allow"}
|
|
|
|
else:
|
|
|
|
class Config:
|
|
extra = "allow"
|
|
|
|
|
|
_model_rebuild(Schema)
|
|
_model_rebuild(Operation)
|
|
_model_rebuild(Encoding)
|