Browse Source
* 📝 Add docs for Separate OpenAPI Schemas for Input and Output * 🔧 Add new docs page to MkDocs config * ✨ Add separate_input_output_schemas parameter to FastAPI class * 📝 Add source examples for separating OpenAPI schemas * ✅ Add tests for separated OpenAPI schemas * 📝 Add source examples for Python 3.10, 3.9, and 3.7+ * 📝 Update docs for Separate OpenAPI Schemas with new multi-version examples * ✅ Add and update tests for different Python versions * ✅ Add tests for corner cases with separate_input_output_schemas * 📝 Update tutorial to use Union instead of Optional * 🐛 Fix type annotations * 🐛 Fix correct import in test * 💄 Add CSS to simulate browser windows for screenshots * ➕ Add playwright as a dev dependency to automate generating screenshots * 🔨 Add Playwright scripts to generate screenshots for new docs * 📝 Update docs, tweak text to match screenshots * 🍱 Add screenshots for new docspull/10150/head
committed by
GitHub
31 changed files with 1950 additions and 2 deletions
@ -0,0 +1,228 @@ |
|||
# Separate OpenAPI Schemas for Input and Output or Not |
|||
|
|||
When using **Pydantic v2**, the generated OpenAPI is a bit more exact and **correct** than before. 😎 |
|||
|
|||
In fact, in some cases, it will even have **two JSON Schemas** in OpenAPI for the same Pydantic model, for input and output, depending on if they have **default values**. |
|||
|
|||
Let's see how that works and how to change it if you need to do that. |
|||
|
|||
## Pydantic Models for Input and Output |
|||
|
|||
Let's say you have a Pydantic model with default values, like this one: |
|||
|
|||
=== "Python 3.10+" |
|||
|
|||
```Python hl_lines="7" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001_py310.py[ln:1-7]!} |
|||
|
|||
# Code below omitted 👇 |
|||
``` |
|||
|
|||
<details> |
|||
<summary>👀 Full file preview</summary> |
|||
|
|||
```Python |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001_py310.py!} |
|||
``` |
|||
|
|||
</details> |
|||
|
|||
=== "Python 3.9+" |
|||
|
|||
```Python hl_lines="9" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001_py39.py[ln:1-9]!} |
|||
|
|||
# Code below omitted 👇 |
|||
``` |
|||
|
|||
<details> |
|||
<summary>👀 Full file preview</summary> |
|||
|
|||
```Python |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001_py39.py!} |
|||
``` |
|||
|
|||
</details> |
|||
|
|||
=== "Python 3.7+" |
|||
|
|||
```Python hl_lines="9" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001.py[ln:1-9]!} |
|||
|
|||
# Code below omitted 👇 |
|||
``` |
|||
|
|||
<details> |
|||
<summary>👀 Full file preview</summary> |
|||
|
|||
```Python |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001.py!} |
|||
``` |
|||
|
|||
</details> |
|||
|
|||
### Model for Input |
|||
|
|||
If you use this model as an input like here: |
|||
|
|||
=== "Python 3.10+" |
|||
|
|||
```Python hl_lines="14" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001_py310.py[ln:1-15]!} |
|||
|
|||
# Code below omitted 👇 |
|||
``` |
|||
|
|||
<details> |
|||
<summary>👀 Full file preview</summary> |
|||
|
|||
```Python |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001_py310.py!} |
|||
``` |
|||
|
|||
</details> |
|||
|
|||
=== "Python 3.9+" |
|||
|
|||
```Python hl_lines="16" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001_py39.py[ln:1-17]!} |
|||
|
|||
# Code below omitted 👇 |
|||
``` |
|||
|
|||
<details> |
|||
<summary>👀 Full file preview</summary> |
|||
|
|||
```Python |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001_py39.py!} |
|||
``` |
|||
|
|||
</details> |
|||
|
|||
=== "Python 3.7+" |
|||
|
|||
```Python hl_lines="16" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001.py[ln:1-17]!} |
|||
|
|||
# Code below omitted 👇 |
|||
``` |
|||
|
|||
<details> |
|||
<summary>👀 Full file preview</summary> |
|||
|
|||
```Python |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001.py!} |
|||
``` |
|||
|
|||
</details> |
|||
|
|||
...then the `description` field will **not be required**. Because it has a default value of `None`. |
|||
|
|||
### Input Model in Docs |
|||
|
|||
You can confirm that in the docs, the `description` field doesn't have a **red asterisk**, it's not marked as required: |
|||
|
|||
<div class="screenshot"> |
|||
<img src="/img/tutorial/separate-openapi-schemas/image01.png"> |
|||
</div> |
|||
|
|||
### Model for Output |
|||
|
|||
But if you use the same model as an output, like here: |
|||
|
|||
=== "Python 3.10+" |
|||
|
|||
```Python hl_lines="19" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001_py310.py!} |
|||
``` |
|||
|
|||
=== "Python 3.9+" |
|||
|
|||
```Python hl_lines="21" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001_py39.py!} |
|||
``` |
|||
|
|||
=== "Python 3.7+" |
|||
|
|||
```Python hl_lines="21" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial001.py!} |
|||
``` |
|||
|
|||
...then because `description` has a default value, if you **don't return anything** for that field, it will still have that **default value**. |
|||
|
|||
### Model for Output Response Data |
|||
|
|||
If you interact with the docs and check the response, even though the code didn't add anything in one of the `description` fields, the JSON response contains the default value (`null`): |
|||
|
|||
<div class="screenshot"> |
|||
<img src="/img/tutorial/separate-openapi-schemas/image02.png"> |
|||
</div> |
|||
|
|||
This means that it will **always have a value**, it's just that sometimes the value could be `None` (or `null` in JSON). |
|||
|
|||
That means that, clients using your API don't have to check if the value exists or not, they can **asume the field will always be there**, but just that in some cases it will have the default value of `None`. |
|||
|
|||
The way to describe this in OpenAPI, is to mark that field as **required**, because it will always be there. |
|||
|
|||
Because of that, the JSON Schema for a model can be different depending on if it's used for **input or output**: |
|||
|
|||
* for **input** the `description` will **not be required** |
|||
* for **output** it will be **required** (and possibly `None`, or in JSON terms, `null`) |
|||
|
|||
### Model for Output in Docs |
|||
|
|||
You can check the output model in the docs too, **both** `name` and `description` are marked as **required** with a **red asterisk**: |
|||
|
|||
<div class="screenshot"> |
|||
<img src="/img/tutorial/separate-openapi-schemas/image03.png"> |
|||
</div> |
|||
|
|||
### Model for Input and Output in Docs |
|||
|
|||
And if you check all the available Schemas (JSON Schemas) in OpenAPI, you will see that there are two, one `Item-Input` and one `Item-Output`. |
|||
|
|||
For `Item-Input`, `description` is **not required**, it doesn't have a red asterisk. |
|||
|
|||
But for `Item-Output`, `description` is **required**, it has a red asterisk. |
|||
|
|||
<div class="screenshot"> |
|||
<img src="/img/tutorial/separate-openapi-schemas/image04.png"> |
|||
</div> |
|||
|
|||
With this feature from **Pydantic v2**, your API documentation is more **precise**, and if you have autogenerated clients and SDKs, they will be more precise too, with a better **developer experience** and consistency. 🎉 |
|||
|
|||
## Do not Separate Schemas |
|||
|
|||
Now, there are some cases where you might want to have the **same schema for input and output**. |
|||
|
|||
Probably the main use case for this is if you already have some autogenerated client code/SDKs and you don't want to update all the autogenerated client code/SDKs yet, you probably will want to do it at some point, but maybe not right now. |
|||
|
|||
In that case, you can disable this feature in **FastAPI**, with the parameter `separate_input_output_schemas=False`. |
|||
|
|||
=== "Python 3.10+" |
|||
|
|||
```Python hl_lines="10" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial002_py310.py!} |
|||
``` |
|||
|
|||
=== "Python 3.9+" |
|||
|
|||
```Python hl_lines="12" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial002_py39.py!} |
|||
``` |
|||
|
|||
=== "Python 3.7+" |
|||
|
|||
```Python hl_lines="12" |
|||
{!> ../../../docs_src/separate_openapi_schemas/tutorial002.py!} |
|||
``` |
|||
|
|||
### Same Schema for Input and Output Models in Docs |
|||
|
|||
And now there will be one single schema for input and output for the model, only `Item`, and it will have `description` as **not required**: |
|||
|
|||
<div class="screenshot"> |
|||
<img src="/img/tutorial/separate-openapi-schemas/image05.png"> |
|||
</div> |
|||
|
|||
This is the same behavior as in Pydantic v1. 🤓 |
After Width: | Height: | Size: 80 KiB |
After Width: | Height: | Size: 90 KiB |
After Width: | Height: | Size: 83 KiB |
After Width: | Height: | Size: 57 KiB |
After Width: | Height: | Size: 44 KiB |
@ -0,0 +1,28 @@ |
|||
from typing import List, Union |
|||
|
|||
from fastapi import FastAPI |
|||
from pydantic import BaseModel |
|||
|
|||
|
|||
class Item(BaseModel): |
|||
name: str |
|||
description: Union[str, None] = None |
|||
|
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
@app.post("/items/") |
|||
def create_item(item: Item): |
|||
return item |
|||
|
|||
|
|||
@app.get("/items/") |
|||
def read_items() -> List[Item]: |
|||
return [ |
|||
Item( |
|||
name="Portal Gun", |
|||
description="Device to travel through the multi-rick-verse", |
|||
), |
|||
Item(name="Plumbus"), |
|||
] |
@ -0,0 +1,26 @@ |
|||
from fastapi import FastAPI |
|||
from pydantic import BaseModel |
|||
|
|||
|
|||
class Item(BaseModel): |
|||
name: str |
|||
description: str | None = None |
|||
|
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
@app.post("/items/") |
|||
def create_item(item: Item): |
|||
return item |
|||
|
|||
|
|||
@app.get("/items/") |
|||
def read_items() -> list[Item]: |
|||
return [ |
|||
Item( |
|||
name="Portal Gun", |
|||
description="Device to travel through the multi-rick-verse", |
|||
), |
|||
Item(name="Plumbus"), |
|||
] |
@ -0,0 +1,28 @@ |
|||
from typing import Optional |
|||
|
|||
from fastapi import FastAPI |
|||
from pydantic import BaseModel |
|||
|
|||
|
|||
class Item(BaseModel): |
|||
name: str |
|||
description: Optional[str] = None |
|||
|
|||
|
|||
app = FastAPI() |
|||
|
|||
|
|||
@app.post("/items/") |
|||
def create_item(item: Item): |
|||
return item |
|||
|
|||
|
|||
@app.get("/items/") |
|||
def read_items() -> list[Item]: |
|||
return [ |
|||
Item( |
|||
name="Portal Gun", |
|||
description="Device to travel through the multi-rick-verse", |
|||
), |
|||
Item(name="Plumbus"), |
|||
] |
@ -0,0 +1,28 @@ |
|||
from typing import List, Union |
|||
|
|||
from fastapi import FastAPI |
|||
from pydantic import BaseModel |
|||
|
|||
|
|||
class Item(BaseModel): |
|||
name: str |
|||
description: Union[str, None] = None |
|||
|
|||
|
|||
app = FastAPI(separate_input_output_schemas=False) |
|||
|
|||
|
|||
@app.post("/items/") |
|||
def create_item(item: Item): |
|||
return item |
|||
|
|||
|
|||
@app.get("/items/") |
|||
def read_items() -> List[Item]: |
|||
return [ |
|||
Item( |
|||
name="Portal Gun", |
|||
description="Device to travel through the multi-rick-verse", |
|||
), |
|||
Item(name="Plumbus"), |
|||
] |
@ -0,0 +1,26 @@ |
|||
from fastapi import FastAPI |
|||
from pydantic import BaseModel |
|||
|
|||
|
|||
class Item(BaseModel): |
|||
name: str |
|||
description: str | None = None |
|||
|
|||
|
|||
app = FastAPI(separate_input_output_schemas=False) |
|||
|
|||
|
|||
@app.post("/items/") |
|||
def create_item(item: Item): |
|||
return item |
|||
|
|||
|
|||
@app.get("/items/") |
|||
def read_items() -> list[Item]: |
|||
return [ |
|||
Item( |
|||
name="Portal Gun", |
|||
description="Device to travel through the multi-rick-verse", |
|||
), |
|||
Item(name="Plumbus"), |
|||
] |
@ -0,0 +1,28 @@ |
|||
from typing import Optional |
|||
|
|||
from fastapi import FastAPI |
|||
from pydantic import BaseModel |
|||
|
|||
|
|||
class Item(BaseModel): |
|||
name: str |
|||
description: Optional[str] = None |
|||
|
|||
|
|||
app = FastAPI(separate_input_output_schemas=False) |
|||
|
|||
|
|||
@app.post("/items/") |
|||
def create_item(item: Item): |
|||
return item |
|||
|
|||
|
|||
@app.get("/items/") |
|||
def read_items() -> list[Item]: |
|||
return [ |
|||
Item( |
|||
name="Portal Gun", |
|||
description="Device to travel through the multi-rick-verse", |
|||
), |
|||
Item(name="Plumbus"), |
|||
] |
@ -0,0 +1,29 @@ |
|||
import subprocess |
|||
|
|||
from playwright.sync_api import Playwright, sync_playwright |
|||
|
|||
|
|||
def run(playwright: Playwright) -> None: |
|||
browser = playwright.chromium.launch(headless=False) |
|||
context = browser.new_context(viewport={"width": 960, "height": 1080}) |
|||
page = context.new_page() |
|||
page.goto("http://localhost:8000/docs") |
|||
page.get_by_text("POST/items/Create Item").click() |
|||
page.get_by_role("tab", name="Schema").first.click() |
|||
page.screenshot( |
|||
path="docs/en/docs/img/tutorial/separate-openapi-schemas/image01.png" |
|||
) |
|||
|
|||
# --------------------- |
|||
context.close() |
|||
browser.close() |
|||
|
|||
|
|||
process = subprocess.Popen( |
|||
["uvicorn", "docs_src.separate_openapi_schemas.tutorial001:app"] |
|||
) |
|||
try: |
|||
with sync_playwright() as playwright: |
|||
run(playwright) |
|||
finally: |
|||
process.terminate() |
@ -0,0 +1,30 @@ |
|||
import subprocess |
|||
|
|||
from playwright.sync_api import Playwright, sync_playwright |
|||
|
|||
|
|||
def run(playwright: Playwright) -> None: |
|||
browser = playwright.chromium.launch(headless=False) |
|||
context = browser.new_context(viewport={"width": 960, "height": 1080}) |
|||
page = context.new_page() |
|||
page.goto("http://localhost:8000/docs") |
|||
page.get_by_text("GET/items/Read Items").click() |
|||
page.get_by_role("button", name="Try it out").click() |
|||
page.get_by_role("button", name="Execute").click() |
|||
page.screenshot( |
|||
path="docs/en/docs/img/tutorial/separate-openapi-schemas/image02.png" |
|||
) |
|||
|
|||
# --------------------- |
|||
context.close() |
|||
browser.close() |
|||
|
|||
|
|||
process = subprocess.Popen( |
|||
["uvicorn", "docs_src.separate_openapi_schemas.tutorial001:app"] |
|||
) |
|||
try: |
|||
with sync_playwright() as playwright: |
|||
run(playwright) |
|||
finally: |
|||
process.terminate() |
@ -0,0 +1,30 @@ |
|||
import subprocess |
|||
|
|||
from playwright.sync_api import Playwright, sync_playwright |
|||
|
|||
|
|||
def run(playwright: Playwright) -> None: |
|||
browser = playwright.chromium.launch(headless=False) |
|||
context = browser.new_context(viewport={"width": 960, "height": 1080}) |
|||
page = context.new_page() |
|||
page.goto("http://localhost:8000/docs") |
|||
page.get_by_text("GET/items/Read Items").click() |
|||
page.get_by_role("tab", name="Schema").click() |
|||
page.get_by_label("Schema").get_by_role("button", name="Expand all").click() |
|||
page.screenshot( |
|||
path="docs/en/docs/img/tutorial/separate-openapi-schemas/image03.png" |
|||
) |
|||
|
|||
# --------------------- |
|||
context.close() |
|||
browser.close() |
|||
|
|||
|
|||
process = subprocess.Popen( |
|||
["uvicorn", "docs_src.separate_openapi_schemas.tutorial001:app"] |
|||
) |
|||
try: |
|||
with sync_playwright() as playwright: |
|||
run(playwright) |
|||
finally: |
|||
process.terminate() |
@ -0,0 +1,29 @@ |
|||
import subprocess |
|||
|
|||
from playwright.sync_api import Playwright, sync_playwright |
|||
|
|||
|
|||
def run(playwright: Playwright) -> None: |
|||
browser = playwright.chromium.launch(headless=False) |
|||
context = browser.new_context(viewport={"width": 960, "height": 1080}) |
|||
page = context.new_page() |
|||
page.goto("http://localhost:8000/docs") |
|||
page.get_by_role("button", name="Item-Input").click() |
|||
page.get_by_role("button", name="Item-Output").click() |
|||
page.set_viewport_size({"width": 960, "height": 820}) |
|||
page.screenshot( |
|||
path="docs/en/docs/img/tutorial/separate-openapi-schemas/image04.png" |
|||
) |
|||
# --------------------- |
|||
context.close() |
|||
browser.close() |
|||
|
|||
|
|||
process = subprocess.Popen( |
|||
["uvicorn", "docs_src.separate_openapi_schemas.tutorial001:app"] |
|||
) |
|||
try: |
|||
with sync_playwright() as playwright: |
|||
run(playwright) |
|||
finally: |
|||
process.terminate() |
@ -0,0 +1,29 @@ |
|||
import subprocess |
|||
|
|||
from playwright.sync_api import Playwright, sync_playwright |
|||
|
|||
|
|||
def run(playwright: Playwright) -> None: |
|||
browser = playwright.chromium.launch(headless=False) |
|||
context = browser.new_context(viewport={"width": 960, "height": 1080}) |
|||
page = context.new_page() |
|||
page.goto("http://localhost:8000/docs") |
|||
page.get_by_role("button", name="Item", exact=True).click() |
|||
page.set_viewport_size({"width": 960, "height": 700}) |
|||
page.screenshot( |
|||
path="docs/en/docs/img/tutorial/separate-openapi-schemas/image05.png" |
|||
) |
|||
|
|||
# --------------------- |
|||
context.close() |
|||
browser.close() |
|||
|
|||
|
|||
process = subprocess.Popen( |
|||
["uvicorn", "docs_src.separate_openapi_schemas.tutorial002:app"] |
|||
) |
|||
try: |
|||
with sync_playwright() as playwright: |
|||
run(playwright) |
|||
finally: |
|||
process.terminate() |
@ -0,0 +1,490 @@ |
|||
from typing import List, Optional |
|||
|
|||
from fastapi import FastAPI |
|||
from fastapi.testclient import TestClient |
|||
from pydantic import BaseModel |
|||
|
|||
from .utils import needs_pydanticv2 |
|||
|
|||
|
|||
class SubItem(BaseModel): |
|||
subname: str |
|||
sub_description: Optional[str] = None |
|||
tags: List[str] = [] |
|||
|
|||
|
|||
class Item(BaseModel): |
|||
name: str |
|||
description: Optional[str] = None |
|||
sub: Optional[SubItem] = None |
|||
|
|||
|
|||
def get_app_client(separate_input_output_schemas: bool = True) -> TestClient: |
|||
app = FastAPI(separate_input_output_schemas=separate_input_output_schemas) |
|||
|
|||
@app.post("/items/") |
|||
def create_item(item: Item): |
|||
return item |
|||
|
|||
@app.post("/items-list/") |
|||
def create_item_list(item: List[Item]): |
|||
return item |
|||
|
|||
@app.get("/items/") |
|||
def read_items() -> List[Item]: |
|||
return [ |
|||
Item( |
|||
name="Portal Gun", |
|||
description="Device to travel through the multi-rick-verse", |
|||
sub=SubItem(subname="subname"), |
|||
), |
|||
Item(name="Plumbus"), |
|||
] |
|||
|
|||
client = TestClient(app) |
|||
return client |
|||
|
|||
|
|||
def test_create_item(): |
|||
client = get_app_client() |
|||
client_no = get_app_client(separate_input_output_schemas=False) |
|||
response = client.post("/items/", json={"name": "Plumbus"}) |
|||
response2 = client_no.post("/items/", json={"name": "Plumbus"}) |
|||
assert response.status_code == response2.status_code == 200, response.text |
|||
assert ( |
|||
response.json() |
|||
== response2.json() |
|||
== {"name": "Plumbus", "description": None, "sub": None} |
|||
) |
|||
|
|||
|
|||
def test_create_item_with_sub(): |
|||
client = get_app_client() |
|||
client_no = get_app_client(separate_input_output_schemas=False) |
|||
data = { |
|||
"name": "Plumbus", |
|||
"sub": {"subname": "SubPlumbus", "sub_description": "Sub WTF"}, |
|||
} |
|||
response = client.post("/items/", json=data) |
|||
response2 = client_no.post("/items/", json=data) |
|||
assert response.status_code == response2.status_code == 200, response.text |
|||
assert ( |
|||
response.json() |
|||
== response2.json() |
|||
== { |
|||
"name": "Plumbus", |
|||
"description": None, |
|||
"sub": {"subname": "SubPlumbus", "sub_description": "Sub WTF", "tags": []}, |
|||
} |
|||
) |
|||
|
|||
|
|||
def test_create_item_list(): |
|||
client = get_app_client() |
|||
client_no = get_app_client(separate_input_output_schemas=False) |
|||
data = [ |
|||
{"name": "Plumbus"}, |
|||
{ |
|||
"name": "Portal Gun", |
|||
"description": "Device to travel through the multi-rick-verse", |
|||
}, |
|||
] |
|||
response = client.post("/items-list/", json=data) |
|||
response2 = client_no.post("/items-list/", json=data) |
|||
assert response.status_code == response2.status_code == 200, response.text |
|||
assert ( |
|||
response.json() |
|||
== response2.json() |
|||
== [ |
|||
{"name": "Plumbus", "description": None, "sub": None}, |
|||
{ |
|||
"name": "Portal Gun", |
|||
"description": "Device to travel through the multi-rick-verse", |
|||
"sub": None, |
|||
}, |
|||
] |
|||
) |
|||
|
|||
|
|||
def test_read_items(): |
|||
client = get_app_client() |
|||
client_no = get_app_client(separate_input_output_schemas=False) |
|||
response = client.get("/items/") |
|||
response2 = client_no.get("/items/") |
|||
assert response.status_code == response2.status_code == 200, response.text |
|||
assert ( |
|||
response.json() |
|||
== response2.json() |
|||
== [ |
|||
{ |
|||
"name": "Portal Gun", |
|||
"description": "Device to travel through the multi-rick-verse", |
|||
"sub": {"subname": "subname", "sub_description": None, "tags": []}, |
|||
}, |
|||
{"name": "Plumbus", "description": None, "sub": None}, |
|||
] |
|||
) |
|||
|
|||
|
|||
@needs_pydanticv2 |
|||
def test_openapi_schema(): |
|||
client = get_app_client() |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == { |
|||
"openapi": "3.1.0", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"get": { |
|||
"summary": "Read Items", |
|||
"operationId": "read_items_items__get", |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"items": { |
|||
"$ref": "#/components/schemas/Item-Output" |
|||
}, |
|||
"type": "array", |
|||
"title": "Response Read Items Items Get", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"post": { |
|||
"summary": "Create Item", |
|||
"operationId": "create_item_items__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/Item-Input"} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
"/items-list/": { |
|||
"post": { |
|||
"summary": "Create Item List", |
|||
"operationId": "create_item_list_items_list__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"items": { |
|||
"$ref": "#/components/schemas/Item-Input" |
|||
}, |
|||
"type": "array", |
|||
"title": "Item", |
|||
} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"components": { |
|||
"schemas": { |
|||
"HTTPValidationError": { |
|||
"properties": { |
|||
"detail": { |
|||
"items": {"$ref": "#/components/schemas/ValidationError"}, |
|||
"type": "array", |
|||
"title": "Detail", |
|||
} |
|||
}, |
|||
"type": "object", |
|||
"title": "HTTPValidationError", |
|||
}, |
|||
"Item-Input": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
"sub": { |
|||
"anyOf": [ |
|||
{"$ref": "#/components/schemas/SubItem-Input"}, |
|||
{"type": "null"}, |
|||
] |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name"], |
|||
"title": "Item", |
|||
}, |
|||
"Item-Output": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
"sub": { |
|||
"anyOf": [ |
|||
{"$ref": "#/components/schemas/SubItem-Output"}, |
|||
{"type": "null"}, |
|||
] |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name", "description", "sub"], |
|||
"title": "Item", |
|||
}, |
|||
"SubItem-Input": { |
|||
"properties": { |
|||
"subname": {"type": "string", "title": "Subname"}, |
|||
"sub_description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Sub Description", |
|||
}, |
|||
"tags": { |
|||
"items": {"type": "string"}, |
|||
"type": "array", |
|||
"title": "Tags", |
|||
"default": [], |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["subname"], |
|||
"title": "SubItem", |
|||
}, |
|||
"SubItem-Output": { |
|||
"properties": { |
|||
"subname": {"type": "string", "title": "Subname"}, |
|||
"sub_description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Sub Description", |
|||
}, |
|||
"tags": { |
|||
"items": {"type": "string"}, |
|||
"type": "array", |
|||
"title": "Tags", |
|||
"default": [], |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["subname", "sub_description", "tags"], |
|||
"title": "SubItem", |
|||
}, |
|||
"ValidationError": { |
|||
"properties": { |
|||
"loc": { |
|||
"items": { |
|||
"anyOf": [{"type": "string"}, {"type": "integer"}] |
|||
}, |
|||
"type": "array", |
|||
"title": "Location", |
|||
}, |
|||
"msg": {"type": "string", "title": "Message"}, |
|||
"type": {"type": "string", "title": "Error Type"}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["loc", "msg", "type"], |
|||
"title": "ValidationError", |
|||
}, |
|||
} |
|||
}, |
|||
} |
|||
|
|||
|
|||
@needs_pydanticv2 |
|||
def test_openapi_schema_no_separate(): |
|||
client = get_app_client(separate_input_output_schemas=False) |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == { |
|||
"openapi": "3.1.0", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"get": { |
|||
"summary": "Read Items", |
|||
"operationId": "read_items_items__get", |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"items": {"$ref": "#/components/schemas/Item"}, |
|||
"type": "array", |
|||
"title": "Response Read Items Items Get", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"post": { |
|||
"summary": "Create Item", |
|||
"operationId": "create_item_items__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/Item"} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
"/items-list/": { |
|||
"post": { |
|||
"summary": "Create Item List", |
|||
"operationId": "create_item_list_items_list__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"items": {"$ref": "#/components/schemas/Item"}, |
|||
"type": "array", |
|||
"title": "Item", |
|||
} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"components": { |
|||
"schemas": { |
|||
"HTTPValidationError": { |
|||
"properties": { |
|||
"detail": { |
|||
"items": {"$ref": "#/components/schemas/ValidationError"}, |
|||
"type": "array", |
|||
"title": "Detail", |
|||
} |
|||
}, |
|||
"type": "object", |
|||
"title": "HTTPValidationError", |
|||
}, |
|||
"Item": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
"sub": { |
|||
"anyOf": [ |
|||
{"$ref": "#/components/schemas/SubItem"}, |
|||
{"type": "null"}, |
|||
] |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name"], |
|||
"title": "Item", |
|||
}, |
|||
"SubItem": { |
|||
"properties": { |
|||
"subname": {"type": "string", "title": "Subname"}, |
|||
"sub_description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Sub Description", |
|||
}, |
|||
"tags": { |
|||
"items": {"type": "string"}, |
|||
"type": "array", |
|||
"title": "Tags", |
|||
"default": [], |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["subname"], |
|||
"title": "SubItem", |
|||
}, |
|||
"ValidationError": { |
|||
"properties": { |
|||
"loc": { |
|||
"items": { |
|||
"anyOf": [{"type": "string"}, {"type": "integer"}] |
|||
}, |
|||
"type": "array", |
|||
"title": "Location", |
|||
}, |
|||
"msg": {"type": "string", "title": "Message"}, |
|||
"type": {"type": "string", "title": "Error Type"}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["loc", "msg", "type"], |
|||
"title": "ValidationError", |
|||
}, |
|||
} |
|||
}, |
|||
} |
@ -0,0 +1,147 @@ |
|||
import pytest |
|||
from fastapi.testclient import TestClient |
|||
|
|||
from ...utils import needs_pydanticv2 |
|||
|
|||
|
|||
@pytest.fixture(name="client") |
|||
def get_client() -> TestClient: |
|||
from docs_src.separate_openapi_schemas.tutorial001 import app |
|||
|
|||
client = TestClient(app) |
|||
return client |
|||
|
|||
|
|||
def test_create_item(client: TestClient) -> None: |
|||
response = client.post("/items/", json={"name": "Foo"}) |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == {"name": "Foo", "description": None} |
|||
|
|||
|
|||
def test_read_items(client: TestClient) -> None: |
|||
response = client.get("/items/") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == [ |
|||
{ |
|||
"name": "Portal Gun", |
|||
"description": "Device to travel through the multi-rick-verse", |
|||
}, |
|||
{"name": "Plumbus", "description": None}, |
|||
] |
|||
|
|||
|
|||
@needs_pydanticv2 |
|||
def test_openapi_schema(client: TestClient) -> None: |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == { |
|||
"openapi": "3.1.0", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"get": { |
|||
"summary": "Read Items", |
|||
"operationId": "read_items_items__get", |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"items": { |
|||
"$ref": "#/components/schemas/Item-Output" |
|||
}, |
|||
"type": "array", |
|||
"title": "Response Read Items Items Get", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"post": { |
|||
"summary": "Create Item", |
|||
"operationId": "create_item_items__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/Item-Input"} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
"components": { |
|||
"schemas": { |
|||
"HTTPValidationError": { |
|||
"properties": { |
|||
"detail": { |
|||
"items": {"$ref": "#/components/schemas/ValidationError"}, |
|||
"type": "array", |
|||
"title": "Detail", |
|||
} |
|||
}, |
|||
"type": "object", |
|||
"title": "HTTPValidationError", |
|||
}, |
|||
"Item-Input": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name"], |
|||
"title": "Item", |
|||
}, |
|||
"Item-Output": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name", "description"], |
|||
"title": "Item", |
|||
}, |
|||
"ValidationError": { |
|||
"properties": { |
|||
"loc": { |
|||
"items": { |
|||
"anyOf": [{"type": "string"}, {"type": "integer"}] |
|||
}, |
|||
"type": "array", |
|||
"title": "Location", |
|||
}, |
|||
"msg": {"type": "string", "title": "Message"}, |
|||
"type": {"type": "string", "title": "Error Type"}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["loc", "msg", "type"], |
|||
"title": "ValidationError", |
|||
}, |
|||
} |
|||
}, |
|||
} |
@ -0,0 +1,150 @@ |
|||
import pytest |
|||
from fastapi.testclient import TestClient |
|||
|
|||
from ...utils import needs_py310, needs_pydanticv2 |
|||
|
|||
|
|||
@pytest.fixture(name="client") |
|||
def get_client() -> TestClient: |
|||
from docs_src.separate_openapi_schemas.tutorial001_py310 import app |
|||
|
|||
client = TestClient(app) |
|||
return client |
|||
|
|||
|
|||
@needs_py310 |
|||
def test_create_item(client: TestClient) -> None: |
|||
response = client.post("/items/", json={"name": "Foo"}) |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == {"name": "Foo", "description": None} |
|||
|
|||
|
|||
@needs_py310 |
|||
def test_read_items(client: TestClient) -> None: |
|||
response = client.get("/items/") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == [ |
|||
{ |
|||
"name": "Portal Gun", |
|||
"description": "Device to travel through the multi-rick-verse", |
|||
}, |
|||
{"name": "Plumbus", "description": None}, |
|||
] |
|||
|
|||
|
|||
@needs_py310 |
|||
@needs_pydanticv2 |
|||
def test_openapi_schema(client: TestClient) -> None: |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == { |
|||
"openapi": "3.1.0", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"get": { |
|||
"summary": "Read Items", |
|||
"operationId": "read_items_items__get", |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"items": { |
|||
"$ref": "#/components/schemas/Item-Output" |
|||
}, |
|||
"type": "array", |
|||
"title": "Response Read Items Items Get", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"post": { |
|||
"summary": "Create Item", |
|||
"operationId": "create_item_items__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/Item-Input"} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
"components": { |
|||
"schemas": { |
|||
"HTTPValidationError": { |
|||
"properties": { |
|||
"detail": { |
|||
"items": {"$ref": "#/components/schemas/ValidationError"}, |
|||
"type": "array", |
|||
"title": "Detail", |
|||
} |
|||
}, |
|||
"type": "object", |
|||
"title": "HTTPValidationError", |
|||
}, |
|||
"Item-Input": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name"], |
|||
"title": "Item", |
|||
}, |
|||
"Item-Output": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name", "description"], |
|||
"title": "Item", |
|||
}, |
|||
"ValidationError": { |
|||
"properties": { |
|||
"loc": { |
|||
"items": { |
|||
"anyOf": [{"type": "string"}, {"type": "integer"}] |
|||
}, |
|||
"type": "array", |
|||
"title": "Location", |
|||
}, |
|||
"msg": {"type": "string", "title": "Message"}, |
|||
"type": {"type": "string", "title": "Error Type"}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["loc", "msg", "type"], |
|||
"title": "ValidationError", |
|||
}, |
|||
} |
|||
}, |
|||
} |
@ -0,0 +1,150 @@ |
|||
import pytest |
|||
from fastapi.testclient import TestClient |
|||
|
|||
from ...utils import needs_py39, needs_pydanticv2 |
|||
|
|||
|
|||
@pytest.fixture(name="client") |
|||
def get_client() -> TestClient: |
|||
from docs_src.separate_openapi_schemas.tutorial001_py39 import app |
|||
|
|||
client = TestClient(app) |
|||
return client |
|||
|
|||
|
|||
@needs_py39 |
|||
def test_create_item(client: TestClient) -> None: |
|||
response = client.post("/items/", json={"name": "Foo"}) |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == {"name": "Foo", "description": None} |
|||
|
|||
|
|||
@needs_py39 |
|||
def test_read_items(client: TestClient) -> None: |
|||
response = client.get("/items/") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == [ |
|||
{ |
|||
"name": "Portal Gun", |
|||
"description": "Device to travel through the multi-rick-verse", |
|||
}, |
|||
{"name": "Plumbus", "description": None}, |
|||
] |
|||
|
|||
|
|||
@needs_py39 |
|||
@needs_pydanticv2 |
|||
def test_openapi_schema(client: TestClient) -> None: |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == { |
|||
"openapi": "3.1.0", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"get": { |
|||
"summary": "Read Items", |
|||
"operationId": "read_items_items__get", |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"items": { |
|||
"$ref": "#/components/schemas/Item-Output" |
|||
}, |
|||
"type": "array", |
|||
"title": "Response Read Items Items Get", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"post": { |
|||
"summary": "Create Item", |
|||
"operationId": "create_item_items__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/Item-Input"} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
"components": { |
|||
"schemas": { |
|||
"HTTPValidationError": { |
|||
"properties": { |
|||
"detail": { |
|||
"items": {"$ref": "#/components/schemas/ValidationError"}, |
|||
"type": "array", |
|||
"title": "Detail", |
|||
} |
|||
}, |
|||
"type": "object", |
|||
"title": "HTTPValidationError", |
|||
}, |
|||
"Item-Input": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name"], |
|||
"title": "Item", |
|||
}, |
|||
"Item-Output": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name", "description"], |
|||
"title": "Item", |
|||
}, |
|||
"ValidationError": { |
|||
"properties": { |
|||
"loc": { |
|||
"items": { |
|||
"anyOf": [{"type": "string"}, {"type": "integer"}] |
|||
}, |
|||
"type": "array", |
|||
"title": "Location", |
|||
}, |
|||
"msg": {"type": "string", "title": "Message"}, |
|||
"type": {"type": "string", "title": "Error Type"}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["loc", "msg", "type"], |
|||
"title": "ValidationError", |
|||
}, |
|||
} |
|||
}, |
|||
} |
@ -0,0 +1,133 @@ |
|||
import pytest |
|||
from fastapi.testclient import TestClient |
|||
|
|||
from ...utils import needs_pydanticv2 |
|||
|
|||
|
|||
@pytest.fixture(name="client") |
|||
def get_client() -> TestClient: |
|||
from docs_src.separate_openapi_schemas.tutorial002 import app |
|||
|
|||
client = TestClient(app) |
|||
return client |
|||
|
|||
|
|||
def test_create_item(client: TestClient) -> None: |
|||
response = client.post("/items/", json={"name": "Foo"}) |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == {"name": "Foo", "description": None} |
|||
|
|||
|
|||
def test_read_items(client: TestClient) -> None: |
|||
response = client.get("/items/") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == [ |
|||
{ |
|||
"name": "Portal Gun", |
|||
"description": "Device to travel through the multi-rick-verse", |
|||
}, |
|||
{"name": "Plumbus", "description": None}, |
|||
] |
|||
|
|||
|
|||
@needs_pydanticv2 |
|||
def test_openapi_schema(client: TestClient) -> None: |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == { |
|||
"openapi": "3.1.0", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"get": { |
|||
"summary": "Read Items", |
|||
"operationId": "read_items_items__get", |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"items": {"$ref": "#/components/schemas/Item"}, |
|||
"type": "array", |
|||
"title": "Response Read Items Items Get", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"post": { |
|||
"summary": "Create Item", |
|||
"operationId": "create_item_items__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/Item"} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
"components": { |
|||
"schemas": { |
|||
"HTTPValidationError": { |
|||
"properties": { |
|||
"detail": { |
|||
"items": {"$ref": "#/components/schemas/ValidationError"}, |
|||
"type": "array", |
|||
"title": "Detail", |
|||
} |
|||
}, |
|||
"type": "object", |
|||
"title": "HTTPValidationError", |
|||
}, |
|||
"Item": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name"], |
|||
"title": "Item", |
|||
}, |
|||
"ValidationError": { |
|||
"properties": { |
|||
"loc": { |
|||
"items": { |
|||
"anyOf": [{"type": "string"}, {"type": "integer"}] |
|||
}, |
|||
"type": "array", |
|||
"title": "Location", |
|||
}, |
|||
"msg": {"type": "string", "title": "Message"}, |
|||
"type": {"type": "string", "title": "Error Type"}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["loc", "msg", "type"], |
|||
"title": "ValidationError", |
|||
}, |
|||
} |
|||
}, |
|||
} |
@ -0,0 +1,136 @@ |
|||
import pytest |
|||
from fastapi.testclient import TestClient |
|||
|
|||
from ...utils import needs_py310, needs_pydanticv2 |
|||
|
|||
|
|||
@pytest.fixture(name="client") |
|||
def get_client() -> TestClient: |
|||
from docs_src.separate_openapi_schemas.tutorial002_py310 import app |
|||
|
|||
client = TestClient(app) |
|||
return client |
|||
|
|||
|
|||
@needs_py310 |
|||
def test_create_item(client: TestClient) -> None: |
|||
response = client.post("/items/", json={"name": "Foo"}) |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == {"name": "Foo", "description": None} |
|||
|
|||
|
|||
@needs_py310 |
|||
def test_read_items(client: TestClient) -> None: |
|||
response = client.get("/items/") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == [ |
|||
{ |
|||
"name": "Portal Gun", |
|||
"description": "Device to travel through the multi-rick-verse", |
|||
}, |
|||
{"name": "Plumbus", "description": None}, |
|||
] |
|||
|
|||
|
|||
@needs_py310 |
|||
@needs_pydanticv2 |
|||
def test_openapi_schema(client: TestClient) -> None: |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == { |
|||
"openapi": "3.1.0", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"get": { |
|||
"summary": "Read Items", |
|||
"operationId": "read_items_items__get", |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"items": {"$ref": "#/components/schemas/Item"}, |
|||
"type": "array", |
|||
"title": "Response Read Items Items Get", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"post": { |
|||
"summary": "Create Item", |
|||
"operationId": "create_item_items__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/Item"} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
"components": { |
|||
"schemas": { |
|||
"HTTPValidationError": { |
|||
"properties": { |
|||
"detail": { |
|||
"items": {"$ref": "#/components/schemas/ValidationError"}, |
|||
"type": "array", |
|||
"title": "Detail", |
|||
} |
|||
}, |
|||
"type": "object", |
|||
"title": "HTTPValidationError", |
|||
}, |
|||
"Item": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name"], |
|||
"title": "Item", |
|||
}, |
|||
"ValidationError": { |
|||
"properties": { |
|||
"loc": { |
|||
"items": { |
|||
"anyOf": [{"type": "string"}, {"type": "integer"}] |
|||
}, |
|||
"type": "array", |
|||
"title": "Location", |
|||
}, |
|||
"msg": {"type": "string", "title": "Message"}, |
|||
"type": {"type": "string", "title": "Error Type"}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["loc", "msg", "type"], |
|||
"title": "ValidationError", |
|||
}, |
|||
} |
|||
}, |
|||
} |
@ -0,0 +1,136 @@ |
|||
import pytest |
|||
from fastapi.testclient import TestClient |
|||
|
|||
from ...utils import needs_py39, needs_pydanticv2 |
|||
|
|||
|
|||
@pytest.fixture(name="client") |
|||
def get_client() -> TestClient: |
|||
from docs_src.separate_openapi_schemas.tutorial002_py39 import app |
|||
|
|||
client = TestClient(app) |
|||
return client |
|||
|
|||
|
|||
@needs_py39 |
|||
def test_create_item(client: TestClient) -> None: |
|||
response = client.post("/items/", json={"name": "Foo"}) |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == {"name": "Foo", "description": None} |
|||
|
|||
|
|||
@needs_py39 |
|||
def test_read_items(client: TestClient) -> None: |
|||
response = client.get("/items/") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == [ |
|||
{ |
|||
"name": "Portal Gun", |
|||
"description": "Device to travel through the multi-rick-verse", |
|||
}, |
|||
{"name": "Plumbus", "description": None}, |
|||
] |
|||
|
|||
|
|||
@needs_py39 |
|||
@needs_pydanticv2 |
|||
def test_openapi_schema(client: TestClient) -> None: |
|||
response = client.get("/openapi.json") |
|||
assert response.status_code == 200, response.text |
|||
assert response.json() == { |
|||
"openapi": "3.1.0", |
|||
"info": {"title": "FastAPI", "version": "0.1.0"}, |
|||
"paths": { |
|||
"/items/": { |
|||
"get": { |
|||
"summary": "Read Items", |
|||
"operationId": "read_items_items__get", |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"items": {"$ref": "#/components/schemas/Item"}, |
|||
"type": "array", |
|||
"title": "Response Read Items Items Get", |
|||
} |
|||
} |
|||
}, |
|||
} |
|||
}, |
|||
}, |
|||
"post": { |
|||
"summary": "Create Item", |
|||
"operationId": "create_item_items__post", |
|||
"requestBody": { |
|||
"content": { |
|||
"application/json": { |
|||
"schema": {"$ref": "#/components/schemas/Item"} |
|||
} |
|||
}, |
|||
"required": True, |
|||
}, |
|||
"responses": { |
|||
"200": { |
|||
"description": "Successful Response", |
|||
"content": {"application/json": {"schema": {}}}, |
|||
}, |
|||
"422": { |
|||
"description": "Validation Error", |
|||
"content": { |
|||
"application/json": { |
|||
"schema": { |
|||
"$ref": "#/components/schemas/HTTPValidationError" |
|||
} |
|||
} |
|||
}, |
|||
}, |
|||
}, |
|||
}, |
|||
} |
|||
}, |
|||
"components": { |
|||
"schemas": { |
|||
"HTTPValidationError": { |
|||
"properties": { |
|||
"detail": { |
|||
"items": {"$ref": "#/components/schemas/ValidationError"}, |
|||
"type": "array", |
|||
"title": "Detail", |
|||
} |
|||
}, |
|||
"type": "object", |
|||
"title": "HTTPValidationError", |
|||
}, |
|||
"Item": { |
|||
"properties": { |
|||
"name": {"type": "string", "title": "Name"}, |
|||
"description": { |
|||
"anyOf": [{"type": "string"}, {"type": "null"}], |
|||
"title": "Description", |
|||
}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["name"], |
|||
"title": "Item", |
|||
}, |
|||
"ValidationError": { |
|||
"properties": { |
|||
"loc": { |
|||
"items": { |
|||
"anyOf": [{"type": "string"}, {"type": "integer"}] |
|||
}, |
|||
"type": "array", |
|||
"title": "Location", |
|||
}, |
|||
"msg": {"type": "string", "title": "Message"}, |
|||
"type": {"type": "string", "title": "Error Type"}, |
|||
}, |
|||
"type": "object", |
|||
"required": ["loc", "msg", "type"], |
|||
"title": "ValidationError", |
|||
}, |
|||
} |
|||
}, |
|||
} |
Loading…
Reference in new issue