diff --git a/.github/workflows/deploy-docs.yml b/.github/workflows/deploy-docs.yml index 77f1c274c..60f8e4b4a 100644 --- a/.github/workflows/deploy-docs.yml +++ b/.github/workflows/deploy-docs.yml @@ -16,8 +16,8 @@ jobs: run: python3.7 -m pip install flit - name: Install docs extras run: python3.7 -m flit install --extras doc - - name: Build MkDocs - run: python3.7 -m mkdocs build + - name: Build Docs + run: python3.7 ./scripts/docs.py build-all - name: Deploy to Netlify uses: nwtgck/actions-netlify@v1.0.3 with: diff --git a/.gitignore b/.gitignore index f110dc597..3dbfdd44d 100644 --- a/.gitignore +++ b/.gitignore @@ -14,3 +14,4 @@ test.db log.txt Pipfile.lock env3.* +docs_build diff --git a/docs/contributing.md b/docs/contributing.md deleted file mode 100644 index 4ca50db9e..000000000 --- a/docs/contributing.md +++ /dev/null @@ -1,176 +0,0 @@ -First, you might want to see the basic ways to [help FastAPI and get help](help-fastapi.md){.internal-link target=_blank}. - -## Developing - -If you already cloned the repository and you know that you need to deep dive in the code, here are some guidelines to set up your environment. - -### Virtual environment with `venv` - -You can create a virtual environment in a directory using Python's `venv` module: - -```console -$ python -m venv env -``` - -That will create a directory `./env/` with the Python binaries and then you will be able to install packages for that isolated environment. - -### Activate the environment - -Activate the new environment with: - -```console -$ source ./env/bin/activate -``` - -Or in Windows' PowerShell: - -```console -$ .\env\Scripts\Activate.ps1 -``` - -Or if you use Bash for Windows (e.g. Git Bash): - -```console -$ source ./env/Scripts/activate -``` - -To check it worked, use: - -```console -$ which pip - -some/directory/fastapi/env/bin/pip -``` - -If it shows the `pip` binary at `env/bin/pip` then it worked. 🎉 - -Or in Windows PowerShell: - -```console -$ Get-Command pip - -some/directory/fastapi/env/bin/pip -``` - -!!! tip - Every time you install a new package with `pip` under that environment, activate the environment again. - - This makes sure that if you use a terminal program installed by that package (like `flit`), you use the one from your local environment and not any other that could be installed globally. - -### Flit - -**FastAPI** uses Flit to build, package and publish the project. - -After activating the environment as described above, install `flit`: - -```console -$ pip install flit -``` - -Now re-activate the environment to make sure you are using the `flit` you just installed (and not a global one). - -And now use `flit` to install the development dependencies: - -```console -$ flit install --deps develop --symlink -``` - -It will install all the dependencies and your local FastAPI in your local environment. - -#### Using your local FastAPI - -If you create a Python file that imports and uses FastAPI, and run it with the Python from your local environment, it will use your local FastAPI source code. - -And if you update that local FastAPI source code, as it is installed with `--symlink`, when you run that Python file again, it will use the fresh version of FastAPI you just edited. - -That way, you don't have to "install" your local version to be able to test every change. - -### Format - -There is a script that you can run that will format and clean all your code: - -```console -$ bash scripts/format.sh -``` - -It will also auto-sort all your imports. - -For it to sort them correctly, you need to have FastAPI installed locally in your environment, with the command in the section above: - -```console -$ flit install --symlink -``` - -### Format imports - -There is another script that formats all the imports and makes sure you don't have unused imports: - -```console -$ bash scripts/format-imports.sh -``` - -As it runs one command after the other and modifies and reverts many files, it takes a bit longer to run, so it might be easier to use `scripts/format.sh` frequently and `scripts/format-imports.sh` only before committing. - -## Docs - -The documentation uses MkDocs. - -All the documentation is in Markdown format in the directory `./docs`. - -Many of the tutorials have blocks of code. - -In most of the cases, these blocks of code are actual complete applications that can be run as is. - -In fact, those blocks of code are not written inside the Markdown, they are Python files in the `./docs/src/` directory. - -And those Python files are included/injected in the documentation when generating the site. - -### Docs for tests - -Most of the tests actually run against the example source files in the documentation. - -This helps making sure that: - -* The documentation is up to date. -* The documentation examples can be run as is. -* Most of the features are covered by the documentation, ensured by test coverage. - -During local development, there is a script that builds the site and checks for any changes, live-reloading: - -```console -$ bash scripts/docs-live.sh -``` - -It will serve the documentation on `http://0.0.0.0:8008`. - -That way, you can edit the documentation/source files and see the changes live. - -### Apps and docs at the same time - -If you run the examples with, e.g.: - -```console -$ uvicorn tutorial001:app --reload -``` - -as Uvicorn by default will use the port `8000`, the documentation on port `8008` won't clash. - -## Tests - -There is a script that you can run locally to test all the code and generate coverage reports in HTML: - -```console -$ bash scripts/test-cov-html.sh -``` - -This command generates a directory `./htmlcov/`, if you open the file `./htmlcov/index.html` in your browser, you can explore interactively the regions of code that are covered by the tests, and notice if there is any region missing. - -### Tests in your editor - -If you want to use the integrated tests in your editor add `./docs/src` to your `PYTHONPATH` variable. - -For example, in VS Code you can create a file `.env` with: - -```env -PYTHONPATH=./docs/src -``` diff --git a/docs/advanced/additional-responses.md b/docs/en/docs/advanced/additional-responses.md similarity index 97% rename from docs/advanced/additional-responses.md rename to docs/en/docs/advanced/additional-responses.md index 71c4cf61d..4b2abaada 100644 --- a/docs/advanced/additional-responses.md +++ b/docs/en/docs/advanced/additional-responses.md @@ -1,3 +1,5 @@ +# Additional Responses in OpenAPI + !!! warning This is a rather advanced topic. @@ -22,7 +24,7 @@ Each of those response `dict`s can have a key `model`, containing a Pydantic mod For example, to declare another response with a status code `404` and a Pydantic model `Message`, you can write: ```Python hl_lines="18 23" -{!./src/additional_responses/tutorial001.py!} +{!../../../docs_src/additional_responses/tutorial001.py!} ``` !!! note @@ -167,7 +169,7 @@ You can use this same `responses` parameter to add different media types for the For example, you can add an additional media type of `image/png`, declaring that your *path operation* can return a JSON object (with media type `application/json`) or a PNG image: ```Python hl_lines="17 18 19 20 21 22 23 24 28" -{!./src/additional_responses/tutorial002.py!} +{!../../../docs_src/additional_responses/tutorial002.py!} ``` !!! note @@ -191,7 +193,7 @@ For example, you can declare a response with a status code `404` that uses a Pyd And a response with a status code `200` that uses your `response_model`, but includes a custom `example`: ```Python hl_lines="20 21 22 23 24 25 26 27 28 29 30 31" -{!./src/additional_responses/tutorial003.py!} +{!../../../docs_src/additional_responses/tutorial003.py!} ``` It will all be combined and included in your OpenAPI, and shown in the API docs: @@ -227,7 +229,7 @@ You can use that technique to re-use some predefined responses in your *path ope For example: ```Python hl_lines="11 12 13 14 15 24" -{!./src/additional_responses/tutorial004.py!} +{!../../../docs_src/additional_responses/tutorial004.py!} ``` ## More information about OpenAPI responses diff --git a/docs/advanced/additional-status-codes.md b/docs/en/docs/advanced/additional-status-codes.md similarity index 95% rename from docs/advanced/additional-status-codes.md rename to docs/en/docs/advanced/additional-status-codes.md index 14a867c98..b9a32cfa3 100644 --- a/docs/advanced/additional-status-codes.md +++ b/docs/en/docs/advanced/additional-status-codes.md @@ -1,3 +1,5 @@ +# Additional Status Codes + By default, **FastAPI** will return the responses using a `JSONResponse`, putting the content you return from your *path operation* inside of that `JSONResponse`. It will use the default status code or the one you set in your *path operation*. @@ -13,7 +15,7 @@ But you also want it to accept new items. And when the items didn't exist before To achieve that, import `JSONResponse`, and return your content there directly, setting the `status_code` that you want: ```Python hl_lines="2 19" -{!./src/additional_status_codes/tutorial001.py!} +{!../../../docs_src/additional_status_codes/tutorial001.py!} ``` !!! warning diff --git a/docs/advanced/advanced-dependencies.md b/docs/en/docs/advanced/advanced-dependencies.md similarity index 91% rename from docs/advanced/advanced-dependencies.md rename to docs/en/docs/advanced/advanced-dependencies.md index ec796362a..5e79776ca 100644 --- a/docs/advanced/advanced-dependencies.md +++ b/docs/en/docs/advanced/advanced-dependencies.md @@ -1,3 +1,5 @@ +# Advanced Dependencies + ## Parameterized dependencies All the dependencies we have seen are a fixed function or class. @@ -17,7 +19,7 @@ Not the class itself (which is already a callable), but an instance of that clas To do that, we declare a method `__call__`: ```Python hl_lines="10" -{!./src/dependencies/tutorial011.py!} +{!../../../docs_src/dependencies/tutorial011.py!} ``` In this case, this `__call__` is what **FastAPI** will use to check for additional parameters and sub-dependencies, and this is what will be called to pass a value to the parameter in your *path operation function* later. @@ -27,7 +29,7 @@ In this case, this `__call__` is what **FastAPI** will use to check for addition And now, we can use `__init__` to declare the parameters of the instance that we can use to "parameterize" the dependency: ```Python hl_lines="7" -{!./src/dependencies/tutorial011.py!} +{!../../../docs_src/dependencies/tutorial011.py!} ``` In this case, **FastAPI** won't ever touch or care about `__init__`, we will use it directly in our code. @@ -37,7 +39,7 @@ In this case, **FastAPI** won't ever touch or care about `__init__`, we will use We could create an instance of this class with: ```Python hl_lines="16" -{!./src/dependencies/tutorial011.py!} +{!../../../docs_src/dependencies/tutorial011.py!} ``` And that way we are able to "parameterize" our dependency, that now has `"bar"` inside of it, as the attribute `checker.fixed_content`. @@ -55,7 +57,7 @@ checker(q="somequery") ...and pass whatever that returns as the value of the dependency in our *path operation function* as the parameter `fixed_content_included`: ```Python hl_lines="20" -{!./src/dependencies/tutorial011.py!} +{!../../../docs_src/dependencies/tutorial011.py!} ``` !!! tip diff --git a/docs/advanced/async-sql-databases.md b/docs/en/docs/advanced/async-sql-databases.md similarity index 90% rename from docs/advanced/async-sql-databases.md rename to docs/en/docs/advanced/async-sql-databases.md index c7d0b86df..523bc91bf 100644 --- a/docs/advanced/async-sql-databases.md +++ b/docs/en/docs/advanced/async-sql-databases.md @@ -1,3 +1,5 @@ +# Async SQL (Relational) Databases + You can also use `encode/databases` with **FastAPI** to connect to databases using `async` and `await`. It is compatible with: @@ -22,7 +24,7 @@ Later, for your production application, you might want to use a database server * Create a table `notes` using the `metadata` object. ```Python hl_lines="4 14 16 17 18 19 20 21 22" -{!./src/async_sql_databases/tutorial001.py!} +{!../../../docs_src/async_sql_databases/tutorial001.py!} ``` !!! tip @@ -37,7 +39,7 @@ Later, for your production application, you might want to use a database server * Create a `database` object. ```Python hl_lines="3 9 12" -{!./src/async_sql_databases/tutorial001.py!} +{!../../../docs_src/async_sql_databases/tutorial001.py!} ``` !!! tip @@ -53,7 +55,7 @@ Here, this section would run directly, right before starting your **FastAPI** ap * Create all the tables from the `metadata` object. ```Python hl_lines="25 26 27 28" -{!./src/async_sql_databases/tutorial001.py!} +{!../../../docs_src/async_sql_databases/tutorial001.py!} ``` ## Create models @@ -64,7 +66,7 @@ Create Pydantic models for: * Notes to be returned (`Note`). ```Python hl_lines="31 32 33 36 37 38 39" -{!./src/async_sql_databases/tutorial001.py!} +{!../../../docs_src/async_sql_databases/tutorial001.py!} ``` By creating these Pydantic models, the input data will be validated, serialized (converted), and annotated (documented). @@ -77,7 +79,7 @@ So, you will be able to see it all in the interactive API docs. * Create event handlers to connect and disconnect from the database. ```Python hl_lines="42 45 46 47 50 51 52" -{!./src/async_sql_databases/tutorial001.py!} +{!../../../docs_src/async_sql_databases/tutorial001.py!} ``` ## Read notes @@ -85,7 +87,7 @@ So, you will be able to see it all in the interactive API docs. Create the *path operation function* to read notes: ```Python hl_lines="55 56 57 58" -{!./src/async_sql_databases/tutorial001.py!} +{!../../../docs_src/async_sql_databases/tutorial001.py!} ``` !!! Note @@ -102,7 +104,7 @@ That documents (and validates, serializes, filters) the output data, as a `list` Create the *path operation function* to create notes: ```Python hl_lines="61 62 63 64 65" -{!./src/async_sql_databases/tutorial001.py!} +{!../../../docs_src/async_sql_databases/tutorial001.py!} ``` !!! Note diff --git a/docs/advanced/custom-request-and-route.md b/docs/en/docs/advanced/custom-request-and-route.md similarity index 91% rename from docs/advanced/custom-request-and-route.md rename to docs/en/docs/advanced/custom-request-and-route.md index 2e823143a..6c5e97a18 100644 --- a/docs/advanced/custom-request-and-route.md +++ b/docs/en/docs/advanced/custom-request-and-route.md @@ -1,3 +1,5 @@ +# Custom Request and APIRoute class + In some cases, you may want to override the logic used by the `Request` and `APIRoute` classes. In particular, this may be a good alternative to logic in a middleware. @@ -35,7 +37,7 @@ If there's no `gzip` in the header, it will not try to decompress the body. That way, the same route class can handle gzip compressed or uncompressed requests. ```Python hl_lines="8 9 10 11 12 13 14 15" -{!./src/custom_request_and_route/tutorial001.py!} +{!../../../docs_src/custom_request_and_route/tutorial001.py!} ``` ### Create a custom `GzipRoute` class @@ -49,7 +51,7 @@ This method returns a function. And that function is what will receive a request Here we use it to create a `GzipRequest` from the original request. ```Python hl_lines="18 19 20 21 22 23 24 25 26" -{!./src/custom_request_and_route/tutorial001.py!} +{!../../../docs_src/custom_request_and_route/tutorial001.py!} ``` !!! note "Technical Details" @@ -83,13 +85,13 @@ We can also use this same approach to access the request body in an exception ha All we need to do is handle the request inside a `try`/`except` block: ```Python hl_lines="13 15" -{!./src/custom_request_and_route/tutorial002.py!} +{!../../../docs_src/custom_request_and_route/tutorial002.py!} ``` If an exception occurs, the`Request` instance will still be in scope, so we can read and make use of the request body when handling the error: ```Python hl_lines="16 17 18" -{!./src/custom_request_and_route/tutorial002.py!} +{!../../../docs_src/custom_request_and_route/tutorial002.py!} ``` ## Custom `APIRoute` class in a router @@ -97,11 +99,11 @@ If an exception occurs, the`Request` instance will still be in scope, so we can You can also set the `route_class` parameter of an `APIRouter`: ```Python hl_lines="26" -{!./src/custom_request_and_route/tutorial003.py!} +{!../../../docs_src/custom_request_and_route/tutorial003.py!} ``` In this example, the *path operations* under the `router` will use the custom `TimedRoute` class, and will have an extra `X-Response-Time` header in the response with the time it took to generate the response: ```Python hl_lines="13 14 15 16 17 18 19 20" -{!./src/custom_request_and_route/tutorial003.py!} +{!../../../docs_src/custom_request_and_route/tutorial003.py!} ``` diff --git a/docs/advanced/custom-response.md b/docs/en/docs/advanced/custom-response.md similarity index 92% rename from docs/advanced/custom-response.md rename to docs/en/docs/advanced/custom-response.md index 21e7abee4..545a84436 100644 --- a/docs/advanced/custom-response.md +++ b/docs/en/docs/advanced/custom-response.md @@ -1,3 +1,5 @@ +# Custom Response - HTML, Stream, File, others + By default, **FastAPI** will return the responses using `JSONResponse`. You can override it by returning a `Response` directly as seen in [Return a Response directly](response-directly.md){.internal-link target=_blank}. @@ -20,7 +22,7 @@ For example, if you are squeezing performance, you can install and use NoSQL. Here we'll see an example using **Couchbase**, a document based NoSQL database. @@ -18,7 +20,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" -{!./src/nosql_databases/tutorial001.py!} +{!../../../docs_src/nosql_databases/tutorial001.py!} ``` ## Define a constant to use as a "document type" @@ -28,7 +30,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" -{!./src/nosql_databases/tutorial001.py!} +{!../../../docs_src/nosql_databases/tutorial001.py!} ``` ## Add a function to get a `Bucket` @@ -53,7 +55,7 @@ This utility function will: * Return it. ```Python hl_lines="13 14 15 16 17 18 19 20 21 22" -{!./src/nosql_databases/tutorial001.py!} +{!../../../docs_src/nosql_databases/tutorial001.py!} ``` ## Create Pydantic models @@ -65,7 +67,7 @@ As **Couchbase** "documents" are actually just "JSON objects", we can model them First, let's create a `User` model: ```Python hl_lines="25 26 27 28 29" -{!./src/nosql_databases/tutorial001.py!} +{!../../../docs_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`. @@ -79,7 +81,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="32 33 34" -{!./src/nosql_databases/tutorial001.py!} +{!../../../docs_src/nosql_databases/tutorial001.py!} ``` !!! note @@ -99,7 +101,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="37 38 39 40 41 42 43" -{!./src/nosql_databases/tutorial001.py!} +{!../../../docs_src/nosql_databases/tutorial001.py!} ``` ### f-strings @@ -134,7 +136,7 @@ UserInDB(username="johndoe", hashed_password="some_hash") ### Create the `FastAPI` app ```Python hl_lines="47" -{!./src/nosql_databases/tutorial001.py!} +{!../../../docs_src/nosql_databases/tutorial001.py!} ``` ### Create the *path operation function* @@ -144,7 +146,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="50 51 52 53 54" -{!./src/nosql_databases/tutorial001.py!} +{!../../../docs_src/nosql_databases/tutorial001.py!} ``` ## Recap diff --git a/docs/advanced/openapi-callbacks.md b/docs/en/docs/advanced/openapi-callbacks.md similarity index 97% rename from docs/advanced/openapi-callbacks.md rename to docs/en/docs/advanced/openapi-callbacks.md index 8d3b0ac9e..15c0ba2a1 100644 --- a/docs/advanced/openapi-callbacks.md +++ b/docs/en/docs/advanced/openapi-callbacks.md @@ -1,3 +1,5 @@ +# OpenAPI Callbacks + You could create an API with a *path operation* that could trigger a request to an *external API* created by someone else (probably the same developer that would be *using* your API). The process that happens when your API app calls the *external API* is named a "callback". Because the software that the external developer wrote sends a request to your API and then your API *calls back*, sending a request to an *external API* (that was probably created by the same developer). @@ -30,7 +32,7 @@ It will have a *path operation* that will receive an `Invoice` body, and a query This part is pretty normal, most of the code is probably already familiar to you: ```Python hl_lines="8 9 10 11 12 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53" -{!./src/openapi_callbacks/tutorial001.py!} +{!../../../docs_src/openapi_callbacks/tutorial001.py!} ``` !!! tip @@ -91,7 +93,7 @@ Because of that, you need to declare what will be the `default_response_class`, But as we are never calling `app.include_router(some_router)`, we need to set the `default_response_class` during creation of the `APIRouter`. ```Python hl_lines="3 24" -{!./src/openapi_callbacks/tutorial001.py!} +{!../../../docs_src/openapi_callbacks/tutorial001.py!} ``` ### Create the callback *path operation* @@ -104,7 +106,7 @@ It should look just like a normal FastAPI *path operation*: * And it could also have a declaration of the response it should return, e.g. `response_model=InvoiceEventReceived`. ```Python hl_lines="15 16 17 20 21 27 28 29 30 31" -{!./src/openapi_callbacks/tutorial001.py!} +{!../../../docs_src/openapi_callbacks/tutorial001.py!} ``` There are 2 main differences from a normal *path operation*: @@ -171,7 +173,7 @@ At this point you have the *callback path operation(s)* needed (the one(s) that Now use the parameter `callbacks` in *your API's path operation decorator* to pass the attribute `.routes` (that's actually just a `list` of routes/*path operations*) from that callback router: ```Python hl_lines="34" -{!./src/openapi_callbacks/tutorial001.py!} +{!../../../docs_src/openapi_callbacks/tutorial001.py!} ``` !!! tip diff --git a/docs/advanced/path-operation-advanced-configuration.md b/docs/en/docs/advanced/path-operation-advanced-configuration.md similarity index 82% rename from docs/advanced/path-operation-advanced-configuration.md rename to docs/en/docs/advanced/path-operation-advanced-configuration.md index eb5cc6a12..7a3247a64 100644 --- a/docs/advanced/path-operation-advanced-configuration.md +++ b/docs/en/docs/advanced/path-operation-advanced-configuration.md @@ -1,3 +1,5 @@ +# Path Operation Advanced Configuration + ## OpenAPI operationId !!! warning @@ -8,7 +10,7 @@ You can set the OpenAPI `operationId` to be used in your *path operation* with t You would have to make sure that it is unique for each operation. ```Python hl_lines="6" -{!./src/path_operation_advanced_configuration/tutorial001.py!} +{!../../../docs_src/path_operation_advanced_configuration/tutorial001.py!} ``` ### Using the *path operation function* name as the operationId @@ -18,7 +20,7 @@ If you want to use your APIs' function names as `operationId`s, you can iterate You should do it after adding all your *path operations*. ```Python hl_lines="2 12 13 14 15 16 17 18 19 20 21 24" -{!./src/path_operation_advanced_configuration/tutorial002.py!} +{!../../../docs_src/path_operation_advanced_configuration/tutorial002.py!} ``` !!! tip @@ -34,7 +36,7 @@ You should do it after adding all your *path operations*. 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" -{!./src/path_operation_advanced_configuration/tutorial003.py!} +{!../../../docs_src/path_operation_advanced_configuration/tutorial003.py!} ``` ## Advanced description from docstring @@ -46,5 +48,5 @@ Adding an `\f` (an escaped "form feed" character) causes **FastAPI** to truncate It won't show up in the documentation, but other tools (such as Sphinx) will be able to use the rest. ```Python hl_lines="19 20 21 22 23 24 25 26 27 28 29" -{!./src/path_operation_advanced_configuration/tutorial004.py!} +{!../../../docs_src/path_operation_advanced_configuration/tutorial004.py!} ``` diff --git a/docs/advanced/response-change-status-code.md b/docs/en/docs/advanced/response-change-status-code.md similarity index 93% rename from docs/advanced/response-change-status-code.md rename to docs/en/docs/advanced/response-change-status-code.md index 1f10e12da..979cef3f0 100644 --- a/docs/advanced/response-change-status-code.md +++ b/docs/en/docs/advanced/response-change-status-code.md @@ -1,3 +1,5 @@ +# Response - Change Status Code + You probably read before that you can set a default [Response Status Code](../tutorial/response-status-code.md){.internal-link target=_blank}. But in some cases you need to return a different status code than the default. @@ -19,7 +21,7 @@ You can declare a parameter of type `Response` in your *path operation function* And then you can set the `status_code` in that *temporal* response object. ```Python hl_lines="1 9 12" -{!./src/response_change_status_code/tutorial001.py!} +{!../../../docs_src/response_change_status_code/tutorial001.py!} ``` And then you can return any object you need, as you normally would (a `dict`, a database model, etc). diff --git a/docs/advanced/response-cookies.md b/docs/en/docs/advanced/response-cookies.md similarity index 94% rename from docs/advanced/response-cookies.md rename to docs/en/docs/advanced/response-cookies.md index 8ce07bbf3..e46ae755f 100644 --- a/docs/advanced/response-cookies.md +++ b/docs/en/docs/advanced/response-cookies.md @@ -1,3 +1,5 @@ +# Response Cookies + ## Use a `Response` parameter You can declare a parameter of type `Response` in your *path operation function*. @@ -5,7 +7,7 @@ You can declare a parameter of type `Response` in your *path operation function* And then you can set cookies in that *temporal* response object. ```Python hl_lines="1 8 9" -{!./src/response_cookies/tutorial002.py!} +{!../../../docs_src/response_cookies/tutorial002.py!} ``` And then you can return any object you need, as you normally would (a `dict`, a database model, etc). @@ -25,7 +27,7 @@ To do that, you can create a response as described in [Return a Response Directl Then set Cookies in it, and then return it: ```Python hl_lines="10 11 12" -{!./src/response_cookies/tutorial001.py!} +{!../../../docs_src/response_cookies/tutorial001.py!} ``` !!! tip diff --git a/docs/advanced/response-directly.md b/docs/en/docs/advanced/response-directly.md similarity index 95% rename from docs/advanced/response-directly.md rename to docs/en/docs/advanced/response-directly.md index 1b1232808..f1dca1812 100644 --- a/docs/advanced/response-directly.md +++ b/docs/en/docs/advanced/response-directly.md @@ -1,3 +1,5 @@ +# Return a Response Directly + When you create a **FastAPI** *path operation* you can normally return any data from it: a `dict`, a `list`, a Pydantic model, a database model, etc. By default, **FastAPI** would automatically convert that return value to JSON using the `jsonable_encoder` explained in [JSON Compatible Encoder](../tutorial/encoder.md){.internal-link target=_blank}. @@ -30,7 +32,7 @@ For example, you cannot put a Pydantic model in a `JSONResponse` without first c For those cases, you can use the `jsonable_encoder` to convert your data before passing it to a response: ```Python hl_lines="4 6 20 21" -{!./src/response_directly/tutorial001.py!} +{!../../../docs_src/response_directly/tutorial001.py!} ``` !!! note "Technical Details" @@ -49,7 +51,7 @@ Let's say that you want to return an `secrets` to check the username and password: ```Python hl_lines="1 11 12 13" -{!./src/security/tutorial007.py!} +{!../../../docs_src/security/tutorial007.py!} ``` This will ensure that `credentials.username` is `"stanleyjobson"`, and that `credentials.password` is `"swordfish"`. This would be similar to: @@ -101,5 +103,5 @@ That way, using `secrets.compare_digest()` in your application code, it will be After detecting that the credentials are incorrect, return an `HTTPException` with a status code 401 (the same returned when no credentials are provided) and add the header `WWW-Authenticate` to make the browser show the login prompt again: ```Python hl_lines="15 16 17 18 19" -{!./src/security/tutorial007.py!} +{!../../../docs_src/security/tutorial007.py!} ``` diff --git a/docs/advanced/security/index.md b/docs/en/docs/advanced/security/index.md similarity index 95% rename from docs/advanced/security/index.md rename to docs/en/docs/advanced/security/index.md index 560af5a7c..0c94986b5 100644 --- a/docs/advanced/security/index.md +++ b/docs/en/docs/advanced/security/index.md @@ -1,3 +1,5 @@ +# Advanced Security - Intro + ## Additional Features There are some extra features to handle security apart from the ones covered in the [Tutorial - User Guide: Security](../../tutorial/security/){.internal-link target=_blank}. diff --git a/docs/advanced/security/oauth2-scopes.md b/docs/en/docs/advanced/security/oauth2-scopes.md similarity index 97% rename from docs/advanced/security/oauth2-scopes.md rename to docs/en/docs/advanced/security/oauth2-scopes.md index 84c806bf8..a78423221 100644 --- a/docs/advanced/security/oauth2-scopes.md +++ b/docs/en/docs/advanced/security/oauth2-scopes.md @@ -1,3 +1,5 @@ +# OAuth2 scopes + You can use OAuth2 scopes directly with **FastAPI**, they are integrated to work seamlessly. This would allow you to have a more fine-grained permission system, following the OAuth2 standard, integrated into your OpenAPI application (and the API docs). @@ -55,7 +57,7 @@ They are normally used to declare specific security permissions, for example: First, let's quickly see the parts that change from the examples in the main **Tutorial - User Guide** for [OAuth2 with Password (and hashing), Bearer with JWT tokens](../../tutorial/security/oauth2-jwt.md){.internal-link target=_blank}. Now using OAuth2 scopes: ```Python hl_lines="2 5 9 13 47 65 106 108 109 110 111 112 113 114 115 116 122 123 124 125 129 130 131 132 133 134 135 140 154" -{!./src/security/tutorial005.py!} +{!../../../docs_src/security/tutorial005.py!} ``` Now let's review those changes step by step. @@ -67,7 +69,7 @@ The first change is that now we are declaring the OAuth2 security scheme with tw The `scopes` parameter receives a `dict` with each scope as a key and the description as the value: ```Python hl_lines="63 64 65 66" -{!./src/security/tutorial005.py!} +{!../../../docs_src/security/tutorial005.py!} ``` Because we are now declaring those scopes, they will show up in the API docs when you log-in/authorize. @@ -92,7 +94,7 @@ And we return the scopes as part of the JWT token. But in your application, for security, you should make sure you only add the scopes that the user is actually able to have, or the ones you have predefined. ```Python hl_lines="155" -{!./src/security/tutorial005.py!} +{!../../../docs_src/security/tutorial005.py!} ``` ## Declare scopes in *path operations* and dependencies @@ -117,7 +119,7 @@ In this case, it requires the scope `me` (it could require more than one scope). We are doing it here to demonstrate how **FastAPI** handles scopes declared at different levels. ```Python hl_lines="5 140 167" -{!./src/security/tutorial005.py!} +{!../../../docs_src/security/tutorial005.py!} ``` !!! info "Technical Details" @@ -142,7 +144,7 @@ We also declare a special parameter of type `SecurityScopes`, imported from `fas This `SecurityScopes` class is similar to `Request` (`Request` was used to get the request object directly). ```Python hl_lines="9 106" -{!./src/security/tutorial005.py!} +{!../../../docs_src/security/tutorial005.py!} ``` ## Use the `scopes` @@ -158,7 +160,7 @@ We create an `HTTPException` that we can re-use (`raise`) later at several point In this exception, we include the scopes required (if any) as a string separated by spaces (using `scope_str`). We put that string containing the scopes in in the `WWW-Authenticate` header (this is part of the spec). ```Python hl_lines="106 108 109 110 111 112 113 114 115 116" -{!./src/security/tutorial005.py!} +{!../../../docs_src/security/tutorial005.py!} ``` ## Verify the `username` and data shape @@ -176,7 +178,7 @@ Instead of, for example, a `dict`, or something else, as it could break the appl We also verify that we have a user with that username, and if not, we raise that same exception we created before. ```Python hl_lines="47 117 118 119 120 121 122 123 124 125 126 127 128" -{!./src/security/tutorial005.py!} +{!../../../docs_src/security/tutorial005.py!} ``` ## Verify the `scopes` @@ -186,7 +188,7 @@ We now verify that all the scopes required, by this dependency and all the depen For this, we use `security_scopes.scopes`, that contains a `list` with all these scopes as `str`. ```Python hl_lines="129 130 131 132 133 134 135" -{!./src/security/tutorial005.py!} +{!../../../docs_src/security/tutorial005.py!} ``` ## Dependency tree and scopes diff --git a/docs/advanced/sql-databases-peewee.md b/docs/en/docs/advanced/sql-databases-peewee.md similarity index 95% rename from docs/advanced/sql-databases-peewee.md rename to docs/en/docs/advanced/sql-databases-peewee.md index ae957bec6..761029006 100644 --- a/docs/advanced/sql-databases-peewee.md +++ b/docs/en/docs/advanced/sql-databases-peewee.md @@ -1,3 +1,5 @@ +# SQL (Relational) Databases with Peewee + !!! warning If you are just starting, the tutorial [SQL (Relational) Databases](../tutorial/sql-databases.md){.internal-link target=_blank} that uses SQLAlchemy should be enough. @@ -60,7 +62,7 @@ Let's refer to the file `sql_app/database.py`. Let's first check all the normal Peewee code, create a Peewee database: ```Python hl_lines="3 5 22" -{!./src/sql_databases_peewee/sql_app/database.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/database.py!} ``` !!! tip @@ -112,7 +114,7 @@ This might seem a bit complex (and it actually is), you don't really need to com We will create a `PeeweeConnectionState`: ```Python hl_lines="10 11 12 13 14 15 16 17 18 19" -{!./src/sql_databases_peewee/sql_app/database.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/database.py!} ``` This class inherits from a special internal class used by Peewee. @@ -133,7 +135,7 @@ So, we need to do some extra tricks to make it work as if it was just using `thr Now, overwrite the `._state` internal attribute in the Peewee database `db` object using the new `PeeweeConnectionState`: ```Python hl_lines="24" -{!./src/sql_databases_peewee/sql_app/database.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/database.py!} ``` !!! tip @@ -160,7 +162,7 @@ This is the same you would do if you followed the Peewee tutorial and updated th Import `db` from `database` (the file `database.py` from above) and use it here. ```Python hl_lines="3 6 7 8 9 10 11 12 15 16 17 18 19 20 21" -{!./src/sql_databases_peewee/sql_app/models.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/models.py!} ``` !!! tip @@ -188,7 +190,7 @@ Now let's check the file `sql_app/schemas.py`. Create all the same Pydantic models as in the SQLAlchemy tutorial: ```Python hl_lines="16 17 18 21 22 25 26 27 28 29 30 34 35 38 39 42 43 44 45 46 47 48" -{!./src/sql_databases_peewee/sql_app/schemas.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/schemas.py!} ``` !!! tip @@ -213,7 +215,7 @@ But recent versions of Pydantic allow providing a custom class that inherits fro We are going to create a custom `PeeweeGetterDict` class and use it in all the same Pydantic *models* / schemas that use `orm_mode`: ```Python hl_lines="3 8 9 10 11 12 13 31 49" -{!./src/sql_databases_peewee/sql_app/schemas.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/schemas.py!} ``` Here we are checking if the attribute that is being accessed (e.g. `.items` in `some_user.items`) is an instance of `peewee.ModelSelect`. @@ -234,7 +236,7 @@ Now let's see the file `sql_app/crud.py`. Create all the same CRUD utils as in the SQLAlchemy tutorial, all the code is very similar: ```Python hl_lines="1 4 5 8 9 12 13 16 17 18 19 20 23 24 27 28 29 30" -{!./src/sql_databases_peewee/sql_app/crud.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/crud.py!} ``` There are some differences with the code for the SQLAlchemy tutorial. @@ -258,7 +260,7 @@ And now in the file `sql_app/main.py` let's integrate and use all the other part In a very simplistic way create the database tables: ```Python hl_lines="9 10 11" -{!./src/sql_databases_peewee/sql_app/main.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/main.py!} ``` ### Create a dependency @@ -266,7 +268,7 @@ In a very simplistic way create the database tables: Create a dependency that will connect the database right at the beginning of a request and disconnect it at the end: ```Python hl_lines="23 24 25 26 27 28 29" -{!./src/sql_databases_peewee/sql_app/main.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/main.py!} ``` Here we have an empty `yield` because we are actually not using the database object directly. @@ -280,7 +282,7 @@ And then, in each *path operation function* that needs to access the database we But we are not using the value given by this dependency (it actually doesn't give any value, as it has an empty `yield`). So, we don't add it to the *path operation function* but to the *path operation decorator* in the `dependencies` parameter: ```Python hl_lines="32 40 47 59 65 72" -{!./src/sql_databases_peewee/sql_app/main.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/main.py!} ``` ### Context variable sub-dependency @@ -290,7 +292,7 @@ For all the `contextvars` parts to work, we need to make sure we have an indepen For that, we need to create another `async` dependency `reset_db_state()` that is used as a sub-dependency in `get_db()`. It will set the value for the context variable (with just a default `dict`) that will be used as the database state for the whole request. And then the dependency `get_db()` will store in it the database state (connection, transactions, etc). ```Python hl_lines="18 19 20" -{!./src/sql_databases_peewee/sql_app/main.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/main.py!} ``` For the **next request**, as we will reset that context variable again in the `async` dependency `reset_db_state()` and then create a new connection in the `get_db()` dependency, that new request will have its own database state (connection, transactions, etc). @@ -319,7 +321,7 @@ async def reset_db_state(): Now, finally, here's the standard **FastAPI** *path operations* code. ```Python hl_lines="32 33 34 35 36 37 40 41 42 43 46 47 48 49 50 51 52 53 56 57 58 59 60 61 62 65 66 67 68 71 72 73 74 75 76 77 78 79" -{!./src/sql_databases_peewee/sql_app/main.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/main.py!} ``` ### About `def` vs `async def` @@ -436,31 +438,31 @@ Repeat the same process with the 10 tabs. This time all of them will wait and yo * `sql_app/database.py`: ```Python -{!./src/sql_databases_peewee/sql_app/database.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/database.py!} ``` * `sql_app/models.py`: ```Python -{!./src/sql_databases_peewee/sql_app/models.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/models.py!} ``` * `sql_app/schemas.py`: ```Python -{!./src/sql_databases_peewee/sql_app/schemas.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/schemas.py!} ``` * `sql_app/crud.py`: ```Python -{!./src/sql_databases_peewee/sql_app/crud.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/crud.py!} ``` * `sql_app/main.py`: ```Python -{!./src/sql_databases_peewee/sql_app/main.py!} +{!../../../docs_src/sql_databases_peewee/sql_app/main.py!} ``` ## Technical Details diff --git a/docs/advanced/sub-applications-proxy.md b/docs/en/docs/advanced/sub-applications-proxy.md similarity index 94% rename from docs/advanced/sub-applications-proxy.md rename to docs/en/docs/advanced/sub-applications-proxy.md index 333ef9ae2..03a7f9446 100644 --- a/docs/advanced/sub-applications-proxy.md +++ b/docs/en/docs/advanced/sub-applications-proxy.md @@ -1,3 +1,5 @@ +# Sub Applications - Behind a Proxy, Mounts + There are at least two situations where you could need to create your **FastAPI** application using some specific paths. But then you need to set them up to be served with a path prefix. @@ -44,7 +46,7 @@ You could want to do this if you have several "independent" applications that yo First, create the main, top-level, **FastAPI** application, and its *path operations*: ```Python hl_lines="3 6 7 8" -{!./src/sub_applications/tutorial001.py!} +{!../../../docs_src/sub_applications/tutorial001.py!} ``` ### Sub-application @@ -56,7 +58,7 @@ This sub-application is just another standard FastAPI application, but this is t When creating the sub-application, use the parameter `openapi_prefix`. In this case, with a prefix of `/subapi`: ```Python hl_lines="11 14 15 16" -{!./src/sub_applications/tutorial001.py!} +{!../../../docs_src/sub_applications/tutorial001.py!} ``` ### Mount the sub-application @@ -66,7 +68,7 @@ In your top-level application, `app`, mount the sub-application, `subapi`. Here you need to make sure you use the same path that you used for the `openapi_prefix`, in this case, `/subapi`: ```Python hl_lines="11 19" -{!./src/sub_applications/tutorial001.py!} +{!../../../docs_src/sub_applications/tutorial001.py!} ``` ## Check the automatic API docs diff --git a/docs/advanced/templates.md b/docs/en/docs/advanced/templates.md similarity index 91% rename from docs/advanced/templates.md rename to docs/en/docs/advanced/templates.md index da4752e4a..871f3163f 100644 --- a/docs/advanced/templates.md +++ b/docs/en/docs/advanced/templates.md @@ -1,3 +1,5 @@ +# Templates + You can use any template engine you want with **FastAPI**. A common election is Jinja2, the same one used by Flask and other tools. @@ -38,7 +40,7 @@ $ pip install aiofiles * Use the `templates` you created to render and return a `TemplateResponse`, passing the `request` as one of the key-value pairs in the Jinja2 "context". ```Python hl_lines="3 10 14 15" -{!./src/templates/tutorial001.py!} +{!../../../docs_src/templates/tutorial001.py!} ``` !!! note @@ -54,7 +56,7 @@ $ pip install aiofiles Then you can write a template at `templates/item.html` with: ```jinja hl_lines="7" -{!./src/templates/templates/item.html!} +{!../../../docs_src/templates/templates/item.html!} ``` It will show the `id` taken from the "context" `dict` you passed: @@ -68,13 +70,13 @@ It will show the `id` taken from the "context" `dict` you passed: And you can also use `url_for()` inside of the template, and use it, for example, with the `StaticFiles` you mounted. ```jinja hl_lines="4" -{!./src/templates/templates/item.html!} +{!../../../docs_src/templates/templates/item.html!} ``` In this example, it would link to a CSS file at `static/styles.css` with: ```CSS hl_lines="4" -{!./src/templates/static/styles.css!} +{!../../../docs_src/templates/static/styles.css!} ``` And because you are using `StaticFiles`, that CSS file would be served automatically by your **FastAPI** application at the URL `/static/styles.css`. diff --git a/docs/advanced/testing-dependencies.md b/docs/en/docs/advanced/testing-dependencies.md similarity index 96% rename from docs/advanced/testing-dependencies.md rename to docs/en/docs/advanced/testing-dependencies.md index 50d50618c..5ba7e219b 100644 --- a/docs/advanced/testing-dependencies.md +++ b/docs/en/docs/advanced/testing-dependencies.md @@ -1,3 +1,5 @@ +# Testing Dependencies with Overrides + ## Overriding dependencies during testing There are some scenarios where you might want to override a dependency during testing. @@ -39,7 +41,7 @@ To override a dependency for testing, you put as a key the original dependency ( And then **FastAPI** will call that override instead of the original dependency. ```Python hl_lines="24 25 28" -{!./src/dependency_testing/tutorial001.py!} +{!../../../docs_src/dependency_testing/tutorial001.py!} ``` !!! tip diff --git a/docs/advanced/testing-events.md b/docs/en/docs/advanced/testing-events.md similarity index 67% rename from docs/advanced/testing-events.md rename to docs/en/docs/advanced/testing-events.md index a52aabe98..45d8d8686 100644 --- a/docs/advanced/testing-events.md +++ b/docs/en/docs/advanced/testing-events.md @@ -1,7 +1,7 @@ -## Testing Events, `startup` and `shutdown` +# Testing Events: startup - shutdown When you need your event handlers (`startup` and `shutdown`) to run in your tests, you can use the `TestClient` with a `with` statement: ```Python hl_lines="9 10 11 12 20 21 22 23 24" -{!./src/app_testing/tutorial003.py!} -``` \ No newline at end of file +{!../../../docs_src/app_testing/tutorial003.py!} +``` diff --git a/docs/advanced/testing-websockets.md b/docs/en/docs/advanced/testing-websockets.md similarity index 72% rename from docs/advanced/testing-websockets.md rename to docs/en/docs/advanced/testing-websockets.md index 8e9ecf34f..a9d6821be 100644 --- a/docs/advanced/testing-websockets.md +++ b/docs/en/docs/advanced/testing-websockets.md @@ -1,9 +1,9 @@ -## Testing WebSockets +# Testing WebSockets You can use the same `TestClient` to test WebSockets. For this, you use the `TestClient` in a `with` statement, connecting to the WebSocket: ```Python hl_lines="27 28 29 30 31" -{!./src/app_testing/tutorial002.py!} +{!../../../docs_src/app_testing/tutorial002.py!} ``` diff --git a/docs/advanced/using-request-directly.md b/docs/en/docs/advanced/using-request-directly.md similarity index 96% rename from docs/advanced/using-request-directly.md rename to docs/en/docs/advanced/using-request-directly.md index 4d9e84d2c..ac452c25e 100644 --- a/docs/advanced/using-request-directly.md +++ b/docs/en/docs/advanced/using-request-directly.md @@ -1,3 +1,5 @@ +# Using the Request Directly + Up to now, you have been declaring the parts of the request that you need with their types. Taking data from: @@ -28,7 +30,7 @@ Let's imagine you want to get the client's IP address/host inside of your *path For that you need to access the request directly. ```Python hl_lines="1 7 8" -{!./src/using_request_directly/tutorial001.py!} +{!../../../docs_src/using_request_directly/tutorial001.py!} ``` By declaring a *path operation function* parameter with the type being the `Request` **FastAPI** will know to pass the `Request` in that parameter. diff --git a/docs/advanced/websockets.md b/docs/en/docs/advanced/websockets.md similarity index 94% rename from docs/advanced/websockets.md rename to docs/en/docs/advanced/websockets.md index a76eab59c..d473cef07 100644 --- a/docs/advanced/websockets.md +++ b/docs/en/docs/advanced/websockets.md @@ -1,3 +1,4 @@ +# WebSockets You can use WebSockets with **FastAPI**. @@ -24,7 +25,7 @@ In production you would have one of the options above. But it's the simplest way to focus on the server-side of WebSockets and have a working example: ```Python hl_lines="2 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 41 42 43" -{!./src/websockets/tutorial001.py!} +{!../../../docs_src/websockets/tutorial001.py!} ``` ## Create a `websocket` @@ -32,7 +33,7 @@ But it's the simplest way to focus on the server-side of WebSockets and have a w In your **FastAPI** application, create a `websocket`: ```Python hl_lines="1 46 47" -{!./src/websockets/tutorial001.py!} +{!../../../docs_src/websockets/tutorial001.py!} ``` !!! note "Technical Details" @@ -45,7 +46,7 @@ In your **FastAPI** application, create a `websocket`: In your WebSocket route you can `await` for messages and send messages. ```Python hl_lines="48 49 50 51 52" -{!./src/websockets/tutorial001.py!} +{!../../../docs_src/websockets/tutorial001.py!} ``` You can receive and send binary, text, and JSON data. @@ -64,7 +65,7 @@ In WebSocket endpoints you can import from `fastapi` and use: They work the same way as for other FastAPI endpoints/*path operations*: ```Python hl_lines="53 54 55 56 57 58 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76" -{!./src/websockets/tutorial002.py!} +{!../../../docs_src/websockets/tutorial002.py!} ``` !!! info diff --git a/docs/advanced/wsgi.md b/docs/en/docs/advanced/wsgi.md similarity index 92% rename from docs/advanced/wsgi.md rename to docs/en/docs/advanced/wsgi.md index 7d2edfc55..4b81aff2e 100644 --- a/docs/advanced/wsgi.md +++ b/docs/en/docs/advanced/wsgi.md @@ -1,3 +1,5 @@ +# Including WSGI - Flask, Django, others + You can mount WSGI applications as you saw with [Sub Applications - Behind a Proxy, Mounts](./sub-applications-proxy.md){.internal-link target=_blank}. For that, you can use the `WSGIMiddleware` and use it to wrap your WSGI application, for example, Flask, Django, etc. @@ -11,7 +13,7 @@ Then wrap the WSGI (e.g. Flask) app with the middleware. And then mount that under a path. ```Python hl_lines="1 3 22" -{!./src/wsgi/tutorial001.py!} +{!../../../docs_src/wsgi/tutorial001.py!} ``` ## Check it diff --git a/docs/alternatives.md b/docs/en/docs/alternatives.md similarity index 99% rename from docs/alternatives.md rename to docs/en/docs/alternatives.md index 730c10794..3d9e3a55a 100644 --- a/docs/alternatives.md +++ b/docs/en/docs/alternatives.md @@ -1,3 +1,5 @@ +# Alternatives, Inspiration and Comparisons + What inspired **FastAPI**, how it compares to other alternatives and what it learned from them. ## Intro diff --git a/docs/async.md b/docs/en/docs/async.md similarity index 99% rename from docs/async.md rename to docs/en/docs/async.md index 24c2f61d2..dc17e5172 100644 --- a/docs/async.md +++ b/docs/en/docs/async.md @@ -1,3 +1,5 @@ +# Concurrency and async / await + Details about the `async def` syntax for *path operation functions* and some background about asynchronous code, concurrency, and parallelism. ## In a hurry? diff --git a/docs/benchmarks.md b/docs/en/docs/benchmarks.md similarity index 99% rename from docs/benchmarks.md rename to docs/en/docs/benchmarks.md index 95efcab53..5223df81d 100644 --- a/docs/benchmarks.md +++ b/docs/en/docs/benchmarks.md @@ -1,3 +1,5 @@ +# Benchmarks + Independent TechEmpower benchmarks show **FastAPI** applications running under Uvicorn as one of the fastest Python frameworks available, only below Starlette and Uvicorn themselves (used internally by FastAPI). (*) But when checking benchmarks and comparisons you should have the following in mind. diff --git a/docs/en/docs/contributing.md b/docs/en/docs/contributing.md new file mode 100644 index 000000000..a08991a15 --- /dev/null +++ b/docs/en/docs/contributing.md @@ -0,0 +1,441 @@ +# Development - Contributing + +First, you might want to see the basic ways to [help FastAPI and get help](help-fastapi.md){.internal-link target=_blank}. + +## Developing + +If you already cloned the repository and you know that you need to deep dive in the code, here are some guidelines to set up your environment. + +### Virtual environment with `venv` + +You can create a virtual environment in a directory using Python's `venv` module: + +
kwargs
. Even if they don't have a default value.
```Python hl_lines="8"
-{!./src/path_params_numeric_validations/tutorial003.py!}
+{!../../../docs_src/path_params_numeric_validations/tutorial003.py!}
```
## Number validations: greater than or equal
@@ -64,7 +66,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"
-{!./src/path_params_numeric_validations/tutorial004.py!}
+{!../../../docs_src/path_params_numeric_validations/tutorial004.py!}
```
## Number validations: greater than and less than or equal
@@ -75,7 +77,7 @@ The same applies for:
* `le`: `l`ess than or `e`qual
```Python hl_lines="9"
-{!./src/path_params_numeric_validations/tutorial005.py!}
+{!../../../docs_src/path_params_numeric_validations/tutorial005.py!}
```
## Number validations: floats, greater than and less than
@@ -89,7 +91,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"
-{!./src/path_params_numeric_validations/tutorial006.py!}
+{!../../../docs_src/path_params_numeric_validations/tutorial006.py!}
```
## Recap
diff --git a/docs/tutorial/path-params.md b/docs/en/docs/tutorial/path-params.md
similarity index 94%
rename from docs/tutorial/path-params.md
rename to docs/en/docs/tutorial/path-params.md
index b4c405455..1c32108bb 100644
--- a/docs/tutorial/path-params.md
+++ b/docs/en/docs/tutorial/path-params.md
@@ -1,7 +1,9 @@
+# Path Parameters
+
You can declare path "parameters" or "variables" with the same syntax used by Python format strings:
```Python hl_lines="6 7"
-{!./src/path_params/tutorial001.py!}
+{!../../../docs_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 +19,7 @@ So, if you run this example and go to regular expression that the parameter should match:
```Python hl_lines="8"
-{!./src/query_params_str_validations/tutorial004.py!}
+{!../../../docs_src/query_params_str_validations/tutorial004.py!}
```
This specific regular expression checks that the received parameter value:
@@ -85,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"
-{!./src/query_params_str_validations/tutorial005.py!}
+{!../../../docs_src/query_params_str_validations/tutorial005.py!}
```
!!! note
@@ -114,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"
-{!./src/query_params_str_validations/tutorial006.py!}
+{!../../../docs_src/query_params_str_validations/tutorial006.py!}
```
!!! info
@@ -129,7 +131,7 @@ When you define a query parameter explicitly with `Query` you can also declare i
For example, to declare a query parameter `q` that can appear multiple times in the URL, you can write:
```Python hl_lines="9"
-{!./src/query_params_str_validations/tutorial011.py!}
+{!../../../docs_src/query_params_str_validations/tutorial011.py!}
```
Then, with a URL like:
@@ -163,7 +165,7 @@ The interactive API docs will update accordingly, to allow multiple values:
And you can also define a default `list` of values if none are provided:
```Python hl_lines="9"
-{!./src/query_params_str_validations/tutorial012.py!}
+{!../../../docs_src/query_params_str_validations/tutorial012.py!}
```
If you go to:
@@ -188,7 +190,7 @@ the default of `q` will be: `["foo", "bar"]` and your response will be:
You can also use `list` directly instead of `List[str]`:
```Python hl_lines="7"
-{!./src/query_params_str_validations/tutorial013.py!}
+{!../../../docs_src/query_params_str_validations/tutorial013.py!}
```
!!! note
@@ -210,13 +212,13 @@ That information will be included in the generated OpenAPI and used by the docum
You can add a `title`:
```Python hl_lines="7"
-{!./src/query_params_str_validations/tutorial007.py!}
+{!../../../docs_src/query_params_str_validations/tutorial007.py!}
```
And a `description`:
```Python hl_lines="11"
-{!./src/query_params_str_validations/tutorial008.py!}
+{!../../../docs_src/query_params_str_validations/tutorial008.py!}
```
## Alias parameters
@@ -238,7 +240,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"
-{!./src/query_params_str_validations/tutorial009.py!}
+{!../../../docs_src/query_params_str_validations/tutorial009.py!}
```
## Deprecating parameters
@@ -250,7 +252,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"
-{!./src/query_params_str_validations/tutorial010.py!}
+{!../../../docs_src/query_params_str_validations/tutorial010.py!}
```
The docs will show it like this:
diff --git a/docs/tutorial/query-params.md b/docs/en/docs/tutorial/query-params.md
similarity index 92%
rename from docs/tutorial/query-params.md
rename to docs/en/docs/tutorial/query-params.md
index 5e7230110..0efc57e1b 100644
--- a/docs/tutorial/query-params.md
+++ b/docs/en/docs/tutorial/query-params.md
@@ -1,7 +1,9 @@
+# Query Parameters
+
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"
-{!./src/query_params/tutorial001.py!}
+{!../../../docs_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 +64,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"
-{!./src/query_params/tutorial002.py!}
+{!../../../docs_src/query_params/tutorial002.py!}
```
In this case, the function parameter `q` will be optional, and will be `None` by default.
@@ -75,7 +77,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"
-{!./src/query_params/tutorial003.py!}
+{!../../../docs_src/query_params/tutorial003.py!}
```
In this case, if you go to:
@@ -120,7 +122,7 @@ And you don't have to declare them in any specific order.
They will be detected by name:
```Python hl_lines="6 8"
-{!./src/query_params/tutorial004.py!}
+{!../../../docs_src/query_params/tutorial004.py!}
```
## Required query parameters
@@ -132,7 +134,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 not declare any default value:
```Python hl_lines="6 7"
-{!./src/query_params/tutorial005.py!}
+{!../../../docs_src/query_params/tutorial005.py!}
```
Here the query parameter `needy` is a required query parameter of type `str`.
@@ -178,7 +180,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"
-{!./src/query_params/tutorial006.py!}
+{!../../../docs_src/query_params/tutorial006.py!}
```
In this case, there are 3 query parameters:
@@ -220,5 +222,5 @@ limit: Optional[int] = None
In a *path operation* that could look like:
```Python hl_lines="9"
-{!./src/query_params/tutorial007.py!}
+{!../../../docs_src/query_params/tutorial007.py!}
```
diff --git a/docs/tutorial/request-files.md b/docs/en/docs/tutorial/request-files.md
similarity index 96%
rename from docs/tutorial/request-files.md
rename to docs/en/docs/tutorial/request-files.md
index b14f45c70..260a5e190 100644
--- a/docs/tutorial/request-files.md
+++ b/docs/en/docs/tutorial/request-files.md
@@ -1,3 +1,5 @@
+# Request Files
+
You can define files to be uploaded by the client using `File`.
!!! info
@@ -12,7 +14,7 @@ You can define files to be uploaded by the client using `File`.
Import `File` and `UploadFile` from `fastapi`:
```Python hl_lines="1"
-{!./src/request_files/tutorial001.py!}
+{!../../../docs_src/request_files/tutorial001.py!}
```
## Define `File` parameters
@@ -20,7 +22,7 @@ Import `File` and `UploadFile` from `fastapi`:
Create file parameters the same way you would for `Body` or `Form`:
```Python hl_lines="7"
-{!./src/request_files/tutorial001.py!}
+{!../../../docs_src/request_files/tutorial001.py!}
```
!!! info
@@ -44,7 +46,7 @@ But there are several cases in where you might benefit from using `UploadFile`.
Define a `File` parameter with a type of `UploadFile`:
```Python hl_lines="12"
-{!./src/request_files/tutorial001.py!}
+{!../../../docs_src/request_files/tutorial001.py!}
```
Using `UploadFile` has several advantages over `bytes`:
@@ -120,7 +122,7 @@ They would be associated to the same "form field" sent using "form data".
To use that, declare a `List` of `bytes` or `UploadFile`:
```Python hl_lines="10 15"
-{!./src/request_files/tutorial002.py!}
+{!../../../docs_src/request_files/tutorial002.py!}
```
You will receive, as declared, a `list` of `bytes` or `UploadFile`s.
diff --git a/docs/tutorial/request-forms-and-files.md b/docs/en/docs/tutorial/request-forms-and-files.md
similarity index 88%
rename from docs/tutorial/request-forms-and-files.md
rename to docs/en/docs/tutorial/request-forms-and-files.md
index 51ba78638..6844f8c65 100644
--- a/docs/tutorial/request-forms-and-files.md
+++ b/docs/en/docs/tutorial/request-forms-and-files.md
@@ -1,3 +1,5 @@
+# Request Forms and Files
+
You can define files and form fields at the same time using `File` and `Form`.
!!! info
@@ -8,7 +10,7 @@ You can define files and form fields at the same time using `File` and `Form`.
## Import `File` and `Form`
```Python hl_lines="1"
-{!./src/request_forms_and_files/tutorial001.py!}
+{!../../../docs_src/request_forms_and_files/tutorial001.py!}
```
## Define `File` and `Form` parameters
@@ -16,7 +18,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="8"
-{!./src/request_forms_and_files/tutorial001.py!}
+{!../../../docs_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/en/docs/tutorial/request-forms.md
similarity index 95%
rename from docs/tutorial/request-forms.md
rename to docs/en/docs/tutorial/request-forms.md
index edb9af641..b5495a400 100644
--- a/docs/tutorial/request-forms.md
+++ b/docs/en/docs/tutorial/request-forms.md
@@ -1,3 +1,5 @@
+# Form Data
+
When you need to receive form fields instead of JSON, you can use `Form`.
!!! info
@@ -10,7 +12,7 @@ When you need to receive form fields instead of JSON, you can use `Form`.
Import `Form` from `fastapi`:
```Python hl_lines="1"
-{!./src/request_forms/tutorial001.py!}
+{!../../../docs_src/request_forms/tutorial001.py!}
```
## Define `Form` parameters
@@ -18,7 +20,7 @@ Import `Form` from `fastapi`:
Create form parameters the same way you would for `Body` or `Query`:
```Python hl_lines="7"
-{!./src/request_forms/tutorial001.py!}
+{!../../../docs_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/en/docs/tutorial/response-model.md
similarity index 92%
rename from docs/tutorial/response-model.md
rename to docs/en/docs/tutorial/response-model.md
index 1f1288ca7..3f06cd4d5 100644
--- a/docs/tutorial/response-model.md
+++ b/docs/en/docs/tutorial/response-model.md
@@ -1,3 +1,5 @@
+# Response Model
+
You can declare the model used for the response with the parameter `response_model` in any of the *path operations*:
* `@app.get()`
@@ -7,7 +9,7 @@ You can declare the model used for the response with the parameter `response_mod
* etc.
```Python hl_lines="17"
-{!./src/response_model/tutorial001.py!}
+{!../../../docs_src/response_model/tutorial001.py!}
```
!!! note
@@ -34,13 +36,13 @@ But most importantly:
Here we are declaring a `UserIn` model, it will contain a plaintext password:
```Python hl_lines="7 9"
-{!./src/response_model/tutorial002.py!}
+{!../../../docs_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="15 16"
-{!./src/response_model/tutorial002.py!}
+{!../../../docs_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.
@@ -57,19 +59,19 @@ But if we use the same model for another *path operation*, we could be sending o
We can instead create an input model with the plaintext password and an output model without it:
```Python hl_lines="7 9 14"
-{!./src/response_model/tutorial003.py!}
+{!../../../docs_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="22"
-{!./src/response_model/tutorial003.py!}
+{!../../../docs_src/response_model/tutorial003.py!}
```
...we declared the `response_model` to be our model `UserOut`, that doesn't include the password:
```Python hl_lines="20"
-{!./src/response_model/tutorial003.py!}
+{!../../../docs_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).
@@ -89,7 +91,7 @@ And both models will be used for the interactive API documentation:
Your response model could have default values, like:
```Python hl_lines="11 13 14"
-{!./src/response_model/tutorial004.py!}
+{!../../../docs_src/response_model/tutorial004.py!}
```
* `description: str = None` has a default of `None`.
@@ -105,7 +107,7 @@ For example, if you have models with many optional attributes in a NoSQL databas
You can set the *path operation decorator* parameter `response_model_exclude_unset=True`:
```Python hl_lines="24"
-{!./src/response_model/tutorial004.py!}
+{!../../../docs_src/response_model/tutorial004.py!}
```
and those default values won't be included in the response, only the values actually set.
@@ -174,7 +176,7 @@ This can be used as a quick shortcut if you have only one Pydantic model and wan
This is because the JSON Schema generated in your app's OpenAPI (and the docs) will still be the one for the complete model, even if you use `response_model_include` or `response_model_exclude` to omit some attributes.
```Python hl_lines="29 35"
-{!./src/response_model/tutorial005.py!}
+{!../../../docs_src/response_model/tutorial005.py!}
```
!!! tip
@@ -187,7 +189,7 @@ This can be used as a quick shortcut if you have only one Pydantic model and wan
If you forget to use a `set` and use a `list` or `tuple` instead, FastAPI will still convert it to a `set` and it will work correctly:
```Python hl_lines="29 35"
-{!./src/response_model/tutorial006.py!}
+{!../../../docs_src/response_model/tutorial006.py!}
```
## Recap
diff --git a/docs/tutorial/response-status-code.md b/docs/en/docs/tutorial/response-status-code.md
similarity index 94%
rename from docs/tutorial/response-status-code.md
rename to docs/en/docs/tutorial/response-status-code.md
index 3f77272de..29b8521fc 100644
--- a/docs/tutorial/response-status-code.md
+++ b/docs/en/docs/tutorial/response-status-code.md
@@ -1,3 +1,5 @@
+# Response Status Code
+
The same way you can specify a response model, you can also declare the HTTP status code used for the response with the parameter `status_code` in any of the *path operations*:
* `@app.get()`
@@ -7,7 +9,7 @@ The same way you can specify a response model, you can also declare the HTTP sta
* etc.
```Python hl_lines="6"
-{!./src/response_status_code/tutorial001.py!}
+{!../../../docs_src/response_status_code/tutorial001.py!}
```
!!! note
@@ -57,7 +59,7 @@ In short:
Let's see the previous example again:
```Python hl_lines="6"
-{!./src/response_status_code/tutorial001.py!}
+{!../../../docs_src/response_status_code/tutorial001.py!}
```
`201` is the status code for "Created".
@@ -67,7 +69,7 @@ But you don't have to memorize what each of these codes mean.
You can use the convenience variables from `fastapi.status`.
```Python hl_lines="1 6"
-{!./src/response_status_code/tutorial002.py!}
+{!../../../docs_src/response_status_code/tutorial002.py!}
```
They are just a convenience, they hold the same number, but that way you can use the editor's autocomplete to find them:
diff --git a/docs/tutorial/security/first-steps.md b/docs/en/docs/tutorial/security/first-steps.md
similarity index 97%
rename from docs/tutorial/security/first-steps.md
rename to docs/en/docs/tutorial/security/first-steps.md
index 6fef16235..c14ec2ad9 100644
--- a/docs/tutorial/security/first-steps.md
+++ b/docs/en/docs/tutorial/security/first-steps.md
@@ -1,3 +1,5 @@
+# Security - First Steps
+
Let's imagine that you have your **backend** API in some domain.
And you have a **frontend** in another domain or in a different path of the same domain (or in a mobile application).
@@ -19,7 +21,7 @@ Let's first just use the code and see how it works, and then we'll come back to
Copy the example in a file `main.py`:
```Python
-{!./src/security/tutorial001.py!}
+{!../../../docs_src/security/tutorial001.py!}
```
## Run it
@@ -115,7 +117,7 @@ In this example we are going to use **OAuth2**, with the **Password** flow, usin
`OAuth2PasswordBearer` is a class that we create passing a parameter of the URL in where the client (the frontend running in the user's browser) can use to send the `username` and `password` and get a token.
```Python hl_lines="6"
-{!./src/security/tutorial001.py!}
+{!../../../docs_src/security/tutorial001.py!}
```
It doesn't create that endpoint / *path operation*, but declares that that URL is the one that the client should use to get the token. That information is used in OpenAPI, and then in the interactive API documentation systems.
@@ -140,7 +142,7 @@ So, it can be used with `Depends`.
Now you can pass that `oauth2_scheme` in a dependency with `Depends`.
```Python hl_lines="10"
-{!./src/security/tutorial001.py!}
+{!../../../docs_src/security/tutorial001.py!}
```
This dependency will provide a `str` that is assigned to the parameter `token` of the *path operation function*.
diff --git a/docs/tutorial/security/get-current-user.md b/docs/en/docs/tutorial/security/get-current-user.md
similarity index 92%
rename from docs/tutorial/security/get-current-user.md
rename to docs/en/docs/tutorial/security/get-current-user.md
index e2d6b7924..3fc0164c3 100644
--- a/docs/tutorial/security/get-current-user.md
+++ b/docs/en/docs/tutorial/security/get-current-user.md
@@ -1,7 +1,9 @@
+# Get Current User
+
In the previous chapter the security system (which is based on the dependency injection system) was giving the *path operation function* a `token` as a `str`:
```Python hl_lines="10"
-{!./src/security/tutorial001.py!}
+{!../../../docs_src/security/tutorial001.py!}
```
But that is still not that useful.
@@ -15,7 +17,7 @@ First, let's create a Pydantic user model.
The same way we use Pydantic to declare bodies, we can use it anywhere else:
```Python hl_lines="5 12 13 14 15 16"
-{!./src/security/tutorial002.py!}
+{!../../../docs_src/security/tutorial002.py!}
```
## Create a `get_current_user` dependency
@@ -29,7 +31,7 @@ Remember that dependencies can have sub-dependencies?
The same as we were doing before in the *path operation* directly, our new dependency `get_current_user` will receive a `token` as a `str` from the sub-dependency `oauth2_scheme`:
```Python hl_lines="25"
-{!./src/security/tutorial002.py!}
+{!../../../docs_src/security/tutorial002.py!}
```
## Get the user
@@ -37,7 +39,7 @@ The same as we were doing before in the *path operation* directly, our new depen
`get_current_user` will use a (fake) utility function we created, that takes a token as a `str` and returns our Pydantic `User` model:
```Python hl_lines="19 20 21 22 26 27"
-{!./src/security/tutorial002.py!}
+{!../../../docs_src/security/tutorial002.py!}
```
## Inject the current user
@@ -45,7 +47,7 @@ The same as we were doing before in the *path operation* directly, our new depen
So now we can use the same `Depends` with our `get_current_user` in the *path operation*:
```Python hl_lines="31"
-{!./src/security/tutorial002.py!}
+{!../../../docs_src/security/tutorial002.py!}
```
Notice that we declare the type of `current_user` as the Pydantic model `User`.
@@ -97,7 +99,7 @@ And all of them (or any portion of them that you want) can take the advantage of
And all these thousands of *path operations* can be as small as 3 lines:
```Python hl_lines="30 31 32"
-{!./src/security/tutorial002.py!}
+{!../../../docs_src/security/tutorial002.py!}
```
## Recap
diff --git a/docs/tutorial/security/index.md b/docs/en/docs/tutorial/security/index.md
similarity index 99%
rename from docs/tutorial/security/index.md
rename to docs/en/docs/tutorial/security/index.md
index 96c941dd5..9aed2adb5 100644
--- a/docs/tutorial/security/index.md
+++ b/docs/en/docs/tutorial/security/index.md
@@ -1,3 +1,5 @@
+# Security Intro
+
There are many ways to handle security, authentication and authorization.
And it normally is a complex and "difficult" topic.
diff --git a/docs/tutorial/security/oauth2-jwt.md b/docs/en/docs/tutorial/security/oauth2-jwt.md
similarity index 97%
rename from docs/tutorial/security/oauth2-jwt.md
rename to docs/en/docs/tutorial/security/oauth2-jwt.md
index cd15cb9d2..18b4e0e30 100644
--- a/docs/tutorial/security/oauth2-jwt.md
+++ b/docs/en/docs/tutorial/security/oauth2-jwt.md
@@ -1,3 +1,5 @@
+# OAuth2 with Password (and hashing), Bearer with JWT tokens
+
Now that we have all the security flow, let's make the application actually secure, using JWT tokens and secure password hashing.
This code is something you can actually use in your application, save the password hashes in your database, etc.
@@ -99,7 +101,7 @@ And another utility to verify if a received password matches the hash stored.
And another one to authenticate and return a user.
```Python hl_lines="7 48 55 56 59 60 69 70 71 72 73 74 75"
-{!./src/security/tutorial004.py!}
+{!../../../docs_src/security/tutorial004.py!}
```
!!! note
@@ -134,7 +136,7 @@ Define a Pydantic Model that will be used in the token endpoint for the response
Create a utility function to generate a new access token.
```Python hl_lines="3 6 12 13 14 28 29 30 78 79 80 81 82 83 84 85 86"
-{!./src/security/tutorial004.py!}
+{!../../../docs_src/security/tutorial004.py!}
```
## Update the dependencies
@@ -146,7 +148,7 @@ Decode the received token, verify it, and return the current user.
If the token is invalid, return an HTTP error right away.
```Python hl_lines="89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106"
-{!./src/security/tutorial004.py!}
+{!../../../docs_src/security/tutorial004.py!}
```
## Update the `/token` *path operation*
@@ -156,7 +158,7 @@ Create a `timedelta` with the expiration time of the token.
Create a real JWT access token and return it.
```Python hl_lines="115 116 117 118 119 120 121 122 123 124 125 126 127 128"
-{!./src/security/tutorial004.py!}
+{!../../../docs_src/security/tutorial004.py!}
```
### Technical details about the JWT "subject" `sub`
diff --git a/docs/tutorial/security/simple-oauth2.md b/docs/en/docs/tutorial/security/simple-oauth2.md
similarity index 97%
rename from docs/tutorial/security/simple-oauth2.md
rename to docs/en/docs/tutorial/security/simple-oauth2.md
index d7507a1ec..c95c6aa6b 100644
--- a/docs/tutorial/security/simple-oauth2.md
+++ b/docs/en/docs/tutorial/security/simple-oauth2.md
@@ -1,3 +1,5 @@
+# Simple OAuth2 with Password and Bearer
+
Now let's build from the previous chapter and add the missing parts to have a complete security flow.
## Get the `username` and `password`
@@ -48,7 +50,7 @@ Now let's use the utilities provided by **FastAPI** to handle this.
First, import `OAuth2PasswordRequestForm`, and use it as a dependency with `Depends` for the path `/token`:
```Python hl_lines="2 74"
-{!./src/security/tutorial003.py!}
+{!../../../docs_src/security/tutorial003.py!}
```
`OAuth2PasswordRequestForm` is a class dependency that declares a form body with:
@@ -89,7 +91,7 @@ If there is no such user, we return an error saying "incorrect username or passw
For the error, we use the exception `HTTPException`:
```Python hl_lines="1 75 76 77"
-{!./src/security/tutorial003.py!}
+{!../../../docs_src/security/tutorial003.py!}
```
### Check the password
@@ -117,7 +119,7 @@ If your database is stolen, the thief won't have your users' plaintext passwords
So, the thief won't be able to try to use those same passwords in another system (as many users use the same password everywhere, this would be dangerous).
```Python hl_lines="78 79 80 81"
-{!./src/security/tutorial003.py!}
+{!../../../docs_src/security/tutorial003.py!}
```
#### About `**user_dict`
@@ -155,7 +157,7 @@ For this simple example, we are going to just be completely insecure and return
But for now, let's focus on the specific details we need.
```Python hl_lines="83"
-{!./src/security/tutorial003.py!}
+{!../../../docs_src/security/tutorial003.py!}
```
!!! tip
@@ -180,7 +182,7 @@ Both of these dependencies will just return an HTTP error if the user doesn't ex
So, in our endpoint, we will only get a user if the user exists, was correctly authenticated, and is active:
```Python hl_lines="56 57 58 59 60 61 62 63 64 65 67 68 69 70 88"
-{!./src/security/tutorial003.py!}
+{!../../../docs_src/security/tutorial003.py!}
```
!!! info
diff --git a/docs/tutorial/sql-databases.md b/docs/en/docs/tutorial/sql-databases.md
similarity index 95%
rename from docs/tutorial/sql-databases.md
rename to docs/en/docs/tutorial/sql-databases.md
index ba4b83648..179e75f0a 100644
--- a/docs/tutorial/sql-databases.md
+++ b/docs/en/docs/tutorial/sql-databases.md
@@ -1,3 +1,5 @@
+# SQL (Relational) Databases
+
**FastAPI** doesn't require you to use a SQL (relational) database.
But you can use any relational database that you want.
@@ -85,13 +87,13 @@ Let's refer to the file `sql_app/database.py`.
### Import the SQLAlchemy parts
```Python hl_lines="1 2 3"
-{!./src/sql_databases/sql_app/database.py!}
+{!../../../docs_src/sql_databases/sql_app/database.py!}
```
### Create a database URL for SQLAlchemy
```Python hl_lines="5 6"
-{!./src/sql_databases/sql_app/database.py!}
+{!../../../docs_src/sql_databases/sql_app/database.py!}
```
In this example, we are "connecting" to a SQLite database (opening a file with the SQLite database).
@@ -119,7 +121,7 @@ The first step is to create a SQLAlchemy "engine".
We will later use this `engine` in other places.
```Python hl_lines="8 9 10"
-{!./src/sql_databases/sql_app/database.py!}
+{!../../../docs_src/sql_databases/sql_app/database.py!}
```
#### Note
@@ -155,7 +157,7 @@ We will use `Session` (the one imported from SQLAlchemy) later.
To create the `SessionLocal` class, use the function `sessionmaker`:
```Python hl_lines="11"
-{!./src/sql_databases/sql_app/database.py!}
+{!../../../docs_src/sql_databases/sql_app/database.py!}
```
### Create a `Base` class
@@ -165,7 +167,7 @@ Now we will use the function `declarative_base()` that returns a class.
Later we will inherit from this class to create each of the database models or classes (the ORM models):
```Python hl_lines="13"
-{!./src/sql_databases/sql_app/database.py!}
+{!../../../docs_src/sql_databases/sql_app/database.py!}
```
## Create the database models
@@ -188,7 +190,7 @@ Create classes that inherit from it.
These classes are the SQLAlchemy models.
```Python hl_lines="4 7 8 18 19"
-{!./src/sql_databases/sql_app/models.py!}
+{!../../../docs_src/sql_databases/sql_app/models.py!}
```
The `__tablename__` attribute tells SQLAlchemy the name of the table to use in the database for each of these models.
@@ -204,7 +206,7 @@ We use `Column` from SQLAlchemy as the default value.
And we pass a SQLAlchemy class "type", as `Integer`, `String`, and `Boolean`, that defines the type in the database, as an argument.
```Python hl_lines="1 10 11 12 13 21 22 23 24"
-{!./src/sql_databases/sql_app/models.py!}
+{!../../../docs_src/sql_databases/sql_app/models.py!}
```
### Create the relationships
@@ -216,7 +218,7 @@ For this, we use `relationship` provided by SQLAlchemy ORM.
This will become, more or less, a "magic" attribute that will contain the values from other tables related to this one.
```Python hl_lines="2 15 26"
-{!./src/sql_databases/sql_app/models.py!}
+{!../../../docs_src/sql_databases/sql_app/models.py!}
```
When accessing the attribute `items` in a `User`, as in `my_user.items`, it will have a list of `Item` SQLAlchemy models (from the `items` table) that have a foreign key pointing to this record in the `users` table.
@@ -247,7 +249,7 @@ So, the user will also have a `password` when creating it.
But for security, the `password` won't be in other Pydantic *models*, for example, it won't be sent from the API when reading a user.
```Python hl_lines="3 6 7 8 11 12 23 24 27 28"
-{!./src/sql_databases/sql_app/schemas.py!}
+{!../../../docs_src/sql_databases/sql_app/schemas.py!}
```
#### SQLAlchemy style and Pydantic style
@@ -277,7 +279,7 @@ The same way, when reading a user, we can now declare that `items` will contain
Not only the IDs of those items, but all the data that we defined in the Pydantic *model* for reading items: `Item`.
```Python hl_lines="15 16 17 31 32 33 34"
-{!./src/sql_databases/sql_app/schemas.py!}
+{!../../../docs_src/sql_databases/sql_app/schemas.py!}
```
!!! tip
@@ -292,7 +294,7 @@ This Starlette, testing **FastAPI** applications is easy and enjoyable.
It is based on Requests, so it's very familiar and intuitive.
@@ -17,7 +19,7 @@ Use the `TestClient` object the same way as you do with `requests`.
Write simple `assert` statements with the standard Python expressions that you need to check (again, standard `pytest`).
```Python hl_lines="2 12 15 16 17 18"
-{!./src/app_testing/tutorial001.py!}
+{!../../../docs_src/app_testing/tutorial001.py!}
```
!!! tip
@@ -43,7 +45,7 @@ And your **FastAPI** application might also be composed of several files/modules
Let's say you have a file `main.py` with your **FastAPI** app:
```Python
-{!./src/app_testing/main.py!}
+{!../../../docs_src/app_testing/main.py!}
```
### Testing file
@@ -51,7 +53,7 @@ Let's say you have a file `main.py` with your **FastAPI** app:
Then you could have a file `test_main.py` with your tests, and import your `app` from the `main` module (`main.py`):
```Python
-{!./src/app_testing/test_main.py!}
+{!../../../docs_src/app_testing/test_main.py!}
```
## Testing: extended example
@@ -69,7 +71,7 @@ It has a `POST` operation that could return several errors.
Both *path operations* require an `X-Token` header.
```Python
-{!./src/app_testing/main_b.py!}
+{!../../../docs_src/app_testing/main_b.py!}
```
### Extended testing file
@@ -77,7 +79,7 @@ Both *path operations* require an `X-Token` header.
You could then have a `test_main_b.py`, the same as before, with the extended tests:
```Python
-{!./src/app_testing/test_main_b.py!}
+{!../../../docs_src/app_testing/test_main_b.py!}
```
Whenever you need the client to pass information in the request and you don't know how to, you can search (Google) how to do it in `requests`.
diff --git a/docs/en/mkdocs.yml b/docs/en/mkdocs.yml
new file mode 100644
index 000000000..eb3b54e94
--- /dev/null
+++ b/docs/en/mkdocs.yml
@@ -0,0 +1,147 @@
+site_name: FastAPI
+site_description: FastAPI framework, high performance, easy to learn, fast to code, ready for production
+site_url: https://fastapi.tiangolo.com/
+theme:
+ name: material
+ palette:
+ primary: teal
+ accent: amber
+ logo: img/icon-white.svg
+ favicon: img/favicon.png
+ language: en
+repo_name: tiangolo/fastapi
+repo_url: https://github.com/tiangolo/fastapi
+edit_uri: ''
+google_analytics:
+- UA-133183413-1
+- auto
+nav:
+- FastAPI: index.md
+- Languages:
+ - en: /
+ - es: /es/
+- features.md
+- python-types.md
+- Tutorial - User Guide:
+ - tutorial/index.md
+ - tutorial/first-steps.md
+ - tutorial/path-params.md
+ - tutorial/query-params.md
+ - tutorial/body.md
+ - tutorial/query-params-str-validations.md
+ - tutorial/path-params-numeric-validations.md
+ - tutorial/body-multiple-params.md
+ - tutorial/body-fields.md
+ - tutorial/body-nested-models.md
+ - tutorial/extra-data-types.md
+ - tutorial/cookie-params.md
+ - tutorial/header-params.md
+ - tutorial/response-model.md
+ - tutorial/extra-models.md
+ - tutorial/response-status-code.md
+ - tutorial/request-forms.md
+ - tutorial/request-files.md
+ - tutorial/request-forms-and-files.md
+ - tutorial/handling-errors.md
+ - tutorial/path-operation-configuration.md
+ - tutorial/encoder.md
+ - tutorial/body-updates.md
+ - Dependencies:
+ - tutorial/dependencies/index.md
+ - tutorial/dependencies/classes-as-dependencies.md
+ - tutorial/dependencies/sub-dependencies.md
+ - tutorial/dependencies/dependencies-in-path-operation-decorators.md
+ - tutorial/dependencies/dependencies-with-yield.md
+ - Security:
+ - tutorial/security/index.md
+ - tutorial/security/first-steps.md
+ - tutorial/security/get-current-user.md
+ - tutorial/security/simple-oauth2.md
+ - tutorial/security/oauth2-jwt.md
+ - tutorial/middleware.md
+ - tutorial/cors.md
+ - tutorial/sql-databases.md
+ - tutorial/bigger-applications.md
+ - tutorial/background-tasks.md
+ - tutorial/application-configuration.md
+ - tutorial/static-files.md
+ - tutorial/testing.md
+ - tutorial/debugging.md
+- Advanced User Guide:
+ - advanced/index.md
+ - advanced/path-operation-advanced-configuration.md
+ - advanced/additional-status-codes.md
+ - advanced/response-directly.md
+ - advanced/custom-response.md
+ - advanced/additional-responses.md
+ - advanced/response-cookies.md
+ - advanced/response-headers.md
+ - advanced/response-change-status-code.md
+ - advanced/advanced-dependencies.md
+ - Advanced Security:
+ - advanced/security/index.md
+ - advanced/security/oauth2-scopes.md
+ - advanced/security/http-basic-auth.md
+ - advanced/using-request-directly.md
+ - advanced/middleware.md
+ - advanced/sql-databases-peewee.md
+ - advanced/async-sql-databases.md
+ - advanced/nosql-databases.md
+ - advanced/sub-applications-proxy.md
+ - advanced/templates.md
+ - advanced/graphql.md
+ - advanced/websockets.md
+ - advanced/events.md
+ - advanced/custom-request-and-route.md
+ - advanced/testing-websockets.md
+ - advanced/testing-events.md
+ - advanced/testing-dependencies.md
+ - advanced/extending-openapi.md
+ - advanced/openapi-callbacks.md
+ - advanced/wsgi.md
+- async.md
+- deployment.md
+- project-generation.md
+- alternatives.md
+- history-design-future.md
+- external-links.md
+- benchmarks.md
+- help-fastapi.md
+- contributing.md
+- release-notes.md
+markdown_extensions:
+- toc:
+ permalink: true
+- markdown.extensions.codehilite:
+ guess_lang: false
+- markdown_include.include:
+ base_path: docs
+- admonition
+- codehilite
+- extra
+- pymdownx.superfences:
+ custom_fences:
+ - name: mermaid
+ class: mermaid
+ format: !!python/name:pymdownx.superfences.fence_div_format ''
+extra:
+ social:
+ - type: github
+ link: https://github.com/tiangolo/typer
+ - type: twitter
+ link: https://twitter.com/tiangolo
+ - type: linkedin
+ link: https://www.linkedin.com/in/tiangolo
+ - type: rss
+ link: https://dev.to/tiangolo
+ - type: medium
+ link: https://medium.com/@tiangolo
+ - type: globe
+ link: https://tiangolo.com
+extra_css:
+- css/termynal.css
+- css/custom.css
+extra_javascript:
+- https://unpkg.com/mermaid@8.4.6/dist/mermaid.min.js
+- js/termynal.js
+- js/custom.js
diff --git a/docs/es/docs/index.md b/docs/es/docs/index.md
new file mode 100644
index 000000000..11c945372
--- /dev/null
+++ b/docs/es/docs/index.md
@@ -0,0 +1,437 @@
+
++ FastAPI framework, high performance, easy to learn, fast to code, ready for production +
+ + +--- + +**Documentation**: https://fastapi.tiangolo.com + +**Source Code**: https://github.com/tiangolo/fastapi + +--- + +FastAPI is a modern, fast (high-performance), web framework for building APIs with Python 3.6+ based on standard Python type hints. + +The key features are: + +* **Fast**: Very high performance, on par with **NodeJS** and **Go** (thanks to Starlette and Pydantic). [One of the fastest Python frameworks available](#performance). + +* **Fast to code**: Increase the speed to develop features by about 200% to 300% *. +* **Fewer bugs**: Reduce about 40% of human (developer) induced errors. * +* **Intuitive**: Great editor support. Completion everywhere. Less time debugging. +* **Easy**: Designed to be easy to use and learn. Less time reading docs. +* **Short**: Minimize code duplication. Multiple features from each parameter declaration. Fewer bugs. +* **Robust**: Get production-ready code. With automatic interactive documentation. +* **Standards-based**: Based on (and fully compatible with) the open standards for APIs: OpenAPI (previously known as Swagger) and JSON Schema. + +* estimation based on tests on an internal development team, building production applications. + +## Opinions + +"*[...] I'm using **FastAPI** a ton these days. [...] I'm actually planning to use it for all of my team's **ML services at Microsoft**. Some of them are getting integrated into the core **Windows** product and some **Office** products.*" + +async def
...uvicorn main:app --reload
...ujson
- for faster JSON "parsing".
+* email_validator
- for email validation.
+
+Used by Starlette:
+
+* requests
- Required if you want to use the `TestClient`.
+* aiofiles
- Required if you want to use `FileResponse` or `StaticFiles`.
+* jinja2
- Required if you want to use the default template configuration.
+* python-multipart
- Required if you want to support form "parsing", with `request.form()`.
+* itsdangerous
- Required for `SessionMiddleware` support.
+* pyyaml
- Required for Starlette's `SchemaGenerator` support (you probably don't need it with FastAPI).
+* graphene
- Required for `GraphQLApp` support.
+* ujson
- Required if you want to use `UJSONResponse`.
+
+Used by FastAPI / Starlette:
+
+* uvicorn
- for the server that loads and serves your application.
+* orjson
- Required if you want to use `ORJSONResponse`.
+
+You can install all of these with `pip install fastapi[all]`.
+
+## License
+
+This project is licensed under the terms of the MIT license.
diff --git a/docs/es/mkdocs.yml b/docs/es/mkdocs.yml
new file mode 100644
index 000000000..8fb7cf469
--- /dev/null
+++ b/docs/es/mkdocs.yml
@@ -0,0 +1,58 @@
+site_name: FastAPI
+site_description: FastAPI framework, high performance, easy to learn, fast to code, ready for production
+site_url: https://fastapi.tiangolo.com/es/
+theme:
+ name: material
+ palette:
+ primary: teal
+ accent: amber
+ logo: https://fastapi.tiangolo.com/img/icon-white.svg
+ favicon: https://fastapi.tiangolo.com/img/favicon.png
+ language: es
+repo_name: tiangolo/fastapi
+repo_url: https://github.com/tiangolo/fastapi
+edit_uri: ''
+google_analytics:
+- UA-133183413-1
+- auto
+nav:
+- FastAPI: index.md
+- Languages:
+ - en: /
+ - es: /es/
+markdown_extensions:
+- toc:
+ permalink: true
+- markdown.extensions.codehilite:
+ guess_lang: false
+- markdown_include.include:
+ base_path: docs
+- admonition
+- codehilite
+- extra
+- pymdownx.superfences:
+ custom_fences:
+ - name: mermaid
+ class: mermaid
+ format: !!python/name:pymdownx.superfences.fence_div_format ''
+extra:
+ social:
+ - type: github
+ link: https://github.com/tiangolo/typer
+ - type: twitter
+ link: https://twitter.com/tiangolo
+ - type: linkedin
+ link: https://www.linkedin.com/in/tiangolo
+ - type: rss
+ link: https://dev.to/tiangolo
+ - type: medium
+ link: https://medium.com/@tiangolo
+ - type: globe
+ link: https://tiangolo.com
+extra_css:
+- https://fastapi.tiangolo.com/css/termynal.css
+- https://fastapi.tiangolo.com/css/custom.css
+extra_javascript:
+- https://unpkg.com/mermaid@8.4.6/dist/mermaid.min.js
+- https://fastapi.tiangolo.com/js/termynal.js
+- https://fastapi.tiangolo.com/js/custom.js
diff --git a/docs/missing-translation.md b/docs/missing-translation.md
new file mode 100644
index 000000000..32b6016f9
--- /dev/null
+++ b/docs/missing-translation.md
@@ -0,0 +1,4 @@
+!!! warning
+ The current page still doesn't have a translation for this language.
+
+ But you can help translating it: [Contributing](https://fastapi.tiangolo.com/contributing/){.internal-link target=_blank}.
diff --git a/docs/src/additional_responses/tutorial001.py b/docs_src/additional_responses/tutorial001.py
similarity index 100%
rename from docs/src/additional_responses/tutorial001.py
rename to docs_src/additional_responses/tutorial001.py
diff --git a/docs/src/additional_responses/tutorial002.py b/docs_src/additional_responses/tutorial002.py
similarity index 100%
rename from docs/src/additional_responses/tutorial002.py
rename to docs_src/additional_responses/tutorial002.py
diff --git a/docs/src/additional_responses/tutorial003.py b/docs_src/additional_responses/tutorial003.py
similarity index 100%
rename from docs/src/additional_responses/tutorial003.py
rename to docs_src/additional_responses/tutorial003.py
diff --git a/docs/src/additional_responses/tutorial004.py b/docs_src/additional_responses/tutorial004.py
similarity index 100%
rename from docs/src/additional_responses/tutorial004.py
rename to docs_src/additional_responses/tutorial004.py
diff --git a/docs/src/additional_status_codes/tutorial001.py b/docs_src/additional_status_codes/tutorial001.py
similarity index 100%
rename from docs/src/additional_status_codes/tutorial001.py
rename to docs_src/additional_status_codes/tutorial001.py
diff --git a/docs/src/advanced_middleware/tutorial001.py b/docs_src/advanced_middleware/tutorial001.py
similarity index 100%
rename from docs/src/advanced_middleware/tutorial001.py
rename to docs_src/advanced_middleware/tutorial001.py
diff --git a/docs/src/advanced_middleware/tutorial002.py b/docs_src/advanced_middleware/tutorial002.py
similarity index 100%
rename from docs/src/advanced_middleware/tutorial002.py
rename to docs_src/advanced_middleware/tutorial002.py
diff --git a/docs/src/advanced_middleware/tutorial003.py b/docs_src/advanced_middleware/tutorial003.py
similarity index 100%
rename from docs/src/advanced_middleware/tutorial003.py
rename to docs_src/advanced_middleware/tutorial003.py
diff --git a/docs/src/app_testing/__init__.py b/docs_src/app_testing/__init__.py
similarity index 100%
rename from docs/src/app_testing/__init__.py
rename to docs_src/app_testing/__init__.py
diff --git a/docs/src/app_testing/main.py b/docs_src/app_testing/main.py
similarity index 100%
rename from docs/src/app_testing/main.py
rename to docs_src/app_testing/main.py
diff --git a/docs/src/app_testing/main_b.py b/docs_src/app_testing/main_b.py
similarity index 100%
rename from docs/src/app_testing/main_b.py
rename to docs_src/app_testing/main_b.py
diff --git a/docs/src/app_testing/test_main.py b/docs_src/app_testing/test_main.py
similarity index 100%
rename from docs/src/app_testing/test_main.py
rename to docs_src/app_testing/test_main.py
diff --git a/docs/src/app_testing/test_main_b.py b/docs_src/app_testing/test_main_b.py
similarity index 100%
rename from docs/src/app_testing/test_main_b.py
rename to docs_src/app_testing/test_main_b.py
diff --git a/docs/src/app_testing/tutorial001.py b/docs_src/app_testing/tutorial001.py
similarity index 100%
rename from docs/src/app_testing/tutorial001.py
rename to docs_src/app_testing/tutorial001.py
diff --git a/docs/src/app_testing/tutorial002.py b/docs_src/app_testing/tutorial002.py
similarity index 100%
rename from docs/src/app_testing/tutorial002.py
rename to docs_src/app_testing/tutorial002.py
diff --git a/docs/src/app_testing/tutorial003.py b/docs_src/app_testing/tutorial003.py
similarity index 100%
rename from docs/src/app_testing/tutorial003.py
rename to docs_src/app_testing/tutorial003.py
diff --git a/docs/src/application_configuration/tutorial001.py b/docs_src/application_configuration/tutorial001.py
similarity index 100%
rename from docs/src/application_configuration/tutorial001.py
rename to docs_src/application_configuration/tutorial001.py
diff --git a/docs/src/application_configuration/tutorial002.py b/docs_src/application_configuration/tutorial002.py
similarity index 100%
rename from docs/src/application_configuration/tutorial002.py
rename to docs_src/application_configuration/tutorial002.py
diff --git a/docs/src/application_configuration/tutorial003.py b/docs_src/application_configuration/tutorial003.py
similarity index 100%
rename from docs/src/application_configuration/tutorial003.py
rename to docs_src/application_configuration/tutorial003.py
diff --git a/docs/src/async_sql_databases/tutorial001.py b/docs_src/async_sql_databases/tutorial001.py
similarity index 100%
rename from docs/src/async_sql_databases/tutorial001.py
rename to docs_src/async_sql_databases/tutorial001.py
diff --git a/docs/src/background_tasks/tutorial001.py b/docs_src/background_tasks/tutorial001.py
similarity index 100%
rename from docs/src/background_tasks/tutorial001.py
rename to docs_src/background_tasks/tutorial001.py
diff --git a/docs/src/background_tasks/tutorial002.py b/docs_src/background_tasks/tutorial002.py
similarity index 100%
rename from docs/src/background_tasks/tutorial002.py
rename to docs_src/background_tasks/tutorial002.py
diff --git a/docs/src/bigger_applications/__init__.py b/docs_src/bigger_applications/__init__.py
similarity index 100%
rename from docs/src/bigger_applications/__init__.py
rename to docs_src/bigger_applications/__init__.py
diff --git a/docs/src/bigger_applications/app/__init__.py b/docs_src/bigger_applications/app/__init__.py
similarity index 100%
rename from docs/src/bigger_applications/app/__init__.py
rename to docs_src/bigger_applications/app/__init__.py
diff --git a/docs/src/bigger_applications/app/main.py b/docs_src/bigger_applications/app/main.py
similarity index 100%
rename from docs/src/bigger_applications/app/main.py
rename to docs_src/bigger_applications/app/main.py
diff --git a/docs/src/bigger_applications/app/routers/__init__.py b/docs_src/bigger_applications/app/routers/__init__.py
similarity index 100%
rename from docs/src/bigger_applications/app/routers/__init__.py
rename to docs_src/bigger_applications/app/routers/__init__.py
diff --git a/docs/src/bigger_applications/app/routers/items.py b/docs_src/bigger_applications/app/routers/items.py
similarity index 100%
rename from docs/src/bigger_applications/app/routers/items.py
rename to docs_src/bigger_applications/app/routers/items.py
diff --git a/docs/src/bigger_applications/app/routers/users.py b/docs_src/bigger_applications/app/routers/users.py
similarity index 100%
rename from docs/src/bigger_applications/app/routers/users.py
rename to docs_src/bigger_applications/app/routers/users.py
diff --git a/docs/src/body/tutorial001.py b/docs_src/body/tutorial001.py
similarity index 100%
rename from docs/src/body/tutorial001.py
rename to docs_src/body/tutorial001.py
diff --git a/docs/src/body/tutorial002.py b/docs_src/body/tutorial002.py
similarity index 100%
rename from docs/src/body/tutorial002.py
rename to docs_src/body/tutorial002.py
diff --git a/docs/src/body/tutorial003.py b/docs_src/body/tutorial003.py
similarity index 100%
rename from docs/src/body/tutorial003.py
rename to docs_src/body/tutorial003.py
diff --git a/docs/src/body/tutorial004.py b/docs_src/body/tutorial004.py
similarity index 100%
rename from docs/src/body/tutorial004.py
rename to docs_src/body/tutorial004.py
diff --git a/docs/src/body_fields/tutorial001.py b/docs_src/body_fields/tutorial001.py
similarity index 100%
rename from docs/src/body_fields/tutorial001.py
rename to docs_src/body_fields/tutorial001.py
diff --git a/docs/src/body_fields/tutorial002.py b/docs_src/body_fields/tutorial002.py
similarity index 100%
rename from docs/src/body_fields/tutorial002.py
rename to docs_src/body_fields/tutorial002.py
diff --git a/docs/src/body_multiple_params/tutorial001.py b/docs_src/body_multiple_params/tutorial001.py
similarity index 100%
rename from docs/src/body_multiple_params/tutorial001.py
rename to docs_src/body_multiple_params/tutorial001.py
diff --git a/docs/src/body_multiple_params/tutorial002.py b/docs_src/body_multiple_params/tutorial002.py
similarity index 100%
rename from docs/src/body_multiple_params/tutorial002.py
rename to docs_src/body_multiple_params/tutorial002.py
diff --git a/docs/src/body_multiple_params/tutorial003.py b/docs_src/body_multiple_params/tutorial003.py
similarity index 100%
rename from docs/src/body_multiple_params/tutorial003.py
rename to docs_src/body_multiple_params/tutorial003.py
diff --git a/docs/src/body_multiple_params/tutorial004.py b/docs_src/body_multiple_params/tutorial004.py
similarity index 100%
rename from docs/src/body_multiple_params/tutorial004.py
rename to docs_src/body_multiple_params/tutorial004.py
diff --git a/docs/src/body_multiple_params/tutorial005.py b/docs_src/body_multiple_params/tutorial005.py
similarity index 100%
rename from docs/src/body_multiple_params/tutorial005.py
rename to docs_src/body_multiple_params/tutorial005.py
diff --git a/docs/src/body_nested_models/tutorial001.py b/docs_src/body_nested_models/tutorial001.py
similarity index 100%
rename from docs/src/body_nested_models/tutorial001.py
rename to docs_src/body_nested_models/tutorial001.py
diff --git a/docs/src/body_nested_models/tutorial002.py b/docs_src/body_nested_models/tutorial002.py
similarity index 100%
rename from docs/src/body_nested_models/tutorial002.py
rename to docs_src/body_nested_models/tutorial002.py
diff --git a/docs/src/body_nested_models/tutorial003.py b/docs_src/body_nested_models/tutorial003.py
similarity index 100%
rename from docs/src/body_nested_models/tutorial003.py
rename to docs_src/body_nested_models/tutorial003.py
diff --git a/docs/src/body_nested_models/tutorial004.py b/docs_src/body_nested_models/tutorial004.py
similarity index 100%
rename from docs/src/body_nested_models/tutorial004.py
rename to docs_src/body_nested_models/tutorial004.py
diff --git a/docs/src/body_nested_models/tutorial005.py b/docs_src/body_nested_models/tutorial005.py
similarity index 100%
rename from docs/src/body_nested_models/tutorial005.py
rename to docs_src/body_nested_models/tutorial005.py
diff --git a/docs/src/body_nested_models/tutorial006.py b/docs_src/body_nested_models/tutorial006.py
similarity index 100%
rename from docs/src/body_nested_models/tutorial006.py
rename to docs_src/body_nested_models/tutorial006.py
diff --git a/docs/src/body_nested_models/tutorial007.py b/docs_src/body_nested_models/tutorial007.py
similarity index 100%
rename from docs/src/body_nested_models/tutorial007.py
rename to docs_src/body_nested_models/tutorial007.py
diff --git a/docs/src/body_nested_models/tutorial008.py b/docs_src/body_nested_models/tutorial008.py
similarity index 100%
rename from docs/src/body_nested_models/tutorial008.py
rename to docs_src/body_nested_models/tutorial008.py
diff --git a/docs/src/body_nested_models/tutorial009.py b/docs_src/body_nested_models/tutorial009.py
similarity index 100%
rename from docs/src/body_nested_models/tutorial009.py
rename to docs_src/body_nested_models/tutorial009.py
diff --git a/docs/src/body_updates/tutorial001.py b/docs_src/body_updates/tutorial001.py
similarity index 100%
rename from docs/src/body_updates/tutorial001.py
rename to docs_src/body_updates/tutorial001.py
diff --git a/docs/src/body_updates/tutorial002.py b/docs_src/body_updates/tutorial002.py
similarity index 100%
rename from docs/src/body_updates/tutorial002.py
rename to docs_src/body_updates/tutorial002.py
diff --git a/docs/src/cookie_params/tutorial001.py b/docs_src/cookie_params/tutorial001.py
similarity index 100%
rename from docs/src/cookie_params/tutorial001.py
rename to docs_src/cookie_params/tutorial001.py
diff --git a/docs/src/cors/tutorial001.py b/docs_src/cors/tutorial001.py
similarity index 100%
rename from docs/src/cors/tutorial001.py
rename to docs_src/cors/tutorial001.py
diff --git a/docs/src/custom_request_and_route/tutorial001.py b/docs_src/custom_request_and_route/tutorial001.py
similarity index 100%
rename from docs/src/custom_request_and_route/tutorial001.py
rename to docs_src/custom_request_and_route/tutorial001.py
diff --git a/docs/src/custom_request_and_route/tutorial002.py b/docs_src/custom_request_and_route/tutorial002.py
similarity index 100%
rename from docs/src/custom_request_and_route/tutorial002.py
rename to docs_src/custom_request_and_route/tutorial002.py
diff --git a/docs/src/custom_request_and_route/tutorial003.py b/docs_src/custom_request_and_route/tutorial003.py
similarity index 100%
rename from docs/src/custom_request_and_route/tutorial003.py
rename to docs_src/custom_request_and_route/tutorial003.py
diff --git a/docs/src/custom_response/tutorial001.py b/docs_src/custom_response/tutorial001.py
similarity index 100%
rename from docs/src/custom_response/tutorial001.py
rename to docs_src/custom_response/tutorial001.py
diff --git a/docs/src/custom_response/tutorial001b.py b/docs_src/custom_response/tutorial001b.py
similarity index 100%
rename from docs/src/custom_response/tutorial001b.py
rename to docs_src/custom_response/tutorial001b.py
diff --git a/docs/src/custom_response/tutorial002.py b/docs_src/custom_response/tutorial002.py
similarity index 100%
rename from docs/src/custom_response/tutorial002.py
rename to docs_src/custom_response/tutorial002.py
diff --git a/docs/src/custom_response/tutorial003.py b/docs_src/custom_response/tutorial003.py
similarity index 100%
rename from docs/src/custom_response/tutorial003.py
rename to docs_src/custom_response/tutorial003.py
diff --git a/docs/src/custom_response/tutorial004.py b/docs_src/custom_response/tutorial004.py
similarity index 100%
rename from docs/src/custom_response/tutorial004.py
rename to docs_src/custom_response/tutorial004.py
diff --git a/docs/src/custom_response/tutorial005.py b/docs_src/custom_response/tutorial005.py
similarity index 100%
rename from docs/src/custom_response/tutorial005.py
rename to docs_src/custom_response/tutorial005.py
diff --git a/docs/src/custom_response/tutorial006.py b/docs_src/custom_response/tutorial006.py
similarity index 100%
rename from docs/src/custom_response/tutorial006.py
rename to docs_src/custom_response/tutorial006.py
diff --git a/docs/src/custom_response/tutorial007.py b/docs_src/custom_response/tutorial007.py
similarity index 100%
rename from docs/src/custom_response/tutorial007.py
rename to docs_src/custom_response/tutorial007.py
diff --git a/docs/src/custom_response/tutorial008.py b/docs_src/custom_response/tutorial008.py
similarity index 100%
rename from docs/src/custom_response/tutorial008.py
rename to docs_src/custom_response/tutorial008.py
diff --git a/docs/src/custom_response/tutorial009.py b/docs_src/custom_response/tutorial009.py
similarity index 100%
rename from docs/src/custom_response/tutorial009.py
rename to docs_src/custom_response/tutorial009.py
diff --git a/docs/src/debugging/tutorial001.py b/docs_src/debugging/tutorial001.py
similarity index 100%
rename from docs/src/debugging/tutorial001.py
rename to docs_src/debugging/tutorial001.py
diff --git a/docs/src/dependencies/tutorial001.py b/docs_src/dependencies/tutorial001.py
similarity index 100%
rename from docs/src/dependencies/tutorial001.py
rename to docs_src/dependencies/tutorial001.py
diff --git a/docs/src/dependencies/tutorial002.py b/docs_src/dependencies/tutorial002.py
similarity index 100%
rename from docs/src/dependencies/tutorial002.py
rename to docs_src/dependencies/tutorial002.py
diff --git a/docs/src/dependencies/tutorial003.py b/docs_src/dependencies/tutorial003.py
similarity index 100%
rename from docs/src/dependencies/tutorial003.py
rename to docs_src/dependencies/tutorial003.py
diff --git a/docs/src/dependencies/tutorial004.py b/docs_src/dependencies/tutorial004.py
similarity index 100%
rename from docs/src/dependencies/tutorial004.py
rename to docs_src/dependencies/tutorial004.py
diff --git a/docs/src/dependencies/tutorial005.py b/docs_src/dependencies/tutorial005.py
similarity index 100%
rename from docs/src/dependencies/tutorial005.py
rename to docs_src/dependencies/tutorial005.py
diff --git a/docs/src/dependencies/tutorial006.py b/docs_src/dependencies/tutorial006.py
similarity index 100%
rename from docs/src/dependencies/tutorial006.py
rename to docs_src/dependencies/tutorial006.py
diff --git a/docs/src/dependencies/tutorial007.py b/docs_src/dependencies/tutorial007.py
similarity index 100%
rename from docs/src/dependencies/tutorial007.py
rename to docs_src/dependencies/tutorial007.py
diff --git a/docs/src/dependencies/tutorial008.py b/docs_src/dependencies/tutorial008.py
similarity index 100%
rename from docs/src/dependencies/tutorial008.py
rename to docs_src/dependencies/tutorial008.py
diff --git a/docs/src/dependencies/tutorial009.py b/docs_src/dependencies/tutorial009.py
similarity index 100%
rename from docs/src/dependencies/tutorial009.py
rename to docs_src/dependencies/tutorial009.py
diff --git a/docs/src/dependencies/tutorial010.py b/docs_src/dependencies/tutorial010.py
similarity index 100%
rename from docs/src/dependencies/tutorial010.py
rename to docs_src/dependencies/tutorial010.py
diff --git a/docs/src/dependencies/tutorial011.py b/docs_src/dependencies/tutorial011.py
similarity index 100%
rename from docs/src/dependencies/tutorial011.py
rename to docs_src/dependencies/tutorial011.py
diff --git a/docs/src/dependency_testing/tutorial001.py b/docs_src/dependency_testing/tutorial001.py
similarity index 100%
rename from docs/src/dependency_testing/tutorial001.py
rename to docs_src/dependency_testing/tutorial001.py
diff --git a/docs/src/encoder/tutorial001.py b/docs_src/encoder/tutorial001.py
similarity index 100%
rename from docs/src/encoder/tutorial001.py
rename to docs_src/encoder/tutorial001.py
diff --git a/docs/src/events/tutorial001.py b/docs_src/events/tutorial001.py
similarity index 100%
rename from docs/src/events/tutorial001.py
rename to docs_src/events/tutorial001.py
diff --git a/docs/src/events/tutorial002.py b/docs_src/events/tutorial002.py
similarity index 100%
rename from docs/src/events/tutorial002.py
rename to docs_src/events/tutorial002.py
diff --git a/docs/src/extending_openapi/tutorial001.py b/docs_src/extending_openapi/tutorial001.py
similarity index 100%
rename from docs/src/extending_openapi/tutorial001.py
rename to docs_src/extending_openapi/tutorial001.py
diff --git a/docs/src/extending_openapi/tutorial002.py b/docs_src/extending_openapi/tutorial002.py
similarity index 100%
rename from docs/src/extending_openapi/tutorial002.py
rename to docs_src/extending_openapi/tutorial002.py
diff --git a/docs/src/extra_data_types/tutorial001.py b/docs_src/extra_data_types/tutorial001.py
similarity index 100%
rename from docs/src/extra_data_types/tutorial001.py
rename to docs_src/extra_data_types/tutorial001.py
diff --git a/docs/src/extra_models/tutorial001.py b/docs_src/extra_models/tutorial001.py
similarity index 100%
rename from docs/src/extra_models/tutorial001.py
rename to docs_src/extra_models/tutorial001.py
diff --git a/docs/src/extra_models/tutorial002.py b/docs_src/extra_models/tutorial002.py
similarity index 100%
rename from docs/src/extra_models/tutorial002.py
rename to docs_src/extra_models/tutorial002.py
diff --git a/docs/src/extra_models/tutorial003.py b/docs_src/extra_models/tutorial003.py
similarity index 100%
rename from docs/src/extra_models/tutorial003.py
rename to docs_src/extra_models/tutorial003.py
diff --git a/docs/src/extra_models/tutorial004.py b/docs_src/extra_models/tutorial004.py
similarity index 100%
rename from docs/src/extra_models/tutorial004.py
rename to docs_src/extra_models/tutorial004.py
diff --git a/docs/src/extra_models/tutorial005.py b/docs_src/extra_models/tutorial005.py
similarity index 100%
rename from docs/src/extra_models/tutorial005.py
rename to docs_src/extra_models/tutorial005.py
diff --git a/docs/src/first_steps/tutorial001.py b/docs_src/first_steps/tutorial001.py
similarity index 100%
rename from docs/src/first_steps/tutorial001.py
rename to docs_src/first_steps/tutorial001.py
diff --git a/docs/src/first_steps/tutorial002.py b/docs_src/first_steps/tutorial002.py
similarity index 100%
rename from docs/src/first_steps/tutorial002.py
rename to docs_src/first_steps/tutorial002.py
diff --git a/docs/src/first_steps/tutorial003.py b/docs_src/first_steps/tutorial003.py
similarity index 100%
rename from docs/src/first_steps/tutorial003.py
rename to docs_src/first_steps/tutorial003.py
diff --git a/docs/src/graphql/tutorial001.py b/docs_src/graphql/tutorial001.py
similarity index 100%
rename from docs/src/graphql/tutorial001.py
rename to docs_src/graphql/tutorial001.py
diff --git a/docs/src/handling_errors/tutorial001.py b/docs_src/handling_errors/tutorial001.py
similarity index 100%
rename from docs/src/handling_errors/tutorial001.py
rename to docs_src/handling_errors/tutorial001.py
diff --git a/docs/src/handling_errors/tutorial002.py b/docs_src/handling_errors/tutorial002.py
similarity index 100%
rename from docs/src/handling_errors/tutorial002.py
rename to docs_src/handling_errors/tutorial002.py
diff --git a/docs/src/handling_errors/tutorial003.py b/docs_src/handling_errors/tutorial003.py
similarity index 100%
rename from docs/src/handling_errors/tutorial003.py
rename to docs_src/handling_errors/tutorial003.py
diff --git a/docs/src/handling_errors/tutorial004.py b/docs_src/handling_errors/tutorial004.py
similarity index 100%
rename from docs/src/handling_errors/tutorial004.py
rename to docs_src/handling_errors/tutorial004.py
diff --git a/docs/src/handling_errors/tutorial005.py b/docs_src/handling_errors/tutorial005.py
similarity index 100%
rename from docs/src/handling_errors/tutorial005.py
rename to docs_src/handling_errors/tutorial005.py
diff --git a/docs/src/handling_errors/tutorial006.py b/docs_src/handling_errors/tutorial006.py
similarity index 100%
rename from docs/src/handling_errors/tutorial006.py
rename to docs_src/handling_errors/tutorial006.py
diff --git a/docs/src/header_params/tutorial001.py b/docs_src/header_params/tutorial001.py
similarity index 100%
rename from docs/src/header_params/tutorial001.py
rename to docs_src/header_params/tutorial001.py
diff --git a/docs/src/header_params/tutorial002.py b/docs_src/header_params/tutorial002.py
similarity index 100%
rename from docs/src/header_params/tutorial002.py
rename to docs_src/header_params/tutorial002.py
diff --git a/docs/src/header_params/tutorial003.py b/docs_src/header_params/tutorial003.py
similarity index 100%
rename from docs/src/header_params/tutorial003.py
rename to docs_src/header_params/tutorial003.py
diff --git a/docs/src/middleware/tutorial001.py b/docs_src/middleware/tutorial001.py
similarity index 100%
rename from docs/src/middleware/tutorial001.py
rename to docs_src/middleware/tutorial001.py
diff --git a/docs/src/nosql_databases/tutorial001.py b/docs_src/nosql_databases/tutorial001.py
similarity index 100%
rename from docs/src/nosql_databases/tutorial001.py
rename to docs_src/nosql_databases/tutorial001.py
diff --git a/docs/src/openapi_callbacks/tutorial001.py b/docs_src/openapi_callbacks/tutorial001.py
similarity index 100%
rename from docs/src/openapi_callbacks/tutorial001.py
rename to docs_src/openapi_callbacks/tutorial001.py
diff --git a/docs/src/path_operation_advanced_configuration/tutorial001.py b/docs_src/path_operation_advanced_configuration/tutorial001.py
similarity index 100%
rename from docs/src/path_operation_advanced_configuration/tutorial001.py
rename to docs_src/path_operation_advanced_configuration/tutorial001.py
diff --git a/docs/src/path_operation_advanced_configuration/tutorial002.py b/docs_src/path_operation_advanced_configuration/tutorial002.py
similarity index 100%
rename from docs/src/path_operation_advanced_configuration/tutorial002.py
rename to docs_src/path_operation_advanced_configuration/tutorial002.py
diff --git a/docs/src/path_operation_advanced_configuration/tutorial003.py b/docs_src/path_operation_advanced_configuration/tutorial003.py
similarity index 100%
rename from docs/src/path_operation_advanced_configuration/tutorial003.py
rename to docs_src/path_operation_advanced_configuration/tutorial003.py
diff --git a/docs/src/path_operation_advanced_configuration/tutorial004.py b/docs_src/path_operation_advanced_configuration/tutorial004.py
similarity index 100%
rename from docs/src/path_operation_advanced_configuration/tutorial004.py
rename to docs_src/path_operation_advanced_configuration/tutorial004.py
diff --git a/docs/src/path_operation_configuration/tutorial001.py b/docs_src/path_operation_configuration/tutorial001.py
similarity index 100%
rename from docs/src/path_operation_configuration/tutorial001.py
rename to docs_src/path_operation_configuration/tutorial001.py
diff --git a/docs/src/path_operation_configuration/tutorial002.py b/docs_src/path_operation_configuration/tutorial002.py
similarity index 100%
rename from docs/src/path_operation_configuration/tutorial002.py
rename to docs_src/path_operation_configuration/tutorial002.py
diff --git a/docs/src/path_operation_configuration/tutorial003.py b/docs_src/path_operation_configuration/tutorial003.py
similarity index 100%
rename from docs/src/path_operation_configuration/tutorial003.py
rename to docs_src/path_operation_configuration/tutorial003.py
diff --git a/docs/src/path_operation_configuration/tutorial004.py b/docs_src/path_operation_configuration/tutorial004.py
similarity index 100%
rename from docs/src/path_operation_configuration/tutorial004.py
rename to docs_src/path_operation_configuration/tutorial004.py
diff --git a/docs/src/path_operation_configuration/tutorial005.py b/docs_src/path_operation_configuration/tutorial005.py
similarity index 100%
rename from docs/src/path_operation_configuration/tutorial005.py
rename to docs_src/path_operation_configuration/tutorial005.py
diff --git a/docs/src/path_operation_configuration/tutorial006.py b/docs_src/path_operation_configuration/tutorial006.py
similarity index 100%
rename from docs/src/path_operation_configuration/tutorial006.py
rename to docs_src/path_operation_configuration/tutorial006.py
diff --git a/docs/src/path_params/tutorial001.py b/docs_src/path_params/tutorial001.py
similarity index 100%
rename from docs/src/path_params/tutorial001.py
rename to docs_src/path_params/tutorial001.py
diff --git a/docs/src/path_params/tutorial002.py b/docs_src/path_params/tutorial002.py
similarity index 100%
rename from docs/src/path_params/tutorial002.py
rename to docs_src/path_params/tutorial002.py
diff --git a/docs/src/path_params/tutorial003.py b/docs_src/path_params/tutorial003.py
similarity index 100%
rename from docs/src/path_params/tutorial003.py
rename to docs_src/path_params/tutorial003.py
diff --git a/docs/src/path_params/tutorial004.py b/docs_src/path_params/tutorial004.py
similarity index 100%
rename from docs/src/path_params/tutorial004.py
rename to docs_src/path_params/tutorial004.py
diff --git a/docs/src/path_params/tutorial005.py b/docs_src/path_params/tutorial005.py
similarity index 100%
rename from docs/src/path_params/tutorial005.py
rename to docs_src/path_params/tutorial005.py
diff --git a/docs/src/path_params_numeric_validations/tutorial001.py b/docs_src/path_params_numeric_validations/tutorial001.py
similarity index 100%
rename from docs/src/path_params_numeric_validations/tutorial001.py
rename to docs_src/path_params_numeric_validations/tutorial001.py
diff --git a/docs/src/path_params_numeric_validations/tutorial002.py b/docs_src/path_params_numeric_validations/tutorial002.py
similarity index 100%
rename from docs/src/path_params_numeric_validations/tutorial002.py
rename to docs_src/path_params_numeric_validations/tutorial002.py
diff --git a/docs/src/path_params_numeric_validations/tutorial003.py b/docs_src/path_params_numeric_validations/tutorial003.py
similarity index 100%
rename from docs/src/path_params_numeric_validations/tutorial003.py
rename to docs_src/path_params_numeric_validations/tutorial003.py
diff --git a/docs/src/path_params_numeric_validations/tutorial004.py b/docs_src/path_params_numeric_validations/tutorial004.py
similarity index 100%
rename from docs/src/path_params_numeric_validations/tutorial004.py
rename to docs_src/path_params_numeric_validations/tutorial004.py
diff --git a/docs/src/path_params_numeric_validations/tutorial005.py b/docs_src/path_params_numeric_validations/tutorial005.py
similarity index 100%
rename from docs/src/path_params_numeric_validations/tutorial005.py
rename to docs_src/path_params_numeric_validations/tutorial005.py
diff --git a/docs/src/path_params_numeric_validations/tutorial006.py b/docs_src/path_params_numeric_validations/tutorial006.py
similarity index 100%
rename from docs/src/path_params_numeric_validations/tutorial006.py
rename to docs_src/path_params_numeric_validations/tutorial006.py
diff --git a/docs/src/python_types/tutorial001.py b/docs_src/python_types/tutorial001.py
similarity index 100%
rename from docs/src/python_types/tutorial001.py
rename to docs_src/python_types/tutorial001.py
diff --git a/docs/src/python_types/tutorial002.py b/docs_src/python_types/tutorial002.py
similarity index 100%
rename from docs/src/python_types/tutorial002.py
rename to docs_src/python_types/tutorial002.py
diff --git a/docs/src/python_types/tutorial003.py b/docs_src/python_types/tutorial003.py
similarity index 100%
rename from docs/src/python_types/tutorial003.py
rename to docs_src/python_types/tutorial003.py
diff --git a/docs/src/python_types/tutorial004.py b/docs_src/python_types/tutorial004.py
similarity index 100%
rename from docs/src/python_types/tutorial004.py
rename to docs_src/python_types/tutorial004.py
diff --git a/docs/src/python_types/tutorial005.py b/docs_src/python_types/tutorial005.py
similarity index 100%
rename from docs/src/python_types/tutorial005.py
rename to docs_src/python_types/tutorial005.py
diff --git a/docs/src/python_types/tutorial006.py b/docs_src/python_types/tutorial006.py
similarity index 100%
rename from docs/src/python_types/tutorial006.py
rename to docs_src/python_types/tutorial006.py
diff --git a/docs/src/python_types/tutorial007.py b/docs_src/python_types/tutorial007.py
similarity index 100%
rename from docs/src/python_types/tutorial007.py
rename to docs_src/python_types/tutorial007.py
diff --git a/docs/src/python_types/tutorial008.py b/docs_src/python_types/tutorial008.py
similarity index 100%
rename from docs/src/python_types/tutorial008.py
rename to docs_src/python_types/tutorial008.py
diff --git a/docs/src/python_types/tutorial009.py b/docs_src/python_types/tutorial009.py
similarity index 100%
rename from docs/src/python_types/tutorial009.py
rename to docs_src/python_types/tutorial009.py
diff --git a/docs/src/python_types/tutorial010.py b/docs_src/python_types/tutorial010.py
similarity index 100%
rename from docs/src/python_types/tutorial010.py
rename to docs_src/python_types/tutorial010.py
diff --git a/docs/src/query_params/tutorial001.py b/docs_src/query_params/tutorial001.py
similarity index 100%
rename from docs/src/query_params/tutorial001.py
rename to docs_src/query_params/tutorial001.py
diff --git a/docs/src/query_params/tutorial002.py b/docs_src/query_params/tutorial002.py
similarity index 100%
rename from docs/src/query_params/tutorial002.py
rename to docs_src/query_params/tutorial002.py
diff --git a/docs/src/query_params/tutorial003.py b/docs_src/query_params/tutorial003.py
similarity index 100%
rename from docs/src/query_params/tutorial003.py
rename to docs_src/query_params/tutorial003.py
diff --git a/docs/src/query_params/tutorial004.py b/docs_src/query_params/tutorial004.py
similarity index 100%
rename from docs/src/query_params/tutorial004.py
rename to docs_src/query_params/tutorial004.py
diff --git a/docs/src/query_params/tutorial005.py b/docs_src/query_params/tutorial005.py
similarity index 100%
rename from docs/src/query_params/tutorial005.py
rename to docs_src/query_params/tutorial005.py
diff --git a/docs/src/query_params/tutorial006.py b/docs_src/query_params/tutorial006.py
similarity index 100%
rename from docs/src/query_params/tutorial006.py
rename to docs_src/query_params/tutorial006.py
diff --git a/docs/src/query_params/tutorial007.py b/docs_src/query_params/tutorial007.py
similarity index 100%
rename from docs/src/query_params/tutorial007.py
rename to docs_src/query_params/tutorial007.py
diff --git a/docs/src/query_params_str_validations/tutorial001.py b/docs_src/query_params_str_validations/tutorial001.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial001.py
rename to docs_src/query_params_str_validations/tutorial001.py
diff --git a/docs/src/query_params_str_validations/tutorial002.py b/docs_src/query_params_str_validations/tutorial002.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial002.py
rename to docs_src/query_params_str_validations/tutorial002.py
diff --git a/docs/src/query_params_str_validations/tutorial003.py b/docs_src/query_params_str_validations/tutorial003.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial003.py
rename to docs_src/query_params_str_validations/tutorial003.py
diff --git a/docs/src/query_params_str_validations/tutorial004.py b/docs_src/query_params_str_validations/tutorial004.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial004.py
rename to docs_src/query_params_str_validations/tutorial004.py
diff --git a/docs/src/query_params_str_validations/tutorial005.py b/docs_src/query_params_str_validations/tutorial005.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial005.py
rename to docs_src/query_params_str_validations/tutorial005.py
diff --git a/docs/src/query_params_str_validations/tutorial006.py b/docs_src/query_params_str_validations/tutorial006.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial006.py
rename to docs_src/query_params_str_validations/tutorial006.py
diff --git a/docs/src/query_params_str_validations/tutorial007.py b/docs_src/query_params_str_validations/tutorial007.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial007.py
rename to docs_src/query_params_str_validations/tutorial007.py
diff --git a/docs/src/query_params_str_validations/tutorial008.py b/docs_src/query_params_str_validations/tutorial008.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial008.py
rename to docs_src/query_params_str_validations/tutorial008.py
diff --git a/docs/src/query_params_str_validations/tutorial009.py b/docs_src/query_params_str_validations/tutorial009.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial009.py
rename to docs_src/query_params_str_validations/tutorial009.py
diff --git a/docs/src/query_params_str_validations/tutorial010.py b/docs_src/query_params_str_validations/tutorial010.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial010.py
rename to docs_src/query_params_str_validations/tutorial010.py
diff --git a/docs/src/query_params_str_validations/tutorial011.py b/docs_src/query_params_str_validations/tutorial011.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial011.py
rename to docs_src/query_params_str_validations/tutorial011.py
diff --git a/docs/src/query_params_str_validations/tutorial012.py b/docs_src/query_params_str_validations/tutorial012.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial012.py
rename to docs_src/query_params_str_validations/tutorial012.py
diff --git a/docs/src/query_params_str_validations/tutorial013.py b/docs_src/query_params_str_validations/tutorial013.py
similarity index 100%
rename from docs/src/query_params_str_validations/tutorial013.py
rename to docs_src/query_params_str_validations/tutorial013.py
diff --git a/docs/src/request_files/tutorial001.py b/docs_src/request_files/tutorial001.py
similarity index 100%
rename from docs/src/request_files/tutorial001.py
rename to docs_src/request_files/tutorial001.py
diff --git a/docs/src/request_files/tutorial002.py b/docs_src/request_files/tutorial002.py
similarity index 100%
rename from docs/src/request_files/tutorial002.py
rename to docs_src/request_files/tutorial002.py
diff --git a/docs/src/request_forms/tutorial001.py b/docs_src/request_forms/tutorial001.py
similarity index 100%
rename from docs/src/request_forms/tutorial001.py
rename to docs_src/request_forms/tutorial001.py
diff --git a/docs/src/request_forms_and_files/tutorial001.py b/docs_src/request_forms_and_files/tutorial001.py
similarity index 100%
rename from docs/src/request_forms_and_files/tutorial001.py
rename to docs_src/request_forms_and_files/tutorial001.py
diff --git a/docs/src/response_change_status_code/tutorial001.py b/docs_src/response_change_status_code/tutorial001.py
similarity index 100%
rename from docs/src/response_change_status_code/tutorial001.py
rename to docs_src/response_change_status_code/tutorial001.py
diff --git a/docs/src/response_cookies/tutorial001.py b/docs_src/response_cookies/tutorial001.py
similarity index 100%
rename from docs/src/response_cookies/tutorial001.py
rename to docs_src/response_cookies/tutorial001.py
diff --git a/docs/src/response_cookies/tutorial002.py b/docs_src/response_cookies/tutorial002.py
similarity index 100%
rename from docs/src/response_cookies/tutorial002.py
rename to docs_src/response_cookies/tutorial002.py
diff --git a/docs/src/response_directly/tutorial001.py b/docs_src/response_directly/tutorial001.py
similarity index 100%
rename from docs/src/response_directly/tutorial001.py
rename to docs_src/response_directly/tutorial001.py
diff --git a/docs/src/response_directly/tutorial002.py b/docs_src/response_directly/tutorial002.py
similarity index 100%
rename from docs/src/response_directly/tutorial002.py
rename to docs_src/response_directly/tutorial002.py
diff --git a/docs/src/response_headers/tutorial001.py b/docs_src/response_headers/tutorial001.py
similarity index 100%
rename from docs/src/response_headers/tutorial001.py
rename to docs_src/response_headers/tutorial001.py
diff --git a/docs/src/response_headers/tutorial002.py b/docs_src/response_headers/tutorial002.py
similarity index 100%
rename from docs/src/response_headers/tutorial002.py
rename to docs_src/response_headers/tutorial002.py
diff --git a/docs/src/response_model/tutorial001.py b/docs_src/response_model/tutorial001.py
similarity index 100%
rename from docs/src/response_model/tutorial001.py
rename to docs_src/response_model/tutorial001.py
diff --git a/docs/src/response_model/tutorial002.py b/docs_src/response_model/tutorial002.py
similarity index 100%
rename from docs/src/response_model/tutorial002.py
rename to docs_src/response_model/tutorial002.py
diff --git a/docs/src/response_model/tutorial003.py b/docs_src/response_model/tutorial003.py
similarity index 100%
rename from docs/src/response_model/tutorial003.py
rename to docs_src/response_model/tutorial003.py
diff --git a/docs/src/response_model/tutorial004.py b/docs_src/response_model/tutorial004.py
similarity index 100%
rename from docs/src/response_model/tutorial004.py
rename to docs_src/response_model/tutorial004.py
diff --git a/docs/src/response_model/tutorial005.py b/docs_src/response_model/tutorial005.py
similarity index 100%
rename from docs/src/response_model/tutorial005.py
rename to docs_src/response_model/tutorial005.py
diff --git a/docs/src/response_model/tutorial006.py b/docs_src/response_model/tutorial006.py
similarity index 100%
rename from docs/src/response_model/tutorial006.py
rename to docs_src/response_model/tutorial006.py
diff --git a/docs/src/response_status_code/tutorial001.py b/docs_src/response_status_code/tutorial001.py
similarity index 100%
rename from docs/src/response_status_code/tutorial001.py
rename to docs_src/response_status_code/tutorial001.py
diff --git a/docs/src/response_status_code/tutorial002.py b/docs_src/response_status_code/tutorial002.py
similarity index 100%
rename from docs/src/response_status_code/tutorial002.py
rename to docs_src/response_status_code/tutorial002.py
diff --git a/docs/src/security/tutorial001.py b/docs_src/security/tutorial001.py
similarity index 100%
rename from docs/src/security/tutorial001.py
rename to docs_src/security/tutorial001.py
diff --git a/docs/src/security/tutorial002.py b/docs_src/security/tutorial002.py
similarity index 100%
rename from docs/src/security/tutorial002.py
rename to docs_src/security/tutorial002.py
diff --git a/docs/src/security/tutorial003.py b/docs_src/security/tutorial003.py
similarity index 100%
rename from docs/src/security/tutorial003.py
rename to docs_src/security/tutorial003.py
diff --git a/docs/src/security/tutorial004.py b/docs_src/security/tutorial004.py
similarity index 100%
rename from docs/src/security/tutorial004.py
rename to docs_src/security/tutorial004.py
diff --git a/docs/src/security/tutorial005.py b/docs_src/security/tutorial005.py
similarity index 100%
rename from docs/src/security/tutorial005.py
rename to docs_src/security/tutorial005.py
diff --git a/docs/src/security/tutorial006.py b/docs_src/security/tutorial006.py
similarity index 100%
rename from docs/src/security/tutorial006.py
rename to docs_src/security/tutorial006.py
diff --git a/docs/src/security/tutorial007.py b/docs_src/security/tutorial007.py
similarity index 100%
rename from docs/src/security/tutorial007.py
rename to docs_src/security/tutorial007.py
diff --git a/docs/src/sql_databases/__init__.py b/docs_src/sql_databases/__init__.py
similarity index 100%
rename from docs/src/sql_databases/__init__.py
rename to docs_src/sql_databases/__init__.py
diff --git a/docs/src/sql_databases/sql_app/__init__.py b/docs_src/sql_databases/sql_app/__init__.py
similarity index 100%
rename from docs/src/sql_databases/sql_app/__init__.py
rename to docs_src/sql_databases/sql_app/__init__.py
diff --git a/docs/src/sql_databases/sql_app/alt_main.py b/docs_src/sql_databases/sql_app/alt_main.py
similarity index 100%
rename from docs/src/sql_databases/sql_app/alt_main.py
rename to docs_src/sql_databases/sql_app/alt_main.py
diff --git a/docs/src/sql_databases/sql_app/crud.py b/docs_src/sql_databases/sql_app/crud.py
similarity index 100%
rename from docs/src/sql_databases/sql_app/crud.py
rename to docs_src/sql_databases/sql_app/crud.py
diff --git a/docs/src/sql_databases/sql_app/database.py b/docs_src/sql_databases/sql_app/database.py
similarity index 100%
rename from docs/src/sql_databases/sql_app/database.py
rename to docs_src/sql_databases/sql_app/database.py
diff --git a/docs/src/sql_databases/sql_app/main.py b/docs_src/sql_databases/sql_app/main.py
similarity index 100%
rename from docs/src/sql_databases/sql_app/main.py
rename to docs_src/sql_databases/sql_app/main.py
diff --git a/docs/src/sql_databases/sql_app/models.py b/docs_src/sql_databases/sql_app/models.py
similarity index 100%
rename from docs/src/sql_databases/sql_app/models.py
rename to docs_src/sql_databases/sql_app/models.py
diff --git a/docs/src/sql_databases/sql_app/schemas.py b/docs_src/sql_databases/sql_app/schemas.py
similarity index 100%
rename from docs/src/sql_databases/sql_app/schemas.py
rename to docs_src/sql_databases/sql_app/schemas.py
diff --git a/docs/src/sql_databases_peewee/__init__.py b/docs_src/sql_databases_peewee/__init__.py
similarity index 100%
rename from docs/src/sql_databases_peewee/__init__.py
rename to docs_src/sql_databases_peewee/__init__.py
diff --git a/docs/src/sql_databases_peewee/sql_app/__init__.py b/docs_src/sql_databases_peewee/sql_app/__init__.py
similarity index 100%
rename from docs/src/sql_databases_peewee/sql_app/__init__.py
rename to docs_src/sql_databases_peewee/sql_app/__init__.py
diff --git a/docs/src/sql_databases_peewee/sql_app/crud.py b/docs_src/sql_databases_peewee/sql_app/crud.py
similarity index 100%
rename from docs/src/sql_databases_peewee/sql_app/crud.py
rename to docs_src/sql_databases_peewee/sql_app/crud.py
diff --git a/docs/src/sql_databases_peewee/sql_app/database.py b/docs_src/sql_databases_peewee/sql_app/database.py
similarity index 100%
rename from docs/src/sql_databases_peewee/sql_app/database.py
rename to docs_src/sql_databases_peewee/sql_app/database.py
diff --git a/docs/src/sql_databases_peewee/sql_app/main.py b/docs_src/sql_databases_peewee/sql_app/main.py
similarity index 100%
rename from docs/src/sql_databases_peewee/sql_app/main.py
rename to docs_src/sql_databases_peewee/sql_app/main.py
diff --git a/docs/src/sql_databases_peewee/sql_app/models.py b/docs_src/sql_databases_peewee/sql_app/models.py
similarity index 100%
rename from docs/src/sql_databases_peewee/sql_app/models.py
rename to docs_src/sql_databases_peewee/sql_app/models.py
diff --git a/docs/src/sql_databases_peewee/sql_app/schemas.py b/docs_src/sql_databases_peewee/sql_app/schemas.py
similarity index 100%
rename from docs/src/sql_databases_peewee/sql_app/schemas.py
rename to docs_src/sql_databases_peewee/sql_app/schemas.py
diff --git a/docs/src/static_files/tutorial001.py b/docs_src/static_files/tutorial001.py
similarity index 100%
rename from docs/src/static_files/tutorial001.py
rename to docs_src/static_files/tutorial001.py
diff --git a/docs/src/sub_applications/tutorial001.py b/docs_src/sub_applications/tutorial001.py
similarity index 100%
rename from docs/src/sub_applications/tutorial001.py
rename to docs_src/sub_applications/tutorial001.py
diff --git a/docs/src/templates/static/styles.css b/docs_src/templates/static/styles.css
similarity index 100%
rename from docs/src/templates/static/styles.css
rename to docs_src/templates/static/styles.css
diff --git a/docs/src/templates/templates/item.html b/docs_src/templates/templates/item.html
similarity index 100%
rename from docs/src/templates/templates/item.html
rename to docs_src/templates/templates/item.html
diff --git a/docs/src/templates/tutorial001.py b/docs_src/templates/tutorial001.py
similarity index 100%
rename from docs/src/templates/tutorial001.py
rename to docs_src/templates/tutorial001.py
diff --git a/docs/src/using_request_directly/tutorial001.py b/docs_src/using_request_directly/tutorial001.py
similarity index 100%
rename from docs/src/using_request_directly/tutorial001.py
rename to docs_src/using_request_directly/tutorial001.py
diff --git a/docs/src/websockets/__init__.py b/docs_src/websockets/__init__.py
similarity index 100%
rename from docs/src/websockets/__init__.py
rename to docs_src/websockets/__init__.py
diff --git a/docs/src/websockets/tutorial001.py b/docs_src/websockets/tutorial001.py
similarity index 100%
rename from docs/src/websockets/tutorial001.py
rename to docs_src/websockets/tutorial001.py
diff --git a/docs/src/websockets/tutorial002.py b/docs_src/websockets/tutorial002.py
similarity index 100%
rename from docs/src/websockets/tutorial002.py
rename to docs_src/websockets/tutorial002.py
diff --git a/docs/src/wsgi/tutorial001.py b/docs_src/wsgi/tutorial001.py
similarity index 100%
rename from docs/src/wsgi/tutorial001.py
rename to docs_src/wsgi/tutorial001.py
diff --git a/mkdocs.yml b/mkdocs.yml
deleted file mode 100644
index bacae9bac..000000000
--- a/mkdocs.yml
+++ /dev/null
@@ -1,150 +0,0 @@
-site_name: FastAPI
-site_description: FastAPI framework, high performance, easy to learn, fast to code, ready for production
-site_url: https://fastapi.tiangolo.com/
-
-theme:
- name: 'material'
- palette:
- primary: 'teal'
- accent: 'amber'
- logo: 'img/icon-white.svg'
- favicon: 'img/favicon.png'
-
-repo_name: tiangolo/fastapi
-repo_url: https://github.com/tiangolo/fastapi
-edit_uri: ''
-google_analytics:
- - 'UA-133183413-1'
- - 'auto'
-
-nav:
- - FastAPI: 'index.md'
- - Features: 'features.md'
- - Python types intro: 'python-types.md'
- - Tutorial - User Guide:
- - Tutorial - User Guide - Intro: 'tutorial/index.md'
- - First Steps: 'tutorial/first-steps.md'
- - Path Parameters: 'tutorial/path-params.md'
- - Query Parameters: 'tutorial/query-params.md'
- - Request Body: 'tutorial/body.md'
- - Query Parameters and String Validations: 'tutorial/query-params-str-validations.md'
- - Path Parameters and Numeric Validations: 'tutorial/path-params-numeric-validations.md'
- - Body - Multiple Parameters: 'tutorial/body-multiple-params.md'
- - Body - Fields: 'tutorial/body-fields.md'
- - Body - Nested Models: 'tutorial/body-nested-models.md'
- - Extra data types: 'tutorial/extra-data-types.md'
- - Cookie Parameters: 'tutorial/cookie-params.md'
- - Header Parameters: 'tutorial/header-params.md'
- - Response Model: 'tutorial/response-model.md'
- - Extra Models: 'tutorial/extra-models.md'
- - Response Status Code: 'tutorial/response-status-code.md'
- - Form Data: 'tutorial/request-forms.md'
- - Request Files: 'tutorial/request-files.md'
- - Request Forms and Files: 'tutorial/request-forms-and-files.md'
- - Handling Errors: 'tutorial/handling-errors.md'
- - Path Operation Configuration: 'tutorial/path-operation-configuration.md'
- - JSON Compatible Encoder: 'tutorial/encoder.md'
- - Body - updates: 'tutorial/body-updates.md'
- - Dependencies:
- - First Steps: 'tutorial/dependencies/index.md'
- - Classes as Dependencies: 'tutorial/dependencies/classes-as-dependencies.md'
- - Sub-dependencies: 'tutorial/dependencies/sub-dependencies.md'
- - Dependencies in path operation decorators: 'tutorial/dependencies/dependencies-in-path-operation-decorators.md'
- - Dependencies with yield: 'tutorial/dependencies/dependencies-with-yield.md'
- - Security:
- - Security Intro: 'tutorial/security/index.md'
- - First Steps: 'tutorial/security/first-steps.md'
- - Get Current User: 'tutorial/security/get-current-user.md'
- - Simple OAuth2 with Password and Bearer: 'tutorial/security/simple-oauth2.md'
- - OAuth2 with Password (and hashing), Bearer with JWT tokens: 'tutorial/security/oauth2-jwt.md'
- - Middleware: 'tutorial/middleware.md'
- - CORS (Cross-Origin Resource Sharing): 'tutorial/cors.md'
- - SQL (Relational) Databases: 'tutorial/sql-databases.md'
- - Bigger Applications - Multiple Files: 'tutorial/bigger-applications.md'
- - Background Tasks: 'tutorial/background-tasks.md'
- - Application Configuration: 'tutorial/application-configuration.md'
- - Static Files: 'tutorial/static-files.md'
- - Testing: 'tutorial/testing.md'
- - Debugging: 'tutorial/debugging.md'
- - Advanced User Guide:
- - Advanced User Guide - Intro: 'advanced/index.md'
- - Path Operation Advanced Configuration: 'advanced/path-operation-advanced-configuration.md'
- - Additional Status Codes: 'advanced/additional-status-codes.md'
- - Return a Response Directly: 'advanced/response-directly.md'
- - Custom Response - HTML, Stream, File, others: 'advanced/custom-response.md'
- - Additional Responses in OpenAPI: 'advanced/additional-responses.md'
- - Response Cookies: 'advanced/response-cookies.md'
- - Response Headers: 'advanced/response-headers.md'
- - Response - Change Status Code: 'advanced/response-change-status-code.md'
- - Advanced Dependencies: 'advanced/advanced-dependencies.md'
- - Advanced Security:
- - Advanced Security - Intro: 'advanced/security/index.md'
- - OAuth2 scopes: 'advanced/security/oauth2-scopes.md'
- - HTTP Basic Auth: 'advanced/security/http-basic-auth.md'
- - Using the Request Directly: 'advanced/using-request-directly.md'
- - Advanced Middleware: 'advanced/middleware.md'
- - SQL (Relational) Databases with Peewee: 'advanced/sql-databases-peewee.md'
- - Async SQL (Relational) Databases: 'advanced/async-sql-databases.md'
- - NoSQL (Distributed / Big Data) Databases: 'advanced/nosql-databases.md'
- - Sub Applications - Behind a Proxy, Mounts: 'advanced/sub-applications-proxy.md'
- - Templates: 'advanced/templates.md'
- - GraphQL: 'advanced/graphql.md'
- - WebSockets: 'advanced/websockets.md'
- - 'Events: startup - shutdown': 'advanced/events.md'
- - Custom Request and APIRoute class: 'advanced/custom-request-and-route.md'
- - Testing WebSockets: 'advanced/testing-websockets.md'
- - 'Testing Events: startup - shutdown': 'advanced/testing-events.md'
- - Testing Dependencies with Overrides: 'advanced/testing-dependencies.md'
- - Extending OpenAPI: 'advanced/extending-openapi.md'
- - OpenAPI Callbacks: 'advanced/openapi-callbacks.md'
- - Including WSGI - Flask, Django, others: 'advanced/wsgi.md'
- - Concurrency and async / await: 'async.md'
- - Deployment: 'deployment.md'
- - Project Generation - Template: 'project-generation.md'
- - Alternatives, Inspiration and Comparisons: 'alternatives.md'
- - History, Design and Future: 'history-design-future.md'
- - External Links and Articles: 'external-links.md'
- - Benchmarks: 'benchmarks.md'
- - Help FastAPI - Get Help: 'help-fastapi.md'
- - Development - Contributing: 'contributing.md'
- - Release Notes: 'release-notes.md'
-
-markdown_extensions:
- - toc:
- permalink: true
- - markdown.extensions.codehilite:
- guess_lang: false
- - markdown_include.include:
- base_path: docs
- - admonition
- - codehilite
- - extra
- - pymdownx.superfences:
- custom_fences:
- - name: mermaid
- class: mermaid
- format: !!python/name:pymdownx.superfences.fence_div_format
-
-extra:
- social:
- - type: 'github'
- link: 'https://github.com/tiangolo/typer'
- - type: 'twitter'
- link: 'https://twitter.com/tiangolo'
- - type: 'linkedin'
- link: 'https://www.linkedin.com/in/tiangolo'
- - type: 'rss'
- link: 'https://dev.to/tiangolo'
- - type: 'medium'
- link: 'https://medium.com/@tiangolo'
- - type: 'globe'
- link: 'https://tiangolo.com'
-
-extra_css:
- - 'css/termynal.css'
- - 'css/custom.css'
-
-extra_javascript:
- - 'https://unpkg.com/mermaid@8.4.6/dist/mermaid.min.js'
- - 'js/termynal.js'
- - 'js/custom.js'
diff --git a/pyproject.toml b/pyproject.toml
index 58e53c3ee..eadff42bb 100644
--- a/pyproject.toml
+++ b/pyproject.toml
@@ -64,7 +64,10 @@ test = [
doc = [
"mkdocs",
"mkdocs-material",
- "markdown-include"
+ "markdown-include",
+ "typer",
+ "typer-cli",
+ "pyyaml"
]
dev = [
"pyjwt",
diff --git a/scripts/build-docs.sh b/scripts/build-docs.sh
index 4f4ae2f74..383ad3f44 100755
--- a/scripts/build-docs.sh
+++ b/scripts/build-docs.sh
@@ -1,5 +1,6 @@
#!/usr/bin/env bash
-python -m mkdocs build
+set -e
+set -x
-cp ./docs/index.md ./README.md
+python ./scripts/docs.py build-all
diff --git a/scripts/docs.py b/scripts/docs.py
new file mode 100644
index 000000000..490f05264
--- /dev/null
+++ b/scripts/docs.py
@@ -0,0 +1,340 @@
+import os
+import shutil
+from http.server import HTTPServer, SimpleHTTPRequestHandler
+from pathlib import Path
+from typing import Dict, Optional, Tuple
+
+import mkdocs.commands.build
+import mkdocs.commands.serve
+import mkdocs.config
+import mkdocs.utils
+import typer
+import yaml
+
+app = typer.Typer()
+
+mkdocs_name = "mkdocs.yml"
+
+missing_translation_snippet = """
+{!../../../docs/missing-translation.md!}
+"""
+
+docs_path = Path("docs")
+
+
+def lang_callback(lang: Optional[str]):
+ if lang is None:
+ return
+ if not lang.isalpha() or len(lang) != 2:
+ typer.echo("Use a 2 letter language code, like: es")
+ raise typer.Abort()
+ lang = lang.lower()
+ return lang
+
+
+def complete_existing_lang(incomplete: str):
+ lang_path: Path
+ for lang_path in docs_path.iterdir():
+ if lang_path.is_dir() and lang_path.name.startswith(incomplete):
+ yield lang_path.name
+
+
+@app.command()
+def new_lang(lang: str = typer.Argument(..., callback=lang_callback)):
+ """
+ Generate a new docs translation directory for the language LANG.
+
+ LANG should be a 2-letter language code, like: en, es, de, pt, etc.
+ """
+ new_path: Path = Path("docs") / lang
+ if new_path.exists():
+ typer.echo(f"The language was already created: {lang}")
+ raise typer.Abort()
+ new_path.mkdir()
+ en_docs_path = Path("docs/en")
+ en_config_path: Path = en_docs_path / mkdocs_name
+ en_config: dict = mkdocs.utils.yaml_load(en_config_path.read_text())
+ fastapi_url_base = "https://fastapi.tiangolo.com/"
+ new_config = {}
+ new_config["site_name"] = en_config["site_name"]
+ new_config["site_description"] = en_config["site_description"]
+ new_config["site_url"] = en_config["site_url"] + f"{lang}/"
+ new_config["theme"] = en_config["theme"]
+ new_config["theme"]["logo"] = fastapi_url_base + en_config["theme"]["logo"]
+ new_config["theme"]["favicon"] = fastapi_url_base + en_config["theme"]["favicon"]
+ new_config["theme"]["language"] = lang
+ new_config["repo_name"] = en_config["repo_name"]
+ new_config["repo_url"] = en_config["repo_url"]
+ new_config["edit_uri"] = en_config["edit_uri"]
+ new_config["google_analytics"] = en_config["google_analytics"]
+ new_config["nav"] = en_config["nav"][:2]
+
+ new_config["markdown_extensions"] = en_config["markdown_extensions"]
+ new_config["extra"] = en_config["extra"]
+
+ extra_css = []
+ css: str
+ for css in en_config["extra_css"]:
+ if css.startswith("http"):
+ extra_css.append(css)
+ else:
+ extra_css.append(fastapi_url_base + css)
+ new_config["extra_css"] = extra_css
+
+ extra_js = []
+ js: str
+ for js in en_config["extra_javascript"]:
+ if js.startswith("http"):
+ extra_js.append(js)
+ else:
+ extra_js.append(fastapi_url_base + js)
+ new_config["extra_javascript"] = extra_js
+ new_config_path: Path = Path(new_path) / mkdocs_name
+ new_config_path.write_text(yaml.dump(new_config, sort_keys=False, width=200))
+ new_config_docs_path: Path = new_path / "docs"
+ new_config_docs_path.mkdir()
+ en_index_path: Path = en_docs_path / "docs" / "index.md"
+ new_index_path: Path = new_config_docs_path / "index.md"
+ en_index_content = en_index_path.read_text()
+ new_index_content = f"{missing_translation_snippet}\n\n{en_index_content}"
+ new_index_path.write_text(new_index_content)
+ typer.secho(f"Successfully initialized: {new_path}", color=typer.colors.GREEN)
+ update_languages(lang=None)
+
+
+@app.command()
+def build_lang(
+ lang: str = typer.Argument(
+ ..., callback=lang_callback, autocompletion=complete_existing_lang
+ )
+):
+ """
+ Build the docs for a language, filling missing pages with translation notifications.
+ """
+ lang_path: Path = Path("docs") / lang
+ if not lang_path.is_dir():
+ typer.echo(f"The language translation doesn't seem to exist yet: {lang}")
+ raise typer.Abort()
+ typer.echo(f"Building docs for: {lang}")
+ build_dir_path = Path("docs_build")
+ build_dir_path.mkdir(exist_ok=True)
+ build_lang_path = build_dir_path / lang
+ en_lang_path = Path("docs/en")
+ site_path = Path("site").absolute()
+ if lang == "en":
+ dist_path = site_path
+ else:
+ dist_path: Path = site_path / lang
+ shutil.rmtree(build_lang_path, ignore_errors=True)
+ shutil.copytree(lang_path, build_lang_path)
+ en_config_path: Path = en_lang_path / mkdocs_name
+ en_config: dict = mkdocs.utils.yaml_load(en_config_path.read_text())
+ nav = en_config["nav"]
+ lang_config_path: Path = lang_path / mkdocs_name
+ lang_config: dict = mkdocs.utils.yaml_load(lang_config_path.read_text())
+ lang_nav = lang_config["nav"]
+ # Exclude first 2 entries FastAPI and Languages, for custom handling
+ use_nav = nav[2:]
+ lang_use_nav = lang_nav[2:]
+ file_to_nav = get_file_to_nav_map(use_nav)
+ sections = get_sections(use_nav)
+ lang_file_to_nav = get_file_to_nav_map(lang_use_nav)
+ use_lang_file_to_nav = get_file_to_nav_map(lang_use_nav)
+ for file in file_to_nav:
+ file_path = Path(file)
+ lang_file_path: Path = build_lang_path / "docs" / file_path
+ en_file_path: Path = en_lang_path / "docs" / file_path
+ lang_file_path.parent.mkdir(parents=True, exist_ok=True)
+ if not lang_file_path.is_file():
+ en_text = en_file_path.read_text()
+ lang_text = get_text_with_translate_missing(en_text)
+ lang_file_path.write_text(lang_text)
+ file_key = file_to_nav[file]
+ use_lang_file_to_nav[file] = file_key
+ if file_key:
+ composite_key = ()
+ new_key = ()
+ for key_part in file_key:
+ composite_key += (key_part,)
+ key_first_file = sections[composite_key]
+ if key_first_file in lang_file_to_nav:
+ new_key = lang_file_to_nav[key_first_file]
+ else:
+ new_key += (key_part,)
+ use_lang_file_to_nav[file] = new_key
+ key_to_section = {(): []}
+ for file, file_key in use_lang_file_to_nav.items():
+ section = get_key_section(key_to_section=key_to_section, key=file_key)
+ section.append(file)
+ new_nav = key_to_section[()]
+ export_lang_nav = [lang_nav[0], nav[1]] + new_nav
+ lang_config["nav"] = export_lang_nav
+ build_lang_config_path: Path = build_lang_path / mkdocs_name
+ build_lang_config_path.write_text(
+ yaml.dump(lang_config, sort_keys=False, width=200)
+ )
+ current_dir = os.getcwd()
+ os.chdir(build_lang_path)
+ mkdocs.commands.build.build(mkdocs.config.load_config(site_dir=str(dist_path)))
+ os.chdir(current_dir)
+ typer.secho(f"Successfully built docs for: {lang}", color=typer.colors.GREEN)
+
+
+@app.command()
+def build_all():
+ """
+ Build mkdocs site for en, and then build each language inside, end result is located
+ at directory ./site/ with each language inside.
+ """
+ site_path = Path("site").absolute()
+ update_languages(lang=None)
+ en_build_path: Path = docs_path / "en"
+ current_dir = os.getcwd()
+ os.chdir(en_build_path)
+ typer.echo(f"Building docs for: en")
+ mkdocs.commands.build.build(mkdocs.config.load_config(site_dir=str(site_path)))
+ os.chdir(current_dir)
+ for lang in docs_path.iterdir():
+ if lang == en_build_path or not lang.is_dir():
+ continue
+ build_lang(lang.name)
+ typer.echo("Copying en index.md to README.md")
+ en_index = en_build_path / "docs" / "index.md"
+ shutil.copyfile(en_index, "README.md")
+
+
+@app.command()
+def update_languages(
+ lang: str = typer.Argument(
+ None, callback=lang_callback, autocompletion=complete_existing_lang
+ )
+):
+ """
+ Update the mkdocs.yml file Languages section including all the available languages.
+
+ The LANG argument is a 2-letter language code. If it's not provided, update all the
+ mkdocs.yml files (for all the languages).
+ """
+ if lang is None:
+ for lang_path in docs_path.iterdir():
+ if lang_path.is_dir():
+ typer.echo(f"Updating {lang_path.name}")
+ update_config(lang_path.name)
+ else:
+ typer.echo(f"Updating {lang}")
+ update_config(lang)
+
+
+@app.command()
+def serve():
+ """
+ A quick server to preview a built site with translations.
+
+ For development, prefer the command live (or just mkdocs serve).
+
+ This is here only to preview a site with translations already built.
+
+ Make sure you run the build-all command first.
+ """
+ typer.echo(
+ "Warning: this is a very simple server."
+ + "For development, use mkdocs serve instead."
+ )
+ typer.echo("This is here only to preview a site with translations already built.")
+ typer.echo("Make sure you run the build-all command first.")
+ os.chdir("site")
+ server_address = ("", 8008)
+ server = HTTPServer(server_address, SimpleHTTPRequestHandler)
+ typer.echo(f"Serving at: http://0.0.0.0:8008")
+ server.serve_forever()
+
+
+@app.command()
+def live(
+ lang: str = typer.Argument(
+ None, callback=lang_callback, autocompletion=complete_existing_lang
+ )
+):
+ """
+ Serve with livereload a docs site for a specific language.
+
+ This only shows the actual translated files, not the placeholders created with
+ build-all.
+
+ Takes an optional LANG argument with the name of the language to serve, by default
+ en.
+ """
+ if lang is None:
+ lang = "en"
+ lang_path: Path = docs_path / lang
+ os.chdir(lang_path)
+ mkdocs.commands.serve.serve(dev_addr="0.0.0.0:8008")
+
+
+def update_config(lang: str):
+ lang_path: Path = docs_path / lang
+ config_path = lang_path / mkdocs_name
+ config: dict = mkdocs.utils.yaml_load(config_path.read_text())
+ languages = [{"en": "/"}]
+ for lang in docs_path.iterdir():
+ if lang.name == "en" or not lang.is_dir():
+ continue
+ name = lang.name
+ languages.append({name: f"/{name}/"})
+ config["nav"][1] = {"Languages": languages}
+ config_path.write_text(yaml.dump(config, sort_keys=False, width=200))
+
+
+def get_key_section(
+ *, key_to_section: Dict[Tuple[str, ...], list], key: Tuple[str, ...]
+) -> list:
+ if key in key_to_section:
+ return key_to_section[key]
+ super_key = key[:-1]
+ title = key[-1]
+ super_section = get_key_section(key_to_section=key_to_section, key=super_key)
+ new_section = []
+ super_section.append({title: new_section})
+ key_to_section[key] = new_section
+ return new_section
+
+
+def get_text_with_translate_missing(text: str) -> str:
+ lines = text.splitlines()
+ lines.insert(1, missing_translation_snippet)
+ new_text = "\n".join(lines)
+ return new_text
+
+
+def get_file_to_nav_map(nav: list) -> Dict[str, Tuple[str, ...]]:
+ file_to_nav = {}
+ for item in nav:
+ if type(item) is str:
+ file_to_nav[item] = tuple()
+ elif type(item) is dict:
+ item_key = list(item.keys())[0]
+ sub_nav = item[item_key]
+ sub_file_to_nav = get_file_to_nav_map(sub_nav)
+ for k, v in sub_file_to_nav.items():
+ file_to_nav[k] = (item_key,) + v
+ return file_to_nav
+
+
+def get_sections(nav: list) -> Dict[Tuple[str, ...], str]:
+ sections = {}
+ for item in nav:
+ if type(item) is str:
+ continue
+ elif type(item) is dict:
+ item_key = list(item.keys())[0]
+ sub_nav = item[item_key]
+ sections[(item_key,)] = sub_nav[0]
+ sub_sections = get_sections(sub_nav)
+ for k, v in sub_sections.items():
+ new_key = (item_key,) + k
+ sections[new_key] = v
+ return sections
+
+
+if __name__ == "__main__":
+ app()
diff --git a/scripts/format-imports.sh b/scripts/format-imports.sh
index cfa66bbd4..d2c0b7a58 100755
--- a/scripts/format-imports.sh
+++ b/scripts/format-imports.sh
@@ -2,5 +2,5 @@
set -x
# Sort imports one per line, so autoflake can remove unused imports
-isort --recursive --force-single-line-imports --thirdparty fastapi --apply fastapi tests docs/src
+isort --recursive --force-single-line-imports --thirdparty fastapi --apply fastapi tests docs_src scripts
sh ./scripts/format.sh
diff --git a/scripts/format.sh b/scripts/format.sh
index c11eaf749..bbcb04354 100755
--- a/scripts/format.sh
+++ b/scripts/format.sh
@@ -1,6 +1,6 @@
#!/bin/sh -e
set -x
-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 --thirdparty fastapi --apply fastapi tests docs/src
+autoflake --remove-all-unused-imports --recursive --remove-unused-variables --in-place docs_src fastapi tests scripts --exclude=__init__.py
+black fastapi tests docs_src scripts
+isort --multi-line=3 --trailing-comma --force-grid-wrap=0 --combine-as --line-width 88 --recursive --thirdparty fastapi --apply fastapi tests docs_src scripts
diff --git a/scripts/test.sh b/scripts/test.sh
index d1a4cf630..468b2c667 100755
--- a/scripts/test.sh
+++ b/scripts/test.sh
@@ -9,6 +9,6 @@ if [ -f ./test.db ]; then
fi
bash ./scripts/lint.sh
# Check README.md is up to date
-diff --brief docs/index.md README.md
-export PYTHONPATH=./docs/src
-pytest --cov=fastapi --cov=tests --cov=docs/src --cov-report=term-missing ${@}
+diff --brief docs/en/docs/index.md README.md
+export PYTHONPATH=./docs_src
+pytest --cov=fastapi --cov=tests --cov=docs/src --cov-report=term-missing tests ${@}
diff --git a/tests/test_tutorial/test_additional_responses/test_tutorial002.py b/tests/test_tutorial/test_additional_responses/test_tutorial002.py
index 09a06d957..998070b3d 100644
--- a/tests/test_tutorial/test_additional_responses/test_tutorial002.py
+++ b/tests/test_tutorial/test_additional_responses/test_tutorial002.py
@@ -107,7 +107,7 @@ def test_path_operation():
def test_path_operation_img():
- shutil.copy("./docs/img/favicon.png", "./image.png")
+ shutil.copy("./docs/en/docs/img/favicon.png", "./image.png")
response = client.get("/items/foo?img=1")
assert response.status_code == 200
assert response.headers["Content-Type"] == "image/png"
diff --git a/tests/test_tutorial/test_additional_responses/test_tutorial004.py b/tests/test_tutorial/test_additional_responses/test_tutorial004.py
index d1285c8ea..11f1227e4 100644
--- a/tests/test_tutorial/test_additional_responses/test_tutorial004.py
+++ b/tests/test_tutorial/test_additional_responses/test_tutorial004.py
@@ -110,7 +110,7 @@ def test_path_operation():
def test_path_operation_img():
- shutil.copy("./docs/img/favicon.png", "./image.png")
+ shutil.copy("./docs/en/docs/img/favicon.png", "./image.png")
response = client.get("/items/foo?img=1")
assert response.status_code == 200
assert response.headers["Content-Type"] == "image/png"
diff --git a/tests/test_tutorial/test_templates/test_tutorial001.py b/tests/test_tutorial/test_templates/test_tutorial001.py
index 48d6456b4..95c5b1ed3 100644
--- a/tests/test_tutorial/test_templates/test_tutorial001.py
+++ b/tests/test_tutorial/test_templates/test_tutorial001.py
@@ -4,8 +4,8 @@ from fastapi.testclient import TestClient
def test_main():
- shutil.copytree("./docs/src/templates/templates/", "./templates")
- shutil.copytree("./docs/src/templates/static/", "./static")
+ shutil.copytree("./docs_src/templates/templates/", "./templates")
+ shutil.copytree("./docs_src/templates/static/", "./static")
from templates.tutorial001 import app
client = TestClient(app)