DEV Community

Cover image for Python FastAPI: ¿el mejor framework de Python?
Eduardo Zepeda
Eduardo Zepeda

Posted on • Edited on • Originally published at coffeebytes.dev

Python FastAPI: ¿el mejor framework de Python?

Estos últimos días he estado probando una librería para Python que se está volviendo muy popular, FastAPI, un framework para crear APIs. FastAPI promete ayudarnos a crear APIs rápidas de manera sencilla y con muy poco código y con un rendimiento extraordinario, para soportar una alta carga de peticiones web.

FastAPI vs Django vs Flask vs Pyramid

FastAPI queda en primer lugar en respuestas por segundo frente a Frameworks más populares como Django, Pyramid o Flask. Y también queda en los primeros lugares si lo comparamos con Frameworks de otros lenguajes de programación, como PHP o Javascript .

Mira estas comparaciones que usan información de Techempower . He resaltado en azul los frameworks de Python. El número indica la cantidad de respuestas por segundo, mientras más alto mejor.

Alt Text

Número de respuestas por segundo para peticiones que devuelven una fila de la base de datos.

Información tomada de https://www.techempower.com/benchmarks

Alt Text

Número de respuestas por segundo para peticiones que devuelven veinte filas de la base de datos.

Información tomada de https://www.techempower.com/benchmarks

Tipado y asincrónismo en Python

FastAPI es totalmente compatible con el tipado y el asincrónismo de las últimas versiones de Python.

Con la intención de mantener este tutorial lo más sencillo posible voy a usarlas únicamente donde sea necesario, si no es estrictamente necesario incluirlas voy a omitirlas. Menciono lo anterior para que tomes en cuenta que cada fragmento de código donde se use FastAPI puede incorporar asincronismo y tipado, según consideres necesario.

¿Cómo instalar FastAPI?

Para instalarlo vamos a crear un entorno virtual con Pipenv. Además de FastAPI necesitaremos uvicorn; un servidor ASGI, el cual usaremos para servir nuestra API.

Si no sabes usar Pipenv date una vuelta por la entrada donde explico como usar esta herramienta de manejo de entornos virtuales.

pipenv install fastapi uvicorn
Enter fullscreen mode Exit fullscreen mode

Activemos el entorno virtual para tener acceso a los paquetes que acabamos de instalar.

pipenv shell
Enter fullscreen mode Exit fullscreen mode

A continuación vamos a crear un archivo llamado main.py, aquí estará todo el código que usaremos para crear nuestra API.

touch main.py
Enter fullscreen mode Exit fullscreen mode

Y ahora vamos a colocar el código mínimo para tener un servidor.

# main.py
from fastapi import FastAPI

app = FastAPI()

@app.get("/")
def read_root():
    return {"Hello": "World"}
Enter fullscreen mode Exit fullscreen mode

Lo primero será importar la librería, después crearemos una instancia de FastAPI. A continuación escribiremos una función que devuelva un diccionario y usaremos un decorador con la ruta que queremos que capture nuestra aplicación. Es todo.

Uvicorn se encargará de servir la API que acabamos de crear por medio del siguiente comando:

uvicorn main:app --reload
Enter fullscreen mode Exit fullscreen mode

Analicemos brevemente lo que acabamos de ejecutar:

  • main se refiere al nombre de nuestro archivo
  • app es la instancia de FastAPI que creamos
  • --reload le dice a uvicorn que escuche cambios en el código
INFO:     Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit)
Enter fullscreen mode Exit fullscreen mode

Como puedes ver, si todo salió bien, tendremos un servidor corriendo localhost:8000

curl localhost:8000
--{"Hello":"World"}
Enter fullscreen mode Exit fullscreen mode

Y, si realizamos una petición al puerto 8000, obtendremos nuestra respuesta en formato JSON, sin necesidad de haberla convertido desde el diccionario Python.

Capturando parámetros

Ahora pasemos de rutas estáticas a rutas con parámetros.

Para capturar parámetros agregaremos estas lineas al archivo main.py que ya tenemos. Importamos el tipado Optional con la intención de capturar nuestros parámetros opcionales. Para este ejemplo voy a usar el tipado que ofrecen las nuevas versiones de Python, nota como fijamos item_id: int para que acepte únicamente valores de tipo entero.

