Django
Django est un Framework de développement de applications web ou d'APIs REST.
Dans ce qui suite, nous résumons le tutoriel de démarrage proposé par Django.
Exensions VS Code recommandées
Veuillez lire les instructions de configuration de chaque extension:
- Formatteur et linteur de templates: monosans.djlint
- Coloration syntaxique des templates: batisteo.vscode-django
Tutoriel couvrant les bases
Création et démarrage d'un projet
- De préférence, créer un environnement virtuel:
python3 -m venv venv
oupython -m venv venv
- Installer la dernière version de Django
pip install Django==[version]
(cette page permet de trouver la dernière version) - Vérifier que l'installation a réussi:
python -m django --version
- Créer un projet django:
django-admin startproject [nom du projet]
- Démarrer le serveur de développement avec
python manage.py runserver
et ouvrir le lien affiché par la sortie de la commande. - Structure du projet
mysite/
manage.py # Script de gestion du projet (fichier à ne pas modifier)
mysite/ # Informations et paramètres du projet
__init__.py # Permet d'être considéré par Python comme un module
settings.py # Paramètres du projet
urls.py # Les routes que le projet gère
asgi.py # Point d'entée pour les serveurs de type ASGI
wsgi.py # Point d'entée pour les serveurs de type WSGI
Ajout d'une application
- Un projet Django contient plusieurs applications
- Pour Django, une application est un paquet Python qui proposer un ensemble de fonctionnalités (pages Web, Interface d'administration, API REST, Middlewares, etc.)
- Les applications sont censées être autonomes et réutilisables dans différents projets.
- Le projet que vous venons de créer utilise déjà quelques applications. Vous pouvez les vérifier dans
settings.py
- Ajouter une nouvelle application:
python manage.py startapp polls
. Consulter le contenu du dossier créé. -
Nous allons ajouter une page. Pour ce faire, suivons ces étapes:
- Ajouter une page dans
views.py
from django.http import HttpResponse def index(request): return HttpResponse("Hello, world. You're at the polls index.")
- Associer une route à la page que nous venons de créer dans
urls.py
(""
signifie que la page sera associée à la racine de l'application)
from django.urls import path from . import views urlpatterns = [ path("", views.index, name="index"), ]
- Comme nous n'avons pas encore défini la route de l'application au sein du projet, il faut le faire maintenant en mettant à jour le fichier
[nom du projet]/urls.py
- Ajouter une page dans
-
Ouvrir la page que nous venons de créer en utilisant l'url:
[url du servur de dev]/[chemon de l'app]
. Pour notre cas, ce sera:http://localhost:8000/polls/
- Exercice: ajouter une page
http://localhost:8000/polls/hello
qui affiche 'Hello World' dans une baliseh1
Bases de données
- Django utilise un mécanisme de migrations qui permet de suivre les évolutions de la structure la BDD dans le temps
- Ce système est très pratique pour permettre des mises à niveau de la BDD de l'application de production sans avoir à appliquer à la main les changement faits durant le développement.
- C'est similaire à ce que fait Git (qui mémorise les changements faits sur un fichier) qui permet de reconstruire la dernière version d'un fichier en partant de sa version initiale et en appliquant tous les changement faits à travers les différents commits.
- Sur Django, cela fonctionne en trois phases:
- Définir ou mettre à jour les modèles (équivalent d'une table côté code Django)
- Définir le point de migration à partir de l'état actuel des modèles:
python manage.py makemigrations polls
- Appliquer les migrations aux bases de données liées à l'application:
python manage.py migrate
- Django sait gérer les migrations des application enregistrées dans la liste
INSTALLED_APPS
du fichiersettings.py
. - Comme les applications pré-resneignées ont déjà défini leur modèles et points de migration, il ne reste qu'a les appliquer à la BDD du projet:
python manage.py migrate
- Vérifier le contenu de la BDD (par défaut c'est le fichier
db.sqlite3
)- 💡 Il est possible de changer la BDD dans le fichier
settings.py
- 💡 Il est possible de changer la BDD dans le fichier
-
Dans la suite, nous allons ajouter des tables dans l'application que nous avons créé précédemment en suivant donc les trois phases introduites plus haut:
-
Définir les modèles dans
[dossier de l'app]/models.py
from django.db import models class Question(models.Model): question_text = models.CharField(max_length=200) pub_date = models.DateTimeField("date published") class Choice(models.Model): question = models.ForeignKey(Question, on_delete=models.CASCADE) choice_text = models.CharField(max_length=200) votes = models.IntegerField(default=0)
-
Dans settings.py, ajouter
"polls.apps.PollsConfig"
dans la listeINSTALLTED_APPS
. Ceci nous permet d'utiliser les migrations. Définissons le points de migration:python manage.py makemigrations polls
. Surveiller la sortie de la console.- Pour vérifier qui sera généré si on exécute cette migration par la suite:
python manage.py sqlmigrate polls 0001
- Pour vérifier qui sera généré si on exécute cette migration par la suite:
- Appliquer les migrations à la BDD:
python manage.py migrate
-
Shell Django
- Il est possible de tester les API de Django et agir sur notre projet depuis le terminal
- Lancer la commande :
python manage.py shell
qui ouvre un invite interactif (affiche les résultat après chaque commande) Python lié à notre projet Django. - Exécuter les commandes suivantes une par une:
from polls.models import Choice, Question
# Devrait afficher <QuerySet []> car la table est vide
Question.objects.all()
from django.utils import timezone
q = Question(question_text="What's new?", pub_date=timezone.now())
q.save() # enregistrer dans la BDD
q.id # devrait afficher 1
q.question_text
q.pub_date
q.question_text = "What's up?"
q.save()
# Devrait affichier <QuerySet [<Question: Question object (1)>]>
# Noter le Question object (1) qui n'est pas très lisible
Question.objects.all()
q.choice_set.create(choice_text= "Choix 1", votes = 2)
q.save()
- Afin d'améliorer l'affichage des objets au sein de notre projet, définir la méthode
__str__(self):
dans chacune des classes modèles et réessayer d'appeler unQuestion.objects.all()
.
Affichage des données dans des pages web
- Ce code permet d'afficher les questions et les choix sous forme de listes à puces.
def index(request):
content = "<h1>Questions</h1>"
content += "<ul>"
for question in Question.objects.all():
content += "<li>"
content += f"{question.question_text} - {question.pub_date}. {question.choice_set.count()} choix"
content += "<ul>"
str_choices = [
f"<li>{c.choice_text} / {c.votes}</li>" for c in question.choice_set.all()
]
content += "".join(str_choices)
content += "</ul>"
content += "</li>"
content += "</ul>"
return HttpResponse(content)
Compréhension de listes
Cette Syntaxe permet de générer une nouvelle liste à partir d'une liste existante.
Par exemple [x+1 for x in [1, 2, 3, 4]]
générera la liste [1, 2, 3, 4, 5]
.
Sur le même principe, [f"<li>{c.choice_text} / {c.votes}</li>" for c in question.choice_set.all()]
gérera une liste de chaines de caractères.
Utilisation d'un template
- Jusqu'à présent, le HTML des vues est codé "en dur" dans le code
- La technique recommandée est de faire les traitement dans le code et déléguer l'affichage à un fichier HTML particulier qu'on appelle "template"
- Ce modèle est appelé modèle MVC (Model View Controller)
- Les templates doivent être placés dans
[dossier de l'app]/templates
from django.shortcuts import render
def showQuestionsWithTemplate(request):
context = {"questions": Question.objects.all()}
return render(request, "questions.html", context)
# ajouter path("questions", views.showQuestionsWithTemplate, name="questions"), dans urls.py
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Questions</title>
</head>
<body>
<a href="prompt-input">Enter your prompt</a>
<h1>Prompt list</h1>
<ul>
{% for question in questions %}
<li>
{{ question.question_text }}. Date: {{ question.pub_date }}
<ul>
{% for choice in question.choice_set() %}
<li></li>
{% endfor %}
</ul>
</li>
{% endfor %}
</ul>
</body>
</html>
Exercices
- Ajouter une page http://localhost:8000/ynav/hello qui affiche 'Hello World' dans une balise h1
- Ajouter une page http://localhost:8000/ynav/user/profile qui répond avec un html complet.
- Exemple de contenu:
- Astuce: Utiliser la fonction
HttpResponse
de Django et"""
pour mettre du texte multi-lignes.- Exemple
- Créer une page qui affiche l'heure côté serveur. Exemple:
Il est 12:00
.- Astuce: Utiliser la librairie
datetime
de Python. - Exemple:
- Astuce: Utiliser la librairie
- Créer une page qui utilise un template pour afficher la valeur du query param p. Exemple:
http://localhost:8000/ynav/query?p=hello
affichehello
.- Astuce: Utiliser
request.GET.get('p')
pour récupérer la valeur du paramètre p. - Exemple:
- Astuce: Utiliser
- Créer une page qui utiliser un template pour afficher le profil de l'utilisateur avec les valeurs des query params
first_name
etlast_name
. Exemple:http://localhost:8000/ynav/query/profile?first_name=John&last_name=Doe
affiche John Doe dans la page. - Créer une page qui affiche tous entiers compris entre un mix et un max passés en query params. Exemple:
http://localhost:8000/ynav/query/range?min=1&max=5
affiche1 2 3 4 5
.- Afficher les entiers dans un liste à puce (
<ul>
).
- Afficher les entiers dans un liste à puce (
- Créer une page qui affiche un formulaire avec un champ
name
et un boutonsubmit
. Lorsque le formulaire est soumis, la page afficheHello <name>
. Exemple:http://localhost:8000/ynav/form
affiche un formulaire avec un champname
et un boutonsubmit
. Lorsque le formulaire est soumis avec le nomJohn
, la page afficheHello John
. (astuce: créer une classNameForm
qui hérite deforms.Form
et ajouter un champname
de typeforms.CharField
). - Afficher une table de multiplication. Exemple:
http://localhost:8000/ynav/multiplication?n=5
affiche la table de multiplication de 5. - Créer une table
Pakiman
avec les champsname
,type
,level
(entier) etattack
(entier). Créer une page qui permet d'ajouter desPakiman
à la base de données. Créer une page qui affiche tous lesPakiman
de la base de données. Créer une page qui affiche lesPakiman
dont le niveau est supérieur au query param passé dans la requête. Exemple:http://localhost:8000/ynav/pakiman?level=5
affiche lesPakiman
dont le niveau est supérieur à 5.- Astuce: Utiliser
models.Model
pour définir le modèlePakiman
.
- Astuce: Utiliser
L'application admin
- L'application admin propose une interface web basique de type CRUD
- Accessible depuis
[URL du serveur]/admin
. Elle propose une page de connexion- Les utilisateurs sont gérés via l'application auth. On y reviendra plus tard.
- Ajoutons un super utilisateur (comme un root dans Linux)
python manage.py createsuperuser
- Se connecter à l'interface d'admin avec le compte nouvellement créé et faire quelques manipulations
- Les tables proposées par l'interface d'admin proviennent de l'application auth qui a enregistré la possibilité d'éditer ses tables depuis l'interface d'admin
- Faisons pareil avec les tables de l'application que nous avons créé. Dans le fichier
[app]/admin.py
ajouter une ligneadmin.site.register([modèle(s)])
- Pour notre cas ce sera:
admin.site.register([Question, Choice])
- Pour notre cas ce sera:
Utilisation d'une librairie de composants
Material UI
ça semble compliqué à priori
- Télécharger la librairie: Ajouter dans
requirements.txt
et relancer unpip install -r requirements.txt
- Activer la librairie: dans
settings.py
ajouter cette app juste avant nos propres apps"theme_material_kit"
- Appliquer la migration comme on a ajouté une nouvelle app
pythom manage.py migrate
- Ajout des urls proposées par l'app en dernière priorité: Dans le
urls.py
global, ajouterpath("", include('theme_material_kit.urls'))
Bulma
- Suivre la doc officielle
- Voir exemple d'intégration dans le projet d'exemple
- Ajouter les styles bulma dans les formulaires avec django-bulma-form-templates
Formulaires d'enregistrement et de connexion
django.contrib.auth.forms
propose des formulaires prédéfinis comme les:AuthenticationForm et UserCreationForm
-
Les custom template tags permettent d'agir sur le rendu html. Cela peut être utile pour insérer des styles dans les formulaires avant d'en faire le rendu.
- Templates de formulaires