diff --git a/docs/en/docs/release-notes.md b/docs/en/docs/release-notes.md
index 67971820e..a15f0f735 100644
--- a/docs/en/docs/release-notes.md
+++ b/docs/en/docs/release-notes.md
@@ -7,6 +7,10 @@ hide:
## Latest Changes
+### Refactors
+
+* โป๏ธ Update tests and internals for compatibility with Pydantic >=2.10. PR [#12971](https://github.com/fastapi/fastapi/pull/12971) by [@tamird](https://github.com/tamird).
+
### Docs
* ๐ Update includes format in docs with an automated script. PR [#12950](https://github.com/fastapi/fastapi/pull/12950) by [@tiangolo](https://github.com/tiangolo).
@@ -15,6 +19,7 @@ hide:
### Translations
+* ๐ Add Korean translation for `docs/ko/docs/tutorial/query-param-models.md`. PR [#12940](https://github.com/fastapi/fastapi/pull/12940) by [@jts8257](https://github.com/jts8257).
* ๐ฅ Remove obsolete tutorial translation to Chinese for `docs/zh/docs/tutorial/sql-databases.md`, it references files that are no longer on the repo. PR [#12949](https://github.com/fastapi/fastapi/pull/12949) by [@tiangolo](https://github.com/tiangolo).
### Internal
diff --git a/docs/ko/docs/tutorial/query-param-models.md b/docs/ko/docs/tutorial/query-param-models.md
new file mode 100644
index 000000000..2ca65a331
--- /dev/null
+++ b/docs/ko/docs/tutorial/query-param-models.md
@@ -0,0 +1,68 @@
+# ์ฟผ๋ฆฌ ๋งค๊ฐ๋ณ์ ๋ชจ๋ธ
+
+์ฐ๊ด๋ ์ฟผ๋ฆฌ **๋งค๊ฐ๋ณ์** ๊ทธ๋ฃน์ด ์๋ค๋ฉด **Pydantic ๋ชจ๋ธ** ์ ์ฌ์ฉํด ์ ์ธํ ์ ์์ต๋๋ค.
+
+์ด๋ ๊ฒ ํ๋ฉด **์ฌ๋ฌ ๊ณณ**์์ **๋ชจ๋ธ์ ์ฌ์ฌ์ฉ**ํ ์ ์์ ๋ฟ๋ง ์๋๋ผ, ๋งค๊ฐ๋ณ์์ ๋ํ ๊ฒ์ฆ ๋ฐ ๋ฉํ๋ฐ์ดํฐ๋ ํ ๋ฒ์ ์ ์ธํ ์ ์์ต๋๋ค. ๐
+
+/// note | ์ฐธ๊ณ
+
+์ด ๊ธฐ๋ฅ์ FastAPI ๋ฒ์ `0.115.0`๋ถํฐ ์ ๊ณต๋ฉ๋๋ค. ๐ค
+
+///
+
+## ์ฟผ๋ฆฌ ๋งค๊ฐ๋ณ์์ Pydantic ๋ชจ๋ธ
+
+ํ์ํ **์ฟผ๋ฆฌ ๋งค๊ฐ๋ณ์**๋ฅผ **Pydantic ๋ชจ๋ธ** ์์ ์ ์ธํ ๋ค์, ๋ชจ๋ธ์ `Query`๋ก ์ ์ธํฉ๋๋ค.
+
+{* ../../docs_src/query_param_models/tutorial001_an_py310.py hl[9:13,17] *}
+
+**FastAPI**๋ ์์ฒญ์ **์ฟผ๋ฆฌ ๋งค๊ฐ๋ณ์**์์ **๊ฐ ํ๋**์ ๋ฐ์ดํฐ๋ฅผ **์ถ์ถ**ํด ์ ์ํ Pydantic ๋ชจ๋ธ๋ก ์ ๊ณตํฉ๋๋ค.
+
+## ๋ฌธ์ ํ์ธํ๊ธฐ
+
+`/docs` ๊ฒฝ๋ก์ API ๋ฌธ์์์ ๋งค๊ฐ๋ณ์๋ฅผ ํ์ธํ ์ ์์ต๋๋ค.
+
+
+

+
+
+## ์ถ๊ฐ ์ฟผ๋ฆฌ ๋งค๊ฐ๋ณ์ ๊ธ์ง
+
+๋ช๋ช์ ํน์ดํ ๊ฒฝ์ฐ์ (ํ์น ์์ง๋ง), ํ์ฉํ ์ฟผ๋ฆฌ ๋งค๊ฐ๋ณ์๋ฅผ **์ ํ**ํด์ผํ ์ ์์ต๋๋ค.
+
+Pydantic ๋ชจ๋ธ ์ค์ ์์ `extra` ํ๋๋ฅผ `forbid` ๋ก ์ค์ ํ ์ ์์ต๋๋ค.
+
+{* ../../docs_src/query_param_models/tutorial002_an_py310.py hl[10] *}
+
+๋ง์ฝ ํด๋ผ์ด์ธํธ๊ฐ ์ฟผ๋ฆฌ ๋งค๊ฐ๋ณ์๋ก **์ถ๊ฐ์ ์ธ** ๋ฐ์ดํฐ๋ฅผ ๋ณด๋ด๋ ค๊ณ ํ๋ฉด, ํด๋ผ์ด์ธํธ๋ **์๋ฌ** ์๋ต์ ๋ฐ๊ฒ ๋ฉ๋๋ค.
+
+์๋ฅผ ๋ค์ด, ์๋์ ๊ฐ์ด ๋ง์ฝ ํด๋ผ์ด์ธํธ๊ฐ `tool` ์ฟผ๋ฆฌ ๋งค๊ฐ๋ณ์์ `plumbus` ๋ผ๋ ๊ฐ์ ์ถ๊ฐํด์ ๋ณด๋ด๋ ค๊ณ ํ๋ฉด,
+
+```http
+https://example.com/items/?limit=10&tool=plumbus
+```
+
+ํด๋ผ์ด์ธํธ๋ ์ฟผ๋ฆฌ ๋งค๊ฐ๋ณ์ `tool` ์ด ํ์ฉ๋์ง ์๋๋ค๋ **์๋ฌ** ์๋ต์ ๋ฐ๊ฒ ๋ฉ๋๋ค.
+
+```json
+{
+ "detail": [
+ {
+ "type": "extra_forbidden",
+ "loc": ["query", "tool"],
+ "msg": "Extra inputs are not permitted",
+ "input": "plumbus"
+ }
+ ]
+}
+```
+
+## ์์ฝ
+
+**FastAPI** ์์ **์ฟผ๋ฆฌ ๋งค๊ฐ๋ณ์** ๋ฅผ ์ ์ธํ ๋ **Pydantic ๋ชจ๋ธ** ์ ์ฌ์ฉํ ์ ์์ต๋๋ค. ๐
+
+/// tip | ํ
+
+์คํฌ์ผ๋ฌ ๊ฒฝ๊ณ : Pydantic ๋ชจ๋ธ์ ์ฟ ํค์ ํค๋์๋ ์ ์ฉํ ์ ์์ต๋๋ค. ์ด์ ๋ํด์๋ ์ดํ ํํ ๋ฆฌ์ผ์์ ๋ค๋ฃฐ ์์ ์
๋๋ค. ๐คซ
+
+///
diff --git a/fastapi/_compat.py b/fastapi/_compat.py
index 2b4d3e725..c07e4a3b0 100644
--- a/fastapi/_compat.py
+++ b/fastapi/_compat.py
@@ -21,12 +21,10 @@ from typing import (
from fastapi.exceptions import RequestErrorModel
from fastapi.types import IncEx, ModelNameMap, UnionType
from pydantic import BaseModel, create_model
-from pydantic.version import VERSION as P_VERSION
+from pydantic.version import VERSION as PYDANTIC_VERSION
from starlette.datastructures import UploadFile
from typing_extensions import Annotated, Literal, get_args, get_origin
-# Reassign variable to make it reexported for mypy
-PYDANTIC_VERSION = P_VERSION
PYDANTIC_VERSION_MINOR_TUPLE = tuple(int(x) for x in PYDANTIC_VERSION.split(".")[:2])
PYDANTIC_V2 = PYDANTIC_VERSION_MINOR_TUPLE[0] == 2
@@ -47,6 +45,8 @@ sequence_annotation_to_type = {
sequence_types = tuple(sequence_annotation_to_type.keys())
+Url: Type[Any]
+
if PYDANTIC_V2:
from pydantic import PydanticSchemaGenerationError as PydanticSchemaGenerationError
from pydantic import TypeAdapter
diff --git a/tests/test_tutorial/test_query_params_str_validations/test_tutorial010.py b/tests/test_tutorial/test_query_params_str_validations/test_tutorial010.py
index 945cee3d2..4f52d6ff7 100644
--- a/tests/test_tutorial/test_query_params_str_validations/test_tutorial010.py
+++ b/tests/test_tutorial/test_query_params_str_validations/test_tutorial010.py
@@ -1,5 +1,6 @@
import pytest
from dirty_equals import IsDict
+from fastapi._compat import PYDANTIC_VERSION_MINOR_TUPLE
from fastapi.testclient import TestClient
@@ -107,6 +108,12 @@ def test_openapi_schema(client: TestClient):
],
"title": "Query string",
"description": "Query string for the items to search in the database that have a good match",
+ # See https://github.com/pydantic/pydantic/blob/80353c29a824c55dea4667b328ba8f329879ac9f/tests/test_fastapi.sh#L25-L34.
+ **(
+ {"deprecated": True}
+ if PYDANTIC_VERSION_MINOR_TUPLE >= (2, 10)
+ else {}
+ ),
}
)
| IsDict(
diff --git a/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an.py b/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an.py
index 23951a9aa..5daca1e70 100644
--- a/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an.py
+++ b/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an.py
@@ -1,5 +1,6 @@
import pytest
from dirty_equals import IsDict
+from fastapi._compat import PYDANTIC_VERSION_MINOR_TUPLE
from fastapi.testclient import TestClient
@@ -107,6 +108,12 @@ def test_openapi_schema(client: TestClient):
],
"title": "Query string",
"description": "Query string for the items to search in the database that have a good match",
+ # See https://github.com/pydantic/pydantic/blob/80353c29a824c55dea4667b328ba8f329879ac9f/tests/test_fastapi.sh#L25-L34.
+ **(
+ {"deprecated": True}
+ if PYDANTIC_VERSION_MINOR_TUPLE >= (2, 10)
+ else {}
+ ),
}
)
| IsDict(
diff --git a/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an_py310.py b/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an_py310.py
index 2968af563..89da4d82e 100644
--- a/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an_py310.py
+++ b/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an_py310.py
@@ -1,5 +1,6 @@
import pytest
from dirty_equals import IsDict
+from fastapi._compat import PYDANTIC_VERSION_MINOR_TUPLE
from fastapi.testclient import TestClient
from ...utils import needs_py310
@@ -114,6 +115,12 @@ def test_openapi_schema(client: TestClient):
],
"title": "Query string",
"description": "Query string for the items to search in the database that have a good match",
+ # See https://github.com/pydantic/pydantic/blob/80353c29a824c55dea4667b328ba8f329879ac9f/tests/test_fastapi.sh#L25-L34.
+ **(
+ {"deprecated": True}
+ if PYDANTIC_VERSION_MINOR_TUPLE >= (2, 10)
+ else {}
+ ),
}
)
| IsDict(
diff --git a/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an_py39.py b/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an_py39.py
index 534ba8759..f5f692b06 100644
--- a/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an_py39.py
+++ b/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_an_py39.py
@@ -1,5 +1,6 @@
import pytest
from dirty_equals import IsDict
+from fastapi._compat import PYDANTIC_VERSION_MINOR_TUPLE
from fastapi.testclient import TestClient
from ...utils import needs_py39
@@ -114,6 +115,12 @@ def test_openapi_schema(client: TestClient):
],
"title": "Query string",
"description": "Query string for the items to search in the database that have a good match",
+ # See https://github.com/pydantic/pydantic/blob/80353c29a824c55dea4667b328ba8f329879ac9f/tests/test_fastapi.sh#L25-L34.
+ **(
+ {"deprecated": True}
+ if PYDANTIC_VERSION_MINOR_TUPLE >= (2, 10)
+ else {}
+ ),
}
)
| IsDict(
diff --git a/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_py310.py b/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_py310.py
index 886bceca2..5b62c969f 100644
--- a/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_py310.py
+++ b/tests/test_tutorial/test_query_params_str_validations/test_tutorial010_py310.py
@@ -1,5 +1,6 @@
import pytest
from dirty_equals import IsDict
+from fastapi._compat import PYDANTIC_VERSION_MINOR_TUPLE
from fastapi.testclient import TestClient
from ...utils import needs_py310
@@ -114,6 +115,12 @@ def test_openapi_schema(client: TestClient):
],
"title": "Query string",
"description": "Query string for the items to search in the database that have a good match",
+ # See https://github.com/pydantic/pydantic/blob/80353c29a824c55dea4667b328ba8f329879ac9f/tests/test_fastapi.sh#L25-L34.
+ **(
+ {"deprecated": True}
+ if PYDANTIC_VERSION_MINOR_TUPLE >= (2, 10)
+ else {}
+ ),
}
)
| IsDict(