FastAPI¶
FastAPI framework, alto rendimiento, fácil de aprender, rápido de programar, listo para producción
Documentación: https://fastapi.tiangolo.com
Código Fuente: https://github.com/fastapi/fastapi
FastAPI es un framework web moderno, rápido (de alto rendimiento), para construir APIs con Python basado en las anotaciones de tipos estándar de Python.
Las características clave son:
- Rápido: Muy alto rendimiento, a la par con NodeJS y Go (gracias a Starlette y Pydantic). Uno de los frameworks Python más rápidos disponibles.
- Rápido de programar: Aumenta la velocidad para desarrollar funcionalidades en aproximadamente un 200% a 300%. *
- Menos bugs: Reduce en aproximadamente un 40% los errores inducidos por humanos (desarrolladores). *
- Intuitivo: Gran soporte para editores. Autocompletado en todas partes. Menos tiempo depurando.
- Fácil: Diseñado para ser fácil de usar y aprender. Menos tiempo leyendo documentación.
- Corto: Minimiza la duplicación de código. Múltiples funcionalidades desde cada declaración de parámetro. Menos bugs.
- Robusto: Obtén código listo para producción. Con documentación interactiva automática.
- Basado en estándares: Basado (y completamente compatible) con los estándares abiertos para APIs: OpenAPI (anteriormente conocido como Swagger) y JSON Schema.
* estimación basada en pruebas con un equipo de desarrollo interno, construyendo aplicaciones de producción.
Sponsors¶
Opiniones¶
"[...] Estoy usando FastAPI un montón estos días. [...] De hecho, estoy planeando usarlo para todos los servicios de ML de mi equipo en Microsoft. Algunos de ellos se están integrando en el núcleo del producto Windows y algunos productos de Office."
"Adoptamos el paquete FastAPI para crear un servidor REST que pueda ser consultado para obtener predicciones. [para Ludwig]"
"Netflix se complace en anunciar el lanzamiento de código abierto de nuestro framework de orquestación de gestión de crisis: Dispatch! [construido con FastAPI]"
"Estoy súper emocionado con FastAPI. ¡Es tan divertido!"
"Honestamente, lo que has construido parece súper sólido y pulido. En muchos aspectos, es lo que quería que Hug fuera; es realmente inspirador ver a alguien construir eso."
"Si estás buscando aprender un framework moderno para construir APIs REST, échale un vistazo a FastAPI [...] Es rápido, fácil de usar y fácil de aprender [...]"
"Nos hemos cambiado a FastAPI para nuestras APIs [...] Creo que te gustará [...]"
"Si alguien está buscando construir una API de Python para producción, altamente recomendaría FastAPI. Está hermosamente diseñado, es simple de usar y altamente escalable, se ha convertido en un componente clave en nuestra estrategia de desarrollo API primero y está impulsando muchas automatizaciones y servicios como nuestro Ingeniero Virtual TAC."
Typer, el FastAPI de las CLIs¶
Si estás construyendo una aplicación de CLI para ser usada en el terminal en lugar de una API web, revisa Typer.
Typer es el hermano pequeño de FastAPI. Y está destinado a ser el FastAPI de las CLIs. ⌨️ 🚀
Requisitos¶
FastAPI se apoya en hombros de gigantes:
Instalación¶
Crea y activa un entorno virtual y luego instala FastAPI:
$ pip install "fastapi[standard]"
---> 100%
Nota: Asegúrate de poner "fastapi[standard]"
entre comillas para asegurar que funcione en todas las terminales.
Ejemplo¶
Créalo¶
- Crea un archivo
main.py
con:
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}
O usa async def
...
Si tu código usa async
/ await
, usa async def
:
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}
Nota:
Si no lo sabes, revisa la sección "¿Con prisa?" sobre async
y await
en la documentación.
Córrelo¶
Corre el servidor con:
$ fastapi dev main.py
╭────────── FastAPI CLI - Development mode ───────────╮
│ │
│ Serving at: http://127.0.0.1:8000 │
│ │
│ API docs: http://127.0.0.1:8000/docs │
│ │
│ Running in development mode, for production use: │
│ │
│ fastapi run │
│ │
╰─────────────────────────────────────────────────────╯
INFO: Will watch for changes in these directories: ['/home/user/code/awesomeapp']
INFO: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit)
INFO: Started reloader process [2248755] using WatchFiles
INFO: Started server process [2248757]
INFO: Waiting for application startup.
INFO: Application startup complete.
Acerca del comando fastapi dev main.py
...
El comando fastapi dev
lee tu archivo main.py
, detecta la app FastAPI en él y arranca un servidor usando Uvicorn.
Por defecto, fastapi dev
comenzará con auto-recarga habilitada para el desarrollo local.
Puedes leer más sobre esto en la documentación del CLI de FastAPI.
Revísalo¶
Abre tu navegador en http://127.0.0.1:8000/items/5?q=somequery.
Verás el response JSON como:
{"item_id": 5, "q": "somequery"}
Ya creaste una API que:
- Recibe requests HTTP en los paths
/
y/items/{item_id}
. - Ambos paths toman operaciones
GET
(también conocidas como métodos HTTP). - El path
/items/{item_id}
tiene un parámetro de pathitem_id
que debe ser unint
. - El path
/items/{item_id}
tiene un parámetro de queryq
opcional que es unstr
.
Documentación interactiva de la API¶
Ahora ve a http://127.0.0.1:8000/docs.
Verás la documentación interactiva automática de la API (proporcionada por Swagger UI):
Documentación de API Alternativa¶
Y ahora, ve a http://127.0.0.1:8000/redoc.
Verás la documentación alternativa automática (proporcionada por ReDoc):
Actualización del Ejemplo¶
Ahora modifica el archivo main.py
para recibir un body desde un request PUT
.
Declara el body usando tipos estándar de Python, gracias a Pydantic.
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}
El servidor fastapi dev
debería recargarse automáticamente.
Actualización de la Documentación Interactiva de la API¶
Ahora ve a http://127.0.0.1:8000/docs.
- La documentación interactiva de la API se actualizará automáticamente, incluyendo el nuevo body:
- Haz clic en el botón "Try it out", te permite llenar los parámetros e interactuar directamente con la API:
- Luego haz clic en el botón "Execute", la interfaz de usuario se comunicará con tu API, enviará los parámetros, obtendrá los resultados y los mostrará en la pantalla:
Actualización de la Documentación Alternativa de la API¶
Y ahora, ve a http://127.0.0.1:8000/redoc.
- La documentación alternativa también reflejará el nuevo parámetro de query y body:
Resumen¶
En resumen, declaras una vez los tipos de parámetros, body, etc. como parámetros de función.
Lo haces con tipos estándar modernos de Python.
No tienes que aprender una nueva sintaxis, los métodos o clases de un paquete específico, etc.
Solo Python estándar.
Por ejemplo, para un int
:
item_id: int
o para un modelo Item
más complejo:
item: Item
...y con esa única declaración obtienes:
- Soporte para editores, incluyendo:
- Autocompletado.
- Chequeo de tipos.
- Validación de datos:
- Errores automáticos y claros cuando los datos son inválidos.
- Validación incluso para objetos JSON profundamente anidados.
- Conversión de datos de entrada: de la red a los datos y tipos de Python. Leyendo desde:
- JSON.
- Parámetros de path.
- Parámetros de query.
- Cookies.
- Headers.
- Forms.
- Archivos.
- Conversión de datos de salida: convirtiendo de datos y tipos de Python a datos de red (como JSON):
- Convertir tipos de Python (
str
,int
,float
,bool
,list
, etc). - Objetos
datetime
. - Objetos
UUID
. - Modelos de base de datos.
- ...y muchos más.
- Convertir tipos de Python (
- Documentación interactiva automática de la API, incluyendo 2 interfaces de usuario alternativas:
- Swagger UI.
- ReDoc.
Volviendo al ejemplo de código anterior, FastAPI:
- Validará que haya un
item_id
en el path para requestsGET
yPUT
. - Validará que el
item_id
sea del tipoint
para requestsGET
yPUT
.- Si no lo es, el cliente verá un error útil y claro.
- Comprobará si hay un parámetro de query opcional llamado
q
(como enhttp://127.0.0.1:8000/items/foo?q=somequery
) para requestsGET
.- Como el parámetro
q
está declarado con= None
, es opcional. - Sin el
None
sería requerido (como lo es el body en el caso conPUT
).
- Como el parámetro
- Para requests
PUT
a/items/{item_id}
, leerá el body como JSON:- Comprobará que tiene un atributo requerido
name
que debe ser unstr
. - Comprobará que tiene un atributo requerido
price
que debe ser unfloat
. - Comprobará que tiene un atributo opcional
is_offer
, que debe ser unbool
, si está presente. - Todo esto también funcionaría para objetos JSON profundamente anidados.
- Comprobará que tiene un atributo requerido
- Convertirá de y a JSON automáticamente.
- Documentará todo con OpenAPI, que puede ser usado por:
- Sistemas de documentación interactiva.
- Sistemas de generación automática de código cliente, para muchos lenguajes.
- Proporcionará 2 interfaces web de documentación interactiva directamente.
Solo tocamos los conceptos básicos, pero ya te haces una idea de cómo funciona todo.
Intenta cambiar la línea con:
return {"item_name": item.name, "item_id": item_id}
...desde:
... "item_name": item.name ...
...a:
... "item_price": item.price ...
...y observa cómo tu editor autocompleta los atributos y conoce sus tipos:
Para un ejemplo más completo incluyendo más funcionalidades, ve al Tutorial - Guía del Usuario.
Alerta de spoilers: el tutorial - guía del usuario incluye:
- Declaración de parámetros desde otros lugares diferentes como: headers, cookies, campos de formulario y archivos.
- Cómo establecer restricciones de validación como
maximum_length
oregex
. - Un sistema de Inyección de Dependencias muy poderoso y fácil de usar.
- Seguridad y autenticación, incluyendo soporte para OAuth2 con tokens JWT y autenticación HTTP Basic.
- Técnicas más avanzadas (pero igualmente fáciles) para declarar modelos JSON profundamente anidados (gracias a Pydantic).
- Integración con GraphQL usando Strawberry y otros paquetes.
- Muchas funcionalidades extra (gracias a Starlette) como:
- WebSockets
- pruebas extremadamente fáciles basadas en HTTPX y
pytest
- CORS
- Sesiones de Cookies
- ...y más.
Rendimiento¶
Benchmarks independientes de TechEmpower muestran aplicaciones FastAPI ejecutándose bajo Uvicorn como uno de los frameworks Python más rápidos disponibles, solo por debajo de Starlette y Uvicorn (usados internamente por FastAPI). (*)
Para entender más sobre esto, ve la sección Benchmarks.
Dependencias¶
FastAPI depende de Pydantic y Starlette.
Dependencias standard
¶
Cuando instalas FastAPI con pip install "fastapi[standard]"
viene con el grupo standard
de dependencias opcionales:
Usadas por Pydantic:
email-validator
- para validación de correos electrónicos.
Usadas por Starlette:
httpx
- Requerido si deseas usar elTestClient
.jinja2
- Requerido si deseas usar la configuración de plantilla predeterminada.python-multipart
- Requerido si deseas soportar "parsing" de forms, conrequest.form()
.
Usadas por FastAPI / Starlette:
uvicorn
- para el servidor que carga y sirve tu aplicación. Esto incluyeuvicorn[standard]
, que incluye algunas dependencias (por ejemplo,uvloop
) necesarias para servir con alto rendimiento.fastapi-cli
- para proporcionar el comandofastapi
.
Sin Dependencias standard
¶
Si no deseas incluir las dependencias opcionales standard
, puedes instalar con pip install fastapi
en lugar de pip install "fastapi[standard]"
.
Dependencias Opcionales Adicionales¶
Existen algunas dependencias adicionales que podrías querer instalar.
Dependencias opcionales adicionales de Pydantic:
pydantic-settings
- para la gestión de configuraciones.pydantic-extra-types
- para tipos extra para ser usados con Pydantic.
Dependencias opcionales adicionales de FastAPI:
Licencia¶
Este proyecto tiene licencia bajo los términos de la licencia MIT.