From 6dc975da9dd4804c1fa3a9c8533bd339b63eee0e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sebasti=C3=A1n=20Ram=C3=ADrez?= Date: Tue, 20 Jun 2023 18:22:03 +0200 Subject: [PATCH] =?UTF-8?q?=E2=99=BB=EF=B8=8F=20Refactor=20parts=20that=20?= =?UTF-8?q?use=20optional=20requirements=20to=20make=20them=20compatible?= =?UTF-8?q?=20with=20installations=20without=20them=20(#9707)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * ♻️ Refactor parts that use optional requirements to make them compatible with installations without them * ♻️ Update JSON Schema for email field without email-validator installed --- fastapi/openapi/models.py | 33 ++++++++++++++++++++++++++++++--- fastapi/utils.py | 5 +++-- 2 files changed, 33 insertions(+), 5 deletions(-) diff --git a/fastapi/openapi/models.py b/fastapi/openapi/models.py index 7284fa5ba..7176e59dd 100644 --- a/fastapi/openapi/models.py +++ b/fastapi/openapi/models.py @@ -1,7 +1,14 @@ from enum import Enum -from typing import Any, Callable, Dict, Iterable, List, Optional, Union - -from fastapi._compat import PYDANTIC_V2, _model_rebuild +from typing import Any, Callable, Dict, Iterable, List, Optional, 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 Literal @@ -26,6 +33,26 @@ except ImportError: # pragma: no cover ) 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 diff --git a/fastapi/utils.py b/fastapi/utils.py index 18ae2365d..2efe7f15a 100644 --- a/fastapi/utils.py +++ b/fastapi/utils.py @@ -15,7 +15,6 @@ from typing import ( from weakref import WeakKeyDictionary import fastapi -from dirty_equals import IsStr from fastapi._compat import ( PYDANTIC_V2, BaseConfig, @@ -219,5 +218,7 @@ def get_value_or_default( return first_item -def match_pydantic_error_url(error_type: str) -> IsStr: +def match_pydantic_error_url(error_type: str) -> Any: + from dirty_equals import IsStr + return IsStr(regex=rf"^https://errors\.pydantic\.dev/.*/v/{error_type}")