diff --git a/docs/img/tutorial/body/image01.png b/docs/img/tutorial/body/image01.png
new file mode 100644
index 000000000..819bf9c78
Binary files /dev/null and b/docs/img/tutorial/body/image01.png differ
diff --git a/docs/img/tutorial/body/image02.png b/docs/img/tutorial/body/image02.png
new file mode 100644
index 000000000..27a683fce
Binary files /dev/null and b/docs/img/tutorial/body/image02.png differ
diff --git a/docs/img/tutorial/body/image03.png b/docs/img/tutorial/body/image03.png
new file mode 100644
index 000000000..1301aafa8
Binary files /dev/null and b/docs/img/tutorial/body/image03.png differ
diff --git a/docs/img/tutorial/body/image04.png b/docs/img/tutorial/body/image04.png
new file mode 100644
index 000000000..f6c25c153
Binary files /dev/null and b/docs/img/tutorial/body/image04.png differ
diff --git a/docs/img/tutorial/body/image05.png b/docs/img/tutorial/body/image05.png
new file mode 100644
index 000000000..03a98a5f3
Binary files /dev/null and b/docs/img/tutorial/body/image05.png differ
diff --git a/docs/tutorial/body.md b/docs/tutorial/body.md
new file mode 100644
index 000000000..72a287930
--- /dev/null
+++ b/docs/tutorial/body.md
@@ -0,0 +1,131 @@
+To declare a request body, you use Pydantic 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 JSON Schema 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 UIs.
+
+## 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:
+
+
+
+And will be also used in the API docs inside each endpoint that needs them:
+
+
+
+## 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):
+
+
+
+You also get error checks for incorrect type operations:
+
+
+
+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 Visual Studio Code.
+
+But you would get the same editor support with PyCharm and most of the other Python editors:
+
+
+
+
+## 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**.
\ No newline at end of file
diff --git a/docs/tutorial/query-params.md b/docs/tutorial/query-params.md
index e58fe2749..6109cb742 100644
--- a/docs/tutorial/query-params.md
+++ b/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:
-```Python hl_lines="6 8"
+```Python hl_lines="6 7"
{!./tutorial/src/query-params/tutorial005.py!}
```
diff --git a/docs/tutorial/src/all/tutorial022.py b/docs/tutorial/src/all/tutorial022.py
index 3f4f53265..1165fd7a0 100644
--- a/docs/tutorial/src/all/tutorial022.py
+++ b/docs/tutorial/src/all/tutorial022.py
@@ -1,5 +1,5 @@
from fastapi import Body, FastAPI
-from pydantic import BaseModel, Schema
+from pydantic import BaseModel
app = FastAPI()
diff --git a/docs/tutorial/src/query-params/tutorial005.py b/docs/tutorial/src/query-params/tutorial005.py
index 62da63fc7..0979ad390 100644
--- a/docs/tutorial/src/query-params/tutorial005.py
+++ b/docs/tutorial/src/query-params/tutorial005.py
@@ -4,8 +4,6 @@ app = FastAPI()
@app.get("/users/{user_id}/items/{item_id}")
-async def read_user_item(
- user_id: int, item_id: str, needy: str
-):
+async def read_user_item(user_id: int, item_id: str, needy: str):
item = {"item_id": item_id, "owner_id": user_id, "needy": needy}
return item
diff --git a/fastapi/dependencies/utils.py b/fastapi/dependencies/utils.py
index d757f6b1f..ba1e5efee 100644
--- a/fastapi/dependencies/utils.py
+++ b/fastapi/dependencies/utils.py
@@ -254,7 +254,11 @@ def request_params_to_args(
if value is None:
if field.required:
errors.append(
- ErrorWrapper(MissingError(), loc=(schema.in_.value, field.alias), config=BaseConfig)
+ ErrorWrapper(
+ MissingError(),
+ loc=(schema.in_.value, field.alias),
+ config=BaseConfig,
+ )
)
else:
values[field.name] = deepcopy(field.default)
diff --git a/mkdocs.yml b/mkdocs.yml
index 1e4b9000a..089fe9492 100644
--- a/mkdocs.yml
+++ b/mkdocs.yml
@@ -18,9 +18,10 @@ nav:
- Features: 'features.md'
- Tutorial:
- Tutorial Intro: 'tutorial/intro.md'
- - First steps: 'tutorial/first-steps.md'
+ - First Steps: 'tutorial/first-steps.md'
- Path Parameters: 'tutorial/path-params.md'
- Query Parameters: 'tutorial/query-params.md'
+ - Request Body: 'tutorial/body.md'
- Concurrency and async / await: 'async.md'
- Deployment: 'deployment.md'