from fastapi import FastAPI
from typing import Optional

app = FastAPI()

@app.get("/")
def read_root():
    return {"Hello": "World"}

@app.get("/items/{item_id}")
def read_item(item_id: int, q: Optional[str] = None):
    # otra_funcion_para_item_id(item_id)
    # otra_function_para_q(q)
    return {"item_id": item_id, "q": q}
Enter fullscreen mode Exit fullscreen mode

Dentro del decorador colocamos entre corchetes el nombre de la variable que queremos capturar. Esta variable se la pasaremos como parámetro a nuestra función

Como segundo parámetro, opcional, esperaremos un parámetro GET, de nombre "q".

En este caso, lo único que hará nuestra función es devolver ambos valores en formato JSON. Aunque, seguramente ya te diste cuenta cuenta de que, en lugar de simplemente devolverlos, puedes usar esos datos para buscar en una base de datos, ingresar esa información como parámetros a otra función y retornar otra cosa totalmente distinta.

curl localhost:8000/items/42
--{"item_id":42,"q":null}
Enter fullscreen mode Exit fullscreen mode

Como no especificamos un parámetro GET opcional nos devuelve un null en su lugar. Mira lo que sucede si intentamos enviar un valor inválido, es decir, que no sea un entero.

curl localhost:8000/items/texto -i
--HTTP/1.1 422 Unprocessable Entity
--date: Sun, 11 Oct 2020 00:13:38 GMT
--server: uvicorn
--content-length: 104
--content-type: application/json
--
--{"detail":[{"loc":["path","item_id"],"msg":"value is not a valid integer","type":"type_error.integer"}]}
Enter fullscreen mode Exit fullscreen mode

FastAPI se encarga de hacernos saber que el valor enviado es inválido mediante una respuesta HTTP 422. Ahora hagamos una petición que incluya valores correctos para ambos parámetros de nuestra funcion read_item()

curl localhost:8000/items/42?q=larespuesta
--{"item_id":42,"q":"larespuesta"}
Enter fullscreen mode Exit fullscreen mode

Observa como nos regresa el número que le pasemos, sea cual sea, así como nuestro parámetro GET opcional llamado "q".

REST

FastAPI se encarga de manejar los métodos HTTP de manera bastante intuitiva, simplemente cambiando la función de nuestro decorador por su respectivo método de petición HTTP

@app.get()
@app.post()
@app.put()
@app.delete()
Enter fullscreen mode Exit fullscreen mode

Además es posible especificar un código de respuesta opcional como parámetro en cada una de estas rutas.

@app.post(status_code=201)
Enter fullscreen mode Exit fullscreen mode

Para corroborarlo creemos otra función, esta en lugar de usar el decorador @app.get, usará @app.post y devolverá un código 201.

from fastapi import FastAPI
from typing import Optional

app = FastAPI()

# {...código anterior}

@app.post("/items/", status_code=201)
def post_item():
    return {"item": "our_item"}
Enter fullscreen mode Exit fullscreen mode

Este decorador capturará cualquier petición POST a la url /items/. Mira lo que sucede si intentamos hacer una petición GET, en lugar de una POST.

curl localhost:8000/items/ -i
--HTTP/1.1 405 Method Not Allowed
--date: Sun, 11 Oct 2020 00:56:06 GMT
--server: uvicorn
--content-length: 31
--content-type: application/json
--
--{"detail":"Method Not Allowed"}
Enter fullscreen mode Exit fullscreen mode

Así es, cualquier otro método no soportado recibirá una respuesta 405 (Método no permitido). Ahora hagamos la petición correcta, con POST.

curl -X POST localhost:8000/items/
--HTTP/1.1 201 Created
--date: Sun, 11 Oct 2020 00:57:05 GMT
--server: uvicorn
--content-length: 19
--content-type: application/json
--
--{"item":"our_item"}
Enter fullscreen mode Exit fullscreen mode

Observa que recibimos un código 201 como respuesta, así como nuestra respuesta en formato JSON.

Cookies

Lectura de cookies

