diff --git a/docs/tutorial/src/application-configuration/tutorial001.py b/docs/src/application-configuration/tutorial001.py similarity index 100% rename from docs/tutorial/src/application-configuration/tutorial001.py rename to docs/src/application-configuration/tutorial001.py diff --git a/docs/tutorial/src/application-configuration/tutorial002.py b/docs/src/application-configuration/tutorial002.py similarity index 100% rename from docs/tutorial/src/application-configuration/tutorial002.py rename to docs/src/application-configuration/tutorial002.py diff --git a/docs/tutorial/src/application-configuration/tutorial003.py b/docs/src/application-configuration/tutorial003.py similarity index 100% rename from docs/tutorial/src/application-configuration/tutorial003.py rename to docs/src/application-configuration/tutorial003.py diff --git a/docs/tutorial/src/bigger_applications/__init__.py b/docs/src/bigger_applications/__init__.py similarity index 100% rename from docs/tutorial/src/bigger_applications/__init__.py rename to docs/src/bigger_applications/__init__.py diff --git a/docs/tutorial/src/bigger_applications/app/__init__.py b/docs/src/bigger_applications/app/__init__.py similarity index 100% rename from docs/tutorial/src/bigger_applications/app/__init__.py rename to docs/src/bigger_applications/app/__init__.py diff --git a/docs/tutorial/src/bigger_applications/app/routers/__init__.py b/docs/src/bigger_applications/app/routers/__init__.py similarity index 100% rename from docs/tutorial/src/bigger_applications/app/routers/__init__.py rename to docs/src/bigger_applications/app/routers/__init__.py diff --git a/docs/tutorial/src/bigger_applications/app/routers/tutorial001.py b/docs/src/bigger_applications/app/routers/tutorial001.py similarity index 100% rename from docs/tutorial/src/bigger_applications/app/routers/tutorial001.py rename to docs/src/bigger_applications/app/routers/tutorial001.py diff --git a/docs/tutorial/src/bigger_applications/app/routers/tutorial002.py b/docs/src/bigger_applications/app/routers/tutorial002.py similarity index 100% rename from docs/tutorial/src/bigger_applications/app/routers/tutorial002.py rename to docs/src/bigger_applications/app/routers/tutorial002.py diff --git a/docs/tutorial/src/bigger_applications/app/tutorial003.py b/docs/src/bigger_applications/app/tutorial003.py similarity index 100% rename from docs/tutorial/src/bigger_applications/app/tutorial003.py rename to docs/src/bigger_applications/app/tutorial003.py diff --git a/docs/tutorial/src/body/tutorial001.py b/docs/src/body/tutorial001.py similarity index 100% rename from docs/tutorial/src/body/tutorial001.py rename to docs/src/body/tutorial001.py diff --git a/docs/tutorial/src/body/tutorial002.py b/docs/src/body/tutorial002.py similarity index 100% rename from docs/tutorial/src/body/tutorial002.py rename to docs/src/body/tutorial002.py diff --git a/docs/tutorial/src/body/tutorial003.py b/docs/src/body/tutorial003.py similarity index 100% rename from docs/tutorial/src/body/tutorial003.py rename to docs/src/body/tutorial003.py diff --git a/docs/tutorial/src/body/tutorial004.py b/docs/src/body/tutorial004.py similarity index 100% rename from docs/tutorial/src/body/tutorial004.py rename to docs/src/body/tutorial004.py diff --git a/docs/tutorial/src/body_multiple_params/tutorial001.py b/docs/src/body_multiple_params/tutorial001.py similarity index 100% rename from docs/tutorial/src/body_multiple_params/tutorial001.py rename to docs/src/body_multiple_params/tutorial001.py diff --git a/docs/tutorial/src/body_multiple_params/tutorial002.py b/docs/src/body_multiple_params/tutorial002.py similarity index 100% rename from docs/tutorial/src/body_multiple_params/tutorial002.py rename to docs/src/body_multiple_params/tutorial002.py diff --git a/docs/tutorial/src/body_multiple_params/tutorial003.py b/docs/src/body_multiple_params/tutorial003.py similarity index 100% rename from docs/tutorial/src/body_multiple_params/tutorial003.py rename to docs/src/body_multiple_params/tutorial003.py diff --git a/docs/tutorial/src/body_multiple_params/tutorial004.py b/docs/src/body_multiple_params/tutorial004.py similarity index 100% rename from docs/tutorial/src/body_multiple_params/tutorial004.py rename to docs/src/body_multiple_params/tutorial004.py diff --git a/docs/tutorial/src/body_multiple_params/tutorial005.py b/docs/src/body_multiple_params/tutorial005.py similarity index 100% rename from docs/tutorial/src/body_multiple_params/tutorial005.py rename to docs/src/body_multiple_params/tutorial005.py diff --git a/docs/tutorial/src/body_nested_models/tutorial001.py b/docs/src/body_nested_models/tutorial001.py similarity index 100% rename from docs/tutorial/src/body_nested_models/tutorial001.py rename to docs/src/body_nested_models/tutorial001.py diff --git a/docs/tutorial/src/body_nested_models/tutorial002.py b/docs/src/body_nested_models/tutorial002.py similarity index 100% rename from docs/tutorial/src/body_nested_models/tutorial002.py rename to docs/src/body_nested_models/tutorial002.py diff --git a/docs/tutorial/src/body_nested_models/tutorial003.py b/docs/src/body_nested_models/tutorial003.py similarity index 100% rename from docs/tutorial/src/body_nested_models/tutorial003.py rename to docs/src/body_nested_models/tutorial003.py diff --git a/docs/tutorial/src/body_nested_models/tutorial004.py b/docs/src/body_nested_models/tutorial004.py similarity index 100% rename from docs/tutorial/src/body_nested_models/tutorial004.py rename to docs/src/body_nested_models/tutorial004.py diff --git a/docs/tutorial/src/body_nested_models/tutorial005.py b/docs/src/body_nested_models/tutorial005.py similarity index 100% rename from docs/tutorial/src/body_nested_models/tutorial005.py rename to docs/src/body_nested_models/tutorial005.py diff --git a/docs/tutorial/src/body_nested_models/tutorial006.py b/docs/src/body_nested_models/tutorial006.py similarity index 100% rename from docs/tutorial/src/body_nested_models/tutorial006.py rename to docs/src/body_nested_models/tutorial006.py diff --git a/docs/tutorial/src/body_nested_models/tutorial007.py b/docs/src/body_nested_models/tutorial007.py similarity index 100% rename from docs/tutorial/src/body_nested_models/tutorial007.py rename to docs/src/body_nested_models/tutorial007.py diff --git a/docs/tutorial/src/body_nested_models/tutorial008.py b/docs/src/body_nested_models/tutorial008.py similarity index 100% rename from docs/tutorial/src/body_nested_models/tutorial008.py rename to docs/src/body_nested_models/tutorial008.py diff --git a/docs/tutorial/src/body_schema/tutorial001.py b/docs/src/body_schema/tutorial001.py similarity index 100% rename from docs/tutorial/src/body_schema/tutorial001.py rename to docs/src/body_schema/tutorial001.py diff --git a/docs/tutorial/src/body_schema/tutorial002.py b/docs/src/body_schema/tutorial002.py similarity index 100% rename from docs/tutorial/src/body_schema/tutorial002.py rename to docs/src/body_schema/tutorial002.py diff --git a/docs/tutorial/src/cookie_params/tutorial001.py b/docs/src/cookie_params/tutorial001.py similarity index 100% rename from docs/tutorial/src/cookie_params/tutorial001.py rename to docs/src/cookie_params/tutorial001.py diff --git a/docs/tutorial/src/custom_response/tutorial001.py b/docs/src/custom_response/tutorial001.py similarity index 100% rename from docs/tutorial/src/custom_response/tutorial001.py rename to docs/src/custom_response/tutorial001.py diff --git a/docs/tutorial/src/custom_response/tutorial002.py b/docs/src/custom_response/tutorial002.py similarity index 100% rename from docs/tutorial/src/custom_response/tutorial002.py rename to docs/src/custom_response/tutorial002.py diff --git a/docs/tutorial/src/custom_response/tutorial003.py b/docs/src/custom_response/tutorial003.py similarity index 100% rename from docs/tutorial/src/custom_response/tutorial003.py rename to docs/src/custom_response/tutorial003.py diff --git a/docs/tutorial/src/custom_response/tutorial004.py b/docs/src/custom_response/tutorial004.py similarity index 100% rename from docs/tutorial/src/custom_response/tutorial004.py rename to docs/src/custom_response/tutorial004.py diff --git a/docs/tutorial/src/dependencies/tutorial001.py b/docs/src/dependencies/tutorial001.py similarity index 100% rename from docs/tutorial/src/dependencies/tutorial001.py rename to docs/src/dependencies/tutorial001.py diff --git a/docs/tutorial/src/dependencies/tutorial002.py b/docs/src/dependencies/tutorial002.py similarity index 100% rename from docs/tutorial/src/dependencies/tutorial002.py rename to docs/src/dependencies/tutorial002.py diff --git a/docs/tutorial/src/dependencies/tutorial003.py b/docs/src/dependencies/tutorial003.py similarity index 100% rename from docs/tutorial/src/dependencies/tutorial003.py rename to docs/src/dependencies/tutorial003.py diff --git a/docs/tutorial/src/dependencies/tutorial004.py b/docs/src/dependencies/tutorial004.py similarity index 100% rename from docs/tutorial/src/dependencies/tutorial004.py rename to docs/src/dependencies/tutorial004.py diff --git a/docs/tutorial/src/extra_models/tutorial001.py b/docs/src/extra_models/tutorial001.py similarity index 100% rename from docs/tutorial/src/extra_models/tutorial001.py rename to docs/src/extra_models/tutorial001.py diff --git a/docs/tutorial/src/extra_models/tutorial002.py b/docs/src/extra_models/tutorial002.py similarity index 100% rename from docs/tutorial/src/extra_models/tutorial002.py rename to docs/src/extra_models/tutorial002.py diff --git a/docs/tutorial/src/first_steps/tutorial001.py b/docs/src/first_steps/tutorial001.py similarity index 100% rename from docs/tutorial/src/first_steps/tutorial001.py rename to docs/src/first_steps/tutorial001.py diff --git a/docs/tutorial/src/first_steps/tutorial002.py b/docs/src/first_steps/tutorial002.py similarity index 100% rename from docs/tutorial/src/first_steps/tutorial002.py rename to docs/src/first_steps/tutorial002.py diff --git a/docs/tutorial/src/first_steps/tutorial003.py b/docs/src/first_steps/tutorial003.py similarity index 100% rename from docs/tutorial/src/first_steps/tutorial003.py rename to docs/src/first_steps/tutorial003.py diff --git a/docs/tutorial/src/header_params/tutorial001.py b/docs/src/header_params/tutorial001.py similarity index 100% rename from docs/tutorial/src/header_params/tutorial001.py rename to docs/src/header_params/tutorial001.py diff --git a/docs/tutorial/src/header_params/tutorial002.py b/docs/src/header_params/tutorial002.py similarity index 100% rename from docs/tutorial/src/header_params/tutorial002.py rename to docs/src/header_params/tutorial002.py diff --git a/docs/tutorial/src/nosql_databases/tutorial001.py b/docs/src/nosql_databases/tutorial001.py similarity index 100% rename from docs/tutorial/src/nosql_databases/tutorial001.py rename to docs/src/nosql_databases/tutorial001.py diff --git a/docs/tutorial/src/path_operation_advanced_configuration/tutorial001.py b/docs/src/path_operation_advanced_configuration/tutorial001.py similarity index 100% rename from docs/tutorial/src/path_operation_advanced_configuration/tutorial001.py rename to docs/src/path_operation_advanced_configuration/tutorial001.py diff --git a/docs/tutorial/src/path_operation_advanced_configuration/tutorial002.py b/docs/src/path_operation_advanced_configuration/tutorial002.py similarity index 100% rename from docs/tutorial/src/path_operation_advanced_configuration/tutorial002.py rename to docs/src/path_operation_advanced_configuration/tutorial002.py diff --git a/docs/tutorial/src/path_operation_configuration/tutorial001.py b/docs/src/path_operation_configuration/tutorial001.py similarity index 100% rename from docs/tutorial/src/path_operation_configuration/tutorial001.py rename to docs/src/path_operation_configuration/tutorial001.py diff --git a/docs/tutorial/src/path_operation_configuration/tutorial002.py b/docs/src/path_operation_configuration/tutorial002.py similarity index 100% rename from docs/tutorial/src/path_operation_configuration/tutorial002.py rename to docs/src/path_operation_configuration/tutorial002.py diff --git a/docs/tutorial/src/path_operation_configuration/tutorial003.py b/docs/src/path_operation_configuration/tutorial003.py similarity index 100% rename from docs/tutorial/src/path_operation_configuration/tutorial003.py rename to docs/src/path_operation_configuration/tutorial003.py diff --git a/docs/tutorial/src/path_operation_configuration/tutorial004.py b/docs/src/path_operation_configuration/tutorial004.py similarity index 100% rename from docs/tutorial/src/path_operation_configuration/tutorial004.py rename to docs/src/path_operation_configuration/tutorial004.py diff --git a/docs/tutorial/src/path_operation_configuration/tutorial005.py b/docs/src/path_operation_configuration/tutorial005.py similarity index 100% rename from docs/tutorial/src/path_operation_configuration/tutorial005.py rename to docs/src/path_operation_configuration/tutorial005.py diff --git a/docs/tutorial/src/path_operation_configuration/tutorial006.py b/docs/src/path_operation_configuration/tutorial006.py similarity index 100% rename from docs/tutorial/src/path_operation_configuration/tutorial006.py rename to docs/src/path_operation_configuration/tutorial006.py diff --git a/docs/tutorial/src/path_params/tutorial001.py b/docs/src/path_params/tutorial001.py similarity index 100% rename from docs/tutorial/src/path_params/tutorial001.py rename to docs/src/path_params/tutorial001.py diff --git a/docs/tutorial/src/path_params/tutorial002.py b/docs/src/path_params/tutorial002.py similarity index 100% rename from docs/tutorial/src/path_params/tutorial002.py rename to docs/src/path_params/tutorial002.py diff --git a/docs/tutorial/src/path_params/tutorial003.py b/docs/src/path_params/tutorial003.py similarity index 100% rename from docs/tutorial/src/path_params/tutorial003.py rename to docs/src/path_params/tutorial003.py diff --git a/docs/tutorial/src/path_params_numeric_validations/tutorial001.py b/docs/src/path_params_numeric_validations/tutorial001.py similarity index 100% rename from docs/tutorial/src/path_params_numeric_validations/tutorial001.py rename to docs/src/path_params_numeric_validations/tutorial001.py diff --git a/docs/tutorial/src/path_params_numeric_validations/tutorial002.py b/docs/src/path_params_numeric_validations/tutorial002.py similarity index 100% rename from docs/tutorial/src/path_params_numeric_validations/tutorial002.py rename to docs/src/path_params_numeric_validations/tutorial002.py diff --git a/docs/tutorial/src/path_params_numeric_validations/tutorial003.py b/docs/src/path_params_numeric_validations/tutorial003.py similarity index 100% rename from docs/tutorial/src/path_params_numeric_validations/tutorial003.py rename to docs/src/path_params_numeric_validations/tutorial003.py diff --git a/docs/tutorial/src/path_params_numeric_validations/tutorial004.py b/docs/src/path_params_numeric_validations/tutorial004.py similarity index 100% rename from docs/tutorial/src/path_params_numeric_validations/tutorial004.py rename to docs/src/path_params_numeric_validations/tutorial004.py diff --git a/docs/tutorial/src/path_params_numeric_validations/tutorial005.py b/docs/src/path_params_numeric_validations/tutorial005.py similarity index 100% rename from docs/tutorial/src/path_params_numeric_validations/tutorial005.py rename to docs/src/path_params_numeric_validations/tutorial005.py diff --git a/docs/tutorial/src/path_params_numeric_validations/tutorial006.py b/docs/src/path_params_numeric_validations/tutorial006.py similarity index 100% rename from docs/tutorial/src/path_params_numeric_validations/tutorial006.py rename to docs/src/path_params_numeric_validations/tutorial006.py diff --git a/docs/tutorial/src/python_types/tutorial001.py b/docs/src/python_types/tutorial001.py similarity index 100% rename from docs/tutorial/src/python_types/tutorial001.py rename to docs/src/python_types/tutorial001.py diff --git a/docs/tutorial/src/python_types/tutorial002.py b/docs/src/python_types/tutorial002.py similarity index 100% rename from docs/tutorial/src/python_types/tutorial002.py rename to docs/src/python_types/tutorial002.py diff --git a/docs/tutorial/src/python_types/tutorial003.py b/docs/src/python_types/tutorial003.py similarity index 100% rename from docs/tutorial/src/python_types/tutorial003.py rename to docs/src/python_types/tutorial003.py diff --git a/docs/tutorial/src/python_types/tutorial004.py b/docs/src/python_types/tutorial004.py similarity index 100% rename from docs/tutorial/src/python_types/tutorial004.py rename to docs/src/python_types/tutorial004.py diff --git a/docs/tutorial/src/python_types/tutorial005.py b/docs/src/python_types/tutorial005.py similarity index 100% rename from docs/tutorial/src/python_types/tutorial005.py rename to docs/src/python_types/tutorial005.py diff --git a/docs/tutorial/src/python_types/tutorial006.py b/docs/src/python_types/tutorial006.py similarity index 100% rename from docs/tutorial/src/python_types/tutorial006.py rename to docs/src/python_types/tutorial006.py diff --git a/docs/tutorial/src/python_types/tutorial007.py b/docs/src/python_types/tutorial007.py similarity index 100% rename from docs/tutorial/src/python_types/tutorial007.py rename to docs/src/python_types/tutorial007.py diff --git a/docs/tutorial/src/python_types/tutorial008.py b/docs/src/python_types/tutorial008.py similarity index 100% rename from docs/tutorial/src/python_types/tutorial008.py rename to docs/src/python_types/tutorial008.py diff --git a/docs/tutorial/src/python_types/tutorial009.py b/docs/src/python_types/tutorial009.py similarity index 100% rename from docs/tutorial/src/python_types/tutorial009.py rename to docs/src/python_types/tutorial009.py diff --git a/docs/tutorial/src/python_types/tutorial010.py b/docs/src/python_types/tutorial010.py similarity index 100% rename from docs/tutorial/src/python_types/tutorial010.py rename to docs/src/python_types/tutorial010.py diff --git a/docs/tutorial/src/query_params/tutorial001.py b/docs/src/query_params/tutorial001.py similarity index 100% rename from docs/tutorial/src/query_params/tutorial001.py rename to docs/src/query_params/tutorial001.py diff --git a/docs/tutorial/src/query_params/tutorial002.py b/docs/src/query_params/tutorial002.py similarity index 100% rename from docs/tutorial/src/query_params/tutorial002.py rename to docs/src/query_params/tutorial002.py diff --git a/docs/tutorial/src/query_params/tutorial003.py b/docs/src/query_params/tutorial003.py similarity index 100% rename from docs/tutorial/src/query_params/tutorial003.py rename to docs/src/query_params/tutorial003.py diff --git a/docs/tutorial/src/query_params/tutorial004.py b/docs/src/query_params/tutorial004.py similarity index 100% rename from docs/tutorial/src/query_params/tutorial004.py rename to docs/src/query_params/tutorial004.py diff --git a/docs/tutorial/src/query_params/tutorial005.py b/docs/src/query_params/tutorial005.py similarity index 100% rename from docs/tutorial/src/query_params/tutorial005.py rename to docs/src/query_params/tutorial005.py diff --git a/docs/tutorial/src/query_params/tutorial006.py b/docs/src/query_params/tutorial006.py similarity index 100% rename from docs/tutorial/src/query_params/tutorial006.py rename to docs/src/query_params/tutorial006.py diff --git a/docs/tutorial/src/query_params_str_validations/tutorial001.py b/docs/src/query_params_str_validations/tutorial001.py similarity index 100% rename from docs/tutorial/src/query_params_str_validations/tutorial001.py rename to docs/src/query_params_str_validations/tutorial001.py diff --git a/docs/tutorial/src/query_params_str_validations/tutorial002.py b/docs/src/query_params_str_validations/tutorial002.py similarity index 100% rename from docs/tutorial/src/query_params_str_validations/tutorial002.py rename to docs/src/query_params_str_validations/tutorial002.py diff --git a/docs/tutorial/src/query_params_str_validations/tutorial003.py b/docs/src/query_params_str_validations/tutorial003.py similarity index 100% rename from docs/tutorial/src/query_params_str_validations/tutorial003.py rename to docs/src/query_params_str_validations/tutorial003.py diff --git a/docs/tutorial/src/query_params_str_validations/tutorial004.py b/docs/src/query_params_str_validations/tutorial004.py similarity index 100% rename from docs/tutorial/src/query_params_str_validations/tutorial004.py rename to docs/src/query_params_str_validations/tutorial004.py diff --git a/docs/tutorial/src/query_params_str_validations/tutorial005.py b/docs/src/query_params_str_validations/tutorial005.py similarity index 100% rename from docs/tutorial/src/query_params_str_validations/tutorial005.py rename to docs/src/query_params_str_validations/tutorial005.py diff --git a/docs/tutorial/src/query_params_str_validations/tutorial006.py b/docs/src/query_params_str_validations/tutorial006.py similarity index 100% rename from docs/tutorial/src/query_params_str_validations/tutorial006.py rename to docs/src/query_params_str_validations/tutorial006.py diff --git a/docs/tutorial/src/query_params_str_validations/tutorial007.py b/docs/src/query_params_str_validations/tutorial007.py similarity index 100% rename from docs/tutorial/src/query_params_str_validations/tutorial007.py rename to docs/src/query_params_str_validations/tutorial007.py diff --git a/docs/tutorial/src/query_params_str_validations/tutorial008.py b/docs/src/query_params_str_validations/tutorial008.py similarity index 100% rename from docs/tutorial/src/query_params_str_validations/tutorial008.py rename to docs/src/query_params_str_validations/tutorial008.py diff --git a/docs/tutorial/src/query_params_str_validations/tutorial009.py b/docs/src/query_params_str_validations/tutorial009.py similarity index 100% rename from docs/tutorial/src/query_params_str_validations/tutorial009.py rename to docs/src/query_params_str_validations/tutorial009.py diff --git a/docs/tutorial/src/query_params_str_validations/tutorial010.py b/docs/src/query_params_str_validations/tutorial010.py similarity index 100% rename from docs/tutorial/src/query_params_str_validations/tutorial010.py rename to docs/src/query_params_str_validations/tutorial010.py diff --git a/docs/tutorial/src/request_files/tutorial001.py b/docs/src/request_files/tutorial001.py similarity index 100% rename from docs/tutorial/src/request_files/tutorial001.py rename to docs/src/request_files/tutorial001.py diff --git a/docs/tutorial/src/request_forms/tutorial001.py b/docs/src/request_forms/tutorial001.py similarity index 100% rename from docs/tutorial/src/request_forms/tutorial001.py rename to docs/src/request_forms/tutorial001.py diff --git a/docs/tutorial/src/request_forms_and_files/tutorial001.py b/docs/src/request_forms_and_files/tutorial001.py similarity index 100% rename from docs/tutorial/src/request_forms_and_files/tutorial001.py rename to docs/src/request_forms_and_files/tutorial001.py diff --git a/docs/tutorial/src/response_model/tutorial001.py b/docs/src/response_model/tutorial001.py similarity index 100% rename from docs/tutorial/src/response_model/tutorial001.py rename to docs/src/response_model/tutorial001.py diff --git a/docs/tutorial/src/response_model/tutorial002.py b/docs/src/response_model/tutorial002.py similarity index 100% rename from docs/tutorial/src/response_model/tutorial002.py rename to docs/src/response_model/tutorial002.py diff --git a/docs/tutorial/src/response_model/tutorial003.py b/docs/src/response_model/tutorial003.py similarity index 100% rename from docs/tutorial/src/response_model/tutorial003.py rename to docs/src/response_model/tutorial003.py diff --git a/docs/tutorial/src/security/tutorial001.py b/docs/src/security/tutorial001.py similarity index 100% rename from docs/tutorial/src/security/tutorial001.py rename to docs/src/security/tutorial001.py diff --git a/docs/tutorial/src/security/tutorial002.py b/docs/src/security/tutorial002.py similarity index 100% rename from docs/tutorial/src/security/tutorial002.py rename to docs/src/security/tutorial002.py diff --git a/docs/tutorial/src/security/tutorial003.py b/docs/src/security/tutorial003.py similarity index 100% rename from docs/tutorial/src/security/tutorial003.py rename to docs/src/security/tutorial003.py diff --git a/docs/tutorial/src/security/tutorial004.py b/docs/src/security/tutorial004.py similarity index 100% rename from docs/tutorial/src/security/tutorial004.py rename to docs/src/security/tutorial004.py diff --git a/docs/tutorial/src/sql_databases/tutorial001.py b/docs/src/sql_databases/tutorial001.py similarity index 100% rename from docs/tutorial/src/sql_databases/tutorial001.py rename to docs/src/sql_databases/tutorial001.py diff --git a/docs/tutorial/application-configuration.md b/docs/tutorial/application-configuration.md index d9b970fc4..f9e3b83f0 100644 --- a/docs/tutorial/application-configuration.md +++ b/docs/tutorial/application-configuration.md @@ -1,13 +1,13 @@ Coming soon... ```Python -{!./tutorial/src/application-configuration/tutorial001.py!} +{!./src/application-configuration/tutorial001.py!} ``` ```Python -{!./tutorial/src/application-configuration/tutorial002.py!} +{!./src/application-configuration/tutorial002.py!} ``` ```Python -{!./tutorial/src/application-configuration/tutorial003.py!} +{!./src/application-configuration/tutorial003.py!} ``` diff --git a/docs/tutorial/bigger-applications.md b/docs/tutorial/bigger-applications.md index b82dd6125..056dcbbbf 100644 --- a/docs/tutorial/bigger-applications.md +++ b/docs/tutorial/bigger-applications.md @@ -1,13 +1,13 @@ Coming soon... ```Python -{!./tutorial/src/bigger_applications/app/routers/tutorial001.py!} +{!./src/bigger_applications/app/routers/tutorial001.py!} ``` ```Python -{!./tutorial/src/bigger_applications/app/routers/tutorial002.py!} +{!./src/bigger_applications/app/routers/tutorial002.py!} ``` ```Python -{!./tutorial/src/bigger_applications/app/tutorial003.py!} +{!./src/bigger_applications/app/tutorial003.py!} ``` diff --git a/docs/tutorial/body-multiple-params.md b/docs/tutorial/body-multiple-params.md index 60005978c..0884d1f90 100644 --- a/docs/tutorial/body-multiple-params.md +++ b/docs/tutorial/body-multiple-params.md @@ -7,7 +7,7 @@ First, of course, you can mix `Path`, `Query` and request body parameter declara And you can also declare body parameters as optional, by setting the default to `None`: ```Python hl_lines="17 18 19" -{!./tutorial/src/body_multiple_params/tutorial001.py!} +{!./src/body_multiple_params/tutorial001.py!} ``` !!! note @@ -30,7 +30,7 @@ In the previous example, the path operations would expect a JSON body with the a But you can also declare multiple body parameters, e.g. `item` and `user`: ```Python hl_lines="20" -{!./tutorial/src/body_multiple_params/tutorial002.py!} +{!./src/body_multiple_params/tutorial002.py!} ``` In this case, **FastAPI** will notice that there are more than one body parameter in the function (two parameters that are Pydantic models). @@ -72,7 +72,7 @@ But you can instruct **FastAPI** to treat it as another body key using `Body`: ```Python hl_lines="21" -{!./tutorial/src/body_multiple_params/tutorial003.py!} +{!./src/body_multiple_params/tutorial003.py!} ``` In this case, **FastAPI** will expect a body like: @@ -109,7 +109,7 @@ q: str = None as in: ```Python hl_lines="25" -{!./tutorial/src/body_multiple_params/tutorial004.py!} +{!./src/body_multiple_params/tutorial004.py!} ``` !!! info @@ -131,7 +131,7 @@ item: Item = Body(..., embed=True) as in: ```Python hl_lines="15" -{!./tutorial/src/body_multiple_params/tutorial005.py!} +{!./src/body_multiple_params/tutorial005.py!} ``` In this case **FastAPI** will expect a body like: diff --git a/docs/tutorial/body-nested-models.md b/docs/tutorial/body-nested-models.md index b5d969d67..27934dbd5 100644 --- a/docs/tutorial/body-nested-models.md +++ b/docs/tutorial/body-nested-models.md @@ -5,7 +5,7 @@ With **FastAPI**, you can define, validate, document, and use arbitrarily deeply You can define an attribute to be a subtype. For example, a Python `list`: ```Python hl_lines="12" -{!./tutorial/src/body_nested_models/tutorial001.py!} +{!./src/body_nested_models/tutorial001.py!} ``` This will make `tags` be a list of items. Although it doesn't declare the type of each of the items. @@ -19,7 +19,7 @@ But Python has a specific way to declare lists with subtypes: First, import `List` from standard Python's `typing` module: ```Python hl_lines="1" -{!./tutorial/src/body_nested_models/tutorial002.py!} +{!./src/body_nested_models/tutorial002.py!} ``` ### Declare a `List` with a subtype @@ -42,7 +42,7 @@ Use that same standard syntax for model attributes with subtypes. So, in our example, we can make `tags` be specifically a "list of strings": ```Python hl_lines="14" -{!./tutorial/src/body_nested_models/tutorial002.py!} +{!./src/body_nested_models/tutorial002.py!} ``` ## Set types @@ -54,7 +54,7 @@ And Python has a special data type for sets of unique items, the `set`. Then we can import `Set` and declare `tags` as a `set` of `str`: ```Python hl_lines="1 14" -{!./tutorial/src/body_nested_models/tutorial003.py!} +{!./src/body_nested_models/tutorial003.py!} ``` With this, even if you receive a request with duplicate data, it will be converted to a set of unique items. @@ -78,7 +78,7 @@ All that, arbitrarily nested. For example, we can define an `Image` model: ```Python hl_lines="9 10 11" -{!./tutorial/src/body_nested_models/tutorial004.py!} +{!./src/body_nested_models/tutorial004.py!} ``` ### Use the submodel as a type @@ -86,7 +86,7 @@ For example, we can define an `Image` model: And then we can use it as the type of an attribute: ```Python hl_lines="20" -{!./tutorial/src/body_nested_models/tutorial004.py!} +{!./src/body_nested_models/tutorial004.py!} ``` This would mean that **FastAPI** would expect a body similar to: @@ -121,7 +121,7 @@ To see all the options you have, checkout the docs for JSON Schema example field to a body request JSON Schema: ```Python hl_lines="20 21 22 23 24 25" -{!./tutorial/src/body_schema/tutorial002.py!} +{!./src/body_schema/tutorial002.py!} ``` ## Recap diff --git a/docs/tutorial/body.md b/docs/tutorial/body.md index ee7a540ac..d77510f46 100644 --- a/docs/tutorial/body.md +++ b/docs/tutorial/body.md @@ -5,7 +5,7 @@ To declare a request body, you use diff --git a/docs/tutorial/extra-models.md b/docs/tutorial/extra-models.md index e4ab41731..1cf727df8 100644 --- a/docs/tutorial/extra-models.md +++ b/docs/tutorial/extra-models.md @@ -14,7 +14,7 @@ This is especially the case for user models, because: Here's a general idea of how the models could look like with their password fields and the places where they are used: ```Python hl_lines="8 10 15 21 23 32 34 39 40" -{!./tutorial/src/extra_models/tutorial001.py!} +{!./src/extra_models/tutorial001.py!} ``` !!! warning @@ -37,7 +37,7 @@ All the data conversion, validation, documentation, etc. will still work as norm That way, we can declare just the differences between the models (with plaintext `password`, with `hashed_password` and without password): ```Python hl_lines="8 14 15 18 19 22 23" -{!./tutorial/src/extra_models/tutorial002.py!} +{!./src/extra_models/tutorial002.py!} ``` ## Recap diff --git a/docs/tutorial/first-steps.md b/docs/tutorial/first-steps.md index 7ae275ea8..40c4d1ba3 100644 --- a/docs/tutorial/first-steps.md +++ b/docs/tutorial/first-steps.md @@ -1,7 +1,7 @@ The simplest FastAPI file could look like this: ```Python -{!tutorial/src/first_steps/tutorial001.py!} +{!./src/first_steps/tutorial001.py!} ``` Copy that to a file `main.py`. @@ -89,7 +89,7 @@ It will show a JSON starting with something like: ### Step 1: import `FastAPI` ```Python hl_lines="1" -{!tutorial/src/first_steps/tutorial001.py!} +{!./src/first_steps/tutorial001.py!} ``` `FastAPI` is a Python class that provides all the functionality for your API. @@ -102,7 +102,7 @@ It will show a JSON starting with something like: ### Step 2: create a `FastAPI` "instance" ```Python hl_lines="3" -{!tutorial/src/first_steps/tutorial001.py!} +{!./src/first_steps/tutorial001.py!} ``` Here the `app` variable will be an "instance" of the class `FastAPI`. @@ -118,7 +118,7 @@ uvicorn main:app --debug If you create your app like: ```Python hl_lines="3" -{!tutorial/src/first_steps/tutorial002.py!} +{!./src/first_steps/tutorial002.py!} ``` And put it in a file `main.py`, then you would call `uvicorn` like: @@ -188,7 +188,7 @@ We are going to call them "operations" too. #### Define a path operation function ```Python hl_lines="6" -{!tutorial/src/first_steps/tutorial001.py!} +{!./src/first_steps/tutorial001.py!} ``` The `@app.get("/")` tells **FastAPI** that the function right below is in charge of handling requests that go to: @@ -221,7 +221,7 @@ And the more exotic ones: ### Step 4: define the path operation function ```Python hl_lines="7" -{!tutorial/src/first_steps/tutorial001.py!} +{!./src/first_steps/tutorial001.py!} ``` This is a Python function. @@ -235,7 +235,7 @@ In this case, it is an `async` function. You could also define it as a normal function instead of `async def`: ```Python hl_lines="7" -{!tutorial/src/first_steps/tutorial003.py!} +{!./src/first_steps/tutorial003.py!} ``` To know the difference, read the section about [Concurrency and `async` / `await`](/async/). @@ -243,7 +243,7 @@ To know the difference, read the section about [Concurrency and `async` / `await ### Step 5: return the content ```Python hl_lines="8" -{!tutorial/src/first_steps/tutorial001.py!} +{!./src/first_steps/tutorial001.py!} ``` You can return a `dict`, `list`, singular values as `str`, `int`, etc. diff --git a/docs/tutorial/header-params.md b/docs/tutorial/header-params.md index 67c2026b9..cc6263265 100644 --- a/docs/tutorial/header-params.md +++ b/docs/tutorial/header-params.md @@ -5,7 +5,7 @@ You can define Header parameters the same way you define `Query`, `Path` and `Co First import `Header`: ```Python hl_lines="1" -{!./tutorial/src/header_params/tutorial001.py!} +{!./src/header_params/tutorial001.py!} ``` ## Declare `Header` parameteres @@ -15,7 +15,7 @@ Then declare the header parameters using the same structure as with `Path`, `Que The first value is the default value, you can pass all the extra validation or annotation parameteres: ```Python hl_lines="7" -{!./tutorial/src/header_params/tutorial001.py!} +{!./src/header_params/tutorial001.py!} ``` !!! info @@ -41,7 +41,7 @@ So, you can use `user_agent` as you normally would in Python code, instead of ne If for some reason you need to disable automatic conversion of underscores to hyphens, set the parameter `convert_underscores` of `Header` to `False`: ```Python hl_lines="7" -{!./tutorial/src/header_params/tutorial002.py!} +{!./src/header_params/tutorial002.py!} ``` !!! warning diff --git a/docs/tutorial/nosql-databases.md b/docs/tutorial/nosql-databases.md index 38a56158c..09ee64dc6 100644 --- a/docs/tutorial/nosql-databases.md +++ b/docs/tutorial/nosql-databases.md @@ -15,7 +15,7 @@ You can adapt it to any other NoSQL database like: For now, don't pay attention to the rest, only the imports: ```Python hl_lines="6 7 8" -{!./tutorial/src/nosql_databases/tutorial001.py!} +{!./src/nosql_databases/tutorial001.py!} ``` ## Define a constant to use as a "document type" @@ -25,7 +25,7 @@ We will use it later as a fixed field `type` in our documents. This is not required by Couchbase, but is a good practice that will help you afterwards. ```Python hl_lines="10" -{!./tutorial/src/nosql_databases/tutorial001.py!} +{!./src/nosql_databases/tutorial001.py!} ``` ## Add a function to get a `Bucket` @@ -50,7 +50,7 @@ This utility function will: * Return it. ```Python hl_lines="13 14 15 16 17 18 19 20" -{!./tutorial/src/nosql_databases/tutorial001.py!} +{!./src/nosql_databases/tutorial001.py!} ``` ## Create Pydantic models @@ -62,7 +62,7 @@ As **Couchbase** "documents" are actually just "JSON objects", we can model them First, let's create a `User` model: ```Python hl_lines="23 24 25 26 27" -{!./tutorial/src/nosql_databases/tutorial001.py!} +{!./src/nosql_databases/tutorial001.py!} ``` We will use this model in our path operation function, so, we don't include in it the `hashed_password`. @@ -76,7 +76,7 @@ This will have the data that is actually stored in the database. We don't create it as a subclass of Pydantic's `BaseModel` but as a subclass of our own `User`, because it will have all the attributes in `User` plus a couple more: ```Python hl_lines="30 31 32" -{!./tutorial/src/nosql_databases/tutorial001.py!} +{!./src/nosql_databases/tutorial001.py!} ``` !!! note @@ -97,7 +97,7 @@ Now create a function that will: By creating a function that is only dedicated to getting your user from a `username` (or any other parameter) independent of your path operation function, you can more easily re-use it in multiple parts and also add unit tests for it: ```Python hl_lines="35 36 37 38 39 40 41" -{!./tutorial/src/nosql_databases/tutorial001.py!} +{!./src/nosql_databases/tutorial001.py!} ``` ### f-strings @@ -132,7 +132,7 @@ UserInDB(username="johndoe", hashed_password="some_hash") ### Create the `FastAPI` app ```Python hl_lines="45" -{!./tutorial/src/nosql_databases/tutorial001.py!} +{!./src/nosql_databases/tutorial001.py!} ``` ### Create the path operation function @@ -142,7 +142,7 @@ As our code is calling Couchbase and we are not using the threads", so, we can get just get the bucket directly and pass it to our utility functions: ```Python hl_lines="48 49 50 51 52" -{!./tutorial/src/nosql_databases/tutorial001.py!} +{!./src/nosql_databases/tutorial001.py!} ``` ## Recap diff --git a/docs/tutorial/path-operation-advanced-configuration.md b/docs/tutorial/path-operation-advanced-configuration.md index 635533b05..7d1b8e734 100644 --- a/docs/tutorial/path-operation-advanced-configuration.md +++ b/docs/tutorial/path-operation-advanced-configuration.md @@ -8,7 +8,7 @@ You can set the OpenAPI `operationId` to be used in your path operation with the You would have to make sure that it is unique for each operation. ```Python hl_lines="6" -{!./tutorial/src/path_operation_advanced_configuration/tutorial001.py!} +{!./src/path_operation_advanced_configuration/tutorial001.py!} ``` ## Exclude from OpenAPI @@ -16,5 +16,5 @@ You would have to make sure that it is unique for each operation. To exclude a path operation from the generated OpenAPI schema (and thus, from the automatic documentation systems), use the parameter `include_in_schema` and set it to `False`; ```Python hl_lines="6" -{!./tutorial/src/path_operation_advanced_configuration/tutorial002.py!} +{!./src/path_operation_advanced_configuration/tutorial002.py!} ``` diff --git a/docs/tutorial/path-operation-configuration.md b/docs/tutorial/path-operation-configuration.md index e70909867..ecbde5734 100644 --- a/docs/tutorial/path-operation-configuration.md +++ b/docs/tutorial/path-operation-configuration.md @@ -12,7 +12,7 @@ You can pass directly the `int` code, like `404`. But if you don't remember what each number code is for, you can use the shortcut constants from `starlette`: ```Python hl_lines="5 18" -{!./tutorial/src/path_operation_configuration/tutorial001.py!} +{!./src/path_operation_configuration/tutorial001.py!} ``` That status code will be used in the response and will be added to the OpenAPI schema. @@ -23,7 +23,7 @@ That status code will be used in the response and will be added to the OpenAPI s You can add tags to your path operation, pass the parameter `tags` with a `list` of `str` (commonly just one `str`): ```Python hl_lines="17 22 27" -{!./tutorial/src/path_operation_configuration/tutorial002.py!} +{!./src/path_operation_configuration/tutorial002.py!} ``` They will be added to the OpenAPI schema and used by the automatic documentation interfaces: @@ -35,7 +35,7 @@ They will be added to the OpenAPI schema and used by the automatic documentation You can add a `summary` and `description`: ```Python hl_lines="20 21" -{!./tutorial/src/path_operation_configuration/tutorial003.py!} +{!./src/path_operation_configuration/tutorial003.py!} ``` ## Description from docstring @@ -43,7 +43,7 @@ You can add a `summary` and `description`: As descriptions tend to be long and cover multiple lines, you can declare the path operation description in the function docstring and **FastAPI** will read it from there. ```Python hl_lines="19 20 21 22 23 24 25 26 27" -{!./tutorial/src/path_operation_configuration/tutorial004.py!} +{!./src/path_operation_configuration/tutorial004.py!} ``` It will be used in the interactive docs: @@ -58,7 +58,7 @@ It will be used in the interactive docs: You can specify the response description with the parameter `response_description`: ```Python hl_lines="21" -{!./tutorial/src/path_operation_configuration/tutorial005.py!} +{!./src/path_operation_configuration/tutorial005.py!} ``` !!! info @@ -77,7 +77,7 @@ If you need to mark a path operation as kwargs. Even if they don't have a default value. ```Python hl_lines="8" -{!./tutorial/src/path_params_numeric_validations/tutorial003.py!} +{!./src/path_params_numeric_validations/tutorial003.py!} ``` ## Number validations: greater than or equal @@ -64,7 +64,7 @@ With `Query` and `Path` (and other's you'll see later) you can declare string co Here, with `ge=1`, `item_id` will need to be an integer number "`g`reater than or `e`qual" to `1`. ```Python hl_lines="8" -{!./tutorial/src/path_params_numeric_validations/tutorial004.py!} +{!./src/path_params_numeric_validations/tutorial004.py!} ``` ## Number validations: greater than and less than or equal @@ -74,7 +74,7 @@ The same applies for: * `le`: `l`ess than or `e`qual ```Python hl_lines="9" -{!./tutorial/src/path_params_numeric_validations/tutorial005.py!} +{!./src/path_params_numeric_validations/tutorial005.py!} ``` ## Number validations: floats, greater than and less than @@ -88,7 +88,7 @@ So, `0.5` would be a valid value. But `0.0` or `0` would not. And the same for lt. ```Python hl_lines="11" -{!./tutorial/src/path_params_numeric_validations/tutorial006.py!} +{!./src/path_params_numeric_validations/tutorial006.py!} ``` ## Recap diff --git a/docs/tutorial/path-params.md b/docs/tutorial/path-params.md index 68d9973f9..77d963256 100644 --- a/docs/tutorial/path-params.md +++ b/docs/tutorial/path-params.md @@ -1,7 +1,7 @@ You can declare path "parameters" or "variables" with the same syntax used by Python format strings: ```Python hl_lines="6 7" -{!./tutorial/src/path_params/tutorial001.py!} +{!./src/path_params/tutorial001.py!} ``` The value of the path parameter `item_id` will be passed to your function as the argument `item_id`. @@ -17,7 +17,7 @@ So, if you run this example and go to type of a variable. - -By declaring types for your variables, editors and tools can give you better support. - -This is just a **quick tutorial / refresher** about Python type hints. It covers only the minimum necessary to use them with **FastAPI**... which is actually very little. - -**FastAPI** is all based on these type hints, they give it many advantages and benefits. - -But even if you never use **FastAPI**, you would benefit from learning a bit about them. - -!!! note - If you are a Python expert, and you already know everything about type hints, skip to the next chapter. - -## Motivation - -Let's start with a simple example: - -```Python -{!./tutorial/src/python_types/tutorial001.py!} -``` - -Calling this program outputs: - -``` -John Doe -``` - -The function does the following: - -* Takes a `fist_name` and `last_name`. -* Converts the first letter of each one to upper case with `title()`. -* Concatenates them with a space in the middle. - -```Python hl_lines="2" -{!./tutorial/src/python_types/tutorial001.py!} -``` - -### Edit it - -It's a very simple program. - -But now imagine that you were writing it from scratch. - -At some point you would have started the definition of the function, you had the parameters ready... - -But then you have to call "that method that converts the first letter to upper case". - -Was it `upper`? Was it `uppercase`? `first_uppercase`? `capitalize`? - -Then, you try with the old programer's friend, editor autocompletion. - -You type the first parameter of the function, `first_name`, then a dot (`.`) and then hit `Ctrl+Space` to trigger the completion. - -But, sadly, you get nothing useful: - - - -### Add types - -Let's modify a single line from the previous version. - -We will change exactly this fragment, the parameters of the function, from: - -```Python - first_name, last_name -``` - -to: - -```Python - first_name: str, last_name: str -``` - -That's it. - -Those are the "type hints": - -```Python hl_lines="1" -{!./tutorial/src/python_types/tutorial002.py!} -``` - -That is not the same as declaring default values like would be with: - -```Python - first_name="john", last_name="doe" -``` - -It's a different thing. - -We are using colons (`:`), not equals (`=`). - -And adding type hints normally doesn't change what happens from what would happen without them. - -But now, imagine you are again in the middle of creating that function, but with type hints. - -At the same point, you try to trigger the autocomplete with `Ctrl+Space` and you see: - - - -With that, you can scroll, seeing the options, until you find the one that "rings a bell": - - - -## More motivation - -Check this function, it already has type hints: - -```Python hl_lines="1" -{!./tutorial/src/python_types/tutorial003.py!} -``` - -Because the editor knows the types of the variables, you don't only get completion, you also get error checks: - - - -Now you know that you have to fix it, convert `age` to a string with `str(age)`: - -```Python hl_lines="2" -{!./tutorial/src/python_types/tutorial004.py!} -``` - - -## Declaring types - -You just saw the main place to declare type hints. As function parameters. - -This is also the main place you would use them with **FastAPI**. - -### Simple types - -You can declare all the standard Python types, not only `str`. - -You can use, for example: - -* `int` -* `float` -* `bool` -* `bytes` - -```Python hl_lines="1" -{!./tutorial/src/python_types/tutorial005.py!} -``` - -### Types with subtypes - -There are some data structures that can contain other values, like `dict`, `list`, `set` and `tuple`. And the internal values can have their own type too. - -To declare those types and the subtypes, you can use the standard Python module `typing`. - -It exists specifically to support these type hints. - -#### Lists - -For example, let's define a variable to be a `list` of `str`. - -From `typing`, import `List` (with a capital `L`): - -```Python hl_lines="1" -{!./tutorial/src/python_types/tutorial006.py!} -``` - -Declare the variable, with the same colon (`:`) syntax. - -As the type, put the `List`. - -As the list is a type that takes a "subtype", you put the subtype in square brackets: - -```Python hl_lines="4" -{!./tutorial/src/python_types/tutorial006.py!} -``` - -That means: "the variable `items` is a `list`, and each of the items in this list is a `str`". - -By doing that, your editor can provide support even while processing items from the list. - -Without types, that's almost impossible to achieve: - - - -Notice that the variable `item` is one of the elements in the list `items`. - -And still, the editor knows it is a `str`, and provides support for that. - -#### Tuples and Sets - -You would do the same to declare `tuple`s and `set`s: - -```Python hl_lines="1 4" -{!./tutorial/src/python_types/tutorial007.py!} -``` - -This means: - -* The variable `items_t` is a `tuple`, and each of its items is an `int`. -* The variable `items_s` is a `set`, and each of its items is of type `bytes`. - -#### Dicts - -To define a `dict`, you pass 2 subtypes, separated by commas. - -The first subtype is for the keys of the `dict`. - -The second subtype is for the values of the `dict`: - -```Python hl_lines="1 4" -{!./tutorial/src/python_types/tutorial008.py!} -``` - -This means: - -* The variable `prices` is a `dict`: - * The keys of this `dict` are of type `str` (let's say, the name of each item). - * The values of this `dict` are of type `float` (let's say, the price of each item). - - -### Classes as types - -You can also declare a class as the type of a variable. - -Let's say you have a class `Person`, with a name: - -```Python hl_lines="1 2 3" -{!./tutorial/src/python_types/tutorial009.py!} -``` - -Then you can declare a variable to be of type `Person`: - -```Python hl_lines="6" -{!./tutorial/src/python_types/tutorial009.py!} -``` - -And then, again, you get all the editor support: - - - - -## Pydantic models - -Pydantic is a Python library to perform data validation. - -You declare the "shape" of the data as classes with attributes. - -And each attribute has a type. - -Then you create an instance of that class with some values and it will validate the values, convert them to the appropriate type (if that's the case) and give you an object with all the data. - -And you get all the editor support with that resulting object. - -Taken from the official Pydantic docs: - -```Python -{!./tutorial/src/python_types/tutorial010.py!} -``` - -!!! info - To learn more about Pydantic, check its docs. - -**FastAPI** is all based on Pydantic. - -You will see a lot more of all this in practice in the next chapters. - - -## Type hints in **FastAPI** - -**FastAPI** takes advantage of these type hints to do several things. - -With **FastAPI** you declare parameters with type hints and you get: - -* **Editor support**. -* **Type checks**. - -...and **FastAPI** uses the same declarations to: - -* **Define requirements**: from request path parameters, query parameters, headers, bodies, dependencies, etc. -* **Convert data**: from the request to the required type. -* **Validate data**: coming from each request: - * Generating **automatic errors** returned to the client when the data is invalid. -* **Document** the API using OpenAPI: - * which is then used by the automatic interactive documentation user interfaces. - -This might all sound abstract. Don't worry. You'll see all this in action in the next chapters. - -The important thing is that by using standard Python types, in a single place (instead of adding more classes, decorators, etc), **FastAPI** will do a lot of the work for you. - -!!! info - If you already went through all the tutorial and came back to see more about types, a good resource is the "cheat sheet" from `mypy`. \ No newline at end of file diff --git a/docs/tutorial/query-params-str-validations.md b/docs/tutorial/query-params-str-validations.md index afed65be8..e25b414c9 100644 --- a/docs/tutorial/query-params-str-validations.md +++ b/docs/tutorial/query-params-str-validations.md @@ -3,7 +3,7 @@ Let's take this application as example: ```Python hl_lines="7" -{!./tutorial/src/query_params_str_validations/tutorial001.py!} +{!./src/query_params_str_validations/tutorial001.py!} ``` The query parameter `q` is of type `str`, and by default is `None`, so it is optional. @@ -18,7 +18,7 @@ We are going to enforce that even though `q` is optional, whenever it is provide To achieve that, first import `Query` from `fastapi`: ```Python hl_lines="1" -{!./tutorial/src/query_params_str_validations/tutorial002.py!} +{!./src/query_params_str_validations/tutorial002.py!} ``` ## Use `Query` as the default value @@ -26,7 +26,7 @@ To achieve that, first import `Query` from `fastapi`: And now use it as the default value of your parameter, setting the parameter `max_length` to 50: ```Python hl_lines="7" -{!./tutorial/src/query_params_str_validations/tutorial002.py!} +{!./src/query_params_str_validations/tutorial002.py!} ``` As we have to replace the default value `None` with `Query(None)`, the first parameter to `Query` serves the same purpose of defining that default value. @@ -59,7 +59,7 @@ This will validate the data, show a clear error when the data is not valid, and You can also add a parameter `min_length`: ```Python hl_lines="7" -{!./tutorial/src/query_params_str_validations/tutorial003.py!} +{!./src/query_params_str_validations/tutorial003.py!} ``` ## Add regular expressions @@ -67,7 +67,7 @@ You can also add a parameter `min_length`: You can define a regular expression that the parameter should match: ```Python hl_lines="8" -{!./tutorial/src/query_params_str_validations/tutorial004.py!} +{!./src/query_params_str_validations/tutorial004.py!} ``` This specific regular expression checks that the received parameter value: @@ -87,7 +87,7 @@ The same way that you can pass `None` as the first argument to be used as the de Let's say that you want to declare the `q` query parameter to have a `min_length` of `3`, and to have a default value of `"fixedquery"`: ```Python hl_lines="7" -{!./tutorial/src/query_params_str_validations/tutorial005.py!} +{!./src/query_params_str_validations/tutorial005.py!} ``` !!! note @@ -116,7 +116,7 @@ q: str = Query(None, min_length=3) So, when you need to declare a value as required while using `Query`, you can use `...` as the first argument: ```Python hl_lines="7" -{!./tutorial/src/query_params_str_validations/tutorial006.py!} +{!./src/query_params_str_validations/tutorial006.py!} ``` !!! info @@ -133,13 +133,13 @@ That information will be included in the generated OpenAPI and used by the docum You can add a `title`: ```Python hl_lines="7" -{!./tutorial/src/query_params_str_validations/tutorial007.py!} +{!./src/query_params_str_validations/tutorial007.py!} ``` And a `description`: ```Python hl_lines="11" -{!./tutorial/src/query_params_str_validations/tutorial008.py!} +{!./src/query_params_str_validations/tutorial008.py!} ``` ## Alias parameters @@ -161,7 +161,7 @@ But you still need it to be exactly `item-query`... Then you can declare an `alias`, and that alias is what will be used to find the parameter value: ```Python hl_lines="7" -{!./tutorial/src/query_params_str_validations/tutorial009.py!} +{!./src/query_params_str_validations/tutorial009.py!} ``` ## Deprecating parameters @@ -173,7 +173,7 @@ You have to leave it there a while because there are clients using it, but you w Then pass the parameter `deprecated=True` to `Query`: ```Python hl_lines="16" -{!./tutorial/src/query_params_str_validations/tutorial010.py!} +{!./src/query_params_str_validations/tutorial010.py!} ``` The docs will show it like this: diff --git a/docs/tutorial/query-params.md b/docs/tutorial/query-params.md index 90e9a8595..153956df9 100644 --- a/docs/tutorial/query-params.md +++ b/docs/tutorial/query-params.md @@ -1,7 +1,7 @@ When you declare other function parameters that are not part of the path parameters, they are automatically interpreted as "query" parameters. ```Python hl_lines="9" -{!./tutorial/src/query_params/tutorial001.py!} +{!./src/query_params/tutorial001.py!} ``` The query is the set of key-value pairs that go after the `?` in a URL, separated by `&` characters. @@ -62,7 +62,7 @@ The parameter values in your function will be: The same way, you can declare optional query parameters, by setting their default to `None`: ```Python hl_lines="7" -{!./tutorial/src/query_params/tutorial002.py!} +{!./src/query_params/tutorial002.py!} ``` In this case, the function parameter `q` will be optional, and will be `None` by default. @@ -75,7 +75,7 @@ In this case, the function parameter `q` will be optional, and will be `None` by You can also declare `bool` types, and they will be converted: ```Python hl_lines="7" -{!./tutorial/src/query_params/tutorial003.py!} +{!./src/query_params/tutorial003.py!} ``` In this case, if you go to: @@ -120,7 +120,7 @@ And you don't have to declare them in any specific order. They will be detected by name: ```Python hl_lines="6 8" -{!./tutorial/src/query_params/tutorial004.py!} +{!./src/query_params/tutorial004.py!} ``` ## Required query parameters @@ -132,7 +132,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 7" -{!./tutorial/src/query_params/tutorial005.py!} +{!./src/query_params/tutorial005.py!} ``` Here the query parameter `needy` is a required query parameter of type `str`. @@ -178,7 +178,7 @@ http://127.0.0.1:8000/items/foo-item?needy=sooooneedy And of course, you can define some parameters as required, some as having a default value, and some entirely optional: ```Python hl_lines="7" -{!./tutorial/src/query_params/tutorial006.py!} +{!./src/query_params/tutorial006.py!} ``` In this case, there are 3 query parameters: diff --git a/docs/tutorial/request-files.md b/docs/tutorial/request-files.md index aecde928a..eacf1862d 100644 --- a/docs/tutorial/request-files.md +++ b/docs/tutorial/request-files.md @@ -5,7 +5,7 @@ You can define files to be uploaded by the client using `File`. Import `File` from `fastapi`: ```Python hl_lines="1" -{!./tutorial/src/request_files/tutorial001.py!} +{!./src/request_files/tutorial001.py!} ``` ## Define `File` parameters @@ -13,7 +13,7 @@ Import `File` from `fastapi`: Create file parameters the same way you would for `Body` or `Form`: ```Python hl_lines="7" -{!./tutorial/src/request_files/tutorial001.py!} +{!./src/request_files/tutorial001.py!} ``` The files will be uploaded as form data and you will receive the contents as `bytes`. diff --git a/docs/tutorial/request-forms-and-files.md b/docs/tutorial/request-forms-and-files.md index 3c32d13ad..00bbead32 100644 --- a/docs/tutorial/request-forms-and-files.md +++ b/docs/tutorial/request-forms-and-files.md @@ -3,7 +3,7 @@ You can define files and form fields at the same time using `File` and `Form`. ## Import `File` and `Form` ```Python hl_lines="1" -{!./tutorial/src/request_forms_and_files/tutorial001.py!} +{!./src/request_forms_and_files/tutorial001.py!} ``` ## Define `File` and `Form` parameters @@ -11,7 +11,7 @@ You can define files and form fields at the same time using `File` and `Form`. Create file and form parameters the same way you would for `Body` or `Query`: ```Python hl_lines="7" -{!./tutorial/src/request_forms_and_files/tutorial001.py!} +{!./src/request_forms_and_files/tutorial001.py!} ``` The files and form fields will be uploaded as form data and you will receive the files and form fields. diff --git a/docs/tutorial/request-forms.md b/docs/tutorial/request-forms.md index 7c506fc23..2ee5d84ae 100644 --- a/docs/tutorial/request-forms.md +++ b/docs/tutorial/request-forms.md @@ -5,7 +5,7 @@ When you need to receive form fields instead of JSON, you can use `Form`. Import `Form` from `fastapi`: ```Python hl_lines="1" -{!./tutorial/src/request_forms/tutorial001.py!} +{!./src/request_forms/tutorial001.py!} ``` ## Define `Form` parameters @@ -13,7 +13,7 @@ Import `Form` from `fastapi`: Create form parameters the same way you would for `Body` or `Query`: ```Python hl_lines="7" -{!./tutorial/src/request_forms/tutorial001.py!} +{!./src/request_forms/tutorial001.py!} ``` For example, in one of the ways the OAuth2 specification can be used (called "password flow") it is required to send a `username` and `password` as form fields. diff --git a/docs/tutorial/response-model.md b/docs/tutorial/response-model.md index 3ae20c3ed..cae00caf0 100644 --- a/docs/tutorial/response-model.md +++ b/docs/tutorial/response-model.md @@ -7,7 +7,7 @@ You can declare the model used for the response with the parameter `response_mod * etc. ```Python hl_lines="17" -{!./tutorial/src/response_model/tutorial001.py!} +{!./src/response_model/tutorial001.py!} ``` !!! note @@ -29,13 +29,13 @@ But most importantly: Here we are declaring a `UserIn` model, it will contain a plaintext password: ```Python hl_lines="8 10" -{!./tutorial/src/response_model/tutorial002.py!} +{!./src/response_model/tutorial002.py!} ``` And we are using this model to declare our input and the same model to declare our output: ```Python hl_lines="16 17" -{!./tutorial/src/response_model/tutorial002.py!} +{!./src/response_model/tutorial002.py!} ``` Now, whenever a browser is creating a user with a password, the API will return the same password in the response. @@ -52,19 +52,19 @@ But if we use sthe same model for another path operation, we could be sending th We can instead create an input model with the plaintext password and an output model without it: ```Python hl_lines="8 10 15" -{!./tutorial/src/response_model/tutorial003.py!} +{!./src/response_model/tutorial003.py!} ``` Here, even though our path operation function is returning the same input user that contains the password: ```Python hl_lines="23" -{!./tutorial/src/response_model/tutorial003.py!} +{!./src/response_model/tutorial003.py!} ``` ...we declared the `response_model` to be our model `UserOut`, that doesn't include the password: ```Python hl_lines="21" -{!./tutorial/src/response_model/tutorial003.py!} +{!./src/response_model/tutorial003.py!} ``` So, **FastAPI** will take care of filtering out all the data that is not declared in the output model (using Pydantic). diff --git a/docs/tutorial/security/first-steps.md b/docs/tutorial/security/first-steps.md index 3cea5cb59..4472d5028 100644 --- a/docs/tutorial/security/first-steps.md +++ b/docs/tutorial/security/first-steps.md @@ -1,5 +1,5 @@ Coming soon... ```Python -{!./tutorial/src/security/tutorial002.py!} +{!./src/security/tutorial002.py!} ``` diff --git a/docs/tutorial/security/intro.md b/docs/tutorial/security/intro.md index e6891dad2..6a0577ea5 100644 --- a/docs/tutorial/security/intro.md +++ b/docs/tutorial/security/intro.md @@ -1,5 +1,5 @@ Coming soon... ```Python -{!./tutorial/src/security/tutorial001.py!} +{!./src/security/tutorial001.py!} ``` diff --git a/docs/tutorial/security/oauth2-jwt.md b/docs/tutorial/security/oauth2-jwt.md index 7ceef9df2..de4294d1b 100644 --- a/docs/tutorial/security/oauth2-jwt.md +++ b/docs/tutorial/security/oauth2-jwt.md @@ -1,5 +1,5 @@ Coming soon... ```Python -{!./tutorial/src/security/tutorial004.py!} +{!./src/security/tutorial004.py!} ``` diff --git a/docs/tutorial/security/simple-oauth2.md b/docs/tutorial/security/simple-oauth2.md index 5dff95b11..871c46926 100644 --- a/docs/tutorial/security/simple-oauth2.md +++ b/docs/tutorial/security/simple-oauth2.md @@ -1,5 +1,5 @@ Coming soon... ```Python -{!./tutorial/src/security/tutorial003.py!} +{!./src/security/tutorial003.py!} ``` diff --git a/docs/tutorial/sql-databases.md b/docs/tutorial/sql-databases.md index bca7f5eb9..7b83a1565 100644 --- a/docs/tutorial/sql-databases.md +++ b/docs/tutorial/sql-databases.md @@ -24,7 +24,7 @@ In this example, we'll use **PostgreSQL**. For now, don't pay attention to the rest, only the imports: ```Python hl_lines="3 4 5" -{!./tutorial/src/sql_databases/tutorial001.py!} +{!./src/sql_databases/tutorial001.py!} ``` ## Define the database @@ -32,7 +32,7 @@ For now, don't pay attention to the rest, only the imports: Define the database that SQLAlchemy should connect to: ```Python hl_lines="8" -{!./tutorial/src/sql_databases/tutorial001.py!} +{!./src/sql_databases/tutorial001.py!} ``` !!! tip @@ -41,13 +41,13 @@ Define the database that SQLAlchemy should connect to: ## Create the SQLAlchemy `engine` ```Python hl_lines="10" -{!./tutorial/src/sql_databases/tutorial001.py!} +{!./src/sql_databases/tutorial001.py!} ``` ## Create a `scoped_session` ```Python hl_lines="11 12 13" -{!./tutorial/src/sql_databases/tutorial001.py!} +{!./src/sql_databases/tutorial001.py!} ``` !!! note "Very Technical Details" @@ -70,13 +70,13 @@ That way you don't have to declare them explicitly. So, your models will behave very similarly to, for example, Flask-SQLAlchemy. ```Python hl_lines="15 16 17 18 19" -{!./tutorial/src/sql_databases/tutorial001.py!} +{!./src/sql_databases/tutorial001.py!} ``` ## Create the SQLAlchemy `Base` model ```Python hl_lines="22" -{!./tutorial/src/sql_databases/tutorial001.py!} +{!./src/sql_databases/tutorial001.py!} ``` ## Create your application data model @@ -86,7 +86,7 @@ Now this is finally code specific to your app. Here's a user model that will be a table in the database: ```Python hl_lines="25 26 27 28 29" -{!./tutorial/src/sql_databases/tutorial001.py!} +{!./src/sql_databases/tutorial001.py!} ``` ## Get a user @@ -94,7 +94,7 @@ Here's a user model that will be a table in the database: By creating a function that is only dedicated to getting your user from a `username` (or any other parameter) independent of your path operation function, you can more easily re-use it in multiple parts and also add unit tests for it: ```Python hl_lines="32 33" -{!./tutorial/src/sql_databases/tutorial001.py!} +{!./src/sql_databases/tutorial001.py!} ``` ## Create your **FastAPI** code @@ -104,7 +104,7 @@ Now, finally, here's the standard **FastAPI** code. Create your app and path operation function: ```Python hl_lines="37 40 41 42 43" -{!./tutorial/src/sql_databases/tutorial001.py!} +{!./src/sql_databases/tutorial001.py!} ``` As we are using SQLAlchemy's `scoped_session`, we don't even have to create a dependency with `Depends`. @@ -132,7 +132,7 @@ user = get_user(username, db_session) Then we should declare the path operation without `async def`, just with a normal `def`: ```Python hl_lines="41" -{!./tutorial/src/sql_databases/tutorial001.py!} +{!./src/sql_databases/tutorial001.py!} ``` ## Migrations diff --git a/scripts/lint.sh b/scripts/lint.sh index 90a3df995..0d268800a 100644 --- a/scripts/lint.sh +++ b/scripts/lint.sh @@ -1,6 +1,6 @@ #!/bin/sh -e set -x -autoflake --remove-all-unused-imports --recursive --remove-unused-variables --in-place docs/tutorial/src/ fastapi tests --exclude=__init__.py -black fastapi tests docs/tutorial/src -isort --multi-line=3 --trailing-comma --force-grid-wrap=0 --combine-as --line-width 88 --recursive --apply fastapi tests docs/tutorial/src +autoflake --remove-all-unused-imports --recursive --remove-unused-variables --in-place docs/src/ fastapi tests --exclude=__init__.py +black fastapi tests docs/src +isort --multi-line=3 --trailing-comma --force-grid-wrap=0 --combine-as --line-width 88 --recursive --apply fastapi tests docs/src diff --git a/scripts/rename_tutorial_src_files.py b/scripts/rename_tutorial_src_files.py index e095fdfb7..f1185654d 100644 --- a/scripts/rename_tutorial_src_files.py +++ b/scripts/rename_tutorial_src_files.py @@ -2,7 +2,7 @@ from pathlib import Path, PurePath from string import digits -directory = Path("./docs/tutorial/src") +directory = Path("./docs/src") skip_names = {"bigger_applications"} skip_dirs = {directory / name for name in skip_names} dirs = sorted([Path(f) for f in directory.iterdir() if f not in skip_dirs]) diff --git a/scripts/test.sh b/scripts/test.sh index 16add06c2..290fa0cc7 100644 --- a/scripts/test.sh +++ b/scripts/test.sh @@ -6,9 +6,9 @@ set -x export VERSION_SCRIPT="import sys; print('%s.%s' % sys.version_info[0:2])" export PYTHON_VERSION=`python -c "$VERSION_SCRIPT"` -export PYTHONPATH=./docs/tutorial/src +export PYTHONPATH=./docs/src # PYTHONPATH=. pytest --cov=fastapi --cov=tests --cov-fail-under=100 --cov-report=term-missing ${@} --cov-report=html -pytest --cov=fastapi --cov=tests --cov=docs/tutorial/src --cov-report=term-missing ${@} --cov-report=html +pytest --cov=fastapi --cov=tests --cov=docs/src --cov-report=term-missing ${@} --cov-report=html mypy fastapi --disallow-untyped-defs if [ "${PYTHON_VERSION}" = '3.7' ]; then echo "Skipping 'black' on 3.7. See issue https://github.com/ambv/black/issues/494"