committed by
GitHub
10 changed files with 562 additions and 0 deletions
@ -0,0 +1,98 @@ |
|||||
|
# Using Dataclasses |
||||
|
|
||||
|
FastAPI is built on top of **Pydantic**, and I have been showing you how to use Pydantic models to declare requests and responses. |
||||
|
|
||||
|
But FastAPI also supports using <a href="https://docs.python.org/3/library/dataclasses.html" class="external-link" target="_blank">`dataclasses`</a> the same way: |
||||
|
|
||||
|
```Python hl_lines="1 7-12 19-20" |
||||
|
{!../../../docs_src/dataclasses/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
This is still thanks to **Pydantic**, as it has <a href="https://pydantic-docs.helpmanual.io/usage/dataclasses/#use-of-stdlib-dataclasses-with-basemodel" class="external-link" target="_blank">internal support for `dataclasses`</a>. |
||||
|
|
||||
|
So, even with the code above that doesn't use Pydantic explicitly, FastAPI is using Pydantic to convert those standard dataclasses to Pydantic's own flavor of dataclasses. |
||||
|
|
||||
|
And of course, it supports the same: |
||||
|
|
||||
|
* data validation |
||||
|
* data serialization |
||||
|
* data documentation, etc. |
||||
|
|
||||
|
This works the same way as with Pydantic models. And it is actually achieved in the same way underneath, using Pydantic. |
||||
|
|
||||
|
!!! info |
||||
|
Have in mind that dataclasses can't do everything Pydantic models can do. |
||||
|
|
||||
|
So, you might still need to use Pydantic models. |
||||
|
|
||||
|
But if you have a bunch of dataclasses laying around, this is a nice trick to use them to power a web API using FastAPI. 🤓 |
||||
|
|
||||
|
## Dataclasses in `response_model` |
||||
|
|
||||
|
You can also use `dataclasses` in the `response_model` parameter: |
||||
|
|
||||
|
```Python hl_lines="1 7-13 19" |
||||
|
{!../../../docs_src/dataclasses/tutorial002.py!} |
||||
|
``` |
||||
|
|
||||
|
The dataclass will be automatically converted to a Pydantic dataclass. |
||||
|
|
||||
|
This way, its schema will show up in the API docs user interface: |
||||
|
|
||||
|
<img src="/img/tutorial/dataclasses/image01.png"> |
||||
|
|
||||
|
## Dataclasses in Nested Data Structures |
||||
|
|
||||
|
You can also combine `dataclasses` with other type annotations to make nested data structures. |
||||
|
|
||||
|
In some cases, you might still have to use Pydantic's version of `dataclasses`. For example, if you have errors with the automatically generated API documentation. |
||||
|
|
||||
|
In that case, you can simply swap the standard `dataclasses` with `pydantic.dataclasses`, which is a drop-in replacement: |
||||
|
|
||||
|
```{ .python .annotate hl_lines="1 5 8-11 14-17 23-25 28" } |
||||
|
{!../../../docs_src/dataclasses/tutorial003.py!} |
||||
|
``` |
||||
|
|
||||
|
1. We still import `field` from standard `dataclasses`. |
||||
|
|
||||
|
2. `pydantic.dataclasses` is a drop-in replacement for `dataclasses`. |
||||
|
|
||||
|
3. The `Author` dataclass includes a list of `Item` dataclasses. |
||||
|
|
||||
|
4. The `Author` dataclass is used as the `response_model` parameter. |
||||
|
|
||||
|
5. You can use other standard type annotations with dataclasses as the request body. |
||||
|
|
||||
|
In this case, it's a list of `Item` dataclasses. |
||||
|
|
||||
|
6. Here we are returning a dictionary that contains `items` which is a list of dataclasses. |
||||
|
|
||||
|
FastAPI is still capable of <abbr title="converting the data to a format that can be transmitted">serializing</abbr> the data to JSON. |
||||
|
|
||||
|
7. Here the `response_model` is using a type annotation of a list of `Author` dataclasses. |
||||
|
|
||||
|
Again, you can combine `dataclasses` with standard type annotations. |
||||
|
|
||||
|
8. Notice that this *path operation function* uses regular `def` instead of `async def`. |
||||
|
|
||||
|
As always, in FastAPI you can combine `def` and `async def` as needed. |
||||
|
|
||||
|
If you need a refresher about when to use which, check out the section _"In a hurry?"_ in the docs about <a href="https://fastapi.tiangolo.com/async/#in-a-hurry" target="_blank" class="internal-link">`async` and `await`</a>. |
||||
|
|
||||
|
9. This *path operation function* is not returning dataclasses (although it could), but a list of dictionaries with internal data. |
||||
|
|
||||
|
FastAPI will use the `response_model` parameter (that includes dataclasses) to convert the response. |
||||
|
|
||||
|
You can combine `dataclasses` with other type annotations in many different combinations to form complex data structures. |
||||
|
|
||||
|
Check the in-code annotation tips above to see more specific details. |
||||
|
|
||||
|
## Learn More |
||||
|
|
||||
|
You can also combine `dataclasses` with other Pydantic models, inherit from them, include them in your own models, etc. |
||||
|
|
||||
|
To learn more, check the <a href="https://pydantic-docs.helpmanual.io/usage/dataclasses/" class="external-link" target="_blank">Pydantic docs about dataclasses</a>. |
||||
|
|
||||
|
## Version |
||||
|
|
||||
|
This is available since FastAPI version `0.67.0`. 🔖 |
After Width: | Height: | Size: 71 KiB |
@ -0,0 +1,20 @@ |
|||||
|
from dataclasses import dataclass |
||||
|
from typing import Optional |
||||
|
|
||||
|
from fastapi import FastAPI |
||||
|
|
||||
|
|
||||
|
@dataclass |
||||
|
class Item: |
||||
|
name: str |
||||
|
price: float |
||||
|
description: Optional[str] = None |
||||
|
tax: Optional[float] = None |
||||
|
|
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.post("/items/") |
||||
|
async def create_item(item: Item): |
||||
|
return item |
@ -0,0 +1,26 @@ |
|||||
|
from dataclasses import dataclass, field |
||||
|
from typing import List, Optional |
||||
|
|
||||
|
from fastapi import FastAPI |
||||
|
|
||||
|
|
||||
|
@dataclass |
||||
|
class Item: |
||||
|
name: str |
||||
|
price: float |
||||
|
tags: List[str] = field(default_factory=list) |
||||
|
description: Optional[str] = None |
||||
|
tax: Optional[float] = None |
||||
|
|
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/items/next", response_model=Item) |
||||
|
async def read_next_item(): |
||||
|
return { |
||||
|
"name": "Island In The Moon", |
||||
|
"price": 12.99, |
||||
|
"description": "A place to be be playin' and havin' fun", |
||||
|
"tags": ["breater"], |
||||
|
} |
@ -0,0 +1,55 @@ |
|||||
|
from dataclasses import field # (1) |
||||
|
from typing import List, Optional |
||||
|
|
||||
|
from fastapi import FastAPI |
||||
|
from pydantic.dataclasses import dataclass # (2) |
||||
|
|
||||
|
|
||||
|
@dataclass |
||||
|
class Item: |
||||
|
name: str |
||||
|
description: Optional[str] = None |
||||
|
|
||||
|
|
||||
|
@dataclass |
||||
|
class Author: |
||||
|
name: str |
||||
|
items: List[Item] = field(default_factory=list) # (3) |
||||
|
|
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.post("/authors/{author_id}/items/", response_model=Author) # (4) |
||||
|
async def create_author_items(author_id: str, items: List[Item]): # (5) |
||||
|
return {"name": author_id, "items": items} # (6) |
||||
|
|
||||
|
|
||||
|
@app.get("/authors/", response_model=List[Author]) # (7) |
||||
|
def get_authors(): # (8) |
||||
|
return [ # (9) |
||||
|
{ |
||||
|
"name": "Breaters", |
||||
|
"items": [ |
||||
|
{ |
||||
|
"name": "Island In The Moon", |
||||
|
"description": "A place to be be playin' and havin' fun", |
||||
|
}, |
||||
|
{"name": "Holy Buddies"}, |
||||
|
], |
||||
|
}, |
||||
|
{ |
||||
|
"name": "System of an Up", |
||||
|
"items": [ |
||||
|
{ |
||||
|
"name": "Salt", |
||||
|
"description": "The kombucha mushroom people's favorite", |
||||
|
}, |
||||
|
{"name": "Pad Thai"}, |
||||
|
{ |
||||
|
"name": "Lonely Night", |
||||
|
"description": "The mostests lonliest nightiest of allest", |
||||
|
}, |
||||
|
], |
||||
|
}, |
||||
|
] |
@ -0,0 +1,113 @@ |
|||||
|
from fastapi.testclient import TestClient |
||||
|
|
||||
|
from docs_src.dataclasses.tutorial001 import app |
||||
|
|
||||
|
client = TestClient(app) |
||||
|
|
||||
|
openapi_schema = { |
||||
|
"openapi": "3.0.2", |
||||
|
"info": {"title": "FastAPI", "version": "0.1.0"}, |
||||
|
"paths": { |
||||
|
"/items/": { |
||||
|
"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": { |
||||
|
"title": "HTTPValidationError", |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"detail": { |
||||
|
"title": "Detail", |
||||
|
"type": "array", |
||||
|
"items": {"$ref": "#/components/schemas/ValidationError"}, |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
"Item": { |
||||
|
"title": "Item", |
||||
|
"required": ["name", "price"], |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"name": {"title": "Name", "type": "string"}, |
||||
|
"price": {"title": "Price", "type": "number"}, |
||||
|
"description": {"title": "Description", "type": "string"}, |
||||
|
"tax": {"title": "Tax", "type": "number"}, |
||||
|
}, |
||||
|
}, |
||||
|
"ValidationError": { |
||||
|
"title": "ValidationError", |
||||
|
"required": ["loc", "msg", "type"], |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"loc": { |
||||
|
"title": "Location", |
||||
|
"type": "array", |
||||
|
"items": {"type": "string"}, |
||||
|
}, |
||||
|
"msg": {"title": "Message", "type": "string"}, |
||||
|
"type": {"title": "Error Type", "type": "string"}, |
||||
|
}, |
||||
|
}, |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
|
||||
|
|
||||
|
def test_openapi_schema(): |
||||
|
response = client.get("/openapi.json") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == openapi_schema |
||||
|
|
||||
|
|
||||
|
def test_post_item(): |
||||
|
response = client.post("/items/", json={"name": "Foo", "price": 3}) |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == { |
||||
|
"name": "Foo", |
||||
|
"price": 3, |
||||
|
"description": None, |
||||
|
"tax": None, |
||||
|
} |
||||
|
|
||||
|
|
||||
|
def test_post_invalid_item(): |
||||
|
response = client.post("/items/", json={"name": "Foo", "price": "invalid price"}) |
||||
|
assert response.status_code == 422 |
||||
|
assert response.json() == { |
||||
|
"detail": [ |
||||
|
{ |
||||
|
"loc": ["body", "price"], |
||||
|
"msg": "value is not a valid float", |
||||
|
"type": "type_error.float", |
||||
|
} |
||||
|
] |
||||
|
} |
@ -0,0 +1,66 @@ |
|||||
|
from fastapi.testclient import TestClient |
||||
|
|
||||
|
from docs_src.dataclasses.tutorial002 import app |
||||
|
|
||||
|
client = TestClient(app) |
||||
|
|
||||
|
openapi_schema = { |
||||
|
"openapi": "3.0.2", |
||||
|
"info": {"title": "FastAPI", "version": "0.1.0"}, |
||||
|
"paths": { |
||||
|
"/items/next": { |
||||
|
"get": { |
||||
|
"summary": "Read Next Item", |
||||
|
"operationId": "read_next_item_items_next_get", |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": {"$ref": "#/components/schemas/Item"} |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
"components": { |
||||
|
"schemas": { |
||||
|
"Item": { |
||||
|
"title": "Item", |
||||
|
"required": ["name", "price", "tags"], |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"name": {"title": "Name", "type": "string"}, |
||||
|
"price": {"title": "Price", "type": "number"}, |
||||
|
"tags": { |
||||
|
"title": "Tags", |
||||
|
"type": "array", |
||||
|
"items": {"type": "string"}, |
||||
|
}, |
||||
|
"description": {"title": "Description", "type": "string"}, |
||||
|
"tax": {"title": "Tax", "type": "number"}, |
||||
|
}, |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
|
||||
|
|
||||
|
def test_openapi_schema(): |
||||
|
response = client.get("/openapi.json") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == openapi_schema |
||||
|
|
||||
|
|
||||
|
def test_get_item(): |
||||
|
response = client.get("/items/next") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == { |
||||
|
"name": "Island In The Moon", |
||||
|
"price": 12.99, |
||||
|
"description": "A place to be be playin' and havin' fun", |
||||
|
"tags": ["breater"], |
||||
|
"tax": None, |
||||
|
} |
@ -0,0 +1,181 @@ |
|||||
|
from fastapi.testclient import TestClient |
||||
|
|
||||
|
from docs_src.dataclasses.tutorial003 import app |
||||
|
|
||||
|
client = TestClient(app) |
||||
|
|
||||
|
openapi_schema = { |
||||
|
"openapi": "3.0.2", |
||||
|
"info": {"title": "FastAPI", "version": "0.1.0"}, |
||||
|
"paths": { |
||||
|
"/authors/{author_id}/items/": { |
||||
|
"post": { |
||||
|
"summary": "Create Author Items", |
||||
|
"operationId": "create_author_items_authors__author_id__items__post", |
||||
|
"parameters": [ |
||||
|
{ |
||||
|
"required": True, |
||||
|
"schema": {"title": "Author Id", "type": "string"}, |
||||
|
"name": "author_id", |
||||
|
"in": "path", |
||||
|
} |
||||
|
], |
||||
|
"requestBody": { |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": { |
||||
|
"title": "Items", |
||||
|
"type": "array", |
||||
|
"items": {"$ref": "#/components/schemas/Item"}, |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
"required": True, |
||||
|
}, |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": {"$ref": "#/components/schemas/Author"} |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
"422": { |
||||
|
"description": "Validation Error", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": { |
||||
|
"$ref": "#/components/schemas/HTTPValidationError" |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
}, |
||||
|
} |
||||
|
}, |
||||
|
"/authors/": { |
||||
|
"get": { |
||||
|
"summary": "Get Authors", |
||||
|
"operationId": "get_authors_authors__get", |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": { |
||||
|
"title": "Response Get Authors Authors Get", |
||||
|
"type": "array", |
||||
|
"items": {"$ref": "#/components/schemas/Author"}, |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
"components": { |
||||
|
"schemas": { |
||||
|
"Author": { |
||||
|
"title": "Author", |
||||
|
"required": ["name"], |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"name": {"title": "Name", "type": "string"}, |
||||
|
"items": { |
||||
|
"title": "Items", |
||||
|
"type": "array", |
||||
|
"items": {"$ref": "#/components/schemas/Item"}, |
||||
|
}, |
||||
|
}, |
||||
|
}, |
||||
|
"HTTPValidationError": { |
||||
|
"title": "HTTPValidationError", |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"detail": { |
||||
|
"title": "Detail", |
||||
|
"type": "array", |
||||
|
"items": {"$ref": "#/components/schemas/ValidationError"}, |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
"Item": { |
||||
|
"title": "Item", |
||||
|
"required": ["name"], |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"name": {"title": "Name", "type": "string"}, |
||||
|
"description": {"title": "Description", "type": "string"}, |
||||
|
}, |
||||
|
}, |
||||
|
"ValidationError": { |
||||
|
"title": "ValidationError", |
||||
|
"required": ["loc", "msg", "type"], |
||||
|
"type": "object", |
||||
|
"properties": { |
||||
|
"loc": { |
||||
|
"title": "Location", |
||||
|
"type": "array", |
||||
|
"items": {"type": "string"}, |
||||
|
}, |
||||
|
"msg": {"title": "Message", "type": "string"}, |
||||
|
"type": {"title": "Error Type", "type": "string"}, |
||||
|
}, |
||||
|
}, |
||||
|
} |
||||
|
}, |
||||
|
} |
||||
|
|
||||
|
|
||||
|
def test_openapi_schema(): |
||||
|
response = client.get("/openapi.json") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == openapi_schema |
||||
|
|
||||
|
|
||||
|
def test_post_authors_item(): |
||||
|
response = client.post( |
||||
|
"/authors/foo/items/", |
||||
|
json=[{"name": "Bar"}, {"name": "Baz", "description": "Drop the Baz"}], |
||||
|
) |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == { |
||||
|
"name": "foo", |
||||
|
"items": [ |
||||
|
{"name": "Bar", "description": None}, |
||||
|
{"name": "Baz", "description": "Drop the Baz"}, |
||||
|
], |
||||
|
} |
||||
|
|
||||
|
|
||||
|
def test_get_authors(): |
||||
|
response = client.get("/authors/") |
||||
|
assert response.status_code == 200 |
||||
|
assert response.json() == [ |
||||
|
{ |
||||
|
"name": "Breaters", |
||||
|
"items": [ |
||||
|
{ |
||||
|
"name": "Island In The Moon", |
||||
|
"description": "A place to be be playin' and havin' fun", |
||||
|
}, |
||||
|
{"name": "Holy Buddies", "description": None}, |
||||
|
], |
||||
|
}, |
||||
|
{ |
||||
|
"name": "System of an Up", |
||||
|
"items": [ |
||||
|
{ |
||||
|
"name": "Salt", |
||||
|
"description": "The kombucha mushroom people's favorite", |
||||
|
}, |
||||
|
{"name": "Pad Thai", "description": None}, |
||||
|
{ |
||||
|
"name": "Lonely Night", |
||||
|
"description": "The mostests lonliest nightiest of allest", |
||||
|
}, |
||||
|
], |
||||
|
}, |
||||
|
] |
Loading…
Reference in new issue