Si queremos leer cookies usando FastAPI tendremos que importar Cookie y luego definir un parámetro, que será una instancia de esa Cookie. Si todo sale bien podremos mandar una Cookie y FastAPI nos regresará su valor.

from fastapi import Cookie, FastAPI
from typing import Optional

app = FastAPI()

# {...código anterior}

@app.get("/cookie/")
def read_cookie(my_cookie = Cookie(None)):
    return {"my_cookie": my_cookie}
Enter fullscreen mode Exit fullscreen mode

Si mandamos una cookie usando curl.

curl --cookie "my_cookie=home_made" localhost:8000/cookie/ -i
--{"my_cookie":"home_made"}
Enter fullscreen mode Exit fullscreen mode

Colocar cookies

Para colocar cookies es necesario acceder al objeto de respuesta de nuestra petición HTTP, y además necesitamos especificar el tipado de este parámetro. Por favor recuerda importarlo

from fastapi import Cookie, FastAPI, Response
from typing import Optional

app = FastAPI()

# {...código anterior}

@app.get("/setcookie/")
def set_cookie(response: Response):
    response.set_cookie(key="myCookie",
                        value="myValue")
    return {"message": "The delicious cookie has been set"}
Enter fullscreen mode Exit fullscreen mode

Mira la cabecera set-cookie que aparece en nuestra respuesta. La presencia de esta cabecera HTTP o header indica que hemos recibido la instrucción de colocar nuestra cookie correctamente.

curl localhost:8000/setcookie/ -i
--HTTP/1.1 200 OK
--date: Mon, 19 Oct 2020 20:45:08 GMT
--server: uvicorn
--content-length: 31
--content-type: application/json
--set-cookie: myCookie=myValue; Path=/; SameSite=lax
--
--{"message":"Delicious cookies"}
Enter fullscreen mode Exit fullscreen mode

Headers o cabeceras HTTP

Leer headers o cabeceras HTTP

Para leer cabeceras HTTP se hará de la misma manera que con las cookies. Por favor recuerda importar Header.

from fastapi import Cookie, Header, Response, FastAPI
from typing import Optional

app = FastAPI()

# {...código anterior}

@app.get("/headers/")
def return_header(user_agent = Header(None)):
    return {"User-Agent": user_agent}
Enter fullscreen mode Exit fullscreen mode

Listo, ahora tendremos una cabecera que nos devolverá el User-Agent actual, con el que estamos realizando la petición, el cual manda automáticamente curl con cada petición, por lo que deberiamos ser capaces de capturarlo.

curl localhost:8000/headers/ -i
--HTTP/1.1 200 OK
--date: Mon, 19 Oct 2020 19:33:45 GMT
--server: uvicorn
--content-length: 28
--content-type: application/json
--
--{"User-Agent":"curl/7.52.1"}
Enter fullscreen mode Exit fullscreen mode

En este caso, como hicimos la petición con curl, nos retornará la cadena de texto "curl/nuestra_versión". Si hicieramos la petición con un navegador web obtendriamos el valor de User-Agent para ese navegador.

Colocar headers o cabeceras HTTP

Para colocar headers necesitamos acceder al objeto response, este objeto tiene una propiedad llamada headers al que podemos agregarle valores como si fuera un diccionario.

from fastapi import Cookie, Header, Response, FastAPI
from typing import Optional

app = FastAPI()

# {...código anterior}

@app.get("/setheader/")
def set_header(response: Response):
    response.headers["response_header"] = "my_header"
    return {"message": "header set"}
Enter fullscreen mode Exit fullscreen mode

Ahora hacemos una petición a la url que acabamos de crear. Esperamos que la respuesta contenga un header o cabecera HTTP llamada response_header

curl localhost:8000/setheader/ -i
--HTTP/1.1 200 OK
--date: Sat, 24 Oct 2020 16:11:31 GMT
--server: uvicorn
--content-length: 24
--content-type: application/json
--response_header: my_header
--x-my_data: X-my_data
--
--{"message":"header set"}
Enter fullscreen mode Exit fullscreen mode

Middleware

