11 changed files with 141 additions and 7 deletions
After Width: | Height: | Size: 54 KiB |
After Width: | Height: | Size: 79 KiB |
After Width: | Height: | Size: 64 KiB |
After Width: | Height: | Size: 39 KiB |
After Width: | Height: | Size: 52 KiB |
@ -0,0 +1,131 @@ |
|||
To declare a request body, you use <a href="https://pydantic-docs.helpmanual.io/" target="_blank">Pydantic</a> models with all their power and benefits. |
|||
|
|||
## Import Pydantic's `BaseModel` |
|||
|
|||
First, you need to import `BaseModel` from `pydantic`: |
|||
|
|||
```Python hl_lines="2" |
|||
{!./tutorial/src/body/tutorial001.py!} |
|||
``` |
|||
|
|||
## Create your data model |
|||
|
|||
Then you declare your data model as a class that inherits from `BaseModel`. |
|||
|
|||
Use standard Python types for all the attributes: |
|||
|
|||
```Python hl_lines="5 6 7 8 9" |
|||
{!./tutorial/src/body/tutorial001.py!} |
|||
``` |
|||
|
|||
The same as when declaring query parameters, when a model attribute has a default value, it is not required. Otherwise, it is required. Use `None` to make it just optional. |
|||
|
|||
For example, this model above declares a JSON "`object`" (or Python `dict`) like: |
|||
|
|||
```JSON |
|||
{ |
|||
"name": "Foo", |
|||
"description": "An optional description", |
|||
"price": 45.2, |
|||
"tax": 3.5 |
|||
} |
|||
``` |
|||
|
|||
...as `description` and `tax` are optional (with a default value of `None`), this JSON "`object`" would also be valid: |
|||
|
|||
```JSON |
|||
{ |
|||
"name": "Foo", |
|||
"price": 45.2 |
|||
} |
|||
``` |
|||
|
|||
## Declare it as a parameter |
|||
|
|||
To add it to your endpoint, declare it the same way you declared path and query parameters: |
|||
|
|||
```Python hl_lines="16" |
|||
{!./tutorial/src/body/tutorial001.py!} |
|||
``` |
|||
|
|||
...and declare its type as the model you created, `Item`. |
|||
|
|||
## Results |
|||
|
|||
With just that Python type declaration, **FastAPI** will: |
|||
|
|||
* Read the body of the request as JSON. |
|||
* Convert the corresponding types (if needed). |
|||
* Validate the data. |
|||
* If the data is invalid, it will return a nice and clear error, indicating exactly where and what was the incorrect data. |
|||
* Give you the received data in the parameter `item`. |
|||
* As you declared it in the function to be of type `Item`, you will also have all the editor support (completion, etc) for all of the attributes and their types. |
|||
* Generate <a href="http://json-schema.org" target="_blank">JSON Schema</a> definitions for your model, you can also use them anywhere else you like if it makes sense for your project. |
|||
* Those schemas will be part of the generated OpenAPI schema, and used by the automatic documentation <abbr title="User Interfaces">UIs</abbr>. |
|||
|
|||
## Automatic docs |
|||
|
|||
The JSON Schemas of your models will be part of your OpenAPI generated schema, and will be shown in the interactive API docs: |
|||
|
|||
<img src="/img/tutorial/body/image01.png"> |
|||
|
|||
And will be also used in the API docs inside each endpoint that needs them: |
|||
|
|||
<img src="/img/tutorial/body/image02.png"> |
|||
|
|||
## Editor support |
|||
|
|||
In your editor, inside your function you will get type hints and completion everywhere (this wouldn't happen if your received a `dict` instead of a Pydantic model): |
|||
|
|||
<img src="/img/tutorial/body/image03.png"> |
|||
|
|||
You also get error checks for incorrect type operations: |
|||
|
|||
<img src="/img/tutorial/body/image04.png"> |
|||
|
|||
This is not by chance, the whole framework was built around that desing. |
|||
|
|||
And it was thoroughly tested at the design phase, before any implementation, to ensure it would work with all the editors. |
|||
|
|||
There were even some changes to Pydantic itself to support this. |
|||
|
|||
The previous screenshots were taken with <a href="https://code.visualstudio.com" target="_blank">Visual Studio Code</a>. |
|||
|
|||
But you would get the same editor support with <a href="https://www.jetbrains.com/pycharm/" target="_blank">PyCharm</a> and most of the other Python editors: |
|||
|
|||
<img src="/img/tutorial/body/image05.png"> |
|||
|
|||
|
|||
## Use the model |
|||
|
|||
Inside of the function, you can access all the attributes of the model object directly: |
|||
|
|||
```Python hl_lines="19" |
|||
{!./tutorial/src/body/tutorial002.py!} |
|||
``` |
|||
|
|||
## Request body + path parameters |
|||
|
|||
You can declare path parameters and body requests at the same time. |
|||
|
|||
**FastAPI** will recognize that the function parameters that match path parameters should be **taken from the path**, and that function parameters that are declared to be Pydantic models should be **taken from the request body**. |
|||
|
|||
```Python hl_lines="15 16" |
|||
{!./tutorial/src/body/tutorial003.py!} |
|||
``` |
|||
|
|||
## Request body + path + query parameters |
|||
|
|||
You can also declare **body**, **path** and **query** parameters, all at the same time. |
|||
|
|||
**FastAPI** will recognize each of them and take the data from the correct place. |
|||
|
|||
```Python hl_lines="16" |
|||
{!./tutorial/src/body/tutorial004.py!} |
|||
``` |
|||
|
|||
The function parameters will be recognized as follows: |
|||
|
|||
* If the parameter is also declared in the **path**, it will be used as a path parameter. |
|||
* If the parameter is of a **singular type** (like `int`, `float`, `str`, `bool`, etc) it will be interpreted as a **query** parameter. |
|||
* If the parameter is declared to be of the type of a **Pydantic model**, it will be interpreted as a request **body**. |
Loading…
Reference in new issue