Browse Source

📝 Add request body tutorial

pull/11/head
Sebastián Ramírez 6 years ago
parent
commit
ddf20e8977
  1. BIN
      docs/img/tutorial/body/image01.png
  2. BIN
      docs/img/tutorial/body/image02.png
  3. BIN
      docs/img/tutorial/body/image03.png
  4. BIN
      docs/img/tutorial/body/image04.png
  5. BIN
      docs/img/tutorial/body/image05.png
  6. 131
      docs/tutorial/body.md
  7. 2
      docs/tutorial/query-params.md
  8. 2
      docs/tutorial/src/all/tutorial022.py
  9. 4
      docs/tutorial/src/query-params/tutorial005.py
  10. 6
      fastapi/dependencies/utils.py
  11. 3
      mkdocs.yml

BIN
docs/img/tutorial/body/image01.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 54 KiB

BIN
docs/img/tutorial/body/image02.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 79 KiB

BIN
docs/img/tutorial/body/image03.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 64 KiB

BIN
docs/img/tutorial/body/image04.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 39 KiB

BIN
docs/img/tutorial/body/image05.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 52 KiB

131
docs/tutorial/body.md

@ -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**.

2
docs/tutorial/query-params.md

@ -131,7 +131,7 @@ If you don't want to add a specific value but just make it optional, set the def
But when you want to make a query parameter required, you can just do not declare any default value: But when you want to make a query parameter required, you can just do not declare any default value:
```Python hl_lines="6 8" ```Python hl_lines="6 7"
{!./tutorial/src/query-params/tutorial005.py!} {!./tutorial/src/query-params/tutorial005.py!}
``` ```

2
docs/tutorial/src/all/tutorial022.py

@ -1,5 +1,5 @@
from fastapi import Body, FastAPI from fastapi import Body, FastAPI
from pydantic import BaseModel, Schema from pydantic import BaseModel
app = FastAPI() app = FastAPI()

4
docs/tutorial/src/query-params/tutorial005.py

@ -4,8 +4,6 @@ app = FastAPI()
@app.get("/users/{user_id}/items/{item_id}") @app.get("/users/{user_id}/items/{item_id}")
async def read_user_item( async def read_user_item(user_id: int, item_id: str, needy: str):
user_id: int, item_id: str, needy: str
):
item = {"item_id": item_id, "owner_id": user_id, "needy": needy} item = {"item_id": item_id, "owner_id": user_id, "needy": needy}
return item return item

6
fastapi/dependencies/utils.py

@ -254,7 +254,11 @@ def request_params_to_args(
if value is None: if value is None:
if field.required: if field.required:
errors.append( errors.append(
ErrorWrapper(MissingError(), loc=(schema.in_.value, field.alias), config=BaseConfig) ErrorWrapper(
MissingError(),
loc=(schema.in_.value, field.alias),
config=BaseConfig,
)
) )
else: else:
values[field.name] = deepcopy(field.default) values[field.name] = deepcopy(field.default)

3
mkdocs.yml

@ -18,9 +18,10 @@ nav:
- Features: 'features.md' - Features: 'features.md'
- Tutorial: - Tutorial:
- Tutorial Intro: 'tutorial/intro.md' - Tutorial Intro: 'tutorial/intro.md'
- First steps: 'tutorial/first-steps.md' - First Steps: 'tutorial/first-steps.md'
- Path Parameters: 'tutorial/path-params.md' - Path Parameters: 'tutorial/path-params.md'
- Query Parameters: 'tutorial/query-params.md' - Query Parameters: 'tutorial/query-params.md'
- Request Body: 'tutorial/body.md'
- Concurrency and async / await: 'async.md' - Concurrency and async / await: 'async.md'
- Deployment: 'deployment.md' - Deployment: 'deployment.md'

Loading…
Cancel
Save