Sí, aunque FastAPI es bastante simple también incorpora la funcionalidad de usar middleware como parte de su ciclo de petición-respuesta.

¿No sabes que es un middleware? De manera simplista, un middleware es una pieza de código que colocas antes de la petición, para "interceptarla" y hacer (o no) algo con ella. Un middleware funciona de manera similar a esas carreras de relevos donde la petición y la respuesta serían las estafetas que van pasándose de un middleware al otro, cada middleware puede modificar la petición o la respuesta o dejarla intacta para pasarla al siguiente middleware.

Alt Text

Esquema super simplificado de un Middleware en el contexto web

Para usar middleware basta con colocar un decorador @app.middleware('http') sobre una función. Esta función recibe el objeto de la petición web (request) y una función llamada call_next, que recibirá la petición web y retornará una respuesta.

from fastapi import Cookie, Header, Response, FastAPI
from typing import Optional

app = FastAPI()

@app.middleware("http")
async def add_something_to_response_headers(request, call_next):
    response = await call_next(request)
    # Ahora ya tenemos la respuesta que podemos modificar o procesar como querramos
    response.headers["X-my_data"] = "x-my_data"
    return response
Enter fullscreen mode Exit fullscreen mode

La respuesta a nuestra petición la obtenemos después de llamar a call_next pasándole como parámetro el objeto request, por lo que todas las modificaciones a la petición van antes de la llamada a call_next, mientras que todas las modificaciones a la respuesta van despúes de call_next

from fastapi import Cookie, Header, Response, FastAPI
from typing import Optional

app = FastAPI()

# {...código anterior}

@app.middleware("http")
async def my_middleware(request, call_next):
    # modificaciones a REQUEST
    response = await call_next(request)
    # modificaciones a RESPONSE
    return response
Enter fullscreen mode Exit fullscreen mode

Ahora hagamos un curl a localhost:8000 para ver si funcionó. Mira como ahora tenemos un header o cabecera HTTP en la respuesta, y este corresponde a la información que le acabamos de colocar.

curl localhost:8000 -i
--HTTP/1.1 200 OK
--date: Mon, 19 Oct 2020 19:20:35 GMT
--server: uvicorn
--content-length: 17
--content-type: application/json
--x-my_data: X-my_data
--
--{"Hello":"World"}
Enter fullscreen mode Exit fullscreen mode

Mientras el middleware que creamos siga activo, cada nueva respuesta que obtengamos contendrá ese header y su respectivo valor.

Middleware incluidos

FastAPI viene con una serie de middleware incluidos que podemos usar y agregar a la lista de middlewares por los que pasarán nuestras peticiones. Para agregar un middleware basta con usar el método add_middleware(), de nuestra app.

No es necesario que agregues el siguiente código. Es solo para que conozcas algunas de las opciones incluye fastAPI como parte de su código.

from fastapi import Cookie, Header, Response, FastAPI
from typing import Optional

from fastapi.middleware.gzip import GZipMiddleware
# from fastapi.middleware.trustedhost import TrustedHostMiddleware
# from fastapi.middleware.httpsredirect import HTTPSRedirectMiddleware

app = FastAPI()

app.add_middleware(GZipMiddleware, minimum_size=1000)
Enter fullscreen mode Exit fullscreen mode
  • GZipMiddleware: se encarga de usar compresión gzip en tus respuestas
  • TrustedHostMiddleware: con este middleware podemos decirle a fastAPI cuales son los dominios seguros, similar a la variable ALLOWED_HOSTS de Django.
  • HttpsRedirectMiddleware: se encarga de redirigir las peticiones http a su version en https

Manejo de formularios

Lo primero que tenemos que hacer para manejar formularios es instalar la dependencia python-multipart a nuestro entorno virtual. Puedes usar pip o pipenv, yo usaré pipenv.

Asegúrate de estar dentro del entorno virtual en el cual estás trabajando.

pipenv install python-multipart
Enter fullscreen mode Exit fullscreen mode

Una vez instalado nuestro paquete, crearemos una función que reciba un parámetro igual al objeto Form. Recuerda, nuevamente, importar Form desde fastapi

También toma en cuenta que si quieres agregar más campos basta con establecer más parámetros para la función.

