diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml
index 767ef8d9e..05c33a608 100644
--- a/.pre-commit-config.yaml
+++ b/.pre-commit-config.yaml
@@ -14,7 +14,7 @@ repos:
- id: end-of-file-fixer
- id: trailing-whitespace
- repo: https://github.com/astral-sh/ruff-pre-commit
- rev: v0.7.4
+ rev: v0.9.4
hooks:
- id: ruff
args:
diff --git a/docs/en/docs/release-notes.md b/docs/en/docs/release-notes.md
index 72c47d4fb..c900dc918 100644
--- a/docs/en/docs/release-notes.md
+++ b/docs/en/docs/release-notes.md
@@ -7,8 +7,16 @@ hide:
## Latest Changes
+### Translations
+
+* ๐ Add Korean translation for `docs/ko/docs/tutorial/security/oauth2-jwt.md`. PR [#13333](https://github.com/fastapi/fastapi/pull/13333) by [@yes0ng](https://github.com/yes0ng).
+* ๐ Add Vietnamese translation for `docs/vi/docs/deployment/cloud.md`. PR [#13407](https://github.com/fastapi/fastapi/pull/13407) by [@ptt3199](https://github.com/ptt3199).
+
### Internal
+* โฌ Bump sqlmodel from 0.0.22 to 0.0.23. PR [#13437](https://github.com/fastapi/fastapi/pull/13437) by [@dependabot[bot]](https://github.com/apps/dependabot).
+* โฌ Bump black from 24.10.0 to 25.1.0. PR [#13436](https://github.com/fastapi/fastapi/pull/13436) by [@dependabot[bot]](https://github.com/apps/dependabot).
+* โฌ Bump ruff to 0.9.4. PR [#13299](https://github.com/fastapi/fastapi/pull/13299) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ๐ง Update sponsors: pause TestDriven. PR [#13446](https://github.com/fastapi/fastapi/pull/13446) by [@tiangolo](https://github.com/tiangolo).
## 0.115.11
diff --git a/docs/ko/docs/tutorial/security/oauth2-jwt.md b/docs/ko/docs/tutorial/security/oauth2-jwt.md
new file mode 100644
index 000000000..d8bac8346
--- /dev/null
+++ b/docs/ko/docs/tutorial/security/oauth2-jwt.md
@@ -0,0 +1,273 @@
+# ํจ์ค์๋ ํด์ฑ์ ์ด์ฉํ OAuth2, JWT ํ ํฐ์ ์ฌ์ฉํ๋ Bearer ์ธ์ฆ
+
+๋ชจ๋ ๋ณด์ ํ๋ฆ์ ๊ตฌ์ฑํ์ผ๋ฏ๋ก, ์ด์ JWT ํ ํฐ๊ณผ ํจ์ค์๋ ํด์ฑ์ ์ฌ์ฉํด ์ ํ๋ฆฌ์ผ์ด์
์ ์์ ํ๊ฒ ๋ง๋ค ๊ฒ์
๋๋ค.
+
+์ด ์ฝ๋๋ ์ค์ ๋ก ์ ํ๋ฆฌ์ผ์ด์
์์ ํจ์ค์๋๋ฅผ ํด์ฑํ์ฌ DB์ ์ ์ฅํ๋ ๋ฑ์ ์์
์ ํ์ฉํ ์ ์์ต๋๋ค.
+
+์ด์ ์ฅ์ ์ด์ด์ ์์ํด ๋ด
์๋ค.
+
+## JWT
+
+JWT ๋ "JSON Web Tokens" ์ ์๋ฏธํฉ๋๋ค.
+
+JSON ๊ฐ์ฒด๋ฅผ ๊ณต๋ฐฑ์ด ์๋ ๊ธด ๋ฌธ์์ด๋ก ์ธ์ฝ๋ฉํ๋ ํ์ค์ด๋ฉฐ, ๋ค์๊ณผ ๊ฐ์ ํํ์
๋๋ค:
+
+```
+eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c
+```
+
+JWT๋ ์ํธํ๋์ง ์์ ๋๊ตฌ๋ ์ง ํ ํฐ์์ ์ ๋ณด๋ฅผ ๋ณต์ํ ์ ์์ต๋๋ค.
+
+ํ์ง๋ง JWT๋ ์๋ช
๋์ด ์์ต๋๋ค. ๊ทธ๋์ ์์ ์ด ๋ฐ๊ธํ ํ ํฐ์ ๋ฐ์์ ๋, ์ค์ ๋ก ์์ ์ด ๋ฐ๊ธํ๊ฒ ๋ง๋์ง ๊ฒ์ฆํ ์ ์์ต๋๋ค.
+
+๋ง๋ฃ ๊ธฐ๊ฐ์ด ์ผ์ฃผ์ผ์ธ ํ ํฐ์ ๋ฐํํ๋ค๊ณ ๊ฐ์ ํด ๋ด
์๋ค. ๋ค์ ๋ ์ฌ์ฉ์๊ฐ ํ ํฐ์ ๊ฐ์ ธ์์ ๋, ๊ทธ ์ฌ์ฉ์๊ฐ ์์คํ
์ ์ฌ์ ํ ๋ก๊ทธ์ธ๋์ด ์๋ค๋ ๊ฒ์ ์ ์ ์์ต๋๋ค.
+
+์ผ์ฃผ์ผ ๋ค์๋ ํ ํฐ์ด ๋ง๋ฃ๋ ๊ฒ์ด๊ณ , ์ฌ์ฉ์๋ ์ธ๊ฐ๋์ง ์์ ์ ํ ํฐ์ ๋ฐ๊ธฐ ์ํด ๋ค์ ๋ก๊ทธ์ธํด์ผ ํ ๊ฒ์
๋๋ค. ๋ง์ฝ ์ฌ์ฉ์(๋๋ ์ 3์)๊ฐ ํ ํฐ์ ์์ ํ๊ฑฐ๋ ๋ง๋ฃ์ผ์ ๋ณ๊ฒฝํ๋ฉด, ์๋ช
์ด ์ผ์นํ์ง ์๊ธฐ ๋๋ฌธ์ ์์์ฑ ์ ์์ ๊ฒ์
๋๋ค.
+
+๋ง์ฝ JWT ํ ํฐ์ ๋ค๋ค๋ณด๊ณ , ์๋ ๋ฐฉ์๋ ์์๋ณด๊ณ ์ถ๋ค๋ฉด https://jwt.io ์ ํ์ธํ์ญ์์ค.
+
+## `PyJWT` ์ค์น
+
+ํ์ด์ฌ์ผ๋ก JWT ํ ํฐ์ ์์ฑํ๊ณ ๊ฒ์ฆํ๋ ค๋ฉด `PyJWT` ๋ฅผ ์ค์นํด์ผ ํฉ๋๋ค.
+
+[๊ฐ์ํ๊ฒฝ](../../virtual-environments.md){.internal-link target=_blank} ์ ๋ง๋ค๊ณ ํ์ฑํํ ๋ค์ `pyjwt` ๋ฅผ ์ค์นํ์ญ์์ค:
+
+
+
+```console
+$ pip install pyjwt
+
+---> 100%
+```
+
+
+
+/// info | ์ฐธ๊ณ
+
+RSA๋ ECDSA ๊ฐ์ ์ ์ ์๋ช
์๊ณ ๋ฆฌ์ฆ์ ์ฌ์ฉํ๋ ค๋ฉด, `pyjwt[crypto]`๋ผ๋ ์ํธํ ๋ผ์ด๋ธ๋ฌ๋ฆฌ ์์กด์ฑ์ ์ค์นํด์ผ ํฉ๋๋ค.
+
+๋ ์์ธํ ๋ด์ฉ์ PyJWT ์ค์น ์์ ํ์ธํ ์ ์์ต๋๋ค.
+
+///
+
+## ํจ์ค์๋ ํด์ฑ
+
+"ํด์ฑ(Hashing)"์ ์ด๋ค ๋ด์ฉ(์ฌ๊ธฐ์๋ ํจ์ค์๋)์ ํด์ํ ์ ์๋ ์ผ๋ จ์ ๋ฐ์ดํธ ์งํฉ(๋จ์ ๋ฌธ์์ด)์ผ๋ก ๋ณํํ๋ ๊ฒ์ ์๋ฏธํฉ๋๋ค.
+
+๋์ผํ ๋ด์ฉ(๋๊ฐ์ ํจ์ค์๋)์ ํด์ฑํ๋ฉด ๋์ผํ ๋ฌธ์์ด์ ์ป์ต๋๋ค.
+
+ํ์ง๋ง ๊ทธ ๋ฌธ์์ด์ ๋ค์ ํจ์ค์๋๋ก ๋๋๋ฆด ์๋ ์์ต๋๋ค.
+
+### ํจ์ค์๋๋ฅผ ํด์ฑํ๋ ์ด์
+
+๋ฐ์ดํฐ๋ฒ ์ด์ค๋ฅผ ํ์ทจ๋นํ๋๋ผ๋, ์นจ์
์๋ ์ฌ์ฉ์์ ํ๋ฌธ ํจ์ค์๋ ๋์ ํด์ ๊ฐ๋ง ์ป์ ์ ์์ต๋๋ค.
+
+๋ฐ๋ผ์ ์นจ์
์๋ ํ์น ์ฌ์ฉ์ ํจ์ค์๋๋ฅผ ๋ค๋ฅธ ์์คํ
์์ ํ์ฉํ ์ ์์ต๋๋ค. (๋๋ค์ ์ฌ์ฉ์๊ฐ ์ฌ๋ฌ ์์คํ
์์ ๋์ผํ ํจ์ค์๋๋ฅผ ์ฌ์ฉํ๊ธฐ ๋๋ฌธ์ ํ๋ฌธ ํจ์ค์๋๊ฐ ์ ์ถ๋๋ฉด ์ํํฉ๋๋ค.)
+
+## `passlib` ์ค์น
+
+PassLib๋ ํจ์ค์๋ ํด์๋ฅผ ๋ค๋ฃจ๋ ํ๋ฅญํ ํ์ด์ฌ ํจํค์ง์
๋๋ค.
+
+๋ง์ ์์ ํ ํด์ ์๊ณ ๋ฆฌ์ฆ๊ณผ ๋๊ตฌ๋ค์ ์ง์ํฉ๋๋ค.
+
+์ถ์ฒํ๋ ์๊ณ ๋ฆฌ์ฆ์ "Bcrypt"์
๋๋ค.
+
+[๊ฐ์ํ๊ฒฝ](../../virtual-environments.md){.internal-link target=_blank} ์ ๋ง๋ค๊ณ ํ์ฑํํ ๋ค์ PassLib์ Bcrypt๋ฅผ ์ค์นํ์ญ์์ค:
+
+
+
+```console
+$ pip install "passlib[bcrypt]"
+
+---> 100%
+```
+
+
+
+/// tip | ํ
+
+`passlib`๋ฅผ ์ฌ์ฉํ์ฌ, **Django**, **Flask** ์ ๋ณด์ ํ๋ฌ๊ทธ์ธ์ด๋ ๋ค๋ฅธ ๋๊ตฌ๋ก ์์ฑํ ํจ์ค์๋๋ฅผ ์ฝ์ ์ ์๋๋ก ์ค์ ํ ์๋ ์์ต๋๋ค.
+
+์๋ฅผ ๋ค์๋ฉด, FastAPI ์ ํ๋ฆฌ์ผ์ด์
๊ณผ Django ์ ํ๋ฆฌ์ผ์ด์
์ด ๊ฐ์ ๋ฐ์ดํฐ๋ฒ ์ด์ค์์ ๋ฐ์ดํฐ๋ฅผ ๊ณต์ ํ ์ ์์ต๋๋ค. ๋๋ ๊ฐ์ ๋ฐ์ดํฐ๋ฒ ์ด์ค๋ฅผ ์ฌ์ฉํ์ฌ Django ์ ํ๋ฆฌ์ผ์ด์
์ ์ ์ง์ ์ผ๋ก ๋ง์ด๊ทธ๋ ์ด์
ํ ์๋ ์์ต๋๋ค.
+
+๊ทธ๋ฆฌ๊ณ ์ฌ์ฉ์๋ FastAPI ์ ํ๋ฆฌ์ผ์ด์
๊ณผ Django ์ ํ๋ฆฌ์ผ์ด์
์ ๋์์ ๋ก๊ทธ์ธํ ์ ์์ต๋๋ค.
+
+///
+
+## ํจ์ค์๋์ ํด์์ ๊ฒ์ฆ
+
+ํ์ํ ๋๊ตฌ๋ฅผ `passlib`์์ ์ํฌํธํฉ๋๋ค.
+
+PassLib "์ปจํ
์คํธ(context)"๋ฅผ ์์ฑํฉ๋๋ค. ์ด๊ฒ์ ํจ์ค์๋๋ฅผ ํด์ฑํ๊ณ ๊ฒ์ฆํ๋๋ฐ ์ฌ์ฉํฉ๋๋ค.
+
+/// tip | ํ
+
+PassLib ์ปจํ
์คํธ๋ ๋ค์ํ ํด์ฑ ์๊ณ ๋ฆฌ์ฆ์ ์ฌ์ฉํ ์ ์๋ ๊ธฐ๋ฅ์ ์ ๊ณตํ๋ฉฐ, ๋ ์ด์ ์ฌ์ฉ์ด ๊ถ์ฅ๋์ง ์๋ ์ค๋๋ ํด์ฑ ์๊ณ ๋ฆฌ์ฆ์ ๊ฒ์ฆํ๋ ๊ธฐ๋ฅ๋ ํฌํจ๋์ด ์์ต๋๋ค.
+
+์๋ฅผ ๋ค์ด, ๋ค๋ฅธ ์์คํ
(Django ๊ฐ์)์์ ์์ฑํ ํจ์ค์๋๋ฅผ ์ฝ๊ณ ๊ฒ์ฆํ ์ ์์ผ๋ฉฐ, ์๋ก์ด ํจ์ค์๋๋ฅผ Bcrypt ๊ฐ์ ๋ค๋ฅธ ์๊ณ ๋ฆฌ์ฆ์ผ๋ก ํด์ฑํ ์๋ ์์ต๋๋ค.
+
+๊ทธ๋ฆฌ๊ณ ๋์์ ๊ทธ๋ฐ ๋ชจ๋ ์๊ณ ๋ฆฌ์ฆ๊ณผ ํธํ์ฑ์ ์ ์งํฉ๋๋ค.
+
+///
+
+์ฌ์ฉ์๋ก๋ถํฐ ๋ฐ์ ํจ์ค์๋๋ฅผ ํด์ฑํ๋ ์ ํธ๋ฆฌํฐ ํจ์๋ฅผ ์์ฑํฉ๋๋ค.
+
+๊ทธ๋ฆฌ๊ณ ๋ฐ์ ํจ์ค์๋๊ฐ ์ ์ฅ๋ ํด์์ ์ผ์นํ๋์ง ๊ฒ์ฆํ๋ ๋ ๋ค๋ฅธ ์ ํธ๋ฆฌํฐ ํจ์๋ ์์ฑํฉ๋๋ค.
+
+๊ทธ๋ฆฌ๊ณ ์ฌ์ฉ์๋ฅผ ์ธ์ฆํ๊ณ ๋ฐํํ๋ ๋ ๋ค๋ฅธ ํจ์๋ ์์ฑํฉ๋๋ค.
+
+{* ../../docs_src/security/tutorial004_an_py310.py hl[8,49,56:57,60:61,70:76] *}
+
+/// note
+
+์๋ก์ด (๊ฐ์ง) ๋ฐ์ดํฐ๋ฒ ์ด์ค `fake_users_db`๋ฅผ ํ์ธํ๋ฉด, ํด์ ์ฒ๋ฆฌ๋ ํจ์ค์๋๊ฐ ์ด๋ป๊ฒ ์๊ฒผ๋์ง ๋ณผ ์ ์์ต๋๋ค: `"$2b$12$EixZaYVK1fsbw1ZfbX3OXePaWxn96p36WQoeG6Lruj3vjPGga31lW"`.
+
+///
+
+## JWT ํ ํฐ ์ฒ๋ฆฌ
+
+์ค์น๋ ๋ชจ๋์ ์ํฌํธ ํฉ๋๋ค.
+
+JWT ํ ํฐ ์๋ช
์ ์ฌ์ฉ๋ ์์์ ๋น๋ฐํค๋ฅผ ์์ฑํฉ๋๋ค.
+
+์์ ํ ์์์ ๋น๋ฐํค๋ฅผ ์์ฑํ๋ ค๋ฉด ๋ค์ ๋ช
๋ น์ด๋ฅผ ์ฌ์ฉํ์ญ์์ค:
+
+
+
+```console
+$ openssl rand -hex 32
+
+09d25e094faa6ca2556c818166b7a9563b93f7099f6f0f4caa6cf63b88e8d3e7
+```
+
+
+
+๊ทธ๋ฆฌ๊ณ ์์ฑํ ๋น๋ฐํค๋ฅผ ๋ณต์ฌํด ๋ณ์ `SECRET_KEY`์ ๋์
ํฉ๋๋ค. (์ด ์์ ์ ๋ณ์ ๊ฐ์ ๊ทธ๋๋ก ์ฌ์ฉํ์ง ๋ง์ญ์์ค.)
+
+JWT ํ ํฐ์ ์๋ช
ํ๋ ๋ฐ ์ฌ์ฉ๋ ์๊ณ ๋ฆฌ์ฆ์ ์ํ ๋ณ์ `ALGORITHM` ์ ์์ฑํ๊ณ `"HS256"` ์ผ๋ก ์ค์ ํฉ๋๋ค.
+
+ํ ํฐ ๋ง๋ฃ ๊ธฐ๊ฐ์ ์ํ ๋ณ์๋ฅผ ์์ฑํฉ๋๋ค.
+
+์๋ต์ ์ํ ํ ํฐ ์๋ํฌ์ธํธ์ ์ฌ์ฉ๋ Pydantic ๋ชจ๋ธ์ ์ ์ํฉ๋๋ค.
+
+์ ์ก์ธ์ค ํ ํฐ์ ์์ฑํ๊ธฐ ์ํ ์ ํธ๋ฆฌํฐ ํจ์๋ฅผ ์์ฑํฉ๋๋ค.
+
+{* ../../docs_src/security/tutorial004_an_py310.py hl[4,7,13:15,29:31,79:87] *}
+
+## ์์กด์ฑ ์์
+
+`get_current_user` ํจ์๋ฅผ ์ด์ ๊ณผ ๋์ผํ ํ ํฐ์ ๋ฐ๋๋ก ์์ ํ๋, ์ด๋ฒ์๋ JWT ํ ํฐ์ ์ฌ์ฉํ๋๋ก ํฉ๋๋ค.
+
+๋ฐ์ ํ ํฐ์ ๋์ฝ๋ฉํ์ฌ ๊ฒ์ฆํ ํ ํ์ฌ ์ฌ์ฉ์๋ฅผ ๋ฐํํฉ๋๋ค.
+
+ํ ํฐ์ด ์ ํจํ์ง ์๋ค๋ฉด HTTP ์ค๋ฅ๋ฅผ ๋ฐํํฉ๋๋ค.
+
+{* ../../docs_src/security/tutorial004_an_py310.py hl[90:107] *}
+
+## `/token` ๊ฒฝ๋ก ์์
์์
+
+ํ ํฐ์ ๋ง๋ฃ ์๊ฐ์ ์ค์ ํ๊ธฐ ์ํด `timedelta` ๋ฅผ ์์ฑํฉ๋๋ค.
+
+์ค์ JWT ์ก์ธ์ค ํ ํฐ์ ์์ฑํ์ฌ ๋ฐํํฉ๋๋ค.
+
+{* ../../docs_src/security/tutorial004_an_py310.py hl[118:133] *}
+
+### JWT "์ฃผ์ฒด(subject)" `sub`์ ๋ํ ๊ธฐ์ ์ธ๋ถ ์ฌํญ
+
+JWT ๋ช
์ธ์ ๋ฐ๋ฅด๋ฉด ํ ํฐ์ ์ฃผ์ฒด๋ฅผ ํฌํจํ๋ `sub`๋ผ๋ ํค๊ฐ ์์ต๋๋ค.
+
+์ฌ์ฉ ์ฌ๋ถ๋ ์ ํ์ฌํญ์ด์ง๋ง, ์ฌ์ฉ์์ ์๋ณ ์ ๋ณด๋ฅผ ์ ์ฅํ ์ ์์ผ๋ฏ๋ก ์ฌ๊ธฐ์๋ ์ด๋ฅผ ์ฌ์ฉํฉ๋๋ค.
+
+JWT๋ ์ฌ์ฉ์๋ฅผ ์๋ณํ๊ณ ์ฌ์ฉ์๊ฐ API๋ฅผ ์ง์ ์ฌ์ฉํ ์ ์๋๋ก ํ์ฉํ๋ ๊ฒ ์ธ์๋ ๋ค๋ฅธ ์ฉ๋๋ก ์ฌ์ฉ๋ ์๋ ์์ต๋๋ค.
+
+์๋ฅผ ๋ค์ด "์๋์ฐจ"๋ "๋ธ๋ก๊ทธ ๊ฒ์๋ฌผ"์ ์๋ณํ๋ ๋ฐ ์ฌ์ฉํ ์ ์์ต๋๋ค.
+
+๊ทธ๋ฆฌ๊ณ "์๋์ฐจ๋ฅผ ์ด์ ํ๋ค"๋ "๋ธ๋ก๊ทธ ๊ฒ์๋ฌผ์ ์์ ํ๋ค"์ฒ๋ผ ํด๋น ์ํฐํฐ์ ๋ํ ๊ถํ์ ์ถ๊ฐํ ์ ์์ต๋๋ค.
+
+๊ทธ ํ ์ด JWT ํ ํฐ์ ์ฌ์ฉ์(๋๋ ๋ด)์๊ฒ ์ ๊ณตํ๋ฉด, ๊ทธ๋ค์ ๊ณ์ ์ ๋ฐ๋ก ๋ง๋ค ํ์ ์์ด API๊ฐ ์์ฑํ JWT ํ ํฐ๋ง์ผ๋ก ์์
(์๋์ฐจ ์ด์ ๋๋ ๋ธ๋ก๊ทธ ๊ฒ์๋ฌผ ํธ์ง)์ ์ํํ ์ ์์ต๋๋ค.
+
+์ด๋ฌํ ๊ฐ๋
์ ํ์ฉํ๋ฉด JWT๋ ํจ์ฌ ๋ ๋ณต์กํ ์๋๋ฆฌ์ค์๋ ์ฌ์ฉํ ์ ์์ต๋๋ค.
+
+์ด ๊ฒฝ์ฐ ์ฌ๋ฌ ์ํฐํฐ๊ฐ ๋์ผํ ID๋ฅผ ๊ฐ์ง ์ ์์ต๋๋ค. ์๋ฅผ ๋ค์ด foo๋ผ๋ ID๋ฅผ ๊ฐ์ง ์ฌ์ฉ์, ์๋์ฐจ, ๋ธ๋ก๊ทธ ๊ฒ์๋ฌผ์ด ์์ ์ ์์ต๋๋ค.
+
+๊ทธ๋์ ID ์ถฉ๋์ ๋ฐฉ์งํ๊ธฐ ์ํด, ์ฌ์ฉ์์ JWT ํ ํฐ์ ์์ฑํ ๋ ์ ๋์ฌ๋ก `sub` ํค๋ฅผ ์ถ๊ฐํ ์ ์์ต๋๋ค. ์๋ฅผ ๋ค์ด `username:` ์ ๋ถ์ด๋ ๋ฐฉ์์
๋๋ค. ์ด ์์ ์์๋ `sub` ๊ฐ์ด `username:johndoe`์ด ๋ ์ ์์ต๋๋ค.
+
+๊ฐ์ฅ ์ค์ํ ์ ์ `sub` ํค๋ ์ ์ฒด ์ ํ๋ฆฌ์ผ์ด์
์์ ๊ณ ์ ํ ์๋ณ์๊ฐ ๋์ด์ผ ํ๋ฉฐ ๋ฌธ์์ด์ด์ด์ผ ํ๋ค๋ ์ ์
๋๋ค.
+
+## ํ์ธํด๋ด
์๋ค
+
+์๋ฒ๋ฅผ ์คํํ๊ณ ๋ฌธ์๋ก ์ด๋ํ์ญ์์ค: http://127.0.0.1:8000/docs.
+
+๋ค์๊ณผ ๊ฐ์ ์ฌ์ฉ์ ์ธํฐํ์ด์ค๋ฅผ ๋ณผ ์ ์์ต๋๋ค:
+
+
+
+์ด์ ๊ณผ ๊ฐ์ ๋ฐฉ๋ฒ์ผ๋ก ์ ํ๋ฆฌ์ผ์ด์
์ ์ธ์ฆํ์ญ์์ค.
+
+๋ค์ ์ธ์ฆ ์ ๋ณด๋ฅผ ์ฌ์ฉํ์ญ์์ค:
+
+Username: `johndoe`
+Password: `secret`
+
+/// check
+
+์ฝ๋ ์ด๋์๋ ํ๋ฌธ ํจ์ค์๋ "`secret`" ์ด ์๋ค๋ ์ ์ ์ ์ํ์ญ์์ค. ํด์๋ ๋ฒ์ ๋ง ์์ต๋๋ค.
+
+///
+
+
+
+`/users/me/` ๋ฅผ ํธ์ถํ๋ฉด ๋ค์๊ณผ ๊ฐ์ ์๋ต์ ์ป์ ์ ์์ต๋๋ค:
+
+```JSON
+{
+ "username": "johndoe",
+ "email": "johndoe@example.com",
+ "full_name": "John Doe",
+ "disabled": false
+}
+```
+
+
+
+๊ฐ๋ฐ์ ๋๊ตฌ๋ฅผ ์ด์ด๋ณด๋ฉด ์ ์ก๋ ๋ฐ์ดํฐ์ ํ ํฐ๋ง ํฌํจ๋ ๊ฒ์ ํ์ธํ ์ ์์ต๋๋ค. ํจ์ค์๋๋ ์ฌ์ฉ์๋ฅผ ์ธ์ฆํ๊ณ ์ก์ธ์ค ํ ํฐ์ ๋ฐ๊ธฐ ์ํ ์ฒซ ๋ฒ์งธ ์์ฒญ์๋ง ์ ์ก๋๋ฉฐ, ์ดํ์๋ ์ ์ก๋์ง ์์ต๋๋ค:
+
+
+
+/// note
+
+`Bearer `๋ก ์์ํ๋ `Authorization` ํค๋์ ์ฃผ๋ชฉํ์ญ์์ค.
+
+///
+
+## `scopes` ์ ๊ณ ๊ธ ์ฌ์ฉ๋ฒ
+
+OAuth2๋ "์ค์ฝํ(scopes)" ๋ผ๋ ๊ฐ๋
์ ๊ฐ๊ณ ์์ต๋๋ค.
+
+์ด๋ฅผ ์ฌ์ฉํ์ฌ JWT ํ ํฐ์ ํน์ ๊ถํ ์งํฉ์ ์ถ๊ฐํ ์ ์์ต๋๋ค.
+
+๊ทธ ํ ์ด ํ ํฐ์ ์ฌ์ฉ์์๊ฒ ์ง์ ์ ๊ณตํ๊ฑฐ๋ ์ 3์์๊ฒ ์ ๊ณตํ์ฌ, ํน์ ์ ํ์ฌํญ ํ์์๋ API์ ํต์ ํ๋๋ก ํ ์ ์์ต๋๋ค.
+
+**FastAPI** ์์์ ์ฌ์ฉ ๋ฐฉ๋ฒ๊ณผ ํตํฉ ๋ฐฉ์์ **์ฌํ ์ฌ์ฉ์ ์๋ด์** ์์ ์์ธํ ๋ฐฐ์ธ ์ ์์ต๋๋ค.
+
+## ์์ฝ
+
+์ง๊ธ๊น์ง ์ดํด๋ณธ ๋ด์ฉ์ ๋ฐํ์ผ๋ก, OAuth2์ JWT ๊ฐ์ ํ์ค์ ์ฌ์ฉํ์ฌ ์์ ํ **FastAPI** ์ ํ๋ฆฌ์ผ์ด์
์ ๋ง๋ค ์ ์์ต๋๋ค.
+
+๊ฑฐ์ ๋ชจ๋ ํ๋ ์์ํฌ์์ ๋ณด์ ์ฒ๋ฆฌ๋ ์๋นํ ๋ณต์กํ ์ฃผ์ ์
๋๋ค.
+
+์ด๋ฅผ ๋จ์ํํ๋ ๋ง์ ํจํค์ง๋ ๋ฐ์ดํฐ ๋ชจ๋ธ, ๋ฐ์ดํฐ๋ฒ ์ด์ค, ์ฌ์ฉ ๊ฐ๋ฅํ ๊ธฐ๋ฅ๋ค์ ๋ํด ์ฌ๋ฌ ์ ์ฝ์ด ์์ต๋๋ค. ๊ทธ๋ฆฌ๊ณ ์ง๋์น๊ฒ ๋จ์ํํ๋ ์ผ๋ถ ํจํค์ง๋ค์ ์ฌ๊ฐํ ๋ณด์ ๊ฒฐํจ์ ๊ฐ์ง ์๋ ์์ต๋๋ค.
+
+---
+
+**FastAPI** ๋ ์ด๋ค ๋ฐ์ดํฐ๋ฒ ์ด์ค, ๋ฐ์ดํฐ ๋ชจ๋ธ, ๋๊ตฌ๋ ๊ฐ์ํ์ง ์์ต๋๋ค.
+
+ํ๋ก์ ํธ์ ๊ฐ์ฅ ์ ํฉํ ๊ฒ์ ์ ํํ ์ ์๋ ์ ์ฐ์ฑ์ ์ ๊ณตํฉ๋๋ค.
+
+๊ทธ๋ฆฌ๊ณ `passlib` ์ `PyJWT` ์ฒ๋ผ ์ ๊ด๋ฆฌ๋๊ณ ๋๋ฆฌ ์ฌ์ฉ๋๋ ํจํค์ง๋ค์ ๋ฐ๋ก ์ฌ์ฉํ ์ ์์ต๋๋ค. **FastAPI** ๋ ์ธ๋ถ ํจํค์ง ํตํฉ์ ์ํด ๋ณต์กํ ๋ฉ์ปค๋์ฆ์ด ํ์ํ์ง ์๊ธฐ ๋๋ฌธ์
๋๋ค.
+
+๊ทธ๋ฌ๋ ์ ์ฐ์ฑ, ๊ฒฌ๊ณ ์ฑ, ๋ณด์์ฑ์ ํด์น์ง ์์ผ๋ฉด์ ๊ณผ์ ์ ๋จ์ํํ ์ ์๋ ๋๊ตฌ๋ค์ ์ ๊ณตํฉ๋๋ค.
+
+๊ทธ๋ฆฌ๊ณ OAuth2์ ๊ฐ์ ํ์ค ํ๋กํ ์ฝ์ ๋น๊ต์ ๊ฐ๋จํ ๋ฐฉ๋ฒ์ผ๋ก ๊ตฌํํ๊ณ ์ฌ์ฉํ ์ ์์ต๋๋ค.
+
+๋ ์ธ๋ถํ๋ ๊ถํ ์ฒด๊ณ๋ฅผ ์ํด OAuth2์ "์ค์ฝํ"๋ฅผ ์ฌ์ฉํ๋ ๋ฐฉ๋ฒ์ **์ฌํ ์ฌ์ฉ์ ์๋ด์**์์ ๋ ์์ธํ ๋ฐฐ์ธ ์ ์์ต๋๋ค. OAuth2์ ์ค์ฝํ๋ ์ 3์ ์ ํ๋ฆฌ์ผ์ด์
์ด ์ฌ์ฉ์๋ฅผ ๋์ ํด ๊ทธ๋ค์ API์ ์ํธ์์ฉํ๋๋ก ๊ถํ์ ๋ถ์ฌํ๊ธฐ ์ํด, Facebook, Google, GitHub, Microsoft, Twitter ๋ฑ์ ๋ง์ ๋ํ ์ธ์ฆ ์ ๊ณต์
์ฒด๋ค์ด ์ฌ์ฉํ๋ ๋ฉ์ปค๋์ฆ์
๋๋ค.
diff --git a/docs/vi/docs/deployment/cloud.md b/docs/vi/docs/deployment/cloud.md
new file mode 100644
index 000000000..9ab72769d
--- /dev/null
+++ b/docs/vi/docs/deployment/cloud.md
@@ -0,0 +1,17 @@
+# Triแปn khai FastAPI trรชn cรกc Dแปch vแปฅ Cloud
+
+Bแบกn cรณ thแป sแปญ dแปฅng **bแบฅt kแปณ nhร cung cแบฅp dแปch vแปฅ cloud** nร o ฤแป triแปn khai แปฉng dแปฅng FastAPI cแปงa mรฌnh.
+
+Trong hแบงu hแบฟt cรกc trฦฐแปng hแปฃp, cรกc nhร cung cแบฅp dแปch vแปฅ cloud lแปn ฤแปu cรณ hฦฐแปng dแบซn triแปn khai FastAPI vแปi hแป.
+
+## Nhร cung cแบฅp dแปch vแปฅ Cloud - Nhร tร i trแปฃ
+Mแปt vร i nhร cung cแบฅp dแปch vแปฅ cloud โจ [**tร i trแปฃ cho FastAPI**](../help-fastapi.md#sponsor-the-author){.internal-link target=_blank} โจ, ฤiแปu nร y giรบp ฤแบฃm bแบฃo sแปฑ phรกt triแปn liรชn tแปฅc vร khแปe mแบกnh cแปงa FastAPI vร hแป sinh thรกi cแปงa nรณ.
+
+Thรชm nแปฏa, ฤiแปu nร y cลฉng thแป hiแปn cam kแบฟt thแปฑc sแปฑ cแปงa hแป ฤแปi vแปi FastAPI vร **cแปng ฤแปng ngฦฐแปi dรนng** (bแบกn), vรฌ hแป khรดng chแป muแปn cung cแบฅp cho bแบกn mแปt **dแปch vแปฅ tแปt** mร cรฒn muแปn ฤแบฃm bแบฃo rแบฑng bแบกn cรณ mแปt **framework tแปt vร bแปn vแปฏng**, ฤรณ chรญnh lร FastAPI. ๐
+
+Bแบกn cรณ thแป thแปญ cรกc dแปch vแปฅ cแปงa hแป vร lร m theo hฦฐแปng dแบซn cแปงa hแป:
+
+* Platform.sh
+* Porter
+* Coherence
+* Render
diff --git a/fastapi/dependencies/utils.py b/fastapi/dependencies/utils.py
index e2866b488..d205d17fa 100644
--- a/fastapi/dependencies/utils.py
+++ b/fastapi/dependencies/utils.py
@@ -133,9 +133,9 @@ def get_param_sub_dependant(
def get_parameterless_sub_dependant(*, depends: params.Depends, path: str) -> Dependant:
- assert callable(
- depends.dependency
- ), "A parameter-less dependency must have a callable dependency"
+ assert callable(depends.dependency), (
+ "A parameter-less dependency must have a callable dependency"
+ )
return get_sub_dependant(depends=depends, dependency=depends.dependency, path=path)
@@ -302,9 +302,9 @@ def get_dependant(
type_annotation=param_details.type_annotation,
dependant=dependant,
):
- assert (
- param_details.field is None
- ), f"Cannot specify multiple FastAPI annotations for {param_name!r}"
+ assert param_details.field is None, (
+ f"Cannot specify multiple FastAPI annotations for {param_name!r}"
+ )
continue
assert param_details.field is not None
if isinstance(param_details.field.field_info, params.Body):
@@ -439,9 +439,9 @@ def analyze_param(
),
):
assert depends is None, f"Cannot specify `Depends` for type {type_annotation!r}"
- assert (
- field_info is None
- ), f"Cannot specify FastAPI annotation for type {type_annotation!r}"
+ assert field_info is None, (
+ f"Cannot specify FastAPI annotation for type {type_annotation!r}"
+ )
# Handle default assignations, neither field_info nor depends was not found in Annotated nor default value
elif field_info is None and depends is None:
default_value = value if value is not inspect.Signature.empty else RequiredParam
@@ -494,9 +494,9 @@ def analyze_param(
field_info=field_info,
)
if is_path_param:
- assert is_scalar_field(
- field=field
- ), "Path params must be of one of the supported types"
+ assert is_scalar_field(field=field), (
+ "Path params must be of one of the supported types"
+ )
elif isinstance(field_info, params.Query):
assert (
is_scalar_field(field)
@@ -521,9 +521,9 @@ def add_param_to_fields(*, field: ModelField, dependant: Dependant) -> None:
elif field_info_in == params.ParamTypes.header:
dependant.header_params.append(field)
else:
- assert (
- field_info_in == params.ParamTypes.cookie
- ), f"non-body parameters must be in path, query, header or cookie: {field.name}"
+ assert field_info_in == params.ParamTypes.cookie, (
+ f"non-body parameters must be in path, query, header or cookie: {field.name}"
+ )
dependant.cookie_params.append(field)
@@ -782,9 +782,9 @@ def request_params_to_args(
if single_not_embedded_field:
field_info = first_field.field_info
- assert isinstance(
- field_info, params.Param
- ), "Params must be subclasses of Param"
+ assert isinstance(field_info, params.Param), (
+ "Params must be subclasses of Param"
+ )
loc: Tuple[str, ...] = (field_info.in_.value,)
v_, errors_ = _validate_value_with_model_field(
field=first_field, value=params_to_process, values=values, loc=loc
@@ -794,9 +794,9 @@ def request_params_to_args(
for field in fields:
value = _get_multidict_value(field, received_params)
field_info = field.field_info
- assert isinstance(
- field_info, params.Param
- ), "Params must be subclasses of Param"
+ assert isinstance(field_info, params.Param), (
+ "Params must be subclasses of Param"
+ )
loc = (field_info.in_.value, field.alias)
v_, errors_ = _validate_value_with_model_field(
field=field, value=value, values=values, loc=loc
diff --git a/fastapi/openapi/utils.py b/fastapi/openapi/utils.py
index 947eca948..bd8f3c106 100644
--- a/fastapi/openapi/utils.py
+++ b/fastapi/openapi/utils.py
@@ -364,9 +364,9 @@ def get_openapi_path(
openapi_response = operation_responses.setdefault(
status_code_key, {}
)
- assert isinstance(
- process_response, dict
- ), "An additional response must be a dict"
+ assert isinstance(process_response, dict), (
+ "An additional response must be a dict"
+ )
field = route.response_fields.get(additional_status_code)
additional_field_schema: Optional[Dict[str, Any]] = None
if field:
@@ -434,9 +434,9 @@ def get_fields_from_routes(
route, routing.APIRoute
):
if route.body_field:
- assert isinstance(
- route.body_field, ModelField
- ), "A request body must be a Pydantic Field"
+ assert isinstance(route.body_field, ModelField), (
+ "A request body must be a Pydantic Field"
+ )
body_fields_from_routes.append(route.body_field)
if route.response_field:
responses_from_routes.append(route.response_field)
diff --git a/fastapi/routing.py b/fastapi/routing.py
index 8ea4bb219..457481e32 100644
--- a/fastapi/routing.py
+++ b/fastapi/routing.py
@@ -504,9 +504,9 @@ class APIRoute(routing.Route):
status_code = int(status_code)
self.status_code = status_code
if self.response_model:
- assert is_body_allowed_for_status_code(
- status_code
- ), f"Status code {status_code} must not have a response body"
+ assert is_body_allowed_for_status_code(status_code), (
+ f"Status code {status_code} must not have a response body"
+ )
response_name = "Response_" + self.unique_id
self.response_field = create_model_field(
name=response_name,
@@ -537,9 +537,9 @@ class APIRoute(routing.Route):
assert isinstance(response, dict), "An additional response must be a dict"
model = response.get("model")
if model:
- assert is_body_allowed_for_status_code(
- additional_status_code
- ), f"Status code {additional_status_code} must not have a response body"
+ assert is_body_allowed_for_status_code(additional_status_code), (
+ f"Status code {additional_status_code} must not have a response body"
+ )
response_name = f"Response_{additional_status_code}_{self.unique_id}"
response_field = create_model_field(
name=response_name, type_=model, mode="serialization"
@@ -844,9 +844,9 @@ class APIRouter(routing.Router):
)
if prefix:
assert prefix.startswith("/"), "A path prefix must start with '/'"
- assert not prefix.endswith(
- "/"
- ), "A path prefix must not end with '/', as the routes will start with '/'"
+ assert not prefix.endswith("/"), (
+ "A path prefix must not end with '/', as the routes will start with '/'"
+ )
self.prefix = prefix
self.tags: List[Union[str, Enum]] = tags or []
self.dependencies = list(dependencies or [])
@@ -1256,9 +1256,9 @@ class APIRouter(routing.Router):
"""
if prefix:
assert prefix.startswith("/"), "A path prefix must start with '/'"
- assert not prefix.endswith(
- "/"
- ), "A path prefix must not end with '/', as the routes will start with '/'"
+ assert not prefix.endswith("/"), (
+ "A path prefix must not end with '/', as the routes will start with '/'"
+ )
else:
for r in router.routes:
path = getattr(r, "path") # noqa: B009
diff --git a/requirements-docs-tests.txt b/requirements-docs-tests.txt
index 331d2a5b3..e7718e61d 100644
--- a/requirements-docs-tests.txt
+++ b/requirements-docs-tests.txt
@@ -1,4 +1,4 @@
# For mkdocstrings and tests
httpx >=0.23.0,<0.28.0
# For linting and generating docs versions
-ruff ==0.6.4
+ruff ==0.9.4
diff --git a/requirements-docs.txt b/requirements-docs.txt
index cd2e4e58e..8812e27f9 100644
--- a/requirements-docs.txt
+++ b/requirements-docs.txt
@@ -14,6 +14,6 @@ cairosvg==2.7.1
mkdocstrings[python]==0.26.1
griffe-typingdoc==0.2.7
# For griffe, it formats with black
-black==24.10.0
+black==25.1.0
mkdocs-macros-plugin==1.3.7
markdown-include-variants==0.0.4
diff --git a/requirements-tests.txt b/requirements-tests.txt
index 4a15844e4..6a870add6 100644
--- a/requirements-tests.txt
+++ b/requirements-tests.txt
@@ -4,7 +4,7 @@ pytest >=7.1.3,<9.0.0
coverage[toml] >= 6.5.0,< 8.0
mypy ==1.8.0
dirty-equals ==0.8.0
-sqlmodel==0.0.22
+sqlmodel==0.0.23
flask >=1.1.2,<4.0.0
anyio[trio] >=3.2.1,<5.0.0
PyJWT==2.8.0
diff --git a/scripts/translate.py b/scripts/translate.py
index ce11b3877..9a2136d1b 100644
--- a/scripts/translate.py
+++ b/scripts/translate.py
@@ -38,9 +38,9 @@ def get_langs() -> dict[str, str]:
def generate_lang_path(*, lang: str, path: Path) -> Path:
en_docs_path = Path("docs/en/docs")
- assert str(path).startswith(
- str(en_docs_path)
- ), f"Path must be inside {en_docs_path}"
+ assert str(path).startswith(str(en_docs_path)), (
+ f"Path must be inside {en_docs_path}"
+ )
lang_docs_path = Path(f"docs/{lang}/docs")
out_path = Path(str(path).replace(str(en_docs_path), str(lang_docs_path)))
return out_path
@@ -56,9 +56,9 @@ def translate_page(*, lang: str, path: Path) -> None:
lang_prompt_content = lang_prompt_path.read_text()
en_docs_path = Path("docs/en/docs")
- assert str(path).startswith(
- str(en_docs_path)
- ), f"Path must be inside {en_docs_path}"
+ assert str(path).startswith(str(en_docs_path)), (
+ f"Path must be inside {en_docs_path}"
+ )
out_path = generate_lang_path(lang=lang, path=path)
out_path.parent.mkdir(parents=True, exist_ok=True)
original_content = path.read_text()
diff --git a/tests/test_enforce_once_required_parameter.py b/tests/test_enforce_once_required_parameter.py
index b64f8341b..30329282f 100644
--- a/tests/test_enforce_once_required_parameter.py
+++ b/tests/test_enforce_once_required_parameter.py
@@ -48,7 +48,7 @@ expected_schema = {
"type": "array",
},
"msg": {"title": "Message", "type": "string"},
- "type": {"title": "Error " "Type", "type": "string"},
+ "type": {"title": "Error Type", "type": "string"},
},
"required": ["loc", "msg", "type"],
"title": "ValidationError",
@@ -73,7 +73,7 @@ expected_schema = {
"responses": {
"200": {
"content": {"application/json": {"schema": {}}},
- "description": "Successful " "Response",
+ "description": "Successful Response",
},
"422": {
"content": {
@@ -83,7 +83,7 @@ expected_schema = {
}
}
},
- "description": "Validation " "Error",
+ "description": "Validation Error",
},
},
"summary": "Foo Handler",
diff --git a/tests/test_generic_parameterless_depends.py b/tests/test_generic_parameterless_depends.py
index fe13ff89b..5aa35320c 100644
--- a/tests/test_generic_parameterless_depends.py
+++ b/tests/test_generic_parameterless_depends.py
@@ -55,7 +55,7 @@ def test_openapi_schema():
"responses": {
"200": {
"content": {"application/json": {"schema": {}}},
- "description": "Successful " "Response",
+ "description": "Successful Response",
}
},
"summary": "A",
@@ -67,7 +67,7 @@ def test_openapi_schema():
"responses": {
"200": {
"content": {"application/json": {"schema": {}}},
- "description": "Successful " "Response",
+ "description": "Successful Response",
}
},
"summary": "B",
diff --git a/tests/test_repeated_dependency_schema.py b/tests/test_repeated_dependency_schema.py
index d7d0dfa05..c21829bd9 100644
--- a/tests/test_repeated_dependency_schema.py
+++ b/tests/test_repeated_dependency_schema.py
@@ -41,7 +41,7 @@ schema = {
"type": "array",
},
"msg": {"title": "Message", "type": "string"},
- "type": {"title": "Error " "Type", "type": "string"},
+ "type": {"title": "Error Type", "type": "string"},
},
"required": ["loc", "msg", "type"],
"title": "ValidationError",
@@ -66,7 +66,7 @@ schema = {
"responses": {
"200": {
"content": {"application/json": {"schema": {}}},
- "description": "Successful " "Response",
+ "description": "Successful Response",
},
"422": {
"content": {
@@ -76,7 +76,7 @@ schema = {
}
}
},
- "description": "Validation " "Error",
+ "description": "Validation Error",
},
},
"summary": "Get Deps",
diff --git a/tests/test_tutorial/test_configure_swagger_ui/test_tutorial001.py b/tests/test_tutorial/test_configure_swagger_ui/test_tutorial001.py
index 72db54bd2..a04dba219 100644
--- a/tests/test_tutorial/test_configure_swagger_ui/test_tutorial001.py
+++ b/tests/test_tutorial/test_configure_swagger_ui/test_tutorial001.py
@@ -8,31 +8,31 @@ client = TestClient(app)
def test_swagger_ui():
response = client.get("/docs")
assert response.status_code == 200, response.text
- assert (
- '"syntaxHighlight": false' in response.text
- ), "syntaxHighlight should be included and converted to JSON"
- assert (
- '"dom_id": "#swagger-ui"' in response.text
- ), "default configs should be preserved"
+ assert '"syntaxHighlight": false' in response.text, (
+ "syntaxHighlight should be included and converted to JSON"
+ )
+ assert '"dom_id": "#swagger-ui"' in response.text, (
+ "default configs should be preserved"
+ )
assert "presets: [" in response.text, "default configs should be preserved"
- assert (
- "SwaggerUIBundle.presets.apis," in response.text
- ), "default configs should be preserved"
- assert (
- "SwaggerUIBundle.SwaggerUIStandalonePreset" in response.text
- ), "default configs should be preserved"
- assert (
- '"layout": "BaseLayout",' in response.text
- ), "default configs should be preserved"
- assert (
- '"deepLinking": true,' in response.text
- ), "default configs should be preserved"
- assert (
- '"showExtensions": true,' in response.text
- ), "default configs should be preserved"
- assert (
- '"showCommonExtensions": true,' in response.text
- ), "default configs should be preserved"
+ assert "SwaggerUIBundle.presets.apis," in response.text, (
+ "default configs should be preserved"
+ )
+ assert "SwaggerUIBundle.SwaggerUIStandalonePreset" in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"layout": "BaseLayout",' in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"deepLinking": true,' in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"showExtensions": true,' in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"showCommonExtensions": true,' in response.text, (
+ "default configs should be preserved"
+ )
def test_get_users():
diff --git a/tests/test_tutorial/test_configure_swagger_ui/test_tutorial002.py b/tests/test_tutorial/test_configure_swagger_ui/test_tutorial002.py
index d06a385b5..ea56b6f21 100644
--- a/tests/test_tutorial/test_configure_swagger_ui/test_tutorial002.py
+++ b/tests/test_tutorial/test_configure_swagger_ui/test_tutorial002.py
@@ -8,34 +8,34 @@ client = TestClient(app)
def test_swagger_ui():
response = client.get("/docs")
assert response.status_code == 200, response.text
- assert (
- '"syntaxHighlight": false' not in response.text
- ), "not used parameters should not be included"
- assert (
- '"syntaxHighlight": {"theme": "obsidian"}' in response.text
- ), "parameters with middle dots should be included in a JSON compatible way"
- assert (
- '"dom_id": "#swagger-ui"' in response.text
- ), "default configs should be preserved"
+ assert '"syntaxHighlight": false' not in response.text, (
+ "not used parameters should not be included"
+ )
+ assert '"syntaxHighlight": {"theme": "obsidian"}' in response.text, (
+ "parameters with middle dots should be included in a JSON compatible way"
+ )
+ assert '"dom_id": "#swagger-ui"' in response.text, (
+ "default configs should be preserved"
+ )
assert "presets: [" in response.text, "default configs should be preserved"
- assert (
- "SwaggerUIBundle.presets.apis," in response.text
- ), "default configs should be preserved"
- assert (
- "SwaggerUIBundle.SwaggerUIStandalonePreset" in response.text
- ), "default configs should be preserved"
- assert (
- '"layout": "BaseLayout",' in response.text
- ), "default configs should be preserved"
- assert (
- '"deepLinking": true,' in response.text
- ), "default configs should be preserved"
- assert (
- '"showExtensions": true,' in response.text
- ), "default configs should be preserved"
- assert (
- '"showCommonExtensions": true,' in response.text
- ), "default configs should be preserved"
+ assert "SwaggerUIBundle.presets.apis," in response.text, (
+ "default configs should be preserved"
+ )
+ assert "SwaggerUIBundle.SwaggerUIStandalonePreset" in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"layout": "BaseLayout",' in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"deepLinking": true,' in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"showExtensions": true,' in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"showCommonExtensions": true,' in response.text, (
+ "default configs should be preserved"
+ )
def test_get_users():
diff --git a/tests/test_tutorial/test_configure_swagger_ui/test_tutorial003.py b/tests/test_tutorial/test_configure_swagger_ui/test_tutorial003.py
index 187e89ace..926bbb14f 100644
--- a/tests/test_tutorial/test_configure_swagger_ui/test_tutorial003.py
+++ b/tests/test_tutorial/test_configure_swagger_ui/test_tutorial003.py
@@ -8,34 +8,34 @@ client = TestClient(app)
def test_swagger_ui():
response = client.get("/docs")
assert response.status_code == 200, response.text
- assert (
- '"deepLinking": false,' in response.text
- ), "overridden configs should be preserved"
- assert (
- '"deepLinking": true' not in response.text
- ), "overridden configs should not include the old value"
- assert (
- '"syntaxHighlight": false' not in response.text
- ), "not used parameters should not be included"
- assert (
- '"dom_id": "#swagger-ui"' in response.text
- ), "default configs should be preserved"
+ assert '"deepLinking": false,' in response.text, (
+ "overridden configs should be preserved"
+ )
+ assert '"deepLinking": true' not in response.text, (
+ "overridden configs should not include the old value"
+ )
+ assert '"syntaxHighlight": false' not in response.text, (
+ "not used parameters should not be included"
+ )
+ assert '"dom_id": "#swagger-ui"' in response.text, (
+ "default configs should be preserved"
+ )
assert "presets: [" in response.text, "default configs should be preserved"
- assert (
- "SwaggerUIBundle.presets.apis," in response.text
- ), "default configs should be preserved"
- assert (
- "SwaggerUIBundle.SwaggerUIStandalonePreset" in response.text
- ), "default configs should be preserved"
- assert (
- '"layout": "BaseLayout",' in response.text
- ), "default configs should be preserved"
- assert (
- '"showExtensions": true,' in response.text
- ), "default configs should be preserved"
- assert (
- '"showCommonExtensions": true,' in response.text
- ), "default configs should be preserved"
+ assert "SwaggerUIBundle.presets.apis," in response.text, (
+ "default configs should be preserved"
+ )
+ assert "SwaggerUIBundle.SwaggerUIStandalonePreset" in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"layout": "BaseLayout",' in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"showExtensions": true,' in response.text, (
+ "default configs should be preserved"
+ )
+ assert '"showCommonExtensions": true,' in response.text, (
+ "default configs should be preserved"
+ )
def test_get_users():
diff --git a/tests/test_tutorial/test_sql_databases/test_tutorial002.py b/tests/test_tutorial/test_sql_databases/test_tutorial002.py
index 68c1966f5..79e48c1c3 100644
--- a/tests/test_tutorial/test_sql_databases/test_tutorial002.py
+++ b/tests/test_tutorial/test_sql_databases/test_tutorial002.py
@@ -71,9 +71,9 @@ def test_crud_app(client: TestClient):
assert response.json() == snapshot(
{"age": 30, "id": IsInt(), "name": "Dead Pond"}
)
- assert (
- response.json()["id"] != 9000
- ), "The ID should be generated by the database"
+ assert response.json()["id"] != 9000, (
+ "The ID should be generated by the database"
+ )
# Read a hero
hero_id = response.json()["id"]