committed by
GitHub
126 changed files with 6959 additions and 835 deletions
@ -2,3 +2,15 @@ blank_issues_enabled: false |
|||||
contact_links: |
contact_links: |
||||
- name: Security Contact |
- name: Security Contact |
||||
about: Please report security vulnerabilities to [email protected] |
about: Please report security vulnerabilities to [email protected] |
||||
|
- name: Question or Problem |
||||
|
about: Ask a question or ask about a problem in GitHub Discussions. |
||||
|
url: https://github.com/tiangolo/fastapi/discussions/categories/questions |
||||
|
- name: Feature Request |
||||
|
about: To suggest an idea or ask about a feature, please start with a question saying what you would like to achieve. There might be a way to do it already. |
||||
|
url: https://github.com/tiangolo/fastapi/discussions/categories/questions |
||||
|
- name: Show and tell |
||||
|
about: Show what you built with FastAPI or to be used with FastAPI. |
||||
|
url: https://github.com/tiangolo/fastapi/discussions/categories/show-and-tell |
||||
|
- name: Translations |
||||
|
about: Coordinate translations in GitHub Discussions. |
||||
|
url: https://github.com/tiangolo/fastapi/discussions/categories/translations |
||||
|
@ -1,181 +0,0 @@ |
|||||
name: Feature Request |
|
||||
description: Suggest an idea or ask for a feature that you would like to have in FastAPI |
|
||||
labels: [enhancement] |
|
||||
body: |
|
||||
- type: markdown |
|
||||
attributes: |
|
||||
value: | |
|
||||
Thanks for your interest in FastAPI! 🚀 |
|
||||
|
|
||||
Please follow these instructions, fill every question, and do every step. 🙏 |
|
||||
|
|
||||
I'm asking this because answering questions and solving problems in GitHub issues is what consumes most of the time. |
|
||||
|
|
||||
I end up not being able to add new features, fix bugs, review pull requests, etc. as fast as I wish because I have to spend too much time handling issues. |
|
||||
|
|
||||
All that, on top of all the incredible help provided by a bunch of community members, the [FastAPI Experts](https://fastapi.tiangolo.com/fastapi-people/#experts), that give a lot of their time to come here and help others. |
|
||||
|
|
||||
That's a lot of work they are doing, but if more FastAPI users came to help others like them just a little bit more, it would be much less effort for them (and you and me 😅). |
|
||||
|
|
||||
By asking questions in a structured way (following this) it will be much easier to help you. |
|
||||
|
|
||||
And there's a high chance that you will find the solution along the way and you won't even have to submit it and wait for an answer. 😎 |
|
||||
|
|
||||
As there are too many issues with questions, I'll have to close the incomplete ones. That will allow me (and others) to focus on helping people like you that follow the whole process and help us help you. 🤓 |
|
||||
- type: checkboxes |
|
||||
id: checks |
|
||||
attributes: |
|
||||
label: First Check |
|
||||
description: Please confirm and check all the following options. |
|
||||
options: |
|
||||
- label: I added a very descriptive title to this issue. |
|
||||
required: true |
|
||||
- label: I used the GitHub search to find a similar issue and didn't find it. |
|
||||
required: true |
|
||||
- label: I searched the FastAPI documentation, with the integrated search. |
|
||||
required: true |
|
||||
- label: I already searched in Google "How to X in FastAPI" and didn't find any information. |
|
||||
required: true |
|
||||
- label: I already read and followed all the tutorial in the docs and didn't find an answer. |
|
||||
required: true |
|
||||
- label: I already checked if it is not related to FastAPI but to [Pydantic](https://github.com/samuelcolvin/pydantic). |
|
||||
required: true |
|
||||
- label: I already checked if it is not related to FastAPI but to [Swagger UI](https://github.com/swagger-api/swagger-ui). |
|
||||
required: true |
|
||||
- label: I already checked if it is not related to FastAPI but to [ReDoc](https://github.com/Redocly/redoc). |
|
||||
required: true |
|
||||
- type: checkboxes |
|
||||
id: help |
|
||||
attributes: |
|
||||
label: Commit to Help |
|
||||
description: | |
|
||||
After submitting this, I commit to one of: |
|
||||
|
|
||||
* Read open issues with questions until I find 2 issues where I can help someone and add a comment to help there. |
|
||||
* I already hit the "watch" button in this repository to receive notifications and I commit to help at least 2 people that ask questions in the future. |
|
||||
* Implement a Pull Request for a confirmed bug. |
|
||||
|
|
||||
options: |
|
||||
- label: I commit to help with one of those options 👆 |
|
||||
required: true |
|
||||
- type: textarea |
|
||||
id: example |
|
||||
attributes: |
|
||||
label: Example Code |
|
||||
description: | |
|
||||
Please add a self-contained, [minimal, reproducible, example](https://stackoverflow.com/help/minimal-reproducible-example) with your use case. |
|
||||
|
|
||||
If I (or someone) can copy it, run it, and see it right away, there's a much higher chance I (or someone) will be able to help you. |
|
||||
|
|
||||
placeholder: | |
|
||||
from fastapi import FastAPI |
|
||||
|
|
||||
app = FastAPI() |
|
||||
|
|
||||
|
|
||||
@app.get("/") |
|
||||
def read_root(): |
|
||||
return {"Hello": "World"} |
|
||||
render: python |
|
||||
validations: |
|
||||
required: true |
|
||||
- type: textarea |
|
||||
id: description |
|
||||
attributes: |
|
||||
label: Description |
|
||||
description: | |
|
||||
What is your feature request? |
|
||||
|
|
||||
Write a short description telling me what you are trying to solve and what you are currently doing. |
|
||||
placeholder: | |
|
||||
* Open the browser and call the endpoint `/`. |
|
||||
* It returns a JSON with `{"Hello": "World"}`. |
|
||||
* I would like it to have an extra parameter to teleport me to the moon and back. |
|
||||
validations: |
|
||||
required: true |
|
||||
- type: textarea |
|
||||
id: wanted-solution |
|
||||
attributes: |
|
||||
label: Wanted Solution |
|
||||
description: | |
|
||||
Tell me what's the solution you would like. |
|
||||
placeholder: | |
|
||||
I would like it to have a `teleport_to_moon` parameter that defaults to `False`, and can be set to `True` to teleport me. |
|
||||
validations: |
|
||||
required: true |
|
||||
- type: textarea |
|
||||
id: wanted-code |
|
||||
attributes: |
|
||||
label: Wanted Code |
|
||||
description: Show me an example of how you would want the code to look like. |
|
||||
placeholder: | |
|
||||
from fastapi import FastAPI |
|
||||
|
|
||||
app = FastAPI() |
|
||||
|
|
||||
|
|
||||
@app.get("/", teleport_to_moon=True) |
|
||||
def read_root(): |
|
||||
return {"Hello": "World"} |
|
||||
render: python |
|
||||
validations: |
|
||||
required: true |
|
||||
- type: textarea |
|
||||
id: alternatives |
|
||||
attributes: |
|
||||
label: Alternatives |
|
||||
description: | |
|
||||
Tell me about alternatives you've considered. |
|
||||
placeholder: | |
|
||||
To wait for Space X moon travel plans to drop down long after they release them. But I would rather teleport. |
|
||||
- type: dropdown |
|
||||
id: os |
|
||||
attributes: |
|
||||
label: Operating System |
|
||||
description: What operating system are you on? |
|
||||
multiple: true |
|
||||
options: |
|
||||
- Linux |
|
||||
- Windows |
|
||||
- macOS |
|
||||
- Other |
|
||||
validations: |
|
||||
required: true |
|
||||
- type: textarea |
|
||||
id: os-details |
|
||||
attributes: |
|
||||
label: Operating System Details |
|
||||
description: You can add more details about your operating system here, in particular if you chose "Other". |
|
||||
- type: input |
|
||||
id: fastapi-version |
|
||||
attributes: |
|
||||
label: FastAPI Version |
|
||||
description: | |
|
||||
What FastAPI version are you using? |
|
||||
|
|
||||
You can find the FastAPI version with: |
|
||||
|
|
||||
```bash |
|
||||
python -c "import fastapi; print(fastapi.__version__)" |
|
||||
``` |
|
||||
validations: |
|
||||
required: true |
|
||||
- type: input |
|
||||
id: python-version |
|
||||
attributes: |
|
||||
label: Python Version |
|
||||
description: | |
|
||||
What Python version are you using? |
|
||||
|
|
||||
You can find the Python version with: |
|
||||
|
|
||||
```bash |
|
||||
python --version |
|
||||
``` |
|
||||
validations: |
|
||||
required: true |
|
||||
- type: textarea |
|
||||
id: context |
|
||||
attributes: |
|
||||
label: Additional Context |
|
||||
description: Add any additional context information or screenshots you think are useful. |
|
@ -0,0 +1,22 @@ |
|||||
|
name: Privileged |
||||
|
description: You are @tiangolo or he asked you directly to create an issue here. If not, check the other options. 👇 |
||||
|
body: |
||||
|
- type: markdown |
||||
|
attributes: |
||||
|
value: | |
||||
|
Thanks for your interest in FastAPI! 🚀 |
||||
|
|
||||
|
If you are not @tiangolo or he didn't ask you directly to create an issue here, please start the conversation in a [Question in GitHub Discussions](https://github.com/tiangolo/fastapi/discussions/categories/questions) instead. |
||||
|
- type: checkboxes |
||||
|
id: privileged |
||||
|
attributes: |
||||
|
label: Privileged issue |
||||
|
description: Confirm that you are allowed to create an issue here. |
||||
|
options: |
||||
|
- label: I'm @tiangolo or he asked me directly to create an issue here. |
||||
|
required: true |
||||
|
- type: textarea |
||||
|
id: content |
||||
|
attributes: |
||||
|
label: Issue Content |
||||
|
description: Add the content of the issue here. |
@ -1,21 +0,0 @@ |
|||||
pt: 1211 |
|
||||
es: 1218 |
|
||||
zh: 1228 |
|
||||
ru: 1362 |
|
||||
it: 1556 |
|
||||
ja: 1572 |
|
||||
uk: 1748 |
|
||||
tr: 1892 |
|
||||
fr: 1972 |
|
||||
ko: 2017 |
|
||||
fa: 2041 |
|
||||
pl: 3169 |
|
||||
de: 3716 |
|
||||
id: 3717 |
|
||||
az: 3994 |
|
||||
nl: 4701 |
|
||||
uz: 4883 |
|
||||
sv: 5146 |
|
||||
he: 5157 |
|
||||
ta: 5434 |
|
||||
ar: 3349 |
|
@ -16,7 +16,7 @@ jobs: |
|||||
rm -rf ./site |
rm -rf ./site |
||||
mkdir ./site |
mkdir ./site |
||||
- name: Download Artifact Docs |
- name: Download Artifact Docs |
||||
uses: dawidd6/[email protected]4.2 |
uses: dawidd6/[email protected]6.0 |
||||
with: |
with: |
||||
github_token: ${{ secrets.GITHUB_TOKEN }} |
github_token: ${{ secrets.GITHUB_TOKEN }} |
||||
workflow: build-docs.yml |
workflow: build-docs.yml |
||||
|
@ -20,7 +20,7 @@ jobs: |
|||||
|
|
||||
- run: pip install smokeshow |
- run: pip install smokeshow |
||||
|
|
||||
- uses: dawidd6/[email protected]4.2 |
- uses: dawidd6/[email protected]6.0 |
||||
with: |
with: |
||||
workflow: test.yml |
workflow: test.yml |
||||
commit: ${{ github.event.workflow_run.head_sha }} |
commit: ${{ github.event.workflow_run.head_sha }} |
||||
|
After Width: | Height: | Size: 15 KiB |
After Width: | Height: | Size: 22 KiB |
@ -0,0 +1,168 @@ |
|||||
|
# Configuration avancée des paramètres de chemin |
||||
|
|
||||
|
## ID d'opération OpenAPI |
||||
|
|
||||
|
!!! Attention |
||||
|
Si vous n'êtes pas un "expert" en OpenAPI, vous n'en avez probablement pas besoin. |
||||
|
|
||||
|
Dans OpenAPI, les chemins sont des ressources, tels que /users/ ou /items/, exposées par votre API, et les opérations sont les méthodes HTTP utilisées pour manipuler ces chemins, telles que GET, POST ou DELETE. Les operationId sont des chaînes uniques facultatives utilisées pour identifier une opération d'un chemin. Vous pouvez définir l'OpenAPI `operationId` à utiliser dans votre *opération de chemin* avec le paramètre `operation_id`. |
||||
|
|
||||
|
Vous devez vous assurer qu'il est unique pour chaque opération. |
||||
|
|
||||
|
```Python hl_lines="6" |
||||
|
{!../../../docs_src/path_operation_advanced_configuration/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
### Utilisation du nom *path operation function* comme operationId |
||||
|
|
||||
|
Si vous souhaitez utiliser les noms de fonction de vos API comme `operationId`, vous pouvez les parcourir tous et remplacer chaque `operation_id` de l'*opération de chemin* en utilisant leur `APIRoute.name`. |
||||
|
|
||||
|
Vous devriez le faire après avoir ajouté toutes vos *paramètres de chemin*. |
||||
|
|
||||
|
```Python hl_lines="2 12-21 24" |
||||
|
{!../../../docs_src/path_operation_advanced_configuration/tutorial002.py!} |
||||
|
``` |
||||
|
|
||||
|
!!! Astuce |
||||
|
Si vous appelez manuellement `app.openapi()`, vous devez mettre à jour les `operationId` avant. |
||||
|
|
||||
|
!!! Attention |
||||
|
Pour faire cela, vous devez vous assurer que chacun de vos *chemin* ait un nom unique. |
||||
|
|
||||
|
Même s'ils se trouvent dans des modules différents (fichiers Python). |
||||
|
|
||||
|
## Exclusion d'OpenAPI |
||||
|
|
||||
|
Pour exclure un *chemin* du schéma OpenAPI généré (et donc des systèmes de documentation automatiques), utilisez le paramètre `include_in_schema` et assignez-lui la valeur `False` : |
||||
|
|
||||
|
```Python hl_lines="6" |
||||
|
{!../../../docs_src/path_operation_advanced_configuration/tutorial003.py!} |
||||
|
``` |
||||
|
|
||||
|
## Description avancée de docstring |
||||
|
|
||||
|
Vous pouvez limiter le texte utilisé de la docstring d'une *fonction de chemin* qui sera affiché sur OpenAPI. |
||||
|
|
||||
|
L'ajout d'un `\f` (un caractère d'échappement "form feed") va permettre à **FastAPI** de tronquer la sortie utilisée pour OpenAPI à ce stade. |
||||
|
|
||||
|
Il n'apparaîtra pas dans la documentation, mais d'autres outils (tel que Sphinx) pourront utiliser le reste. |
||||
|
|
||||
|
```Python hl_lines="19-29" |
||||
|
{!../../../docs_src/path_operation_advanced_configuration/tutorial004.py!} |
||||
|
``` |
||||
|
|
||||
|
## Réponses supplémentaires |
||||
|
|
||||
|
Vous avez probablement vu comment déclarer le `response_model` et le `status_code` pour une *opération de chemin*. |
||||
|
|
||||
|
Cela définit les métadonnées sur la réponse principale d'une *opération de chemin*. |
||||
|
|
||||
|
Vous pouvez également déclarer des réponses supplémentaires avec leurs modèles, codes de statut, etc. |
||||
|
|
||||
|
Il y a un chapitre entier ici dans la documentation à ce sujet, vous pouvez le lire sur [Réponses supplémentaires dans OpenAPI](./additional-responses.md){.internal-link target=_blank}. |
||||
|
|
||||
|
## OpenAPI supplémentaire |
||||
|
|
||||
|
Lorsque vous déclarez un *chemin* dans votre application, **FastAPI** génère automatiquement les métadonnées concernant ce *chemin* à inclure dans le schéma OpenAPI. |
||||
|
|
||||
|
!!! note "Détails techniques" |
||||
|
La spécification OpenAPI appelle ces métaonnées des <a href="https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md#operation-object" class="external-link" target="_blank">Objets d'opération</a>. |
||||
|
|
||||
|
Il contient toutes les informations sur le *chemin* et est utilisé pour générer automatiquement la documentation. |
||||
|
|
||||
|
Il inclut les `tags`, `parameters`, `requestBody`, `responses`, etc. |
||||
|
|
||||
|
Ce schéma OpenAPI spécifique aux *operations* est normalement généré automatiquement par **FastAPI**, mais vous pouvez également l'étendre. |
||||
|
|
||||
|
!!! Astuce |
||||
|
Si vous avez seulement besoin de déclarer des réponses supplémentaires, un moyen plus pratique de le faire est d'utiliser les [réponses supplémentaires dans OpenAPI](./additional-responses.md){.internal-link target=_blank}. |
||||
|
|
||||
|
Vous pouvez étendre le schéma OpenAPI pour une *opération de chemin* en utilisant le paramètre `openapi_extra`. |
||||
|
|
||||
|
### Extensions OpenAPI |
||||
|
|
||||
|
Cet `openapi_extra` peut être utile, par exemple, pour déclarer [OpenAPI Extensions](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md#specificationExtensions) : |
||||
|
|
||||
|
```Python hl_lines="6" |
||||
|
{!../../../docs_src/path_operation_advanced_configuration/tutorial005.py!} |
||||
|
``` |
||||
|
|
||||
|
Si vous ouvrez la documentation automatique de l'API, votre extension apparaîtra au bas du *chemin* spécifique. |
||||
|
|
||||
|
<img src="/img/tutorial/path-operation-advanced-configuration/image01.png"> |
||||
|
|
||||
|
Et dans le fichier openapi généré (`/openapi.json`), vous verrez également votre extension dans le cadre du *chemin* spécifique : |
||||
|
|
||||
|
```JSON hl_lines="22" |
||||
|
{ |
||||
|
"openapi": "3.0.2", |
||||
|
"info": { |
||||
|
"title": "FastAPI", |
||||
|
"version": "0.1.0" |
||||
|
}, |
||||
|
"paths": { |
||||
|
"/items/": { |
||||
|
"get": { |
||||
|
"summary": "Read Items", |
||||
|
"operationId": "read_items_items__get", |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
"schema": {} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
}, |
||||
|
"x-aperture-labs-portal": "blue" |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
``` |
||||
|
|
||||
|
### Personnalisation du Schéma OpenAPI pour un chemin |
||||
|
|
||||
|
Le dictionnaire contenu dans la variable `openapi_extra` sera fusionné avec le schéma OpenAPI généré automatiquement pour l'*opération de chemin*. |
||||
|
|
||||
|
Ainsi, vous pouvez ajouter des données supplémentaires au schéma généré automatiquement. |
||||
|
|
||||
|
Par exemple, vous pouvez décider de lire et de valider la requête avec votre propre code, sans utiliser les fonctionnalités automatiques de validation proposée par Pydantic, mais vous pouvez toujours définir la requête dans le schéma OpenAPI. |
||||
|
|
||||
|
Vous pouvez le faire avec `openapi_extra` : |
||||
|
|
||||
|
```Python hl_lines="20-37 39-40" |
||||
|
{!../../../docs_src/path_operation_advanced_configuration/tutorial006.py !} |
||||
|
``` |
||||
|
|
||||
|
Dans cet exemple, nous n'avons déclaré aucun modèle Pydantic. En fait, le corps de la requête n'est même pas <abbr title="converti d'un format simple, comme des octets, en objets Python">parsé</abbr> en tant que JSON, il est lu directement en tant que `bytes`, et la fonction `magic_data_reader()` serait chargé de l'analyser d'une manière ou d'une autre. |
||||
|
|
||||
|
Néanmoins, nous pouvons déclarer le schéma attendu pour le corps de la requête. |
||||
|
|
||||
|
### Type de contenu OpenAPI personnalisé |
||||
|
|
||||
|
En utilisant cette même astuce, vous pouvez utiliser un modèle Pydantic pour définir le schéma JSON qui est ensuite inclus dans la section de schéma OpenAPI personnalisée pour le *chemin* concerné. |
||||
|
|
||||
|
Et vous pouvez le faire même si le type de données dans la requête n'est pas au format JSON. |
||||
|
|
||||
|
Dans cet exemple, nous n'utilisons pas les fonctionnalités de FastAPI pour extraire le schéma JSON des modèles Pydantic ni la validation automatique pour JSON. En fait, nous déclarons le type de contenu de la requête en tant que YAML, et non JSON : |
||||
|
|
||||
|
```Python hl_lines="17-22 24" |
||||
|
{!../../../docs_src/path_operation_advanced_configuration/tutorial007.py!} |
||||
|
``` |
||||
|
|
||||
|
Néanmoins, bien que nous n'utilisions pas la fonctionnalité par défaut, nous utilisons toujours un modèle Pydantic pour générer manuellement le schéma JSON pour les données que nous souhaitons recevoir en YAML. |
||||
|
|
||||
|
Ensuite, nous utilisons directement la requête et extrayons son contenu en tant qu'octets. Cela signifie que FastAPI n'essaiera même pas d'analyser le payload de la requête en tant que JSON. |
||||
|
|
||||
|
Et nous analysons directement ce contenu YAML, puis nous utilisons à nouveau le même modèle Pydantic pour valider le contenu YAML : |
||||
|
|
||||
|
```Python hl_lines="26-33" |
||||
|
{!../../../docs_src/path_operation_advanced_configuration/tutorial007.py!} |
||||
|
``` |
||||
|
|
||||
|
!!! Astuce |
||||
|
Ici, nous réutilisons le même modèle Pydantic. |
||||
|
|
||||
|
Mais nous aurions pu tout aussi bien pu le valider d'une autre manière. |
@ -0,0 +1,153 @@ |
|||||
|
# Exécuter un serveur manuellement - Uvicorn |
||||
|
|
||||
|
La principale chose dont vous avez besoin pour exécuter une application **FastAPI** sur une machine serveur distante est un programme serveur ASGI tel que **Uvicorn**. |
||||
|
|
||||
|
Il existe 3 principales alternatives : |
||||
|
|
||||
|
* <a href="https://www.uvicorn.org/" class="external-link" target="_blank">Uvicorn</a> : un serveur ASGI haute performance. |
||||
|
* <a href="https://pgjones.gitlab.io/hypercorn/" class="external-link" target="_blank">Hypercorn</a> : un serveur |
||||
|
ASGI compatible avec HTTP/2 et Trio entre autres fonctionnalités. |
||||
|
* <a href="https://github.com/django/daphne" class="external-link" target="_blank">Daphne</a> : le serveur ASGI |
||||
|
conçu pour Django Channels. |
||||
|
|
||||
|
## Machine serveur et programme serveur |
||||
|
|
||||
|
Il y a un petit détail sur les noms à garder à l'esprit. 💡 |
||||
|
|
||||
|
Le mot "**serveur**" est couramment utilisé pour désigner à la fois l'ordinateur distant/cloud (la machine physique ou virtuelle) et également le programme qui s'exécute sur cette machine (par exemple, Uvicorn). |
||||
|
|
||||
|
Gardez cela à l'esprit lorsque vous lisez "serveur" en général, cela pourrait faire référence à l'une de ces deux choses. |
||||
|
|
||||
|
Lorsqu'on se réfère à la machine distante, il est courant de l'appeler **serveur**, mais aussi **machine**, **VM** (machine virtuelle), **nœud**. Tout cela fait référence à un type de machine distante, exécutant Linux, en règle générale, sur laquelle vous exécutez des programmes. |
||||
|
|
||||
|
|
||||
|
## Installer le programme serveur |
||||
|
|
||||
|
Vous pouvez installer un serveur compatible ASGI avec : |
||||
|
|
||||
|
=== "Uvicorn" |
||||
|
|
||||
|
* <a href="https://www.uvicorn.org/" class="external-link" target="_blank">Uvicorn</a>, un serveur ASGI rapide comme l'éclair, basé sur uvloop et httptools. |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ pip install "uvicorn[standard]" |
||||
|
|
||||
|
---> 100% |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
!!! tip "Astuce" |
||||
|
En ajoutant `standard`, Uvicorn va installer et utiliser quelques dépendances supplémentaires recommandées. |
||||
|
|
||||
|
Cela inclut `uvloop`, le remplaçant performant de `asyncio`, qui fournit le gros gain de performance en matière de concurrence. |
||||
|
|
||||
|
=== "Hypercorn" |
||||
|
|
||||
|
* <a href="https://gitlab.com/pgjones/hypercorn" class="external-link" target="_blank">Hypercorn</a>, un serveur ASGI également compatible avec HTTP/2. |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ pip install hypercorn |
||||
|
|
||||
|
---> 100% |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
...ou tout autre serveur ASGI. |
||||
|
|
||||
|
## Exécutez le programme serveur |
||||
|
|
||||
|
Vous pouvez ensuite exécuter votre application de la même manière que vous l'avez fait dans les tutoriels, mais sans l'option `--reload`, par exemple : |
||||
|
|
||||
|
=== "Uvicorn" |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ uvicorn main:app --host 0.0.0.0 --port 80 |
||||
|
|
||||
|
<span style="color: green;">INFO</span>: Uvicorn running on http://0.0.0.0:80 (Press CTRL+C to quit) |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
=== "Hypercorn" |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ hypercorn main:app --bind 0.0.0.0:80 |
||||
|
|
||||
|
Running on 0.0.0.0:8080 over http (CTRL + C to quit) |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
!!! warning |
||||
|
N'oubliez pas de supprimer l'option `--reload` si vous l'utilisiez. |
||||
|
|
||||
|
L'option `--reload` consomme beaucoup plus de ressources, est plus instable, etc. |
||||
|
|
||||
|
Cela aide beaucoup pendant le **développement**, mais vous **ne devriez pas** l'utiliser en **production**. |
||||
|
|
||||
|
## Hypercorn avec Trio |
||||
|
|
||||
|
Starlette et **FastAPI** sont basés sur |
||||
|
<a href="https://anyio.readthedocs.io/en/stable/" class="external-link" target="_blank">AnyIO</a>, qui les rend |
||||
|
compatibles avec <a href="https://docs.python.org/3/library/asyncio-task.html" class="external-link" target="_blank">asyncio</a>, de la bibliothèque standard Python et |
||||
|
<a href="https://trio.readthedocs.io/en/stable/" class="external-link" target="_blank">Trio</a>. |
||||
|
|
||||
|
Néanmoins, Uvicorn n'est actuellement compatible qu'avec asyncio, et il utilise normalement <a href="https://github. |
||||
|
com/MagicStack/uvloop" class="external-link" target="_blank">`uvloop`</a >, le remplaçant hautes performances de `asyncio`. |
||||
|
|
||||
|
Mais si vous souhaitez utiliser directement **Trio**, vous pouvez utiliser **Hypercorn** car il le prend en charge. ✨ |
||||
|
|
||||
|
### Installer Hypercorn avec Trio |
||||
|
|
||||
|
Vous devez d'abord installer Hypercorn avec le support Trio : |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ pip install "hypercorn[trio]" |
||||
|
---> 100% |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
### Exécuter avec Trio |
||||
|
|
||||
|
Ensuite, vous pouvez passer l'option de ligne de commande `--worker-class` avec la valeur `trio` : |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ hypercorn main:app --worker-class trio |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Et cela démarrera Hypercorn avec votre application en utilisant Trio comme backend. |
||||
|
|
||||
|
Vous pouvez désormais utiliser Trio en interne dans votre application. Ou mieux encore, vous pouvez utiliser AnyIO pour que votre code reste compatible avec Trio et asyncio. 🎉 |
||||
|
|
||||
|
## Concepts de déploiement |
||||
|
|
||||
|
Ces exemples lancent le programme serveur (e.g. Uvicorn), démarrant **un seul processus**, sur toutes les IPs (`0.0. |
||||
|
0.0`) sur un port prédéfini (par example, `80`). |
||||
|
|
||||
|
C'est l'idée de base. Mais vous vous préoccuperez probablement de certains concepts supplémentaires, tels que ... : |
||||
|
|
||||
|
* la sécurité - HTTPS |
||||
|
* l'exécution au démarrage |
||||
|
* les redémarrages |
||||
|
* la réplication (le nombre de processus en cours d'exécution) |
||||
|
* la mémoire |
||||
|
* les étapes précédant le démarrage |
||||
|
|
||||
|
Je vous en dirai plus sur chacun de ces concepts, sur la façon de les aborder, et donnerai quelques exemples concrets avec des stratégies pour les traiter dans les prochains chapitres. 🚀 |
@ -0,0 +1,112 @@ |
|||||
|
# <abbr title="En anglais: Debugging">Débogage</abbr> |
||||
|
|
||||
|
Vous pouvez connecter le <abbr title="En anglais: debugger">débogueur</abbr> dans votre éditeur, par exemple avec Visual Studio Code ou PyCharm. |
||||
|
|
||||
|
## Faites appel à `uvicorn` |
||||
|
|
||||
|
Dans votre application FastAPI, importez et exécutez directement `uvicorn` : |
||||
|
|
||||
|
```Python hl_lines="1 15" |
||||
|
{!../../../docs_src/debugging/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
### À propos de `__name__ == "__main__"` |
||||
|
|
||||
|
Le but principal de `__name__ == "__main__"` est d'avoir du code qui est exécuté lorsque votre fichier est appelé avec : |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ python myapp.py |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
mais qui n'est pas appelé lorsqu'un autre fichier l'importe, comme dans : |
||||
|
|
||||
|
```Python |
||||
|
from myapp import app |
||||
|
``` |
||||
|
|
||||
|
#### Pour davantage de détails |
||||
|
|
||||
|
Imaginons que votre fichier s'appelle `myapp.py`. |
||||
|
|
||||
|
Si vous l'exécutez avec : |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ python myapp.py |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
alors la variable interne `__name__` de votre fichier, créée automatiquement par Python, aura pour valeur la chaîne de caractères `"__main__"`. |
||||
|
|
||||
|
Ainsi, la section : |
||||
|
|
||||
|
```Python |
||||
|
uvicorn.run(app, host="0.0.0.0", port=8000) |
||||
|
``` |
||||
|
|
||||
|
va s'exécuter. |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
Cela ne se produira pas si vous importez ce module (fichier). |
||||
|
|
||||
|
Par exemple, si vous avez un autre fichier `importer.py` qui contient : |
||||
|
|
||||
|
```Python |
||||
|
from myapp import app |
||||
|
|
||||
|
# Code supplémentaire |
||||
|
``` |
||||
|
|
||||
|
dans ce cas, la variable automatique `__name__` à l'intérieur de `myapp.py` n'aura pas la valeur `"__main__"`. |
||||
|
|
||||
|
Ainsi, la ligne : |
||||
|
|
||||
|
```Python |
||||
|
uvicorn.run(app, host="0.0.0.0", port=8000) |
||||
|
``` |
||||
|
|
||||
|
ne sera pas exécutée. |
||||
|
|
||||
|
!!! info |
||||
|
Pour plus d'informations, consultez <a href="https://docs.python.org/3/library/__main__.html" class="external-link" target="_blank">la documentation officielle de Python</a>. |
||||
|
|
||||
|
## Exécutez votre code avec votre <abbr title="En anglais: debugger">débogueur</abbr> |
||||
|
|
||||
|
Parce que vous exécutez le serveur Uvicorn directement depuis votre code, vous pouvez appeler votre programme Python (votre application FastAPI) directement depuis le <abbr title="En anglais: debugger">débogueur</abbr>. |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
Par exemple, dans Visual Studio Code, vous pouvez : |
||||
|
|
||||
|
- Cliquer sur l'onglet "Debug" de la barre d'activités de Visual Studio Code. |
||||
|
- "Add configuration...". |
||||
|
- Sélectionnez "Python". |
||||
|
- Lancez le <abbr title="En anglais: debugger">débogueur</abbr> avec l'option "`Python: Current File (Integrated Terminal)`". |
||||
|
|
||||
|
Il démarrera alors le serveur avec votre code **FastAPI**, s'arrêtera à vos points d'arrêt, etc. |
||||
|
|
||||
|
Voici à quoi cela pourrait ressembler : |
||||
|
|
||||
|
<img src="/img/tutorial/debugging/image01.png"> |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
Si vous utilisez Pycharm, vous pouvez : |
||||
|
|
||||
|
- Ouvrir le menu "Run". |
||||
|
- Sélectionnez l'option "Debug...". |
||||
|
- Un menu contextuel s'affiche alors. |
||||
|
- Sélectionnez le fichier à déboguer (dans ce cas, `main.py`). |
||||
|
|
||||
|
Il démarrera alors le serveur avec votre code **FastAPI**, s'arrêtera à vos points d'arrêt, etc. |
||||
|
|
||||
|
Voici à quoi cela pourrait ressembler : |
||||
|
|
||||
|
<img src="/img/tutorial/debugging/image02.png"> |
@ -0,0 +1,467 @@ |
|||||
|
|
||||
|
{!../../../docs/missing-translation.md!} |
||||
|
|
||||
|
|
||||
|
<p align="center"> |
||||
|
<a href="https://fastapi.tiangolo.com"><img src="https://fastapi.tiangolo.com/img/logo-margin/logo-teal.png" alt="FastAPI"></a> |
||||
|
</p> |
||||
|
<p align="center"> |
||||
|
<em>FastAPI framework, high performance, easy to learn, fast to code, ready for production</em> |
||||
|
</p> |
||||
|
<p align="center"> |
||||
|
<a href="https://github.com/tiangolo/fastapi/actions?query=workflow%3ATest+event%3Apush+branch%3Amaster" target="_blank"> |
||||
|
<img src="https://github.com/tiangolo/fastapi/workflows/Test/badge.svg?event=push&branch=master" alt="Test"> |
||||
|
</a> |
||||
|
<a href="https://coverage-badge.samuelcolvin.workers.dev/redirect/tiangolo/fastapi" target="_blank"> |
||||
|
<img src="https://coverage-badge.samuelcolvin.workers.dev/tiangolo/fastapi.svg" alt="Coverage"> |
||||
|
</a> |
||||
|
<a href="https://pypi.org/project/fastapi" target="_blank"> |
||||
|
<img src="https://img.shields.io/pypi/v/fastapi?color=%2334D058&label=pypi%20package" alt="Package version"> |
||||
|
</a> |
||||
|
<a href="https://pypi.org/project/fastapi" target="_blank"> |
||||
|
<img src="https://img.shields.io/pypi/pyversions/fastapi.svg?color=%2334D058" alt="Supported Python versions"> |
||||
|
</a> |
||||
|
</p> |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
**Documentation**: <a href="https://fastapi.tiangolo.com" target="_blank">https://fastapi.tiangolo.com</a> |
||||
|
|
||||
|
**Source Code**: <a href="https://github.com/tiangolo/fastapi" target="_blank">https://github.com/tiangolo/fastapi</a> |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
FastAPI is a modern, fast (high-performance), web framework for building APIs with Python 3.7+ 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. <abbr title="also known as auto-complete, autocompletion, IntelliSense">Completion</abbr> 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: <a href="https://github.com/OAI/OpenAPI-Specification" class="external-link" target="_blank">OpenAPI</a> (previously known as Swagger) and <a href="https://json-schema.org/" class="external-link" target="_blank">JSON Schema</a>. |
||||
|
|
||||
|
<small>* estimation based on tests on an internal development team, building production applications.</small> |
||||
|
|
||||
|
## Sponsors |
||||
|
|
||||
|
<!-- sponsors --> |
||||
|
|
||||
|
{% if sponsors %} |
||||
|
{% for sponsor in sponsors.gold -%} |
||||
|
<a href="{{ sponsor.url }}" target="_blank" title="{{ sponsor.title }}"><img src="{{ sponsor.img }}" style="border-radius:15px"></a> |
||||
|
{% endfor -%} |
||||
|
{%- for sponsor in sponsors.silver -%} |
||||
|
<a href="{{ sponsor.url }}" target="_blank" title="{{ sponsor.title }}"><img src="{{ sponsor.img }}" style="border-radius:15px"></a> |
||||
|
{% endfor %} |
||||
|
{% endif %} |
||||
|
|
||||
|
<!-- /sponsors --> |
||||
|
|
||||
|
<a href="https://fastapi.tiangolo.com/fastapi-people/#sponsors" class="external-link" target="_blank">Other sponsors</a> |
||||
|
|
||||
|
## 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._" |
||||
|
|
||||
|
<div style="text-align: right; margin-right: 10%;">Kabir Khan - <strong>Microsoft</strong> <a href="https://github.com/tiangolo/fastapi/pull/26" target="_blank"><small>(ref)</small></a></div> |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
"_We adopted the **FastAPI** library to spawn a **REST** server that can be queried to obtain **predictions**. [for Ludwig]_" |
||||
|
|
||||
|
<div style="text-align: right; margin-right: 10%;">Piero Molino, Yaroslav Dudin, and Sai Sumanth Miryala - <strong>Uber</strong> <a href="https://eng.uber.com/ludwig-v0-2/" target="_blank"><small>(ref)</small></a></div> |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
"_**Netflix** is pleased to announce the open-source release of our **crisis management** orchestration framework: **Dispatch**! [built with **FastAPI**]_" |
||||
|
|
||||
|
<div style="text-align: right; margin-right: 10%;">Kevin Glisson, Marc Vilanova, Forest Monsen - <strong>Netflix</strong> <a href="https://netflixtechblog.com/introducing-dispatch-da4b8a2a8072" target="_blank"><small>(ref)</small></a></div> |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
"_I’m over the moon excited about **FastAPI**. It’s so fun!_" |
||||
|
|
||||
|
<div style="text-align: right; margin-right: 10%;">Brian Okken - <strong><a href="https://pythonbytes.fm/episodes/show/123/time-to-right-the-py-wrongs?time_in_sec=855" target="_blank">Python Bytes</a> podcast host</strong> <a href="https://twitter.com/brianokken/status/1112220079972728832" target="_blank"><small>(ref)</small></a></div> |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
"_Honestly, what you've built looks super solid and polished. In many ways, it's what I wanted **Hug** to be - it's really inspiring to see someone build that._" |
||||
|
|
||||
|
<div style="text-align: right; margin-right: 10%;">Timothy Crosley - <strong><a href="https://www.hug.rest/" target="_blank">Hug</a> creator</strong> <a href="https://news.ycombinator.com/item?id=19455465" target="_blank"><small>(ref)</small></a></div> |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
"_If you're looking to learn one **modern framework** for building REST APIs, check out **FastAPI** [...] It's fast, easy to use and easy to learn [...]_" |
||||
|
|
||||
|
"_We've switched over to **FastAPI** for our **APIs** [...] I think you'll like it [...]_" |
||||
|
|
||||
|
<div style="text-align: right; margin-right: 10%;">Ines Montani - Matthew Honnibal - <strong><a href="https://explosion.ai" target="_blank">Explosion AI</a> founders - <a href="https://spacy.io" target="_blank">spaCy</a> creators</strong> <a href="https://twitter.com/_inesmontani/status/1144173225322143744" target="_blank"><small>(ref)</small></a> - <a href="https://twitter.com/honnibal/status/1144031421859655680" target="_blank"><small>(ref)</small></a></div> |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
## **Typer**, the FastAPI of CLIs |
||||
|
|
||||
|
<a href="https://typer.tiangolo.com" target="_blank"><img src="https://typer.tiangolo.com/img/logo-margin/logo-margin-vector.svg" style="width: 20%;"></a> |
||||
|
|
||||
|
If you are building a <abbr title="Command Line Interface">CLI</abbr> app to be used in the terminal instead of a web API, check out <a href="https://typer.tiangolo.com/" class="external-link" target="_blank">**Typer**</a>. |
||||
|
|
||||
|
**Typer** is FastAPI's little sibling. And it's intended to be the **FastAPI of CLIs**. ⌨️ 🚀 |
||||
|
|
||||
|
## Requirements |
||||
|
|
||||
|
Python 3.7+ |
||||
|
|
||||
|
FastAPI stands on the shoulders of giants: |
||||
|
|
||||
|
* <a href="https://www.starlette.io/" class="external-link" target="_blank">Starlette</a> for the web parts. |
||||
|
* <a href="https://pydantic-docs.helpmanual.io/" class="external-link" target="_blank">Pydantic</a> for the data parts. |
||||
|
|
||||
|
## Installation |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ pip install fastapi |
||||
|
|
||||
|
---> 100% |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
You will also need an ASGI server, for production such as <a href="https://www.uvicorn.org" class="external-link" target="_blank">Uvicorn</a> or <a href="https://github.com/pgjones/hypercorn" class="external-link" target="_blank">Hypercorn</a>. |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ pip install "uvicorn[standard]" |
||||
|
|
||||
|
---> 100% |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
## Example |
||||
|
|
||||
|
### Create it |
||||
|
|
||||
|
* Create a file `main.py` with: |
||||
|
|
||||
|
```Python |
||||
|
from typing import Union |
||||
|
|
||||
|
from fastapi import FastAPI |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/") |
||||
|
def read_root(): |
||||
|
return {"Hello": "World"} |
||||
|
|
||||
|
|
||||
|
@app.get("/items/{item_id}") |
||||
|
def read_item(item_id: int, q: Union[str, None] = None): |
||||
|
return {"item_id": item_id, "q": q} |
||||
|
``` |
||||
|
|
||||
|
<details markdown="1"> |
||||
|
<summary>Or use <code>async def</code>...</summary> |
||||
|
|
||||
|
If your code uses `async` / `await`, use `async def`: |
||||
|
|
||||
|
```Python hl_lines="9 14" |
||||
|
from typing import Union |
||||
|
|
||||
|
from fastapi import FastAPI |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/") |
||||
|
async def read_root(): |
||||
|
return {"Hello": "World"} |
||||
|
|
||||
|
|
||||
|
@app.get("/items/{item_id}") |
||||
|
async def read_item(item_id: int, q: Union[str, None] = None): |
||||
|
return {"item_id": item_id, "q": q} |
||||
|
``` |
||||
|
|
||||
|
**Note**: |
||||
|
|
||||
|
If you don't know, check the _"In a hurry?"_ section about <a href="https://fastapi.tiangolo.com/async/#in-a-hurry" target="_blank">`async` and `await` in the docs</a>. |
||||
|
|
||||
|
</details> |
||||
|
|
||||
|
### Run it |
||||
|
|
||||
|
Run the server with: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ uvicorn main:app --reload |
||||
|
|
||||
|
INFO: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit) |
||||
|
INFO: Started reloader process [28720] |
||||
|
INFO: Started server process [28722] |
||||
|
INFO: Waiting for application startup. |
||||
|
INFO: Application startup complete. |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
<details markdown="1"> |
||||
|
<summary>About the command <code>uvicorn main:app --reload</code>...</summary> |
||||
|
|
||||
|
The command `uvicorn main:app` refers to: |
||||
|
|
||||
|
* `main`: the file `main.py` (the Python "module"). |
||||
|
* `app`: the object created inside of `main.py` with the line `app = FastAPI()`. |
||||
|
* `--reload`: make the server restart after code changes. Only do this for development. |
||||
|
|
||||
|
</details> |
||||
|
|
||||
|
### Check it |
||||
|
|
||||
|
Open your browser at <a href="http://127.0.0.1:8000/items/5?q=somequery" class="external-link" target="_blank">http://127.0.0.1:8000/items/5?q=somequery</a>. |
||||
|
|
||||
|
You will see the JSON response as: |
||||
|
|
||||
|
```JSON |
||||
|
{"item_id": 5, "q": "somequery"} |
||||
|
``` |
||||
|
|
||||
|
You already created an API that: |
||||
|
|
||||
|
* Receives HTTP requests in the _paths_ `/` and `/items/{item_id}`. |
||||
|
* Both _paths_ take `GET` <em>operations</em> (also known as HTTP _methods_). |
||||
|
* The _path_ `/items/{item_id}` has a _path parameter_ `item_id` that should be an `int`. |
||||
|
* The _path_ `/items/{item_id}` has an optional `str` _query parameter_ `q`. |
||||
|
|
||||
|
### Interactive API docs |
||||
|
|
||||
|
Now go to <a href="http://127.0.0.1:8000/docs" class="external-link" target="_blank">http://127.0.0.1:8000/docs</a>. |
||||
|
|
||||
|
You will see the automatic interactive API documentation (provided by <a href="https://github.com/swagger-api/swagger-ui" class="external-link" target="_blank">Swagger UI</a>): |
||||
|
|
||||
|
 |
||||
|
|
||||
|
### Alternative API docs |
||||
|
|
||||
|
And now, go to <a href="http://127.0.0.1:8000/redoc" class="external-link" target="_blank">http://127.0.0.1:8000/redoc</a>. |
||||
|
|
||||
|
You will see the alternative automatic documentation (provided by <a href="https://github.com/Rebilly/ReDoc" class="external-link" target="_blank">ReDoc</a>): |
||||
|
|
||||
|
 |
||||
|
|
||||
|
## Example upgrade |
||||
|
|
||||
|
Now modify the file `main.py` to receive a body from a `PUT` request. |
||||
|
|
||||
|
Declare the body using standard Python types, thanks to Pydantic. |
||||
|
|
||||
|
```Python hl_lines="4 9-12 25-27" |
||||
|
from typing import Union |
||||
|
|
||||
|
from fastapi import FastAPI |
||||
|
from pydantic import BaseModel |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
class Item(BaseModel): |
||||
|
name: str |
||||
|
price: float |
||||
|
is_offer: Union[bool, None] = None |
||||
|
|
||||
|
|
||||
|
@app.get("/") |
||||
|
def read_root(): |
||||
|
return {"Hello": "World"} |
||||
|
|
||||
|
|
||||
|
@app.get("/items/{item_id}") |
||||
|
def read_item(item_id: int, q: Union[str, None] = None): |
||||
|
return {"item_id": item_id, "q": q} |
||||
|
|
||||
|
|
||||
|
@app.put("/items/{item_id}") |
||||
|
def update_item(item_id: int, item: Item): |
||||
|
return {"item_name": item.name, "item_id": item_id} |
||||
|
``` |
||||
|
|
||||
|
The server should reload automatically (because you added `--reload` to the `uvicorn` command above). |
||||
|
|
||||
|
### Interactive API docs upgrade |
||||
|
|
||||
|
Now go to <a href="http://127.0.0.1:8000/docs" class="external-link" target="_blank">http://127.0.0.1:8000/docs</a>. |
||||
|
|
||||
|
* The interactive API documentation will be automatically updated, including the new body: |
||||
|
|
||||
|
 |
||||
|
|
||||
|
* Click on the button "Try it out", it allows you to fill the parameters and directly interact with the API: |
||||
|
|
||||
|
 |
||||
|
|
||||
|
* Then click on the "Execute" button, the user interface will communicate with your API, send the parameters, get the results and show them on the screen: |
||||
|
|
||||
|
 |
||||
|
|
||||
|
### Alternative API docs upgrade |
||||
|
|
||||
|
And now, go to <a href="http://127.0.0.1:8000/redoc" class="external-link" target="_blank">http://127.0.0.1:8000/redoc</a>. |
||||
|
|
||||
|
* The alternative documentation will also reflect the new query parameter and body: |
||||
|
|
||||
|
 |
||||
|
|
||||
|
### Recap |
||||
|
|
||||
|
In summary, you declare **once** the types of parameters, body, etc. as function parameters. |
||||
|
|
||||
|
You do that with standard modern Python types. |
||||
|
|
||||
|
You don't have to learn a new syntax, the methods or classes of a specific library, etc. |
||||
|
|
||||
|
Just standard **Python 3.7+**. |
||||
|
|
||||
|
For example, for an `int`: |
||||
|
|
||||
|
```Python |
||||
|
item_id: int |
||||
|
``` |
||||
|
|
||||
|
or for a more complex `Item` model: |
||||
|
|
||||
|
```Python |
||||
|
item: Item |
||||
|
``` |
||||
|
|
||||
|
...and with that single declaration you get: |
||||
|
|
||||
|
* Editor support, including: |
||||
|
* Completion. |
||||
|
* Type checks. |
||||
|
* Validation of data: |
||||
|
* Automatic and clear errors when the data is invalid. |
||||
|
* Validation even for deeply nested JSON objects. |
||||
|
* <abbr title="also known as: serialization, parsing, marshalling">Conversion</abbr> of input data: coming from the network to Python data and types. Reading from: |
||||
|
* JSON. |
||||
|
* Path parameters. |
||||
|
* Query parameters. |
||||
|
* Cookies. |
||||
|
* Headers. |
||||
|
* Forms. |
||||
|
* Files. |
||||
|
* <abbr title="also known as: serialization, parsing, marshalling">Conversion</abbr> of output data: converting from Python data and types to network data (as JSON): |
||||
|
* Convert Python types (`str`, `int`, `float`, `bool`, `list`, etc). |
||||
|
* `datetime` objects. |
||||
|
* `UUID` objects. |
||||
|
* Database models. |
||||
|
* ...and many more. |
||||
|
* Automatic interactive API documentation, including 2 alternative user interfaces: |
||||
|
* Swagger UI. |
||||
|
* ReDoc. |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
Coming back to the previous code example, **FastAPI** will: |
||||
|
|
||||
|
* Validate that there is an `item_id` in the path for `GET` and `PUT` requests. |
||||
|
* Validate that the `item_id` is of type `int` for `GET` and `PUT` requests. |
||||
|
* If it is not, the client will see a useful, clear error. |
||||
|
* Check if there is an optional query parameter named `q` (as in `http://127.0.0.1:8000/items/foo?q=somequery`) for `GET` requests. |
||||
|
* As the `q` parameter is declared with `= None`, it is optional. |
||||
|
* Without the `None` it would be required (as is the body in the case with `PUT`). |
||||
|
* For `PUT` requests to `/items/{item_id}`, Read the body as JSON: |
||||
|
* Check that it has a required attribute `name` that should be a `str`. |
||||
|
* Check that it has a required attribute `price` that has to be a `float`. |
||||
|
* Check that it has an optional attribute `is_offer`, that should be a `bool`, if present. |
||||
|
* All this would also work for deeply nested JSON objects. |
||||
|
* Convert from and to JSON automatically. |
||||
|
* Document everything with OpenAPI, that can be used by: |
||||
|
* Interactive documentation systems. |
||||
|
* Automatic client code generation systems, for many languages. |
||||
|
* Provide 2 interactive documentation web interfaces directly. |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
We just scratched the surface, but you already get the idea of how it all works. |
||||
|
|
||||
|
Try changing the line with: |
||||
|
|
||||
|
```Python |
||||
|
return {"item_name": item.name, "item_id": item_id} |
||||
|
``` |
||||
|
|
||||
|
...from: |
||||
|
|
||||
|
```Python |
||||
|
... "item_name": item.name ... |
||||
|
``` |
||||
|
|
||||
|
...to: |
||||
|
|
||||
|
```Python |
||||
|
... "item_price": item.price ... |
||||
|
``` |
||||
|
|
||||
|
...and see how your editor will auto-complete the attributes and know their types: |
||||
|
|
||||
|
 |
||||
|
|
||||
|
For a more complete example including more features, see the <a href="https://fastapi.tiangolo.com/tutorial/">Tutorial - User Guide</a>. |
||||
|
|
||||
|
**Spoiler alert**: the tutorial - user guide includes: |
||||
|
|
||||
|
* Declaration of **parameters** from other different places as: **headers**, **cookies**, **form fields** and **files**. |
||||
|
* How to set **validation constraints** as `maximum_length` or `regex`. |
||||
|
* A very powerful and easy to use **<abbr title="also known as components, resources, providers, services, injectables">Dependency Injection</abbr>** system. |
||||
|
* Security and authentication, including support for **OAuth2** with **JWT tokens** and **HTTP Basic** auth. |
||||
|
* More advanced (but equally easy) techniques for declaring **deeply nested JSON models** (thanks to Pydantic). |
||||
|
* **GraphQL** integration with <a href="https://strawberry.rocks" class="external-link" target="_blank">Strawberry</a> and other libraries. |
||||
|
* Many extra features (thanks to Starlette) as: |
||||
|
* **WebSockets** |
||||
|
* extremely easy tests based on HTTPX and `pytest` |
||||
|
* **CORS** |
||||
|
* **Cookie Sessions** |
||||
|
* ...and more. |
||||
|
|
||||
|
## Performance |
||||
|
|
||||
|
Independent TechEmpower benchmarks show **FastAPI** applications running under Uvicorn as <a href="https://www.techempower.com/benchmarks/#section=test&runid=7464e520-0dc2-473d-bd34-dbdfd7e85911&hw=ph&test=query&l=zijzen-7" class="external-link" target="_blank">one of the fastest Python frameworks available</a>, only below Starlette and Uvicorn themselves (used internally by FastAPI). (*) |
||||
|
|
||||
|
To understand more about it, see the section <a href="https://fastapi.tiangolo.com/benchmarks/" class="internal-link" target="_blank">Benchmarks</a>. |
||||
|
|
||||
|
## Optional Dependencies |
||||
|
|
||||
|
Used by Pydantic: |
||||
|
|
||||
|
* <a href="https://github.com/esnme/ultrajson" target="_blank"><code>ujson</code></a> - for faster JSON <abbr title="converting the string that comes from an HTTP request into Python data">"parsing"</abbr>. |
||||
|
* <a href="https://github.com/JoshData/python-email-validator" target="_blank"><code>email_validator</code></a> - for email validation. |
||||
|
|
||||
|
Used by Starlette: |
||||
|
|
||||
|
* <a href="https://www.python-httpx.org" target="_blank"><code>httpx</code></a> - Required if you want to use the `TestClient`. |
||||
|
* <a href="https://jinja.palletsprojects.com" target="_blank"><code>jinja2</code></a> - Required if you want to use the default template configuration. |
||||
|
* <a href="https://andrew-d.github.io/python-multipart/" target="_blank"><code>python-multipart</code></a> - Required if you want to support form <abbr title="converting the string that comes from an HTTP request into Python data">"parsing"</abbr>, with `request.form()`. |
||||
|
* <a href="https://pythonhosted.org/itsdangerous/" target="_blank"><code>itsdangerous</code></a> - Required for `SessionMiddleware` support. |
||||
|
* <a href="https://pyyaml.org/wiki/PyYAMLDocumentation" target="_blank"><code>pyyaml</code></a> - Required for Starlette's `SchemaGenerator` support (you probably don't need it with FastAPI). |
||||
|
* <a href="https://github.com/esnme/ultrajson" target="_blank"><code>ujson</code></a> - Required if you want to use `UJSONResponse`. |
||||
|
|
||||
|
Used by FastAPI / Starlette: |
||||
|
|
||||
|
* <a href="https://www.uvicorn.org" target="_blank"><code>uvicorn</code></a> - for the server that loads and serves your application. |
||||
|
* <a href="https://github.com/ijl/orjson" target="_blank"><code>orjson</code></a> - 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. |
@ -0,0 +1,148 @@ |
|||||
|
site_name: FastAPI |
||||
|
site_description: FastAPI framework, high performance, easy to learn, fast to code, ready for production |
||||
|
site_url: https://fastapi.tiangolo.com/hy/ |
||||
|
theme: |
||||
|
name: material |
||||
|
custom_dir: overrides |
||||
|
palette: |
||||
|
- media: '(prefers-color-scheme: light)' |
||||
|
scheme: default |
||||
|
primary: teal |
||||
|
accent: amber |
||||
|
toggle: |
||||
|
icon: material/lightbulb |
||||
|
name: Switch to light mode |
||||
|
- media: '(prefers-color-scheme: dark)' |
||||
|
scheme: slate |
||||
|
primary: teal |
||||
|
accent: amber |
||||
|
toggle: |
||||
|
icon: material/lightbulb-outline |
||||
|
name: Switch to dark mode |
||||
|
features: |
||||
|
- search.suggest |
||||
|
- search.highlight |
||||
|
- content.tabs.link |
||||
|
icon: |
||||
|
repo: fontawesome/brands/github-alt |
||||
|
logo: https://fastapi.tiangolo.com/img/icon-white.svg |
||||
|
favicon: https://fastapi.tiangolo.com/img/favicon.png |
||||
|
language: hy |
||||
|
repo_name: tiangolo/fastapi |
||||
|
repo_url: https://github.com/tiangolo/fastapi |
||||
|
edit_uri: '' |
||||
|
plugins: |
||||
|
- search |
||||
|
- markdownextradata: |
||||
|
data: data |
||||
|
nav: |
||||
|
- FastAPI: index.md |
||||
|
- Languages: |
||||
|
- en: / |
||||
|
- az: /az/ |
||||
|
- de: /de/ |
||||
|
- es: /es/ |
||||
|
- fa: /fa/ |
||||
|
- fr: /fr/ |
||||
|
- he: /he/ |
||||
|
- hy: /hy/ |
||||
|
- id: /id/ |
||||
|
- it: /it/ |
||||
|
- ja: /ja/ |
||||
|
- ko: /ko/ |
||||
|
- nl: /nl/ |
||||
|
- pl: /pl/ |
||||
|
- pt: /pt/ |
||||
|
- ru: /ru/ |
||||
|
- sq: /sq/ |
||||
|
- sv: /sv/ |
||||
|
- tr: /tr/ |
||||
|
- uk: /uk/ |
||||
|
- zh: /zh/ |
||||
|
markdown_extensions: |
||||
|
- toc: |
||||
|
permalink: true |
||||
|
- markdown.extensions.codehilite: |
||||
|
guess_lang: false |
||||
|
- mdx_include: |
||||
|
base_path: docs |
||||
|
- admonition |
||||
|
- codehilite |
||||
|
- extra |
||||
|
- pymdownx.superfences: |
||||
|
custom_fences: |
||||
|
- name: mermaid |
||||
|
class: mermaid |
||||
|
format: !!python/name:pymdownx.superfences.fence_code_format '' |
||||
|
- pymdownx.tabbed: |
||||
|
alternate_style: true |
||||
|
- attr_list |
||||
|
- md_in_html |
||||
|
extra: |
||||
|
analytics: |
||||
|
provider: google |
||||
|
property: UA-133183413-1 |
||||
|
social: |
||||
|
- icon: fontawesome/brands/github-alt |
||||
|
link: https://github.com/tiangolo/fastapi |
||||
|
- icon: fontawesome/brands/discord |
||||
|
link: https://discord.gg/VQjSZaeJmf |
||||
|
- icon: fontawesome/brands/twitter |
||||
|
link: https://twitter.com/fastapi |
||||
|
- icon: fontawesome/brands/linkedin |
||||
|
link: https://www.linkedin.com/in/tiangolo |
||||
|
- icon: fontawesome/brands/dev |
||||
|
link: https://dev.to/tiangolo |
||||
|
- icon: fontawesome/brands/medium |
||||
|
link: https://medium.com/@tiangolo |
||||
|
- icon: fontawesome/solid/globe |
||||
|
link: https://tiangolo.com |
||||
|
alternate: |
||||
|
- link: / |
||||
|
name: en - English |
||||
|
- link: /az/ |
||||
|
name: az |
||||
|
- link: /de/ |
||||
|
name: de |
||||
|
- link: /es/ |
||||
|
name: es - español |
||||
|
- link: /fa/ |
||||
|
name: fa |
||||
|
- link: /fr/ |
||||
|
name: fr - français |
||||
|
- link: /he/ |
||||
|
name: he |
||||
|
- link: /hy/ |
||||
|
name: hy |
||||
|
- link: /id/ |
||||
|
name: id |
||||
|
- link: /it/ |
||||
|
name: it - italiano |
||||
|
- link: /ja/ |
||||
|
name: ja - 日本語 |
||||
|
- link: /ko/ |
||||
|
name: ko - 한국어 |
||||
|
- link: /nl/ |
||||
|
name: nl |
||||
|
- link: /pl/ |
||||
|
name: pl |
||||
|
- link: /pt/ |
||||
|
name: pt - português |
||||
|
- link: /ru/ |
||||
|
name: ru - русский язык |
||||
|
- link: /sq/ |
||||
|
name: sq - shqip |
||||
|
- link: /sv/ |
||||
|
name: sv - svenska |
||||
|
- link: /tr/ |
||||
|
name: tr - Türkçe |
||||
|
- link: /uk/ |
||||
|
name: uk - українська мова |
||||
|
- link: /zh/ |
||||
|
name: zh - 汉语 |
||||
|
extra_css: |
||||
|
- https://fastapi.tiangolo.com/css/termynal.css |
||||
|
- https://fastapi.tiangolo.com/css/custom.css |
||||
|
extra_javascript: |
||||
|
- https://fastapi.tiangolo.com/js/termynal.js |
||||
|
- https://fastapi.tiangolo.com/js/custom.js |
@ -0,0 +1,84 @@ |
|||||
|
# 교차 출처 리소스 공유 |
||||
|
|
||||
|
<a href="https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS" class="external-link" target="_blank">CORS 또는 "교차-출처 리소스 공유"</a>란, 브라우저에서 동작하는 프론트엔드가 자바스크립트로 코드로 백엔드와 통신하고, 백엔드는 해당 프론트엔드와 다른 "출처"에 존재하는 상황을 의미합니다. |
||||
|
|
||||
|
## 출처 |
||||
|
|
||||
|
출처란 프로토콜(`http` , `https`), 도메인(`myapp.com`, `localhost`, `localhost.tiangolo.com` ), 그리고 포트(`80`, `443`, `8080` )의 조합을 의미합니다. |
||||
|
|
||||
|
따라서, 아래는 모두 상이한 출처입니다: |
||||
|
|
||||
|
* `http://localhost` |
||||
|
* `https://localhost` |
||||
|
* `http://localhost:8080` |
||||
|
|
||||
|
모두 `localhost` 에 있지만, 서로 다른 프로토콜과 포트를 사용하고 있으므로 다른 "출처"입니다. |
||||
|
|
||||
|
## 단계 |
||||
|
|
||||
|
브라우저 내 `http://localhost:8080`에서 동작하는 프론트엔드가 있고, 자바스크립트는 `http://localhost`를 통해 백엔드와 통신한다고 가정해봅시다(포트를 명시하지 않는 경우, 브라우저는 `80` 을 기본 포트로 간주합니다). |
||||
|
|
||||
|
그러면 브라우저는 백엔드에 HTTP `OPTIONS` 요청을 보내고, 백엔드에서 이 다른 출처(`http://localhost:8080`)와의 통신을 허가하는 적절한 헤더를 보내면, 브라우저는 프론트엔드의 자바스크립트가 백엔드에 요청을 보낼 수 있도록 합니다. |
||||
|
|
||||
|
이를 위해, 백엔드는 "허용된 출처(allowed origins)" 목록을 가지고 있어야만 합니다. |
||||
|
|
||||
|
이 경우, 프론트엔드가 제대로 동작하기 위해 `http://localhost:8080`을 목록에 포함해야 합니다. |
||||
|
|
||||
|
## 와일드카드 |
||||
|
|
||||
|
모든 출처를 허용하기 위해 목록을 `"*"` ("와일드카드")로 선언하는 것도 가능합니다. |
||||
|
|
||||
|
하지만 이것은 특정한 유형의 통신만을 허용하며, 쿠키 및 액세스 토큰과 사용되는 인증 헤더(Authoriztion header) 등이 포함된 경우와 같이 자격 증명(credentials)이 포함된 통신은 허용되지 않습니다. |
||||
|
|
||||
|
따라서 모든 작업을 의도한대로 실행하기 위해, 허용되는 출처를 명시적으로 지정하는 것이 좋습니다. |
||||
|
|
||||
|
## `CORSMiddleware` 사용 |
||||
|
|
||||
|
`CORSMiddleware` 을 사용하여 **FastAPI** 응용 프로그램의 교차 출처 리소스 공유 환경을 설정할 수 있습니다. |
||||
|
|
||||
|
* `CORSMiddleware` 임포트. |
||||
|
* 허용되는 출처(문자열 형식)의 리스트 생성. |
||||
|
* FastAPI 응용 프로그램에 "미들웨어(middleware)"로 추가. |
||||
|
|
||||
|
백엔드에서 다음의 사항을 허용할지에 대해 설정할 수도 있습니다: |
||||
|
|
||||
|
* 자격증명 (인증 헤더, 쿠키 등). |
||||
|
* 특정한 HTTP 메소드(`POST`, `PUT`) 또는 와일드카드 `"*"` 를 사용한 모든 HTTP 메소드. |
||||
|
* 특정한 HTTP 헤더 또는 와일드카드 `"*"` 를 사용한 모든 HTTP 헤더. |
||||
|
|
||||
|
```Python hl_lines="2 6-11 13-19" |
||||
|
{!../../../docs_src/cors/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
`CORSMiddleware` 에서 사용하는 기본 매개변수는 제한적이므로, 브라우저가 교차-도메인 상황에서 특정한 출처, 메소드, 헤더 등을 사용할 수 있도록 하려면 이들을 명시적으로 허용해야 합니다. |
||||
|
|
||||
|
다음의 인자들이 지원됩니다: |
||||
|
|
||||
|
* `allow_origins` - 교차-출처 요청을 보낼 수 있는 출처의 리스트입니다. 예) `['https://example.org', 'https://www.example.org']`. 모든 출처를 허용하기 위해 `['*']` 를 사용할 수 있습니다. |
||||
|
* `allow_origin_regex` - 교차-출처 요청을 보낼 수 있는 출처를 정규표현식 문자열로 나타냅니다. `'https://.*\.example\.org'`. |
||||
|
* `allow_methods` - 교차-출처 요청을 허용하는 HTTP 메소드의 리스트입니다. 기본값은 `['GET']` 입니다. `['*']` 을 사용하여 모든 표준 메소드들을 허용할 수 있습니다. |
||||
|
* `allow_headers` - 교차-출처를 지원하는 HTTP 요청 헤더의 리스트입니다. 기본값은 `[]` 입니다. 모든 헤더들을 허용하기 위해 `['*']` 를 사용할 수 있습니다. `Accept`, `Accept-Language`, `Content-Language` 그리고 `Content-Type` 헤더는 CORS 요청시 언제나 허용됩니다. |
||||
|
* `allow_credentials` - 교차-출처 요청시 쿠키 지원 여부를 설정합니다. 기본값은 `False` 입니다. 또한 해당 항목을 허용할 경우 `allow_origins` 는 `['*']` 로 설정할 수 없으며, 출처를 반드시 특정해야 합니다. |
||||
|
* `expose_headers` - 브라우저에 접근할 수 있어야 하는 모든 응답 헤더를 가리킵니다. 기본값은 `[]` 입니다. |
||||
|
* `max_age` - 브라우저가 CORS 응답을 캐시에 저장하는 최대 시간을 초 단위로 설정합니다. 기본값은 `600` 입니다. |
||||
|
|
||||
|
미들웨어는 두가지 특정한 종류의 HTTP 요청에 응답합니다... |
||||
|
|
||||
|
### CORS 사전 요청 |
||||
|
|
||||
|
`Origin` 및 `Access-Control-Request-Method` 헤더와 함께 전송하는 모든 `OPTIONS` 요청입니다. |
||||
|
|
||||
|
이 경우 미들웨어는 들어오는 요청을 가로채 적절한 CORS 헤더와, 정보 제공을 위한 `200` 또는 `400` 응답으로 응답합니다. |
||||
|
|
||||
|
### 단순한 요청 |
||||
|
|
||||
|
`Origin` 헤더를 가진 모든 요청. 이 경우 미들웨어는 요청을 정상적으로 전달하지만, 적절한 CORS 헤더를 응답에 포함시킵니다. |
||||
|
|
||||
|
## 더 많은 정보 |
||||
|
|
||||
|
<abbr title="교차-출처 리소스 공유">CORS</abbr>에 대한 더 많은 정보를 알고싶다면, <a href="https://developer.mozilla.org/ko/docs/Web/HTTP/CORS" class="external-link" target="_blank">Mozilla CORS 문서</a>를 참고하기 바랍니다. |
||||
|
|
||||
|
!!! note "기술적 세부 사항" |
||||
|
`from starlette.middleware.cors import CORSMiddleware` 역시 사용할 수 있습니다. |
||||
|
|
||||
|
**FastAPI**는 개발자인 당신의 편의를 위해 `fastapi.middleware` 에서 몇가지의 미들웨어를 제공합니다. 하지만 대부분의 미들웨어가 Stralette으로부터 직접 제공됩니다. |
@ -0,0 +1,42 @@ |
|||||
|
# Codificador Compatível com JSON |
||||
|
|
||||
|
Existem alguns casos em que você pode precisar converter um tipo de dados (como um modelo Pydantic) para algo compatível com JSON (como um `dict`, `list`, etc). |
||||
|
|
||||
|
Por exemplo, se você precisar armazená-lo em um banco de dados. |
||||
|
|
||||
|
Para isso, **FastAPI** fornece uma função `jsonable_encoder()`. |
||||
|
|
||||
|
## Usando a função `jsonable_encoder` |
||||
|
|
||||
|
Vamos imaginar que você tenha um banco de dados `fake_db` que recebe apenas dados compatíveis com JSON. |
||||
|
|
||||
|
Por exemplo, ele não recebe objetos `datetime`, pois estes objetos não são compatíveis com JSON. |
||||
|
|
||||
|
Então, um objeto `datetime` teria que ser convertido em um `str` contendo os dados no formato <a href="https://en.wikipedia.org/wiki/ISO_8601" class="external-link" target="_blank">ISO</a>. |
||||
|
|
||||
|
Da mesma forma, este banco de dados não receberia um modelo Pydantic (um objeto com atributos), apenas um `dict`. |
||||
|
|
||||
|
Você pode usar a função `jsonable_encoder` para resolver isso. |
||||
|
|
||||
|
A função recebe um objeto, como um modelo Pydantic e retorna uma versão compatível com JSON: |
||||
|
|
||||
|
=== "Python 3.6 e acima" |
||||
|
|
||||
|
```Python hl_lines="5 22" |
||||
|
{!> ../../../docs_src/encoder/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
=== "Python 3.10 e acima" |
||||
|
|
||||
|
```Python hl_lines="4 21" |
||||
|
{!> ../../../docs_src/encoder/tutorial001_py310.py!} |
||||
|
``` |
||||
|
|
||||
|
Neste exemplo, ele converteria o modelo Pydantic em um `dict`, e o `datetime` em um `str`. |
||||
|
|
||||
|
O resultado de chamar a função é algo que pode ser codificado com o padrão do Python <a href="https://docs.python.org/3/library/json.html#json.dumps" class="external-link" target="_blank">`json.dumps()`</a>. |
||||
|
|
||||
|
A função não retorna um grande `str` contendo os dados no formato JSON (como uma string). Mas sim, retorna uma estrutura de dados padrão do Python (por exemplo, um `dict`) com valores e subvalores compatíveis com JSON. |
||||
|
|
||||
|
!!! nota |
||||
|
`jsonable_encoder` é realmente usado pelo **FastAPI** internamente para converter dados. Mas também é útil em muitos outros cenários. |
@ -0,0 +1,39 @@ |
|||||
|
# Arquivos Estáticos |
||||
|
|
||||
|
Você pode servir arquivos estáticos automaticamente de um diretório usando `StaticFiles`. |
||||
|
|
||||
|
## Use `StaticFiles` |
||||
|
|
||||
|
* Importe `StaticFiles`. |
||||
|
* "Monte" uma instância de `StaticFiles()` em um caminho específico. |
||||
|
|
||||
|
```Python hl_lines="2 6" |
||||
|
{!../../../docs_src/static_files/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
!!! note "Detalhes técnicos" |
||||
|
Você também pode usar `from starlette.staticfiles import StaticFiles`. |
||||
|
|
||||
|
O **FastAPI** fornece o mesmo que `starlette.staticfiles` como `fastapi.staticfiles` apenas como uma conveniência para você, o desenvolvedor. Mas na verdade vem diretamente da Starlette. |
||||
|
|
||||
|
### O que é "Montagem" |
||||
|
|
||||
|
"Montagem" significa adicionar um aplicativo completamente "independente" em uma rota específica, que então cuida de todas as subrotas. |
||||
|
|
||||
|
Isso é diferente de usar um `APIRouter`, pois um aplicativo montado é completamente independente. A OpenAPI e a documentação do seu aplicativo principal não incluirão nada do aplicativo montado, etc. |
||||
|
|
||||
|
Você pode ler mais sobre isso no **Guia Avançado do Usuário**. |
||||
|
|
||||
|
## Detalhes |
||||
|
|
||||
|
O primeiro `"/static"` refere-se à subrota em que este "subaplicativo" será "montado". Portanto, qualquer caminho que comece com `"/static"` será tratado por ele. |
||||
|
|
||||
|
O `directory="static"` refere-se ao nome do diretório que contém seus arquivos estáticos. |
||||
|
|
||||
|
O `name="static"` dá a ela um nome que pode ser usado internamente pelo FastAPI. |
||||
|
|
||||
|
Todos esses parâmetros podem ser diferentes de "`static`", ajuste-os de acordo com as necessidades e detalhes específicos de sua própria aplicação. |
||||
|
|
||||
|
## Mais informações |
||||
|
|
||||
|
Para mais detalhes e opções, verifique <a href="https://www.starlette.io/staticfiles/" class="external-link" target="_blank">Starlette's docs about Static Files</a>. |
@ -0,0 +1,469 @@ |
|||||
|
# Участие в разработке фреймворка |
||||
|
|
||||
|
Возможно, для начала Вам стоит ознакомиться с основными способами [помочь FastAPI или получить помощь](help-fastapi.md){.internal-link target=_blank}. |
||||
|
|
||||
|
## Разработка |
||||
|
|
||||
|
Если Вы уже склонировали репозиторий и знаете, что Вам нужно более глубокое погружение в код фреймворка, то здесь представлены некоторые инструкции по настройке виртуального окружения. |
||||
|
|
||||
|
### Виртуальное окружение с помощью `venv` |
||||
|
|
||||
|
Находясь в нужной директории, Вы можете создать виртуальное окружение при помощи Python модуля `venv`. |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ python -m venv env |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Эта команда создаст директорию `./env/` с бинарными (двоичными) файлами Python, а затем Вы сможете скачивать и устанавливать необходимые библиотеки в изолированное виртуальное окружение. |
||||
|
|
||||
|
### Активация виртуального окружения |
||||
|
|
||||
|
Активируйте виртуально окружение командой: |
||||
|
|
||||
|
=== "Linux, macOS" |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ source ./env/bin/activate |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
=== "Windows PowerShell" |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ .\env\Scripts\Activate.ps1 |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
=== "Windows Bash" |
||||
|
|
||||
|
Если Вы пользуетесь Bash для Windows (например: <a href="https://gitforwindows.org/" class="external-link" target="_blank">Git Bash</a>): |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ source ./env/Scripts/activate |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Проверьте, что всё сработало: |
||||
|
|
||||
|
=== "Linux, macOS, Windows Bash" |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ which pip |
||||
|
|
||||
|
some/directory/fastapi/env/bin/pip |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
=== "Windows PowerShell" |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ Get-Command pip |
||||
|
|
||||
|
some/directory/fastapi/env/bin/pip |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Ели в терминале появится ответ, что бинарник `pip` расположен по пути `.../env/bin/pip`, значит всё в порядке. 🎉 |
||||
|
|
||||
|
Во избежание ошибок в дальнейших шагах, удостоверьтесь, что в Вашем виртуальном окружении установлена последняя версия `pip`: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ python -m pip install --upgrade pip |
||||
|
|
||||
|
---> 100% |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
!!! tip "Подсказка" |
||||
|
Каждый раз, перед установкой новой библиотеки в виртуальное окружение при помощи `pip`, не забудьте активировать это виртуальное окружение. |
||||
|
|
||||
|
Это гарантирует, что если Вы используете библиотеку, установленную этим пакетом, то Вы используете библиотеку из Вашего локального окружения, а не любую другую, которая может быть установлена глобально. |
||||
|
|
||||
|
### pip |
||||
|
|
||||
|
После активации виртуального окружения, как было указано ранее, введите следующую команду: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ pip install -e ."[dev,doc,test]" |
||||
|
|
||||
|
---> 100% |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Это установит все необходимые зависимости в локальное окружение для Вашего локального FastAPI. |
||||
|
|
||||
|
#### Использование локального FastAPI |
||||
|
|
||||
|
Если Вы создаёте Python файл, который импортирует и использует FastAPI,а затем запускаете его интерпретатором Python из Вашего локального окружения, то он будет использовать код из локального FastAPI. |
||||
|
|
||||
|
И, так как при вводе вышеупомянутой команды был указан флаг `-e`, если Вы измените код локального FastAPI, то при следующем запуске этого файла, он будет использовать свежую версию локального FastAPI, который Вы только что изменили. |
||||
|
|
||||
|
Таким образом, Вам не нужно "переустанавливать" Вашу локальную версию, чтобы протестировать каждое изменение. |
||||
|
|
||||
|
### Форматировние |
||||
|
|
||||
|
Скачанный репозиторий содержит скрипт, который может отформатировать и подчистить Ваш код: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ bash scripts/format.sh |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Заодно он упорядочит Ваши импорты. |
||||
|
|
||||
|
Чтобы он сортировал их правильно, необходимо, чтобы FastAPI был установлен локально в Вашей среде, с помощью команды из раздела выше, использующей флаг `-e`. |
||||
|
|
||||
|
## Документация |
||||
|
|
||||
|
Прежде всего, убедитесь, что Вы настроили своё окружение, как описано выше, для установки всех зависимостей. |
||||
|
|
||||
|
Документация использует <a href="https://www.mkdocs.org/" class="external-link" target="_blank">MkDocs</a>. |
||||
|
|
||||
|
Также существуют дополнительные инструменты/скрипты для работы с переводами в `./scripts/docs.py`. |
||||
|
|
||||
|
!!! tip "Подсказка" |
||||
|
|
||||
|
Нет необходимости заглядывать в `./scripts/docs.py`, просто используйте это в командной строке. |
||||
|
|
||||
|
Вся документация имеет формат Markdown и расположена в директории `./docs/en/`. |
||||
|
|
||||
|
Многие руководства содержат блоки кода. |
||||
|
|
||||
|
В большинстве случаев эти блоки кода представляют собой вполне законченные приложения, которые можно запускать как есть. |
||||
|
|
||||
|
На самом деле, эти блоки кода не написаны внутри Markdown, это Python файлы в директории `./docs_src/`. |
||||
|
|
||||
|
И эти Python файлы включаются/вводятся в документацию при создании сайта. |
||||
|
|
||||
|
### Тестирование документации |
||||
|
|
||||
|
|
||||
|
Фактически, большинство тестов запускаются с примерами исходных файлов в документации. |
||||
|
|
||||
|
Это помогает убедиться, что: |
||||
|
|
||||
|
* Документация находится в актуальном состоянии. |
||||
|
* Примеры из документации могут быть запущены как есть. |
||||
|
* Большинство функций описаны в документации и покрыты тестами. |
||||
|
|
||||
|
Существует скрипт, который во время локальной разработки создаёт сайт и проверяет наличие любых изменений, перезагружая его в реальном времени: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ python ./scripts/docs.py live |
||||
|
|
||||
|
<span style="color: green;">[INFO]</span> Serving on http://127.0.0.1:8008 |
||||
|
<span style="color: green;">[INFO]</span> Start watching changes |
||||
|
<span style="color: green;">[INFO]</span> Start detecting changes |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Он запустит сайт документации по адресу: `http://127.0.0.1:8008`. |
||||
|
|
||||
|
|
||||
|
Таким образом, Вы сможете редактировать файлы с документацией или кодом и наблюдать изменения вживую. |
||||
|
|
||||
|
#### Typer CLI (опционально) |
||||
|
|
||||
|
|
||||
|
Приведенная ранее инструкция показала Вам, как запускать скрипт `./scripts/docs.py` непосредственно через интерпретатор `python` . |
||||
|
|
||||
|
Но также можно использовать <a href="https://typer.tiangolo.com/typer-cli/" class="external-link" target="_blank">Typer CLI</a>, что позволит Вам воспользоваться автозаполнением команд в Вашем терминале. |
||||
|
|
||||
|
Если Вы установили Typer CLI, то для включения функции автозаполнения, введите эту команду: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ typer --install-completion |
||||
|
|
||||
|
zsh completion installed in /home/user/.bashrc. |
||||
|
Completion will take effect once you restart the terminal. |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
### Приложения и документация одновременно |
||||
|
|
||||
|
Если Вы запускаете приложение, например так: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ uvicorn tutorial001:app --reload |
||||
|
|
||||
|
<span style="color: green;">INFO</span>: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit) |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
По умолчанию Uvicorn будет использовать порт `8000` и не будет конфликтовать с сайтом документации, использующим порт `8008`. |
||||
|
|
||||
|
### Переводы на другие языки |
||||
|
|
||||
|
Помощь с переводами ценится КРАЙНЕ ВЫСОКО! И переводы не могут быть сделаны без помощи сообщества. 🌎 🚀 |
||||
|
|
||||
|
Ниже приведены шаги, как помочь с переводами. |
||||
|
|
||||
|
#### Подсказки и инструкции |
||||
|
|
||||
|
* Проверьте <a href="https://github.com/tiangolo/fastapi/pulls" class="external-link" target="_blank">существующие пул-реквесты</a> для Вашего языка. Добавьте отзывы с просьбой внести изменения, если они необходимы, или одобрите их. |
||||
|
|
||||
|
!!! tip "Подсказка" |
||||
|
Вы можете <a href="https://help.github.com/en/github/collaborating-with-issues-and-pull-requests/commenting-on-a-pull-request" class="external-link" target="_blank">добавлять комментарии с предложениями по изменению</a> в существующие пул-реквесты. |
||||
|
|
||||
|
Ознакомьтесь с документацией о <a href="https://help.github.com/en/github/collaborating-with-issues-and-pull-requests/about-pull-request-reviews" class="external-link" target="_blank">добавлении отзыва к пул-реквесту</a>, чтобы утвердить его или запросить изменения. |
||||
|
|
||||
|
* Проверьте <a href="https://github.com/tiangolo/fastapi/issues" class="external-link" target="_blank">проблемы и вопросы</a>, чтобы узнать, есть ли кто-то, координирующий переводы для Вашего языка. |
||||
|
|
||||
|
* Добавляйте один пул-реквест для каждой отдельной переведённой страницы. Это значительно облегчит другим его просмотр. |
||||
|
|
||||
|
Для языков, которые я не знаю, прежде чем добавить перевод в основную ветку, я подожду пока несколько других участников сообщества проверят его. |
||||
|
|
||||
|
* Вы также можете проверить, есть ли переводы для Вашего языка и добавить к ним отзыв, который поможет мне убедиться в правильности перевода. Тогда я смогу объединить его с основной веткой. |
||||
|
|
||||
|
* Используйте те же самые примеры кода Python. Переводите только текст документации. Вам не нужно ничего менять, чтобы эти примеры работали. |
||||
|
|
||||
|
* Используйте те же самые изображения, имена файлов и ссылки. Вы не должны менять ничего для сохранения работоспособности. |
||||
|
|
||||
|
* Чтобы узнать 2-буквенный код языка, на который Вы хотите сделать перевод, Вы можете воспользоваться таблицей <a href="https://en.wikipedia.org/wiki/List_of_ISO_639-1_codes" class="external-link" target="_blank">Список кодов языков ISO 639-1</a>. |
||||
|
|
||||
|
#### Существующий язык |
||||
|
|
||||
|
Допустим, Вы хотите перевести страницу на язык, на котором уже есть какие-то переводы, например, на испанский. |
||||
|
|
||||
|
Кодом испанского языка является `es`. А значит директория для переводов на испанский язык: `docs/es/`. |
||||
|
|
||||
|
!!! tip "Подсказка" |
||||
|
Главный ("официальный") язык - английский, директория для него `docs/en/`. |
||||
|
|
||||
|
Вы можете запустить сервер документации на испанском: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
// Используйте команду "live" и передайте код языка в качестве аргумента командной строки |
||||
|
$ python ./scripts/docs.py live es |
||||
|
|
||||
|
<span style="color: green;">[INFO]</span> Serving on http://127.0.0.1:8008 |
||||
|
<span style="color: green;">[INFO]</span> Start watching changes |
||||
|
<span style="color: green;">[INFO]</span> Start detecting changes |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Теперь Вы можете перейти по адресу: <a href="http://127.0.0.1:8008" class="external-link" target="_blank">http://127.0.0.1:8008</a> и наблюдать вносимые Вами изменения вживую. |
||||
|
|
||||
|
|
||||
|
Если Вы посмотрите на сайт документации FastAPI, то увидите, что все страницы есть на каждом языке. Но некоторые страницы не переведены и имеют уведомление об отсутствующем переводе. |
||||
|
|
||||
|
Но когда Вы запускаете сайт локально, Вы видите только те страницы, которые уже переведены. |
||||
|
|
||||
|
|
||||
|
Предположим, что Вы хотите добавить перевод страницы [Основные свойства](features.md){.internal-link target=_blank}. |
||||
|
|
||||
|
* Скопируйте файл: |
||||
|
|
||||
|
``` |
||||
|
docs/en/docs/features.md |
||||
|
``` |
||||
|
|
||||
|
* Вставьте его точно в то же место, но в директорию языка, на который Вы хотите сделать перевод, например: |
||||
|
|
||||
|
``` |
||||
|
docs/es/docs/features.md |
||||
|
``` |
||||
|
|
||||
|
!!! tip "Подсказка" |
||||
|
Заметьте, что в пути файла мы изменили только код языка с `en` на `es`. |
||||
|
|
||||
|
* Теперь откройте файл конфигурации MkDocs для английского языка, расположенный тут: |
||||
|
|
||||
|
``` |
||||
|
docs/en/mkdocs.yml |
||||
|
``` |
||||
|
|
||||
|
* Найдите в файле конфигурации место, где расположена строка `docs/features.md`. Похожее на это: |
||||
|
|
||||
|
```YAML hl_lines="8" |
||||
|
site_name: FastAPI |
||||
|
# More stuff |
||||
|
nav: |
||||
|
- FastAPI: index.md |
||||
|
- Languages: |
||||
|
- en: / |
||||
|
- es: /es/ |
||||
|
- features.md |
||||
|
``` |
||||
|
|
||||
|
* Откройте файл конфигурации MkDocs для языка, на который Вы переводите, например: |
||||
|
|
||||
|
``` |
||||
|
docs/es/mkdocs.yml |
||||
|
``` |
||||
|
|
||||
|
* Добавьте строку `docs/features.md` точно в то же место, как и в случае для английского, как-то так: |
||||
|
|
||||
|
```YAML hl_lines="8" |
||||
|
site_name: FastAPI |
||||
|
# More stuff |
||||
|
nav: |
||||
|
- FastAPI: index.md |
||||
|
- Languages: |
||||
|
- en: / |
||||
|
- es: /es/ |
||||
|
- features.md |
||||
|
``` |
||||
|
|
||||
|
Убедитесь, что при наличии других записей, новая запись с Вашим переводом находится точно в том же порядке, что и в английской версии. |
||||
|
|
||||
|
Если Вы зайдёте в свой браузер, то увидите, что в документации стал отображаться Ваш новый раздел.🎉 |
||||
|
|
||||
|
Теперь Вы можете переводить эту страницу и смотреть, как она выглядит при сохранении файла. |
||||
|
|
||||
|
#### Новый язык |
||||
|
|
||||
|
Допустим, Вы хотите добавить перевод для языка, на который пока что не переведена ни одна страница. |
||||
|
|
||||
|
Скажем, Вы решили сделать перевод для креольского языка, но его еще нет в документации. |
||||
|
|
||||
|
Перейдите в таблицу кодов языков по ссылке указанной выше, где найдёте, что кодом креольского языка является `ht`. |
||||
|
|
||||
|
Затем запустите скрипт, генерирующий директорию для переводов на новые языки: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
// Используйте команду new-lang и передайте код языка в качестве аргумента командной строки |
||||
|
$ python ./scripts/docs.py new-lang ht |
||||
|
|
||||
|
Successfully initialized: docs/ht |
||||
|
Updating ht |
||||
|
Updating en |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
После чего Вы можете проверить в своем редакторе кода, что появился новый каталог `docs/ht/`. |
||||
|
|
||||
|
!!! tip "Подсказка" |
||||
|
Создайте первый пул-реквест, который будет содержать только пустую директорию для нового языка, прежде чем добавлять переводы. |
||||
|
|
||||
|
Таким образом, другие участники могут переводить другие страницы, пока Вы работаете над одной. 🚀 |
||||
|
|
||||
|
Начните перевод с главной страницы `docs/ht/index.md`. |
||||
|
|
||||
|
В дальнейшем можно действовать, как указано в предыдущих инструкциях для "существующего языка". |
||||
|
|
||||
|
##### Новый язык не поддерживается |
||||
|
|
||||
|
Если при запуске скрипта `./scripts/docs.py live` Вы получаете сообщение об ошибке, что язык не поддерживается, что-то вроде: |
||||
|
|
||||
|
``` |
||||
|
raise TemplateNotFound(template) |
||||
|
jinja2.exceptions.TemplateNotFound: partials/language/xx.html |
||||
|
``` |
||||
|
|
||||
|
Сие означает, что тема не поддерживает этот язык (в данном случае с поддельным 2-буквенным кодом `xx`). |
||||
|
|
||||
|
Но не стоит переживать. Вы можете установить языком темы английский, а затем перевести текст документации. |
||||
|
|
||||
|
Если возникла такая необходимость, отредактируйте `mkdocs.yml` для Вашего нового языка. Это будет выглядеть как-то так: |
||||
|
|
||||
|
```YAML hl_lines="5" |
||||
|
site_name: FastAPI |
||||
|
# More stuff |
||||
|
theme: |
||||
|
# More stuff |
||||
|
language: xx |
||||
|
``` |
||||
|
|
||||
|
Измените `xx` (код Вашего языка) на `en` и перезапустите сервер. |
||||
|
|
||||
|
#### Предпросмотр результата |
||||
|
|
||||
|
Когда Вы запускаете скрипт `./scripts/docs.py` с командой `live`, то будут показаны файлы и переводы для указанного языка. |
||||
|
|
||||
|
Но когда Вы закончите, то можете посмотреть, как это будет выглядеть по-настоящему. |
||||
|
|
||||
|
Для этого сначала создайте всю документацию: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
// Используйте команду "build-all", это займёт немного времени |
||||
|
$ python ./scripts/docs.py build-all |
||||
|
|
||||
|
Updating es |
||||
|
Updating en |
||||
|
Building docs for: en |
||||
|
Building docs for: es |
||||
|
Successfully built docs for: es |
||||
|
Copying en index.md to README.md |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Скрипт сгенерирует `./docs_build/` для каждого языка. Он добавит все файлы с отсутствующими переводами с пометкой о том, что "у этого файла еще нет перевода". Но Вам не нужно ничего делать с этим каталогом. |
||||
|
|
||||
|
Затем он создаст независимые сайты MkDocs для каждого языка, объединит их и сгенерирует конечный результат на `./site/`. |
||||
|
|
||||
|
После чего Вы сможете запустить сервер со всеми языками командой `serve`: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
// Используйте команду "serve" после того, как отработает команда "build-all" |
||||
|
$ python ./scripts/docs.py serve |
||||
|
|
||||
|
Warning: this is a very simple server. For development, use mkdocs serve instead. |
||||
|
This is here only to preview a site with translations already built. |
||||
|
Make sure you run the build-all command first. |
||||
|
Serving at: http://127.0.0.1:8008 |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
## Тесты |
||||
|
|
||||
|
Также в репозитории есть скрипт, который Вы можете запустить локально, чтобы протестировать весь код и сгенерировать отчеты о покрытии тестами в HTML: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ bash scripts/test-cov-html.sh |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Эта команда создаст директорию `./htmlcov/`, в которой будет файл `./htmlcov/index.html`. Открыв его в Вашем браузере, Вы можете в интерактивном режиме изучить, все ли части кода охвачены тестами. |
@ -0,0 +1,180 @@ |
|||||
|
|
||||
|
# Люди, поддерживающие FastAPI |
||||
|
|
||||
|
У FastAPI замечательное сообщество, которое доброжелательно к людям с любым уровнем знаний. |
||||
|
|
||||
|
## Создатель и хранитель |
||||
|
|
||||
|
Ку! 👋 |
||||
|
|
||||
|
Это я: |
||||
|
|
||||
|
{% if people %} |
||||
|
<div class="user-list user-list-center"> |
||||
|
{% for user in people.maintainers %} |
||||
|
|
||||
|
<div class="user"><a href="{{ user.url }}" target="_blank"><div class="avatar-wrapper"><img src="{{ user.avatarUrl }}"/></div><div class="title">@{{ user.login }}</div></a> <div class="count">Answers: {{ user.answers }}</div><div class="count">Pull Requests: {{ user.prs }}</div></div> |
||||
|
{% endfor %} |
||||
|
|
||||
|
</div> |
||||
|
{% endif %} |
||||
|
|
||||
|
Я создал и продолжаю поддерживать **FastAPI**. Узнать обо мне больше можно тут [Помочь FastAPI - Получить помощь - Связаться с автором](help-fastapi.md#connect-with-the-author){.internal-link target=_blank}. |
||||
|
|
||||
|
... но на этой странице я хочу показать вам наше сообщество. |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
**FastAPI** получает огромную поддержку от своего сообщества. И я хочу отметить вклад его участников. |
||||
|
|
||||
|
Это люди, которые: |
||||
|
|
||||
|
* [Помогают другим с их проблемами (вопросами) на GitHub](help-fastapi.md#help-others-with-issues-in-github){.internal-link target=_blank}. |
||||
|
* [Создают пул-реквесты](help-fastapi.md#create-a-pull-request){.internal-link target=_blank}. |
||||
|
* Делают ревью пул-реквестов, [что особенно важно для переводов на другие языки](contributing.md#translations){.internal-link target=_blank}. |
||||
|
|
||||
|
Поаплодируем им! 👏 🙇 |
||||
|
|
||||
|
## Самые активные участники за прошедший месяц |
||||
|
|
||||
|
Эти участники [оказали наибольшую помощь другим с решением их проблем (вопросов) на GitHub](help-fastapi.md#help-others-with-issues-in-github){.internal-link target=_blank} в течение последнего месяца. ☕ |
||||
|
|
||||
|
{% if people %} |
||||
|
<div class="user-list user-list-center"> |
||||
|
{% for user in people.last_month_active %} |
||||
|
|
||||
|
<div class="user"><a href="{{ user.url }}" target="_blank"><div class="avatar-wrapper"><img src="{{ user.avatarUrl }}"/></div><div class="title">@{{ user.login }}</div></a> <div class="count">Issues replied: {{ user.count }}</div></div> |
||||
|
{% endfor %} |
||||
|
|
||||
|
</div> |
||||
|
{% endif %} |
||||
|
|
||||
|
## Эксперты |
||||
|
|
||||
|
Здесь представлены **Эксперты FastAPI**. 🤓 |
||||
|
|
||||
|
Эти участники [оказали наибольшую помощь другим с решением их проблем (вопросов) на GitHub](help-fastapi.md#help-others-with-issues-in-github){.internal-link target=_blank} за *всё время*. |
||||
|
|
||||
|
Оказывая помощь многим другим, они подтвердили свой уровень знаний. ✨ |
||||
|
|
||||
|
{% if people %} |
||||
|
<div class="user-list user-list-center"> |
||||
|
{% for user in people.experts %} |
||||
|
|
||||
|
<div class="user"><a href="{{ user.url }}" target="_blank"><div class="avatar-wrapper"><img src="{{ user.avatarUrl }}"/></div><div class="title">@{{ user.login }}</div></a> <div class="count">Issues replied: {{ user.count }}</div></div> |
||||
|
{% endfor %} |
||||
|
|
||||
|
</div> |
||||
|
{% endif %} |
||||
|
|
||||
|
## Рейтинг участников, внёсших вклад в код |
||||
|
|
||||
|
Здесь представлен **Рейтинг участников, внёсших вклад в код**. 👷 |
||||
|
|
||||
|
Эти люди [сделали наибольшее количество пул-реквестов](help-fastapi.md#create-a-pull-request){.internal-link target=_blank}, *включённых в основной код*. |
||||
|
|
||||
|
Они сделали наибольший вклад в исходный код, документацию, переводы и т.п. 📦 |
||||
|
|
||||
|
{% if people %} |
||||
|
<div class="user-list user-list-center"> |
||||
|
{% for user in people.top_contributors %} |
||||
|
|
||||
|
<div class="user"><a href="{{ user.url }}" target="_blank"><div class="avatar-wrapper"><img src="{{ user.avatarUrl }}"/></div><div class="title">@{{ user.login }}</div></a> <div class="count">Pull Requests: {{ user.count }}</div></div> |
||||
|
{% endfor %} |
||||
|
|
||||
|
</div> |
||||
|
{% endif %} |
||||
|
|
||||
|
На самом деле таких людей довольно много (более сотни), вы можете увидеть всех на этой странице <a href="https://github.com/tiangolo/fastapi/graphs/contributors" class="external-link" target="_blank">FastAPI GitHub Contributors page</a>. 👷 |
||||
|
|
||||
|
## Рейтинг ревьюеров |
||||
|
|
||||
|
Здесь представлен **Рейтинг ревьюеров**. 🕵️ |
||||
|
|
||||
|
### Проверки переводов на другие языки |
||||
|
|
||||
|
Я знаю не очень много языков (и не очень хорошо 😅). |
||||
|
Итак, ревьюеры - это люди, которые могут [**подтвердить предложенный вами перевод** документации](contributing.md#translations){.internal-link target=_blank}. Без них не было бы документации на многих языках. |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
В **Рейтинге ревьюеров** 🕵️ представлены те, кто проверил наибольшее количество пул-реквестов других участников, обеспечивая качество кода, документации и, особенно, **переводов на другие языки**. |
||||
|
|
||||
|
{% if people %} |
||||
|
<div class="user-list user-list-center"> |
||||
|
{% for user in people.top_reviewers %} |
||||
|
|
||||
|
<div class="user"><a href="{{ user.url }}" target="_blank"><div class="avatar-wrapper"><img src="{{ user.avatarUrl }}"/></div><div class="title">@{{ user.login }}</div></a> <div class="count">Reviews: {{ user.count }}</div></div> |
||||
|
{% endfor %} |
||||
|
|
||||
|
</div> |
||||
|
{% endif %} |
||||
|
|
||||
|
## Спонсоры |
||||
|
|
||||
|
Здесь представлены **Спонсоры**. 😎 |
||||
|
|
||||
|
Спонсоры поддерживают мою работу над **FastAPI** (и другими проектами) главным образом через <a href="https://github.com/sponsors/tiangolo" class="external-link" target="_blank">GitHub Sponsors</a>. |
||||
|
|
||||
|
{% if sponsors %} |
||||
|
|
||||
|
{% if sponsors.gold %} |
||||
|
|
||||
|
### Золотые спонсоры |
||||
|
|
||||
|
{% for sponsor in sponsors.gold -%} |
||||
|
<a href="{{ sponsor.url }}" target="_blank" title="{{ sponsor.title }}"><img src="{{ sponsor.img }}" style="border-radius:15px"></a> |
||||
|
{% endfor %} |
||||
|
{% endif %} |
||||
|
|
||||
|
{% if sponsors.silver %} |
||||
|
|
||||
|
### Серебрянные спонсоры |
||||
|
|
||||
|
{% for sponsor in sponsors.silver -%} |
||||
|
<a href="{{ sponsor.url }}" target="_blank" title="{{ sponsor.title }}"><img src="{{ sponsor.img }}" style="border-radius:15px"></a> |
||||
|
{% endfor %} |
||||
|
{% endif %} |
||||
|
|
||||
|
{% if sponsors.bronze %} |
||||
|
|
||||
|
### Бронзовые спонсоры |
||||
|
|
||||
|
{% for sponsor in sponsors.bronze -%} |
||||
|
<a href="{{ sponsor.url }}" target="_blank" title="{{ sponsor.title }}"><img src="{{ sponsor.img }}" style="border-radius:15px"></a> |
||||
|
{% endfor %} |
||||
|
{% endif %} |
||||
|
|
||||
|
{% endif %} |
||||
|
|
||||
|
### Индивидуальные спонсоры |
||||
|
|
||||
|
{% if github_sponsors %} |
||||
|
{% for group in github_sponsors.sponsors %} |
||||
|
|
||||
|
<div class="user-list user-list-center"> |
||||
|
|
||||
|
{% for user in group %} |
||||
|
{% if user.login not in sponsors_badge.logins %} |
||||
|
|
||||
|
<div class="user"><a href="{{ user.url }}" target="_blank"><div class="avatar-wrapper"><img src="{{ user.avatarUrl }}"/></div><div class="title">@{{ user.login }}</div></a></div> |
||||
|
|
||||
|
{% endif %} |
||||
|
{% endfor %} |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
{% endfor %} |
||||
|
{% endif %} |
||||
|
|
||||
|
## О данных - технические детали |
||||
|
|
||||
|
Основная цель этой страницы - подчеркнуть усилия сообщества по оказанию помощи другим. |
||||
|
|
||||
|
Особенно это касается усилий, которые обычно менее заметны и во многих случаях более трудоемки, таких как помощь другим в решении проблем и проверка пул-реквестов с переводами. |
||||
|
|
||||
|
Данные рейтинги подсчитываются каждый месяц, ознакомиться с тем, как это работает можно <a href="https://github.com/tiangolo/fastapi/blob/master/.github/actions/people/app/main.py" class="external-link" target="_blank">тут</a>. |
||||
|
|
||||
|
Кроме того, я также подчеркиваю вклад спонсоров. |
||||
|
|
||||
|
И я оставляю за собой право обновлять алгоритмы подсчёта, виды рейтингов, пороговые значения и т.д. (так, на всякий случай 🤷). |
@ -0,0 +1,257 @@ |
|||||
|
# Помочь FastAPI - Получить помощь |
||||
|
|
||||
|
Нравится ли Вам **FastAPI**? |
||||
|
|
||||
|
Хотели бы Вы помочь FastAPI, его пользователям и автору? |
||||
|
|
||||
|
Может быть у Вас возникли трудности с **FastAPI** и Вам нужна помощь? |
||||
|
|
||||
|
Есть несколько очень простых способов оказания помощи (иногда достаточно всего лишь одного или двух кликов). |
||||
|
|
||||
|
И также есть несколько способов получить помощь. |
||||
|
|
||||
|
## Подписаться на новостную рассылку |
||||
|
|
||||
|
Вы можете подписаться на редкую [новостную рассылку **FastAPI и его друзья**](/newsletter/){.internal-link target=_blank} и быть в курсе о: |
||||
|
|
||||
|
* Новостях о FastAPI и его друзьях 🚀 |
||||
|
* Руководствах 📝 |
||||
|
* Возможностях ✨ |
||||
|
* Исправлениях 🚨 |
||||
|
* Подсказках и хитростях ✅ |
||||
|
|
||||
|
## Подписаться на FastAPI в Twitter |
||||
|
|
||||
|
<a href="https://twitter.com/fastapi" class="external-link" target="_blank">Подписаться на @fastapi в **Twitter**</a> для получения наисвежайших новостей о **FastAPI**. 🐦 |
||||
|
|
||||
|
## Добавить **FastAPI** звезду на GitHub |
||||
|
|
||||
|
Вы можете добавить FastAPI "звезду" на GitHub (кликнуть на кнопку звезды в верхнем правом углу экрана): <a href="https://github.com/tiangolo/fastapi" class="external-link" target="_blank">https://github.com/tiangolo/fastapi</a>. ⭐️ |
||||
|
|
||||
|
Чем больше звёзд, тем легче другим пользователям найти нас и увидеть, что проект уже стал полезным для многих. |
||||
|
|
||||
|
## Отслеживать свежие выпуски в репозитории на GitHub |
||||
|
|
||||
|
Вы можете "отслеживать" FastAPI на GitHub (кликните по кнопке "watch" наверху справа): <a href="https://github.com/tiangolo/fastapi" class="external-link" target="_blank">https://github.com/tiangolo/fastapi</a>. 👀 |
||||
|
|
||||
|
Там же Вы можете указать в настройках - "Releases only". |
||||
|
|
||||
|
С такой настройкой Вы будете получать уведомления на вашу электронную почту каждый раз, когда появится новый релиз (новая версия) **FastAPI** с исправлениями ошибок и новыми возможностями. |
||||
|
|
||||
|
## Связаться с автором |
||||
|
|
||||
|
Можно связаться со <a href="https://tiangolo.com" class="external-link" target="_blank">мной (Себястьян Рамирез / `tiangolo`)</a>, автором FastAPI. |
||||
|
|
||||
|
Вы можете: |
||||
|
|
||||
|
* <a href="https://github.com/tiangolo" class="external-link" target="_blank">Подписаться на меня на **GitHub**</a>. |
||||
|
* Посмотреть другие мои проекты с открытым кодом, которые могут быть полезны Вам. |
||||
|
* Подписавшись на меня Вы сможете получать уведомления, что я создал новый проект с открытым кодом,. |
||||
|
* <a href="https://twitter.com/tiangolo" class="external-link" target="_blank">Подписаться на меня в **Twitter**</a> или в <a href="https://fosstodon.org/@tiangolo" class="external-link" target="_blank">Mastodon</a>. |
||||
|
* Поделиться со мной, как Вы используете FastAPI (я обожаю читать про это). |
||||
|
* Получать уведомления, когда я делаю объявления и представляю новые инструменты. |
||||
|
* Вы также можете <a href="https://twitter.com/fastapi" class="external-link" target="_blank">подписаться на @fastapi в Twitter</a> (это отдельный аккаунт). |
||||
|
* <a href="https://www.linkedin.com/in/tiangolo/" class="external-link" target="_blank">Подписаться на меня в **Linkedin**</a>. |
||||
|
* Получать уведомления, когда я делаю объявления и представляю новые инструменты (правда чаще всего я использую Twitter 🤷♂). |
||||
|
* Читать, что я пишу (или подписаться на меня) в <a href="https://dev.to/tiangolo" class="external-link" target="_blank">**Dev.to**</a> или в <a href="https://medium.com/@tiangolo" class="external-link" target="_blank">**Medium**</a>. |
||||
|
* Читать другие идеи, статьи и читать об инструментах созданных мной. |
||||
|
* Подпишитесь на меня, чтобы прочитать, когда я опубликую что-нибудь новое. |
||||
|
|
||||
|
## Оставить сообщение в Twitter о **FastAPI** |
||||
|
|
||||
|
<a href="https://twitter.com/compose/tweet?text=I'm loving @fastapi because... https://github.com/tiangolo/fastapi" class="external-link" target="_blank">Оставьте сообщение в Twitter о **FastAPI**</a> и позвольте мне и другим узнать - почему он Вам нравится. 🎉 |
||||
|
|
||||
|
Я люблю узнавать о том, как **FastAPI** используется, что Вам понравилось в нём, в каких проектах/компаниях Вы используете его и т.п. |
||||
|
|
||||
|
## Оставить голос за FastAPI |
||||
|
|
||||
|
* <a href="https://www.slant.co/options/34241/~fastapi-review" class="external-link" target="_blank">Голосуйте за **FastAPI** в Slant</a>. |
||||
|
* <a href="https://alternativeto.net/software/fastapi/" class="external-link" target="_blank">Голосуйте за **FastAPI** в AlternativeTo</a>. |
||||
|
* <a href="https://stackshare.io/pypi-fastapi" class="external-link" target="_blank">Расскажите, как Вы используете **FastAPI** на StackShare</a>. |
||||
|
|
||||
|
## Помочь другим с их проблемами на GitHub |
||||
|
|
||||
|
Вы можете посмотреть, какие <a href="https://github.com/tiangolo/fastapi/issues" class="external-link" target="_blank">проблемы</a> испытывают другие люди и попытаться помочь им. Чаще всего это вопросы, на которые, весьма вероятно, Вы уже знаете ответ. 🤓 |
||||
|
|
||||
|
Если Вы будете много помогать людям с решением их проблем, Вы можете стать официальным [Экспертом FastAPI](fastapi-people.md#experts){.internal-link target=_blank}. 🎉 |
||||
|
|
||||
|
Только помните, самое важное при этом - доброта. Столкнувшись с проблемой, люди расстраиваются и часто задают вопросы не лучшим образом, но постарайтесь быть максимально доброжелательным. 🤗 |
||||
|
|
||||
|
Идея сообщества **FastAPI** в том, чтобы быть добродушным и гостеприимными. Не допускайте издевательств или неуважительного поведения по отношению к другим. Мы должны заботиться друг о друге. |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
Как помочь другим с их проблемами: |
||||
|
|
||||
|
### Понять вопрос |
||||
|
|
||||
|
* Удостоверьтесь, что поняли **цель** и обстоятельства случая вопрошающего. |
||||
|
|
||||
|
* Затем проверьте, что вопрос (в подавляющем большинстве - это вопросы) Вам **ясен**. |
||||
|
|
||||
|
* Во многих случаях вопрос касается решения, которое пользователь придумал сам, но может быть и решение **получше**. Если Вы поймёте проблему и обстоятельства случая, то сможете предложить **альтернативное решение**. |
||||
|
|
||||
|
* Ежели вопрос Вам непонятен, запросите больше **деталей**. |
||||
|
|
||||
|
### Воспроизвести проблему |
||||
|
|
||||
|
В большинстве случаев есть что-то связанное с **исходным кодом** вопрошающего. |
||||
|
|
||||
|
И во многих случаях будет предоставлен только фрагмент этого кода, которого недостаточно для **воспроизведения проблемы**. |
||||
|
|
||||
|
* Попросите предоставить <a href="https://stackoverflow.com/help/minimal-reproducible-example" class="external-link" target="_blank">минимальный воспроизводимый пример</a>, который можно **скопировать** и запустить локально дабы увидеть такую же ошибку, или поведение, или лучше понять обстоятельства случая. |
||||
|
|
||||
|
* Если на Вас нахлынуло великодушие, то можете попытаться **создать похожий пример** самостоятельно, основываясь только на описании проблемы. Но имейте в виду, что это может занять много времени и, возможно, стоит сначала позадавать вопросы для прояснения проблемы. |
||||
|
|
||||
|
### Предложить решение |
||||
|
|
||||
|
* После того как Вы поняли вопрос, Вы можете дать **ответ**. |
||||
|
|
||||
|
* Следует понять **основную проблему и обстоятельства случая**, потому что может быть решение лучше, чем то, которое пытались реализовать. |
||||
|
|
||||
|
### Попросить закрыть проблему |
||||
|
|
||||
|
Если Вам ответили, высоки шансы, что Вам удалось решить проблему, поздравляю, **Вы - герой**! 🦸 |
||||
|
|
||||
|
* В таком случае, если вопрос решён, попросите **закрыть проблему**. |
||||
|
|
||||
|
## Отслеживать репозиторий на GitHub |
||||
|
|
||||
|
Вы можете "отслеживать" FastAPI на GitHub (кликните по кнопке "watch" наверху справа): <a href="https://github.com/tiangolo/fastapi" class="external-link" target="_blank">https://github.com/tiangolo/fastapi</a>. 👀 |
||||
|
|
||||
|
Если Вы выберете "Watching" вместо "Releases only", то будете получать уведомления когда кто-либо попросит о помощи с решением его проблемы. |
||||
|
|
||||
|
Тогда Вы можете попробовать решить эту проблему. |
||||
|
|
||||
|
## Запросить помощь с решением проблемы |
||||
|
|
||||
|
Вы можете <a href="https://github.com/tiangolo/fastapi/issues/new/choose" class="external-link" target="_blank">создать новый запрос с просьбой о помощи</a> в репозитории на GitHub, например: |
||||
|
|
||||
|
* Задать **вопрос** или попросить помощи в решении **проблемы**. |
||||
|
* Предложить новое **улучшение**. |
||||
|
|
||||
|
**Заметка**: Если Вы создаёте подобные запросы, то я попрошу Вас также оказывать аналогичную помощь другим. 😉 |
||||
|
|
||||
|
## Проверять пул-реквесты |
||||
|
|
||||
|
Вы можете помочь мне проверять пул-реквесты других участников. |
||||
|
|
||||
|
И повторюсь, постарайтесь быть доброжелательным. 🤗 |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
О том, что нужно иметь в виду при проверке пул-реквестов: |
||||
|
|
||||
|
### Понять проблему |
||||
|
|
||||
|
* Во-первых, убедитесь, что **поняли проблему**, которую пул-реквест пытается решить. Для этого может потребоваться продолжительное обсуждение. |
||||
|
|
||||
|
* Также есть вероятность, что пул-реквест не актуален, так как проблему можно решить **другим путём**. В таком случае Вы можете указать на этот факт. |
||||
|
|
||||
|
### Не переживайте о стиле |
||||
|
|
||||
|
* Не стоит слишком беспокоиться о таких вещах, как стиль сообщений в коммитах или количество коммитов. При слиянии пул-реквеста с основной веткой, я буду сжимать и настраивать всё вручную. |
||||
|
|
||||
|
* Также не беспокойтесь о правилах стиля, для проверки сего есть автоматизированные инструменты. |
||||
|
|
||||
|
И если всё же потребуется какой-то другой стиль, я попрошу Вас об этом напрямую или добавлю сам коммиты с необходимыми изменениями. |
||||
|
|
||||
|
### Проверить код |
||||
|
|
||||
|
* Проверьте и прочитайте код, посмотрите, какой он имеет смысл, **запустите его локально** и посмотрите, действительно ли он решает поставленную задачу. |
||||
|
|
||||
|
* Затем, используя **комментарий**, сообщите, что Вы сделали проверку, тогда я буду знать, что Вы действительно проверили код. |
||||
|
|
||||
|
!!! Информация |
||||
|
К сожалению, я не могу так просто доверять пул-реквестам, у которых уже есть несколько одобрений. |
||||
|
|
||||
|
Бывали случаи, что пул-реквесты имели 3, 5 или больше одобрений, вероятно из-за привлекательного описания, но когда я проверял эти пул-реквесты, они оказывались сломаны, содержали ошибки или вовсе не решали проблему, которую, как они утверждали, должны были решить. 😅 |
||||
|
|
||||
|
Потому это действительно важно - проверять и запускать код, и комментарием уведомлять меня, что Вы проделали эти действия. 🤓 |
||||
|
|
||||
|
* Если Вы считаете, что пул-реквест можно упростить, то можете попросить об этом, но не нужно быть слишком придирчивым, может быть много субъективных точек зрения (и у меня тоже будет своя 🙈), поэтому будет лучше, если Вы сосредоточитесь на фундаментальных вещах. |
||||
|
|
||||
|
### Тестировать |
||||
|
|
||||
|
* Помогите мне проверить, что у пул-реквеста есть **тесты**. |
||||
|
|
||||
|
* Проверьте, что тесты **падали** до пул-реквеста. 🚨 |
||||
|
|
||||
|
* Затем проверьте, что тесты **не валятся** после пул-реквеста. ✅ |
||||
|
|
||||
|
* Многие пул-реквесты не имеют тестов, Вы можете **напомнить** о необходимости добавления тестов или даже **предложить** какие-либо свои тесты. Это одна из тех вещей, которые отнимают много времени и Вы можете помочь с этим. |
||||
|
|
||||
|
* Затем добавьте комментарий, что Вы испробовали в ходе проверки. Таким образом я буду знать, как Вы произвели проверку. 🤓 |
||||
|
|
||||
|
## Создать пул-реквест |
||||
|
|
||||
|
Вы можете [сделать вклад](contributing.md){.internal-link target=_blank} в код фреймворка используя пул-реквесты, например: |
||||
|
|
||||
|
* Исправить опечатку, которую Вы нашли в документации. |
||||
|
* Поделиться статьёй, видео или подкастом о FastAPI, которые Вы создали или нашли <a href="https://github.com/tiangolo/fastapi/edit/master/docs/en/data/external_links.yml" class="external-link" target="_blank">изменив этот файл</a>. |
||||
|
* Убедитесь, что Вы добавили свою ссылку в начало соответствующего раздела. |
||||
|
* Помочь с [переводом документации](contributing.md#translations){.internal-link target=_blank} на Ваш язык. |
||||
|
* Вы также можете проверять переводы сделанные другими. |
||||
|
* Предложить новые разделы документации. |
||||
|
* Исправить существующуе проблемы/баги. |
||||
|
* Убедитесь, что добавили тесты. |
||||
|
* Добавить новую возможность. |
||||
|
* Убедитесь, что добавили тесты. |
||||
|
* Убедитесь, что добавили документацию, если она необходима. |
||||
|
|
||||
|
## Помочь поддерживать FastAPI |
||||
|
|
||||
|
Помогите мне поддерживать **FastAPI**! 🤓 |
||||
|
|
||||
|
Предстоит ещё много работы и, по большей части, **ВЫ** можете её сделать. |
||||
|
|
||||
|
Основные задачи, которые Вы можете выполнить прямо сейчас: |
||||
|
|
||||
|
* [Помочь другим с их проблемами на GitHub](#help-others-with-issues-in-github){.internal-link target=_blank} (смотрите вышестоящую секцию). |
||||
|
* [Проверить пул-реквесты](#review-pull-requests){.internal-link target=_blank} (смотрите вышестоящую секцию). |
||||
|
|
||||
|
Эти две задачи **отнимают больше всего времени**. Это основная работа по поддержке FastAPI. |
||||
|
|
||||
|
Если Вы можете помочь мне с этим, **Вы помогаете поддерживать FastAPI** и следить за тем, чтобы он продолжал **развиваться быстрее и лучше**. 🚀 |
||||
|
|
||||
|
## Подключиться к чату |
||||
|
|
||||
|
Подключайтесь к 👥 <a href="https://discord.gg/VQjSZaeJmf" class="external-link" target="_blank"> чату в Discord</a> 👥 и общайтесь с другими участниками сообщества FastAPI. |
||||
|
|
||||
|
!!! Подсказка |
||||
|
Вопросы по проблемам с фреймворком лучше задавать в <a href="https://github.com/tiangolo/fastapi/issues/new/choose" class="external-link" target="_blank">GitHub issues</a>, так больше шансов, что Вы получите помощь от [Экспертов FastAPI](fastapi-people.md#experts){.internal-link target=_blank}. |
||||
|
|
||||
|
Используйте этот чат только для бесед на отвлечённые темы. |
||||
|
|
||||
|
Существует также <a href="https://gitter.im/tiangolo/fastapi" class="external-link" target="_blank">чат в Gitter</a>, но поскольку в нем нет каналов и расширенных функций, общение в нём сложнее, потому рекомендуемой системой является Discord. |
||||
|
|
||||
|
### Не использовать чаты для вопросов |
||||
|
|
||||
|
Имейте в виду, что чаты позволяют больше "свободного общения", потому там легко задавать вопросы, которые слишком общие и на которые труднее ответить, так что Вы можете не получить нужные Вам ответы. |
||||
|
|
||||
|
В разделе "проблемы" на GitHub, есть шаблон, который поможет Вам написать вопрос правильно, чтобы Вам было легче получить хороший ответ или даже решить проблему самостоятельно, прежде чем Вы зададите вопрос. В GitHub я могу быть уверен, что всегда отвечаю на всё, даже если это займет какое-то время. И я не могу сделать то же самое в чатах. 😅 |
||||
|
|
||||
|
Кроме того, общение в чатах не так легкодоступно для поиска, как в GitHub, потому вопросы и ответы могут потеряться среди другого общения. И только проблемы решаемые на GitHub учитываются в получении лычки [Эксперт FastAPI](fastapi-people.md#experts){.internal-link target=_blank}, так что весьма вероятно, что Вы получите больше внимания на GitHub. |
||||
|
|
||||
|
С другой стороны, в чатах тысячи пользователей, а значит есть большие шансы в любое время найти там кого-то, с кем можно поговорить. 😄 |
||||
|
|
||||
|
## Спонсировать автора |
||||
|
|
||||
|
Вы также можете оказать мне финансовую поддержку посредством <a href="https://github.com/sponsors/tiangolo" class="external-link" target="_blank">спонсорства через GitHub</a>. |
||||
|
|
||||
|
Там можно просто купить мне кофе ☕️ в знак благодарности. 😄 |
||||
|
|
||||
|
А ещё Вы можете стать Серебряным или Золотым спонсором для FastAPI. 🏅🎉 |
||||
|
|
||||
|
## Спонсировать инструменты, на которых зиждется мощь FastAPI |
||||
|
|
||||
|
Как Вы могли заметить в документации, FastAPI опирается на плечи титанов: Starlette и Pydantic. |
||||
|
|
||||
|
Им тоже можно оказать спонсорскую поддержку: |
||||
|
|
||||
|
* <a href="https://github.com/sponsors/samuelcolvin" class="external-link" target="_blank">Samuel Colvin (Pydantic)</a> |
||||
|
* <a href="https://github.com/sponsors/encode" class="external-link" target="_blank">Encode (Starlette, Uvicorn)</a> |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
Благодарствую! 🚀 |
@ -0,0 +1,77 @@ |
|||||
|
# История создания и дальнейшее развитие |
||||
|
|
||||
|
Однажды, <a href="https://github.com/tiangolo/fastapi/issues/3#issuecomment-454956920" class="external-link" target="_blank">один из пользователей **FastAPI** задал вопрос</a>: |
||||
|
|
||||
|
> Какова история этого проекта? Создаётся впечатление, что он явился из ниоткуда и завоевал мир за несколько недель [...] |
||||
|
|
||||
|
Что ж, вот небольшая часть истории проекта. |
||||
|
|
||||
|
## Альтернативы |
||||
|
|
||||
|
В течение нескольких лет я, возглавляя различные команды разработчиков, создавал довольно сложные API для машинного обучения, распределённых систем, асинхронных задач, баз данных NoSQL и т.д. |
||||
|
|
||||
|
В рамках работы над этими проектами я исследовал, проверял и использовал многие фреймворки. |
||||
|
|
||||
|
Во многом история **FastAPI** - история его предшественников. |
||||
|
|
||||
|
Как написано в разделе [Альтернативы](alternatives.md){.internal-link target=_blank}: |
||||
|
|
||||
|
<blockquote markdown="1"> |
||||
|
|
||||
|
**FastAPI** не существовал бы, если б не было более ранних работ других людей. |
||||
|
|
||||
|
Они создали большое количество инструментов, которые и вдохновили меня на создание **FastAPI**. |
||||
|
|
||||
|
Я всячески избегал создания нового фреймворка в течение нескольких лет. Сначала я пытался собрать все нужные возможности, которые ныне есть в **FastAPI**, используя множество различных фреймворков, плагинов и инструментов. |
||||
|
|
||||
|
Но в какой-то момент не осталось другого выбора, кроме как создать что-то, что предоставляло бы все эти возможности сразу. Взять самые лучшие идеи из предыдущих инструментов и, используя введённые в Python подсказки типов (которых не было до версии 3.6), объединить их. |
||||
|
|
||||
|
</blockquote> |
||||
|
|
||||
|
## Исследования |
||||
|
|
||||
|
Благодаря опыту использования существующих альтернатив, мы с коллегами изучили их основные идеи и скомбинировали собранные знания наилучшим образом. |
||||
|
|
||||
|
Например, стало ясно, что необходимо брать за основу стандартные подсказки типов Python, а самым лучшим подходом является использование уже существующих стандартов. |
||||
|
|
||||
|
Итак, прежде чем приступить к написанию **FastAPI**, я потратил несколько месяцев на изучение OpenAPI, JSON Schema, OAuth2, и т.п. для понимания их взаимосвязей, совпадений и различий. |
||||
|
|
||||
|
## Дизайн |
||||
|
|
||||
|
Затем я потратил некоторое время на придумывание "API" разработчика, который я хотел иметь как пользователь (как разработчик, использующий FastAPI). |
||||
|
|
||||
|
Я проверил несколько идей на самых популярных редакторах кода среди Python-разработчиков: PyCharm, VS Code, Jedi. |
||||
|
|
||||
|
Данные по редакторам я взял из <a href="https://www.jetbrains.com/research/python-developers-survey-2018/#development-tools" class="external-link" target="_blank">опроса Python-разработчиков</a>, который охватываает около 80% пользователей. |
||||
|
|
||||
|
Это означает, что **FastAPI** был специально проверен на редакторах, используемых 80% Python-разработчиками. И поскольку большинство других редакторов, как правило, работают аналогичным образом, все его преимущества должны работать практически для всех редакторов. |
||||
|
|
||||
|
Таким образом, я смог найти наилучшие способы сократить дублирование кода, обеспечить повсеместное автодополнение, проверку типов и ошибок и т.д. |
||||
|
|
||||
|
И все это, чтобы все пользователи могли получать наилучший опыт разработки. |
||||
|
|
||||
|
## Зависимости |
||||
|
|
||||
|
Протестировав несколько вариантов, я решил, что в качестве основы буду использовать <a href="https://pydantic-docs.helpmanual.io/" class="external-link" target="_blank">**Pydantic**</a> и его преимущества. |
||||
|
|
||||
|
По моим предложениям был изменён код этого фреймворка, чтобы сделать его полностью совместимым с JSON Schema, поддержать различные способы определения ограничений и улучшить помощь редакторов (проверки типов, автозаполнение). |
||||
|
|
||||
|
В то же время, я принимал участие в разработке <a href="https://www.starlette.io/" class="external-link" target="_blank">**Starlette**</a>, ещё один из основных компонентов FastAPI. |
||||
|
|
||||
|
## Разработка |
||||
|
|
||||
|
К тому времени, когда я начал создавать **FastAPI**, большинство необходимых деталей уже существовало, дизайн был определён, зависимости и прочие инструменты были готовы, а знания о стандартах и спецификациях были четкими и свежими. |
||||
|
|
||||
|
## Будущее |
||||
|
|
||||
|
Сейчас уже ясно, что **FastAPI** со своими идеями стал полезен многим людям. |
||||
|
|
||||
|
При сравнении с альтернативами, выбор падает на него, поскольку он лучше подходит для множества вариантов использования. |
||||
|
|
||||
|
Многие разработчики и команды уже используют **FastAPI** в своих проектах (включая меня и мою команду). |
||||
|
|
||||
|
Но, тем не менее, грядёт добавление ещё многих улучшений и возможностей. |
||||
|
|
||||
|
У **FastAPI** великое будущее. |
||||
|
|
||||
|
И [ваш вклад в это](help-fastapi.md){.internal-link target=_blank} - очень ценнен. |
@ -0,0 +1,69 @@ |
|||||
|
# Body - Поля |
||||
|
|
||||
|
Таким же способом, как вы объявляете дополнительную валидацию и метаданные в параметрах *функции обработки пути* с помощью функций `Query`, `Path` и `Body`, вы можете объявлять валидацию и метаданные внутри Pydantic моделей, используя функцию `Field` из Pydantic. |
||||
|
|
||||
|
## Импорт `Field` |
||||
|
|
||||
|
Сначала вы должны импортировать его: |
||||
|
|
||||
|
=== "Python 3.6 и выше" |
||||
|
|
||||
|
```Python hl_lines="4" |
||||
|
{!> ../../../docs_src/body_fields/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
=== "Python 3.10 и выше" |
||||
|
|
||||
|
```Python hl_lines="2" |
||||
|
{!> ../../../docs_src/body_fields/tutorial001_py310.py!} |
||||
|
``` |
||||
|
|
||||
|
!!! warning "Внимание" |
||||
|
Обратите внимание, что функция `Field` импортируется непосредственно из `pydantic`, а не из `fastapi`, как все остальные функции (`Query`, `Path`, `Body` и т.д.). |
||||
|
|
||||
|
## Объявление атрибутов модели |
||||
|
|
||||
|
Вы можете использовать функцию `Field` с атрибутами модели: |
||||
|
|
||||
|
=== "Python 3.6 и выше" |
||||
|
|
||||
|
```Python hl_lines="11-14" |
||||
|
{!> ../../../docs_src/body_fields/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
=== "Python 3.10 и выше" |
||||
|
|
||||
|
```Python hl_lines="9-12" |
||||
|
{!> ../../../docs_src/body_fields/tutorial001_py310.py!} |
||||
|
``` |
||||
|
|
||||
|
Функция `Field` работает так же, как `Query`, `Path` и `Body`, у ее такие же параметры и т.д. |
||||
|
|
||||
|
!!! note "Технические детали" |
||||
|
На самом деле, `Query`, `Path` и другие функции, которые вы увидите в дальнейшем, создают объекты подклассов общего класса `Param`, который сам по себе является подклассом `FieldInfo` из Pydantic. |
||||
|
|
||||
|
И `Field` (из Pydantic), и `Body`, оба возвращают объекты подкласса `FieldInfo`. |
||||
|
|
||||
|
У класса `Body` есть и другие подклассы, с которыми вы ознакомитесь позже. |
||||
|
|
||||
|
Помните, что когда вы импортируете `Query`, `Path` и другое из `fastapi`, это фактически функции, которые возвращают специальные классы. |
||||
|
|
||||
|
!!! tip "Подсказка" |
||||
|
Обратите внимание, что каждый атрибут модели с типом, значением по умолчанию и `Field` имеет ту же структуру, что и параметр *функции обработки пути* с `Field` вместо `Path`, `Query` и `Body`. |
||||
|
|
||||
|
## Добавление дополнительной информации |
||||
|
|
||||
|
Вы можете объявлять дополнительную информацию в `Field`, `Query`, `Body` и т.п. Она будет включена в сгенерированную JSON схему. |
||||
|
|
||||
|
Вы узнаете больше о добавлении дополнительной информации позже в документации, когда будете изучать, как задавать примеры принимаемых данных. |
||||
|
|
||||
|
|
||||
|
!!! warning "Внимание" |
||||
|
Дополнительные ключи, переданные в функцию `Field`, также будут присутствовать в сгенерированной OpenAPI схеме вашего приложения. |
||||
|
Поскольку эти ключи не являются обязательной частью спецификации OpenAPI, некоторые инструменты OpenAPI, например, [валидатор OpenAPI](https://validator.swagger.io/), могут не работать с вашей сгенерированной схемой. |
||||
|
|
||||
|
## Резюме |
||||
|
|
||||
|
Вы можете использовать функцию `Field` из Pydantic, чтобы задавать дополнительную валидацию и метаданные для атрибутов модели. |
||||
|
|
||||
|
Вы также можете использовать дополнительные ключевые аргументы, чтобы добавить метаданные JSON схемы. |
@ -0,0 +1,49 @@ |
|||||
|
# Параметры Cookie |
||||
|
|
||||
|
Вы можете задать параметры Cookie таким же способом, как `Query` и `Path` параметры. |
||||
|
|
||||
|
## Импорт `Cookie` |
||||
|
|
||||
|
Сначала импортируйте `Cookie`: |
||||
|
|
||||
|
=== "Python 3.6 и выше" |
||||
|
|
||||
|
```Python hl_lines="3" |
||||
|
{!> ../../../docs_src/cookie_params/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
=== "Python 3.10 и выше" |
||||
|
|
||||
|
```Python hl_lines="1" |
||||
|
{!> ../../../docs_src/cookie_params/tutorial001_py310.py!} |
||||
|
``` |
||||
|
|
||||
|
## Объявление параметров `Cookie` |
||||
|
|
||||
|
Затем объявляйте параметры cookie, используя ту же структуру, что и с `Path` и `Query`. |
||||
|
|
||||
|
Первое значение - это значение по умолчанию, вы можете передать все дополнительные параметры проверки или аннотации: |
||||
|
|
||||
|
=== "Python 3.6 и выше" |
||||
|
|
||||
|
```Python hl_lines="9" |
||||
|
{!> ../../../docs_src/cookie_params/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
=== "Python 3.10 и выше" |
||||
|
|
||||
|
```Python hl_lines="7" |
||||
|
{!> ../../../docs_src/cookie_params/tutorial001_py310.py!} |
||||
|
``` |
||||
|
|
||||
|
!!! note "Технические детали" |
||||
|
`Cookie` - это класс, родственный `Path` и `Query`. Он также наследуется от общего класса `Param`. |
||||
|
|
||||
|
Но помните, что когда вы импортируете `Query`, `Path`, `Cookie` и другое из `fastapi`, это фактически функции, которые возвращают специальные классы. |
||||
|
|
||||
|
!!! info "Дополнительная информация" |
||||
|
Для объявления cookies, вам нужно использовать `Cookie`, иначе параметры будут интерпретированы как параметры запроса. |
||||
|
|
||||
|
## Резюме |
||||
|
|
||||
|
Объявляйте cookies с помощью `Cookie`, используя тот же общий шаблон, что и `Query`, и `Path`. |
@ -0,0 +1,148 @@ |
|||||
|
site_name: FastAPI |
||||
|
site_description: FastAPI framework, high performance, easy to learn, fast to code, ready for production |
||||
|
site_url: https://fastapi.tiangolo.com/ta/ |
||||
|
theme: |
||||
|
name: material |
||||
|
custom_dir: overrides |
||||
|
palette: |
||||
|
- media: '(prefers-color-scheme: light)' |
||||
|
scheme: default |
||||
|
primary: teal |
||||
|
accent: amber |
||||
|
toggle: |
||||
|
icon: material/lightbulb |
||||
|
name: Switch to light mode |
||||
|
- media: '(prefers-color-scheme: dark)' |
||||
|
scheme: slate |
||||
|
primary: teal |
||||
|
accent: amber |
||||
|
toggle: |
||||
|
icon: material/lightbulb-outline |
||||
|
name: Switch to dark mode |
||||
|
features: |
||||
|
- search.suggest |
||||
|
- search.highlight |
||||
|
- content.tabs.link |
||||
|
icon: |
||||
|
repo: fontawesome/brands/github-alt |
||||
|
logo: https://fastapi.tiangolo.com/img/icon-white.svg |
||||
|
favicon: https://fastapi.tiangolo.com/img/favicon.png |
||||
|
language: en |
||||
|
repo_name: tiangolo/fastapi |
||||
|
repo_url: https://github.com/tiangolo/fastapi |
||||
|
edit_uri: '' |
||||
|
plugins: |
||||
|
- search |
||||
|
- markdownextradata: |
||||
|
data: data |
||||
|
nav: |
||||
|
- FastAPI: index.md |
||||
|
- Languages: |
||||
|
- en: / |
||||
|
- az: /az/ |
||||
|
- de: /de/ |
||||
|
- es: /es/ |
||||
|
- fa: /fa/ |
||||
|
- fr: /fr/ |
||||
|
- he: /he/ |
||||
|
- id: /id/ |
||||
|
- it: /it/ |
||||
|
- ja: /ja/ |
||||
|
- ko: /ko/ |
||||
|
- nl: /nl/ |
||||
|
- pl: /pl/ |
||||
|
- pt: /pt/ |
||||
|
- ru: /ru/ |
||||
|
- sq: /sq/ |
||||
|
- sv: /sv/ |
||||
|
- ta: /ta/ |
||||
|
- tr: /tr/ |
||||
|
- uk: /uk/ |
||||
|
- zh: /zh/ |
||||
|
markdown_extensions: |
||||
|
- toc: |
||||
|
permalink: true |
||||
|
- markdown.extensions.codehilite: |
||||
|
guess_lang: false |
||||
|
- mdx_include: |
||||
|
base_path: docs |
||||
|
- admonition |
||||
|
- codehilite |
||||
|
- extra |
||||
|
- pymdownx.superfences: |
||||
|
custom_fences: |
||||
|
- name: mermaid |
||||
|
class: mermaid |
||||
|
format: !!python/name:pymdownx.superfences.fence_code_format '' |
||||
|
- pymdownx.tabbed: |
||||
|
alternate_style: true |
||||
|
- attr_list |
||||
|
- md_in_html |
||||
|
extra: |
||||
|
analytics: |
||||
|
provider: google |
||||
|
property: UA-133183413-1 |
||||
|
social: |
||||
|
- icon: fontawesome/brands/github-alt |
||||
|
link: https://github.com/tiangolo/fastapi |
||||
|
- icon: fontawesome/brands/discord |
||||
|
link: https://discord.gg/VQjSZaeJmf |
||||
|
- icon: fontawesome/brands/twitter |
||||
|
link: https://twitter.com/fastapi |
||||
|
- icon: fontawesome/brands/linkedin |
||||
|
link: https://www.linkedin.com/in/tiangolo |
||||
|
- icon: fontawesome/brands/dev |
||||
|
link: https://dev.to/tiangolo |
||||
|
- icon: fontawesome/brands/medium |
||||
|
link: https://medium.com/@tiangolo |
||||
|
- icon: fontawesome/solid/globe |
||||
|
link: https://tiangolo.com |
||||
|
alternate: |
||||
|
- link: / |
||||
|
name: en - English |
||||
|
- link: /az/ |
||||
|
name: az |
||||
|
- link: /de/ |
||||
|
name: de |
||||
|
- link: /es/ |
||||
|
name: es - español |
||||
|
- link: /fa/ |
||||
|
name: fa |
||||
|
- link: /fr/ |
||||
|
name: fr - français |
||||
|
- link: /he/ |
||||
|
name: he |
||||
|
- link: /id/ |
||||
|
name: id |
||||
|
- link: /it/ |
||||
|
name: it - italiano |
||||
|
- link: /ja/ |
||||
|
name: ja - 日本語 |
||||
|
- link: /ko/ |
||||
|
name: ko - 한국어 |
||||
|
- link: /nl/ |
||||
|
name: nl |
||||
|
- link: /pl/ |
||||
|
name: pl |
||||
|
- link: /pt/ |
||||
|
name: pt - português |
||||
|
- link: /ru/ |
||||
|
name: ru - русский язык |
||||
|
- link: /sq/ |
||||
|
name: sq - shqip |
||||
|
- link: /sv/ |
||||
|
name: sv - svenska |
||||
|
- link: /ta/ |
||||
|
name: ta - தமிழ் |
||||
|
- link: /tr/ |
||||
|
name: tr - Türkçe |
||||
|
- link: /uk/ |
||||
|
name: uk - українська мова |
||||
|
- link: /zh/ |
||||
|
name: zh - 汉语 |
||||
|
extra_css: |
||||
|
- https://fastapi.tiangolo.com/css/termynal.css |
||||
|
- https://fastapi.tiangolo.com/css/custom.css |
||||
|
extra_javascript: |
||||
|
- https://fastapi.tiangolo.com/js/termynal.js |
||||
|
- https://fastapi.tiangolo.com/js/custom.js |
@ -0,0 +1,336 @@ |
|||||
|
# İlk Adımlar |
||||
|
|
||||
|
En basit FastAPI dosyası şu şekildedir: |
||||
|
|
||||
|
```Python |
||||
|
{!../../../docs_src/first_steps/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
Bunu bir `main.py` dosyasına kopyalayın. |
||||
|
|
||||
|
Projeyi çalıştırın: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ uvicorn main:app --reload |
||||
|
|
||||
|
<span style="color: green;">INFO</span>: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit) |
||||
|
<span style="color: green;">INFO</span>: Started reloader process [28720] |
||||
|
<span style="color: green;">INFO</span>: Started server process [28722] |
||||
|
<span style="color: green;">INFO</span>: Waiting for application startup. |
||||
|
<span style="color: green;">INFO</span>: Application startup complete. |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
!!! note |
||||
|
`uvicorn main:app` komutu şunu ifade eder: |
||||
|
|
||||
|
* `main`: `main.py` dosyası (the Python "module"). |
||||
|
* `app`: `main.py` dosyası içerisinde `app = FastAPI()` satırıyla oluşturulan nesne. |
||||
|
* `--reload`: Kod değişikliği sonrasında sunucunun yeniden başlatılmasını sağlar. Yalnızca geliştirme için kullanın. |
||||
|
|
||||
|
Çıktıda şu şekilde bir satır vardır: |
||||
|
|
||||
|
```hl_lines="4" |
||||
|
INFO: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit) |
||||
|
``` |
||||
|
|
||||
|
Bu satır, yerel makinenizde uygulamanızın sunulduğu URL'yi gösterir. |
||||
|
|
||||
|
### Kontrol Et |
||||
|
|
||||
|
Tarayıcınızda <a href="http://127.0.0.1:8000" class="external-link" target="_blank">http://127.0.0.1:8000</a> adresini açın. |
||||
|
|
||||
|
Bir JSON yanıtı göreceksiniz: |
||||
|
|
||||
|
```JSON |
||||
|
{"message": "Hello World"} |
||||
|
``` |
||||
|
|
||||
|
### İnteraktif API dokümantasyonu |
||||
|
|
||||
|
<a href="http://127.0.0.1:8000/docs" class="external-link" target="_blank">http://127.0.0.1:8000/docs</a> adresine gidin. |
||||
|
|
||||
|
Otomatik oluşturulmuş( <a href="https://github.com/swagger-api/swagger-ui" class="external-link" target="_blank">Swagger UI</a> tarafından sağlanan) interaktif bir API dokümanı göreceksiniz: |
||||
|
|
||||
|
 |
||||
|
|
||||
|
### Alternatif API dokümantasyonu |
||||
|
|
||||
|
Şimdi, <a href="http://127.0.0.1:8000/redoc" class="external-link" target="_blank">http://127.0.0.1:8000/redoc</a> adresine gidin. |
||||
|
|
||||
|
Otomatik oluşturulmuş(<a href="https://github.com/Rebilly/ReDoc" class="external-link" target="_blank">ReDoc</a> tarafından sağlanan) bir API dokümanı göreceksiniz: |
||||
|
|
||||
|
 |
||||
|
|
||||
|
### OpenAPI |
||||
|
|
||||
|
**FastAPI**, **OpenAPI** standardını kullanarak tüm API'lerinizi açıklayan bir "şema" oluşturur. |
||||
|
|
||||
|
#### "Şema" |
||||
|
|
||||
|
Bir "şema", bir şeyin tanımı veya açıklamasıdır. Soyut bir açıklamadır, uygulayan kod değildir. |
||||
|
|
||||
|
#### API "şemaları" |
||||
|
|
||||
|
Bu durumda, <a href="https://github.com/OAI/OpenAPI-Specification" class="external-link" target="_blank">OpenAPI</a>, API şemasını nasıl tanımlayacağınızı belirten şartnamelerdir. |
||||
|
|
||||
|
Bu şema tanımı, API yollarınızı, aldıkları olası parametreleri vb. içerir. |
||||
|
|
||||
|
#### Data "şema" |
||||
|
|
||||
|
"Şema" terimi, JSON içeriği gibi bazı verilerin şeklini de ifade edebilir. |
||||
|
|
||||
|
Bu durumda, JSON öznitelikleri ve sahip oldukları veri türleri vb. anlamına gelir. |
||||
|
|
||||
|
#### OpenAPI and JSON Şema |
||||
|
|
||||
|
OpenAPI, API'niz için bir API şeması tanımlar. Ve bu şema, JSON veri şemaları standardı olan **JSON Şema** kullanılarak API'niz tarafından gönderilen ve alınan verilerin tanımlarını (veya "şemalarını") içerir. |
||||
|
|
||||
|
#### `openapi.json` kontrol et |
||||
|
|
||||
|
OpenAPI şemasının nasıl göründüğünü merak ediyorsanız, FastAPI otomatik olarak tüm API'nizin açıklamalarını içeren bir JSON (şema) oluşturur. |
||||
|
|
||||
|
Doğrudan şu adreste görebilirsiniz: <a href="http://127.0.0.1:8000/openapi.json" class="external-link" target="_blank">http://127.0.0.1:8000/openapi.json</a>. |
||||
|
|
||||
|
Aşağıdaki gibi bir şeyle başlayan bir JSON gösterecektir: |
||||
|
|
||||
|
```JSON |
||||
|
{ |
||||
|
"openapi": "3.0.2", |
||||
|
"info": { |
||||
|
"title": "FastAPI", |
||||
|
"version": "0.1.0" |
||||
|
}, |
||||
|
"paths": { |
||||
|
"/items/": { |
||||
|
"get": { |
||||
|
"responses": { |
||||
|
"200": { |
||||
|
"description": "Successful Response", |
||||
|
"content": { |
||||
|
"application/json": { |
||||
|
|
||||
|
|
||||
|
|
||||
|
... |
||||
|
``` |
||||
|
|
||||
|
#### OpenAPI ne içindir? |
||||
|
|
||||
|
OpenAPI şeması, dahili olarak bulunan iki etkileşimli dokümantasyon sistemine güç veren şeydir. |
||||
|
|
||||
|
Ve tamamen OpenAPI'ye dayalı düzinelerce alternatif vardır. **FastAPI** ile oluşturulmuş uygulamanıza bu alternatiflerden herhangi birini kolayca ekleyebilirsiniz. |
||||
|
|
||||
|
API'nizle iletişim kuran istemciler için otomatik olarak kod oluşturmak için de kullanabilirsiniz. Örneğin, frontend, mobil veya IoT uygulamaları. |
||||
|
|
||||
|
## Adım adım özet |
||||
|
|
||||
|
### Adım 1: `FastAPI`yi içe aktarın |
||||
|
|
||||
|
```Python hl_lines="1" |
||||
|
{!../../../docs_src/first_steps/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
`FastAPI`, API'niz için tüm fonksiyonları sağlayan bir Python sınıfıdır. |
||||
|
|
||||
|
!!! note "Teknik Detaylar" |
||||
|
`FastAPI` doğrudan `Starlette` kalıtım alan bir sınıftır. |
||||
|
|
||||
|
Tüm <a href="https://www.starlette.io/" class="external-link" target="_blank">Starlette</a> fonksiyonlarını `FastAPI` ile de kullanabilirsiniz. |
||||
|
|
||||
|
### Adım 2: Bir `FastAPI` örneği oluşturun |
||||
|
|
||||
|
```Python hl_lines="3" |
||||
|
{!../../../docs_src/first_steps/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
Burada `app` değişkeni `FastAPI` sınıfının bir örneği olacaktır. |
||||
|
|
||||
|
Bu tüm API'yi oluşturmak için ana etkileşim noktası olacaktır. |
||||
|
|
||||
|
`uvicorn` komutunda atıfta bulunulan `app` ile aynıdır. |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ uvicorn main:app --reload |
||||
|
|
||||
|
<span style="color: green;">INFO</span>: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit) |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
Uygulamanızı aşağıdaki gibi oluşturursanız: |
||||
|
|
||||
|
```Python hl_lines="3" |
||||
|
{!../../../docs_src/first_steps/tutorial002.py!} |
||||
|
``` |
||||
|
|
||||
|
Ve bunu `main.py` dosyasına koyduktan sonra `uvicorn` komutunu şu şekilde çağırabilirsiniz: |
||||
|
|
||||
|
<div class="termy"> |
||||
|
|
||||
|
```console |
||||
|
$ uvicorn main:my_awesome_api --reload |
||||
|
|
||||
|
<span style="color: green;">INFO</span>: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit) |
||||
|
``` |
||||
|
|
||||
|
</div> |
||||
|
|
||||
|
### Adım 3: *Path işlemleri* oluşturmak |
||||
|
|
||||
|
#### Path |
||||
|
|
||||
|
Burada "Path" URL'de ilk "\" ile başlayan son bölümü ifade eder. |
||||
|
|
||||
|
Yani, şu şekilde bir URL'de: |
||||
|
|
||||
|
``` |
||||
|
https://example.com/items/foo |
||||
|
``` |
||||
|
|
||||
|
... path şöyle olabilir: |
||||
|
|
||||
|
``` |
||||
|
/items/foo |
||||
|
``` |
||||
|
|
||||
|
!!! info |
||||
|
Genellikle bir "path", "endpoint" veya "route" olarak adlandırılabilir. |
||||
|
|
||||
|
Bir API oluştururken, "path", "resource" ile "concern" ayırmanın ana yoludur. |
||||
|
|
||||
|
#### İşlemler |
||||
|
|
||||
|
Burada "işlem" HTTP methodlarından birini ifade eder. |
||||
|
|
||||
|
Onlardan biri: |
||||
|
|
||||
|
* `POST` |
||||
|
* `GET` |
||||
|
* `PUT` |
||||
|
* `DELETE` |
||||
|
|
||||
|
... ve daha egzotik olanları: |
||||
|
|
||||
|
* `OPTIONS` |
||||
|
* `HEAD` |
||||
|
* `PATCH` |
||||
|
* `TRACE` |
||||
|
|
||||
|
HTTP protokolünde, bu "methodlardan" birini (veya daha fazlasını) kullanarak her path ile iletişim kurabilirsiniz. |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
API'lerinizi oluştururkan, belirli bir işlemi gerçekleştirirken belirli HTTP methodlarını kullanırsınız. |
||||
|
|
||||
|
Normalde kullanılan: |
||||
|
|
||||
|
* `POST`: veri oluşturmak. |
||||
|
* `GET`: veri okumak. |
||||
|
* `PUT`: veriyi güncellemek. |
||||
|
* `DELETE`: veriyi silmek. |
||||
|
|
||||
|
Bu nedenle, OpenAPI'de HTTP methodlarından her birine "işlem" denir. |
||||
|
|
||||
|
Bizde onlara "**işlemler**" diyeceğiz. |
||||
|
|
||||
|
#### Bir *Path işlem decoratorleri* tanımlanmak |
||||
|
|
||||
|
```Python hl_lines="6" |
||||
|
{!../../../docs_src/first_steps/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
`@app.get("/")` **FastAPI'ye** aşağıdaki fonksiyonun adresine giden istekleri işlemekten sorumlu olduğunu söyler: |
||||
|
|
||||
|
* path `/` |
||||
|
* <abbr title="an HTTP GET method"><code>get</code> işlemi</abbr> kullanılarak |
||||
|
|
||||
|
|
||||
|
!!! info "`@decorator` Bilgisi" |
||||
|
Python `@something` şeklinde ifadeleri "decorator" olarak adlandırır. |
||||
|
|
||||
|
Decoratoru bir fonksiyonun üzerine koyarsınız. Dekoratif bir şapka gibi (Sanırım terim buradan gelmektedir). |
||||
|
|
||||
|
Bir "decorator" fonksiyonu alır ve bazı işlemler gerçekleştir. |
||||
|
|
||||
|
Bizim durumumzda decarator **FastAPI'ye** fonksiyonun bir `get` işlemi ile `/` pathine geldiğini söyler. |
||||
|
|
||||
|
Bu **path işlem decoratordür** |
||||
|
|
||||
|
Ayrıca diğer işlemleri de kullanabilirsiniz: |
||||
|
|
||||
|
* `@app.post()` |
||||
|
* `@app.put()` |
||||
|
* `@app.delete()` |
||||
|
|
||||
|
Ve daha egzotik olanları: |
||||
|
|
||||
|
* `@app.options()` |
||||
|
* `@app.head()` |
||||
|
* `@app.patch()` |
||||
|
* `@app.trace()` |
||||
|
|
||||
|
!!! tip |
||||
|
Her işlemi (HTTP method) istediğiniz gibi kullanmakta özgürsünüz. |
||||
|
|
||||
|
**FastAPI** herhangi bir özel anlamı zorlamaz. |
||||
|
|
||||
|
Buradaki bilgiler bir gereklilik değil, bir kılavuz olarak sunulmaktadır. |
||||
|
|
||||
|
Örneğin, GraphQL kullanırkan normalde tüm işlemleri yalnızca `POST` işlemini kullanarak gerçekleştirirsiniz. |
||||
|
|
||||
|
### Adım 4: **path işlem fonksiyonunu** tanımlayın |
||||
|
|
||||
|
Aşağıdakiler bizim **path işlem fonksiyonlarımızdır**: |
||||
|
|
||||
|
* **path**: `/` |
||||
|
* **işlem**: `get` |
||||
|
* **function**: "decorator"ün altındaki fonksiyondur (`@app.get("/")` altında). |
||||
|
|
||||
|
```Python hl_lines="7" |
||||
|
{!../../../docs_src/first_steps/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
Bu bir Python fonksiyonudur. |
||||
|
|
||||
|
Bir `GET` işlemi kullanarak "`/`" URL'sine bir istek geldiğinde **FastAPI** tarafından çağrılır. |
||||
|
|
||||
|
Bu durumda bir `async` fonksiyonudur. |
||||
|
|
||||
|
--- |
||||
|
|
||||
|
Bunu `async def` yerine normal bir fonksiyon olarakta tanımlayabilirsiniz. |
||||
|
|
||||
|
```Python hl_lines="7" |
||||
|
{!../../../docs_src/first_steps/tutorial003.py!} |
||||
|
``` |
||||
|
|
||||
|
!!! note |
||||
|
|
||||
|
Eğer farkı bilmiyorsanız, [Async: *"Acelesi var?"*](../async.md#in-a-hurry){.internal-link target=_blank} kontrol edebilirsiniz. |
||||
|
|
||||
|
### Adım 5: İçeriği geri döndürün |
||||
|
|
||||
|
|
||||
|
```Python hl_lines="8" |
||||
|
{!../../../docs_src/first_steps/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
Bir `dict`, `list` döndürebilir veya `str`, `int` gibi tekil değerler döndürebilirsiniz. |
||||
|
|
||||
|
Ayrıca, Pydantic modellerini de döndürebilirsiniz. (Bununla ilgili daha sonra ayrıntılı bilgi göreceksiniz.) |
||||
|
|
||||
|
Otomatik olarak JSON'a dönüştürülecek(ORM'ler vb. dahil) başka birçok nesne ve model vardır. En beğendiklerinizi kullanmayı deneyin, yüksek ihtimalle destekleniyordur. |
||||
|
|
||||
|
## Özet |
||||
|
|
||||
|
* `FastAPI`'yi içe aktarın. |
||||
|
* Bir `app` örneği oluşturun. |
||||
|
* **path işlem decorator** yazın. (`@app.get("/")` gibi) |
||||
|
* **path işlem fonksiyonu** yazın. (`def root(): ...` gibi) |
||||
|
* Development sunucunuzu çalıştırın. (`uvicorn main:app --reload` gibi) |
@ -0,0 +1,28 @@ |
|||||
|
from contextlib import asynccontextmanager |
||||
|
|
||||
|
from fastapi import FastAPI |
||||
|
|
||||
|
|
||||
|
def fake_answer_to_everything_ml_model(x: float): |
||||
|
return x * 42 |
||||
|
|
||||
|
|
||||
|
ml_models = {} |
||||
|
|
||||
|
|
||||
|
@asynccontextmanager |
||||
|
async def lifespan(app: FastAPI): |
||||
|
# Load the ML model |
||||
|
ml_models["answer_to_everything"] = fake_answer_to_everything_ml_model |
||||
|
yield |
||||
|
# Clean up the ML models and release the resources |
||||
|
ml_models.clear() |
||||
|
|
||||
|
|
||||
|
app = FastAPI(lifespan=lifespan) |
||||
|
|
||||
|
|
||||
|
@app.get("/predict") |
||||
|
async def predict(x: float): |
||||
|
result = ml_models["answer_to_everything"](x) |
||||
|
return {"result": result} |
@ -0,0 +1,27 @@ |
|||||
|
from typing import List, Union |
||||
|
|
||||
|
from fastapi import FastAPI |
||||
|
from pydantic import BaseModel |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
class Item(BaseModel): |
||||
|
name: str |
||||
|
description: Union[str, None] = None |
||||
|
price: float |
||||
|
tax: Union[float, None] = None |
||||
|
tags: List[str] = [] |
||||
|
|
||||
|
|
||||
|
@app.post("/items/") |
||||
|
async def create_item(item: Item) -> Item: |
||||
|
return item |
||||
|
|
||||
|
|
||||
|
@app.get("/items/") |
||||
|
async def read_items() -> List[Item]: |
||||
|
return [ |
||||
|
Item(name="Portal Gun", price=42.0), |
||||
|
Item(name="Plumbus", price=32.0), |
||||
|
] |
@ -0,0 +1,25 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from pydantic import BaseModel |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
class Item(BaseModel): |
||||
|
name: str |
||||
|
description: str | None = None |
||||
|
price: float |
||||
|
tax: float | None = None |
||||
|
tags: list[str] = [] |
||||
|
|
||||
|
|
||||
|
@app.post("/items/") |
||||
|
async def create_item(item: Item) -> Item: |
||||
|
return item |
||||
|
|
||||
|
|
||||
|
@app.get("/items/") |
||||
|
async def read_items() -> list[Item]: |
||||
|
return [ |
||||
|
Item(name="Portal Gun", price=42.0), |
||||
|
Item(name="Plumbus", price=32.0), |
||||
|
] |
@ -0,0 +1,27 @@ |
|||||
|
from typing import Union |
||||
|
|
||||
|
from fastapi import FastAPI |
||||
|
from pydantic import BaseModel |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
class Item(BaseModel): |
||||
|
name: str |
||||
|
description: Union[str, None] = None |
||||
|
price: float |
||||
|
tax: Union[float, None] = None |
||||
|
tags: list[str] = [] |
||||
|
|
||||
|
|
||||
|
@app.post("/items/") |
||||
|
async def create_item(item: Item) -> Item: |
||||
|
return item |
||||
|
|
||||
|
|
||||
|
@app.get("/items/") |
||||
|
async def read_items() -> list[Item]: |
||||
|
return [ |
||||
|
Item(name="Portal Gun", price=42.0), |
||||
|
Item(name="Plumbus", price=32.0), |
||||
|
] |
@ -0,0 +1,21 @@ |
|||||
|
from typing import Union |
||||
|
|
||||
|
from fastapi import FastAPI |
||||
|
from pydantic import BaseModel, EmailStr |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
class BaseUser(BaseModel): |
||||
|
username: str |
||||
|
email: EmailStr |
||||
|
full_name: Union[str, None] = None |
||||
|
|
||||
|
|
||||
|
class UserIn(BaseUser): |
||||
|
password: str |
||||
|
|
||||
|
|
||||
|
@app.post("/user/") |
||||
|
async def create_user(user: UserIn) -> BaseUser: |
||||
|
return user |
@ -0,0 +1,19 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from pydantic import BaseModel, EmailStr |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
class BaseUser(BaseModel): |
||||
|
username: str |
||||
|
email: EmailStr |
||||
|
full_name: str | None = None |
||||
|
|
||||
|
|
||||
|
class UserIn(BaseUser): |
||||
|
password: str |
||||
|
|
||||
|
|
||||
|
@app.post("/user/") |
||||
|
async def create_user(user: UserIn) -> BaseUser: |
||||
|
return user |
@ -0,0 +1,11 @@ |
|||||
|
from fastapi import FastAPI, Response |
||||
|
from fastapi.responses import JSONResponse, RedirectResponse |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/portal") |
||||
|
async def get_portal(teleport: bool = False) -> Response: |
||||
|
if teleport: |
||||
|
return RedirectResponse(url="https://www.youtube.com/watch?v=dQw4w9WgXcQ") |
||||
|
return JSONResponse(content={"message": "Here's your interdimensional portal."}) |
@ -0,0 +1,9 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from fastapi.responses import RedirectResponse |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/teleport") |
||||
|
async def get_teleport() -> RedirectResponse: |
||||
|
return RedirectResponse(url="https://www.youtube.com/watch?v=dQw4w9WgXcQ") |
@ -0,0 +1,13 @@ |
|||||
|
from typing import Union |
||||
|
|
||||
|
from fastapi import FastAPI, Response |
||||
|
from fastapi.responses import RedirectResponse |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/portal") |
||||
|
async def get_portal(teleport: bool = False) -> Union[Response, dict]: |
||||
|
if teleport: |
||||
|
return RedirectResponse(url="https://www.youtube.com/watch?v=dQw4w9WgXcQ") |
||||
|
return {"message": "Here's your interdimensional portal."} |
@ -0,0 +1,11 @@ |
|||||
|
from fastapi import FastAPI, Response |
||||
|
from fastapi.responses import RedirectResponse |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/portal") |
||||
|
async def get_portal(teleport: bool = False) -> Response | dict: |
||||
|
if teleport: |
||||
|
return RedirectResponse(url="https://www.youtube.com/watch?v=dQw4w9WgXcQ") |
||||
|
return {"message": "Here's your interdimensional portal."} |
@ -0,0 +1,13 @@ |
|||||
|
from typing import Union |
||||
|
|
||||
|
from fastapi import FastAPI, Response |
||||
|
from fastapi.responses import RedirectResponse |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/portal", response_model=None) |
||||
|
async def get_portal(teleport: bool = False) -> Union[Response, dict]: |
||||
|
if teleport: |
||||
|
return RedirectResponse(url="https://www.youtube.com/watch?v=dQw4w9WgXcQ") |
||||
|
return {"message": "Here's your interdimensional portal."} |
@ -0,0 +1,11 @@ |
|||||
|
from fastapi import FastAPI, Response |
||||
|
from fastapi.responses import RedirectResponse |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/portal", response_model=None) |
||||
|
async def get_portal(teleport: bool = False) -> Response | dict: |
||||
|
if teleport: |
||||
|
return RedirectResponse(url="https://www.youtube.com/watch?v=dQw4w9WgXcQ") |
||||
|
return {"message": "Here's your interdimensional portal."} |
Some files were not shown because too many files changed in this diff
Loading…
Reference in new issue