Sí, puede sonar bastante obvio pero es mejor mencionarlo.

from fastapi import Cookie, Header, Response, FastAPI, Form
from typing import Optional

app = FastAPI()

# {...código anterior}

@app.post("/subscribe/")
async def subscribe(email=Form(...)):
    return {"email": email}
Enter fullscreen mode Exit fullscreen mode

Ahora intentemos mandar un dato usando un formulario usando curl.

curl -X POST -F 'email=email@example.org' localhost:8000/subscribe/
--{"email":"email@example.org"}
Enter fullscreen mode Exit fullscreen mode

Veremos como nos regresa un objeto JSON, con el correo que mandamos en el formulario, como respuesta.

Manejo de archivos

De la misma manera que para los formularios, el manejo de archivos requiere la librería python-multipart. Instalalá usando pip o pipenv si aún no lo has hecho. Una vez hecho esto agrega File y UploadFile a las importaciones.

Por favor observa como es necesario usar el tipado de Python para este ejemplo, si no lo haces te devolverá un error.

from fastapi import Cookie, Header, Response, FastAPI, Form, File, UploadFile
from typing import Optional

app = FastAPI()

# {...código anterior}

@app.post("/files/")
async def create_file(file: bytes = File(...)):
    return {"file_size": len(file)}


@app.post("/uploadfile/")
async def create_upload_file(file: UploadFile = File(...)):
    return {"filename": file.filename}
Enter fullscreen mode Exit fullscreen mode

Para este ejemplo vamos a crear un archivo sencillo de texto.

El siguiente comando creará un archivo de extensión txt en nuestra carpeta actual. Si no te sientes cómodo usando la terminal de GNU/Linux visita mi serie de entradas donde explico los comandos básicos.

printf "texto" > archivo.txt
Enter fullscreen mode Exit fullscreen mode

Observa como una petición a /files/ nos devuelve el tamaño de nuestro archivo.

curl -F "file=@archivo.txt" localhost:8000/files/
--{"file_size":5}
Enter fullscreen mode Exit fullscreen mode

Mientras que una petición a /uploadfile/ nos devuelve el nombre de nuestro archivo

curl -F "file=@archivo.txt" localhost:8000/uploadfile/
--{"filename":"archivo.txt"}
Enter fullscreen mode Exit fullscreen mode

Seguramente ya habrás notado que en ningún caso el archivo está siendo guardado, sino que solo se pone a disposición de fastAPI, para que hagamos con él lo que queramos dentro de nuestra función.

Manejo de errores

FastAPI cuenta con una serie de excepciones que podemos usar para manejar los errores de nuestra aplicación.

Para este ejemplo nuestra función solo retornará un error, pero bien podrías colocar esta excepción en la búsqueda fallida de un objeto en la base de datos o alguna otra situación que se te ocurra.

from fastapi import Cookie, Header, Response, FastAPI, Form, File, UploadFile, HTTPException
from typing import Optional

app = FastAPI()

# {...código anterior}

@app.get("/error/")
def generate_error():
    raise HTTPException(status_code=404, detail="Something was not found")
Enter fullscreen mode Exit fullscreen mode

Elegimos el código a devolver con status_code y la información adicional con detail.

Si hacemos una petición web a /error/ recibiremos una respuesta HTTP 404, junto con la respuesta

curl localhost:8000/error/ -i
--HTTP/1.1 404 Not Found
--date: Mon, 19 Oct 2020 20:21:28 GMT
--server: uvicorn
--content-length: 36
--content-type: application/json
--x-my_data: X-my_data
--
--{"detail":"Something was not found"}
Enter fullscreen mode Exit fullscreen mode

Si lo deseamos también podemos agregar cabeceras HTTP o headers a la respuesta directamente como un argumento llamado headers en nuestra excepción.

from fastapi import Cookie, Header, Response, FastAPI, Form, File, UploadFile, HTTPException
from typing import Optional

app = FastAPI()

# {...código anterior}

@app.get("/error/")
def generate_error():
    raise HTTPException(status_code=404,
                        detail="Something was not found",
                        headers={"X-Error": "Header Error"},)
Enter fullscreen mode Exit fullscreen mode

