Dans ce tutoriel, vous allez apprendre à faire une api web en python avec le framework Flask. Le framework flask est un framework python permettant la réalisation d’un site web ou d’une api web. Son principal avantage est d’être simple à utiliser, mais sans perdre de fonctionnalités, de plus il peut quasiment tout faire grâce à de nombreuses extensions.
L’objectif de ce tutoriel est d’apprendre comment faire :
- Une api web en python
- Le traitement des requêtes
Installation
Pour commencer, il vous faut un interpréteur python en version 3, dans mon cas, j’utiliserai python 3.8
Linux - Ubuntu (& toutes distributions utilisant APT comme gestionnaire de paquets)
Sous linux, c’est assez simple.
Depuis un terminal, installation de python3 :
sudo apt install python3
Vous aurez ensuite besoin de pip le gestionnaire de package de python, il est souvent préinstallé avec python, mais dans le doute :
sudo apt install python3-pip
Maintenant installons flask :
pip3 install flask
Si vous avez une erreur vous disant que vous n’avez pas assez de permissions, faites :
pip3 install --user flask
Windows
Sur Windows, ça se complique un peu, commencez par télécharger python3 pour Windows ici et installez-le.
Déplacez-vous dans le dossier où vous avez installé python et faites :
shift + click droit -> ouvrir une fenêtre powershell
(sur Windows 7 pour les
réfractaires au changement ça doit être cmd)
Vous êtes normalement dans un terminal, entrez alors :
.\python.exe -m pip install flask
MacOS
N’ayant pas de Mac, je ne peux pas tester l’installation, il faut toutefois aussi utiliser python et PIP, et suivre les instructions pour linux afin d’installer flask.
Une requête HTTP ?
L’HyperText Transfer Protocol (HTTP, littéralement « protocole de transfert hypertexte ») est un protocole de communication client-serveur développé pour le World Wide Web.
Source Wikipédia.
Il existe 5 principales requêtes HTTP :
- GET, permet d’accéder à une ressource.
- HEAD, permet de récupérer l’entête d’une ressource, pour par exemple connaitre la date de sa dernière modification (utile pour le système de cache d’un navigateur)
- POST, permet d’ajouter une ressource
- PUT, permet de mettre à jour une ressource
- DELETE, permet de supprimer une ressource
Qu’est-ce qu’une API web ?
Une API Web est une interface de programmation composée d’un ou de plusieurs points endpoints exposés publiquement via le Web, le plus souvent au moyen d’un système basé sur serveur web HTTP.
Source Wikipédia.
À ne pas confondre avec une API REST, qui est une api web avec un ensemble de contraintes et de règles prédéfinies à utiliser. Toutes les API web ne sont pas des API REST…
Un premier Endpoint
Créez un fichier app.py
avec le contenu suivant :
from flask import Flask
app = Flask(__name__)
@app.route('/')
def super_endpoint():
return 'Hello World'
Pour lancer votre premier Endpoint :
flask run
Ou sinon :
python -m flask run
Si vous allez sur http://127.0.0.1:5000/
avec votre navigateur web, vous
devriez avoir :
Hello World
Ou alors avec curl
curl http://127.0.0.1:5000/
Hello World
Super, nous avons notre premier “hello world”, mais comment faire pour avoir plusieurs routes possibles ?
Routing
Pour régler ce problème, nous allons utiliser une fonctionnalité intégrée à
Flask pour faire du routing par rapport à notre URL.
On crée un nouvel endpoint qu’on pourra appeler avec
l’URL : http://127.0.0.1:5000/test
@app.route('/test')
def test_endpoint():
return 'test_endpoint'
curl http://127.0.0.1:5000/test
test_endpoint
Passer des paramètres
Dans la vraie vie, il est parfois (même très souvent) nécessaire de passer des
paramètres à notre endpoint.
Pour passer des paramètres avec le routing on utilise les <>
avec un simple
argument de notre fonction
@app.route('/test/<id_test>')
def test_endpoint(id_test):
return 'test ' + id_test
Ce qui retourne :
curl http://127.0.0.1:5000/test/1
test 1
Par défaut le type est un string, mais il est possible de forcer le cast vers
un type en ajoutant un type dans les <>
de la route :
@app.route('/test/<int:id_test>')
def test_endpoint(id_test):
return 'test ' + id_test
Il existe différents convertisseurs intégrés dans flask :
- string
- int
- float
- path
- uuid
Il est également possible de créer ses propres convertisseurs si nécessaire.
Méthodes HTTP
Pour le moment notre API répond à tous les types de requêtes HTTP ce qui peut
poser des problèmes, pour spécifier pour quelles méthodes l’endpoint doit être
disponible, on ajoute dans l’annotation @app.route
un nouveau
paramètre methods
@app.route('/test', methods=["GET"])
def test_endpoint_get():
return 'test_endpoint_get'
curl -X GET http://127.0.0.1:5000/test
test_endpoint_get
Le GET avec l’aide curl renvoie bien la bonne valeur, mais si on tente avec un POST ça ne fonctionne pas ! Car Flask ne sait pas vers quoi rediriger notre requête HTTP.
curl -X POST http://127.0.0.1:5000/test
Traiter une requête POST
Pour traiter une requête POST, on a besoin de récupérer les données de celle-ci.
On importe donc request
pour cela.
from flask import request
@app.route('/test', methods=["POST"])
def test_endpoint_post():
data = request.form
# Traiter la requête
return data
Le module request converti automatiquement les données de la requête en dictionnaire python. Dans l’autre sens si notre API retourne un dictionnaire, flask se charge de le convertir pour nous en json :
curl -X POST http://127.0.0.1:5000/test -d "param1=jeej"
{"param1":"jeej"}
Exemple d’un POST avec un traitement simpliste
@app.route('/exemple', methods=["POST"])
def test2_endpoint_post():
"""
Exemple de traitement
"""
responses = {}
param1 = request.form["param1"]
responses["return1"] = param1 + "AAA"
return responses
curl -X POST http://127.0.0.1:5000/exemple -d "param1=jeej"
{"return1":"jeejAAA"}
Voilà, vous êtes maintenant capable de créer une api web simple, mais performante. D’autres tutoriels sur flask seront prochainement disponibles : par exemple pour interroger une base de données et avoir des données dynamiques ou docker son application flask.
Le code complet de ce tutoriel
from flask import Flask
from flask import request
app = Flask(__name__)
@app.route('/')
def super_endpoint():
return 'Hello World'
@app.route('/test/<id_test>')
def test_endpoint(id_test):
return 'test ' + id_test
@app.route('/test', methods=["GET"])
def test_endpoint_get():
return 'test_endpoint_get'
@app.route('/test', methods=["POST"])
def test_endpoint_post():
data = request.form
# traiter la requête
return data
@app.route('/exemple', methods=["POST"])
def test2_endpoint_post():
"""
Exemple de traitement
"""
responses = {}
param1 = request.form["param1"]
responses["return1"] = param1 + "AAA"
return responses