Testing en FastAPI

FastAPI contiene un cliente con el que podemos hacer testeo. Antes de empezar a realizar el testing vamos a instalar los paquetes necesarios para hacerlo: pytest y requests.

Si quieres profundizar en el testeo en Python tengo una entrada donde expongo algunas de las librerías principales para hacer testing.

pipenv install requests pytest
Enter fullscreen mode Exit fullscreen mode

Ahora que ya los tenemos vamos a crear un archivo de testeo llamado test_api.py y un archivo __init__.py para que python poder tener acceso a nuestros módulos.

touch __init__.py test_apy.py
Enter fullscreen mode Exit fullscreen mode

En nuestro archivo test_api.py vamos a colocar el siguiente código.

from fastapi.testclient import TestClient
from typing import Optional

from .main import app

client = TestClient(app)

def test_read_main():
    response = client.get("/")
    assert response.status_code == 200
    assert response.json() == {"Hello": "World"}
Enter fullscreen mode Exit fullscreen mode

Como puedes apreciar en el código anterior:

  • client.get() se encarga de realizar la petición a root
  • response.status_code contiene el código de respuesta
  • response.json() nos devuelve el cuerpo de la respuesta en formato JSON

Si corremos pytest veremos que se ejecutan los tests correspondientes para corroborar que cada uno de los elementos anteriores sea igual al valor esperando.

pytest
--======== test session starts ===========
--platform linux -- Python 3.7.2, pytest-6.1.1, py-1.9.0, pluggy-0.13.1
--rootdir: /home/usuario/fastAPI
--collected 1 item
--test_api.py .  [100%]
--
--======== 1 passed in 0.17s =============
Enter fullscreen mode Exit fullscreen mode

Documentación en FastAPI

Hasta este momento te he ocultado una de las características más geniales de FastAPI, no me odies por favor. Así es, ya sabes exactamente a que me refiero: ¡Documentación automática!

Sí, como seguramente ya sabías, FastAPI cuenta con documentación automática usando swagger y redoc, no tienes que agregar código, ni establecer una variable para esto, sencillamente abre tu navegador y dirígete a tu localhost:8000/docs/ y localhost:8000/redoc/, respectivamente, y verás la documentación interactiva generada automáticamente.

Alt Text

Deployment sin Docker

El despliegue también es una tarea sencilla de realizar.

Para hacer deployment sin usar Docker basta con correr uvicorn, justo como hicimos al principio de este tutorial.

uvicorn main:app --host 0.0.0.0 --port 80
Enter fullscreen mode Exit fullscreen mode

Deployment con Docker

Hacer un deployment con Docker es super sencillo, el creador de FastAPI ya nos provee de una imagen de Docker personalizada que podemos usar como base para nuestro Dockerfile. Primero creemos un archivo Dockerfile.

touch Dockerfile
Enter fullscreen mode Exit fullscreen mode

Ahora vamos a colocar lo siguiente dentro:

FROM tiangolo/uvicorn-gunicorn-fastapi:python3.7

COPY . /app
Enter fullscreen mode Exit fullscreen mode

Le indicamos a Docker que copie todo el contenido de nuestra carpeta actual en la carpeta /app. Despues de la instrucción Copy puedes agregar más código para personalizar tu imagen.

A continuación vamos a compilar la imagen.

docker build -t fastapi .
Enter fullscreen mode Exit fullscreen mode

Cuando termine de compilar nuestra imagen vamos a correr un contenedor en segundo plano, en el puerto 80. Este contenedor usa gunicorn para servir el contenido.

docker run -d --name fastapicontainer -p 80:80 fastapi
Enter fullscreen mode Exit fullscreen mode

La configuración de uvicorn y el uso de certificados SSL, ya sea usar Cerbot o Traefik o alguna otra opción ya depende de ti.

Con esto concluye mi pequeño tutorial FastAPI. Esta fue únicamente una pequeña introducción con las opciones que yo considero más relevantes, por favor lee la documentación oficial para profundizar en cada una de las opciones que FastAPI tiene disponibles.

¿Te gustaría leer más contenido como este?

Haz click aquí

Top comments (0)