mirror of
https://github.com/ae-utbm/sith.git
synced 2024-12-23 16:21:22 +00:00
411 lines
13 KiB
Markdown
411 lines
13 KiB
Markdown
L'ORM de Django est puissant, très puissant, non par parce qu'il
|
|
est performant (après tout, ce n'est qu'une interface, le gros du boulot,
|
|
c'est la db qui le fait), mais parce qu'il permet d'écrire
|
|
de manière relativement simple un grand panel de requêtes.
|
|
|
|
De manière générale, puisqu'un ORM est un système
|
|
consistant à manipuler avec un code orienté-objet
|
|
une db relationnelle (c'est-à-dire deux paradigmes
|
|
qui ne fonctionnent absolument pas pareil),
|
|
on rencontre un des deux problèmes suivants :
|
|
|
|
- soit l'ORM n'offre pas assez d'abstraction,
|
|
auquel cas, quand on veut faire des requêtes
|
|
plus complexes qu'un `select` avec un `where`,
|
|
on s'emmêle les pinceaux et on se dit que
|
|
ça aurait été plus simple de le faire directement
|
|
en SQL.
|
|
- soit l'ORM offre trop d'abstraction,
|
|
auquel cas, on a tendance à ne pas prêter
|
|
assez attention aux requêtes envoyées en base
|
|
de données et on finit par se rendre compte
|
|
que les temps d'attente explosent
|
|
parce qu'on envoie trop de requêtes.
|
|
|
|
Django est dans ce deuxième cas.
|
|
|
|
C'est pourquoi nous ne parlerons pas ici
|
|
de son fonctionnement exact ni de toutes les fonctions
|
|
que l'on peut utiliser
|
|
(la [doc officielle](https://docs.djangoproject.com/fr/stable/topics/db/queries/) fait déjà ça mieux que nous),
|
|
mais plutôt des pièges courants
|
|
et des astuces pour les éviter.
|
|
|
|
## Les `N+1 queries`
|
|
|
|
### Le problème
|
|
|
|
Normalement, quand on veut récupérer une liste,
|
|
on fait une requête et c'est fini.
|
|
Mais des fois, ça n'est pas si simple.
|
|
Par exemple, supposons que nous voulons
|
|
récupérer les 100 utilisateurs les plus riches,
|
|
avec leurs informations client :
|
|
|
|
```python
|
|
from core.models import User
|
|
|
|
for user in User.objects.order_by("-customer__amount")[:100]:
|
|
print(user.customer.amount)
|
|
```
|
|
|
|
Combien de requêtes le bout de code suivant effectue-t-il ?
|
|
101\.
|
|
En deux pauvres lignes de code, nous avons demandé
|
|
à la base de données d'effectuer 101 requêtes.
|
|
Une requête toute seule n'est déjà une opération anodine,
|
|
alors je vous laisse imaginer ce que ça donne pour 101.
|
|
|
|
Si vous ne comprenez pourquoi ce nombre, c'est très simple :
|
|
|
|
- Une requête pour sélectionner nos 100 utilisateurs
|
|
- Une requête supplémentaire pour récupérer les informations
|
|
client de chaque utilisateur, soit 100 requêtes.
|
|
|
|
En effet, les informations client sont stockées dans une
|
|
autre table, mais le fait d'établir un lien de clef
|
|
étrangère permet de manipuler `customer`
|
|
comme si c'était un membre à part entière de `User`.
|
|
|
|
Il est à noter cependant, que Django n'effectue une requête
|
|
que pour le premier accès à un membre d'une relation
|
|
de clef étrangère.
|
|
Toutes les fois suivantes, l'objet est déjà là,
|
|
et django le récupère :
|
|
|
|
```python
|
|
from core.models import User
|
|
|
|
# l'utilisateur le plus riche
|
|
user = User.objects.order_by("-customer__amount").first() # <-- requête db
|
|
print(user.customer.amount) # <-- requête db
|
|
print(user.customer.account_id) # on a déjà récupéré `customer`, donc pas de requête
|
|
```
|
|
|
|
Ce n'est donc pas gravissime si vous faites cette
|
|
erreur quand vous manipulez un seul objet.
|
|
En revanche, quand vous en manipulez plusieurs,
|
|
il faut régler le problème.
|
|
Pour ça, il y a plusieurs méthodes, en fonction de votre cas.
|
|
|
|
### `select_related`
|
|
|
|
La méthode la plus basique consiste à annoter le queryset,
|
|
avec la méthode `select_related()`.
|
|
En faisant ça, Django fera une jointure sur l'autre table
|
|
et demandera des informations en plus
|
|
à la db lors de la requête.
|
|
|
|
De la sorte, lorsque vous appellerez le membre relié,
|
|
les informations seront déjà là.
|
|
|
|
```python
|
|
from core.models import User
|
|
|
|
richest = User.objects.order_by("-customer__amount")
|
|
for user in richest.select_related("customer")[:100]:
|
|
print(user.customer)
|
|
```
|
|
|
|
Le code ci-dessus effectue une seule requête.
|
|
Chaque fois qu'on veut accéder à `customer`, c'est bon,
|
|
ça a déjà été récupéré à travers le `select_related`.
|
|
|
|
### `prefetch_related`
|
|
|
|
Maintenant, un cas plus compliqué.
|
|
Supposons que vous ne vouliez pas récupérer des informations
|
|
reliées par une relation One-to-One,
|
|
mais par une relation One-to-Many.
|
|
|
|
Par exemple, un utilisateur a un seul compte client,
|
|
mais il peut avoir plusieurs cotisations à son actif.
|
|
Et dans ces cas-là, `annotate` ne marche plus.
|
|
En effet, s'il peut exister plusieurs cotisations,
|
|
comment savoir laquelle on veut ?
|
|
|
|
Il faut alors utiliser un `prefetch_related`.
|
|
C'est un mécanisme un peu différent :
|
|
au lieu de faire une jointure et d'ajouter les informations
|
|
voulues dans la même requête, Django va effectuer
|
|
une deuxième requête pour récupérer les éléments de l'autre table,
|
|
puis, à partir de ces éléments, peupler la relation
|
|
de son côté.
|
|
|
|
C'est un mécanisme qui peut être un peu coûteux en mémoire
|
|
et qui demande une deuxième requête,
|
|
mais qui reste quand même largement préférable
|
|
à faire N requêtes en plus.
|
|
|
|
```python
|
|
from core.models import User
|
|
|
|
for user in User.objects.prefetch_related("subscriptions")[:100]:
|
|
# c'est bon, la méthode prefetch a récupéré en avance les `subscriptions`
|
|
print(user.subscriptions.all())
|
|
```
|
|
|
|
!!! danger
|
|
|
|
La méthode `prefetch_related` ne marche que si vous
|
|
utilisez la méthode `all()` pour accéder au membre.
|
|
Si vous utilisez une autre méthode (comme `filter` ou `annotate`),
|
|
alors Django effectuera une nouvelle requête,
|
|
et vous retomberez dans le problème initial.
|
|
|
|
```python
|
|
from core.models import User
|
|
from django.db.models import Count
|
|
|
|
for user in User.objects.prefetch_related("subscriptions")[:100]:
|
|
# Le prefetch_related ne marche plus !
|
|
print(user.subscriptions.annotate(count=Count("*")))
|
|
```
|
|
|
|
### Récupérer ce dont vous avez besoin
|
|
|
|
Des fois (souvent, même), penser explicitement
|
|
à la jointure est le meilleur choix.
|
|
|
|
En effet, vous remarquerez que dans tous
|
|
les exemples précédents, nous n'utilisions
|
|
qu'une partie des informations
|
|
(par exemple, nous ne récupérions que la somme
|
|
d'argent sur les comptes, et éventuellement le numéro de compte).
|
|
|
|
Nous pouvons utiliser la méthode `annotate`
|
|
pour spécifier explicitement les données que l'on veut
|
|
joindre à notre requête.
|
|
|
|
Quand nous voulions récupérer les informations utilisateur,
|
|
nous aurions tout aussi bien pu écrire :
|
|
|
|
```python
|
|
from core.models import User
|
|
from django.db.models import F
|
|
|
|
richest = User.objects.order_by("-customer__amount")
|
|
for user in richest.annotate(amount=F("customer__amount"))[:100]:
|
|
print(user.amount)
|
|
```
|
|
|
|
On aurait même pu réorganiser ça :
|
|
```python
|
|
|
|
from core.models import User
|
|
from django.db.models import F
|
|
|
|
richest = User.objects.annotate(amount=F("customer__amount")).order_by("-amount")
|
|
for user in richest[:100]:
|
|
print(user.amount)
|
|
```
|
|
|
|
Ça peut sembler moins bien qu'un `select_related`, comme ça.
|
|
Des fois, c'est en effet moins bien, et des fois c'est mieux.
|
|
La comparaison est plus évidente avec le `prefetch_related`.
|
|
|
|
En effet, quand nous voulions récupérer
|
|
le nombre de cotisations des utilisateurs,
|
|
le `prefetch_related` ne marchait plus.
|
|
Pourtant, nous voulions récupérer une seule information.
|
|
|
|
Il aurait donc été suffisant d'écrire :
|
|
```python
|
|
from core.models import User
|
|
from django.db.models import Count
|
|
|
|
for user in User.objects.annotate(nb_subscriptions=Count("subscriptions"))[:100]:
|
|
# Et là ça marche, en une seule requête.
|
|
print(user.nb_subscriptions)
|
|
```
|
|
|
|
Faire une jointure, c'est normal en SQL.
|
|
Et pourtant avec Django on les oublie trop facilement.
|
|
Posez-vous toujours la question des données que vous pourriez
|
|
avoir besoin d'annoter, et vous éviterez beaucoup d'ennuis.
|
|
|
|
## Les aggrégations manquées
|
|
|
|
Il arrive souvent que l'on veuille une information qui
|
|
porte sur un ensemble d'objets de notre db.
|
|
|
|
Imaginons par exemple que nous voulons connaitre
|
|
la somme totale des ventes faites à un comptoir.
|
|
|
|
Nous avons tous suivi nos cours de programmation,
|
|
nous écrivons donc instinctivement :
|
|
|
|
```python
|
|
from counter.models import Counter
|
|
|
|
foyer = Counter.objects.get(name="Foyer")
|
|
total_amount = sum(
|
|
sale.amount * sale.unit_price
|
|
for sale in foyer.sellings.all()
|
|
)
|
|
```
|
|
|
|
On pourrait penser qu'il n'y a pas de problème.
|
|
Après tout, on ne fait qu'une seule requête.
|
|
Eh bien si, il y a un problème :
|
|
on fait beaucoup de choses en trop.
|
|
|
|
Concrètement, on demande à la base de données
|
|
de renvoyer toutes les informations,
|
|
ce qui rallonge inutilement la durée
|
|
de l'échange entre le serveur et la db,
|
|
puis on perd du temps à convertir ces informations
|
|
en objets Python (opération qui a un coût également),
|
|
et enfin on reperd du temps à calculer en Python
|
|
quelque chose que la db aurait pu calculer
|
|
à notre plus bien plus vite.
|
|
|
|
Nous aurions dû aggréger la requête,
|
|
avec la méthode `aggregate` :
|
|
|
|
```python
|
|
from counter.models import Counter
|
|
from django.db.models import Sum, F
|
|
|
|
foyer = Counter.objects.get(name="Foyer")
|
|
total_amount = (
|
|
foyer.sellings.aggregate(amount=Sum(F("amount") * F("unit_price"), default=0))
|
|
)["amount__sum"]
|
|
```
|
|
|
|
En effectuant cette requête, la base de données nous renverra exactement
|
|
l'information dont nous avons besoin.
|
|
Et de notre côté, nous n'aurons pas à faire de traitement en plus.
|
|
|
|
|
|
## Benchmark
|
|
|
|
### Ce qu'il faut mesurer
|
|
|
|
Quand on parle d'interaction avec une base de données,
|
|
la question de la performance est cruciale.
|
|
Et quand on parle de performance, on en vient
|
|
forcément à parler d'optimisation.
|
|
|
|
Or, pour optimiser, il faut savoir quoi optimiser.
|
|
C'est-à-dire qu'il nous faut un benchmark pour
|
|
étudier les performances réelles de notre code.
|
|
En ce qui concerne des requêtes à une base de données,
|
|
deux aspects sont étudiables :
|
|
|
|
- le nombre de requêtes qu'une vue ou une fonction
|
|
effectue pour son fonctionnement.
|
|
- le temps d'exécution individuel des requêtes les plus longues.
|
|
|
|
Le premier aspect est celui qui nous intéresse le plus,
|
|
puisqu'il est relié au problème le plus fréquent
|
|
et le plus facile à mesurer.
|
|
Le second aspect, au contraire, est bien moins fréquent
|
|
(dans 99% des cas, une requête complexe prendra
|
|
moins de temps que deux requêtes, même simples)
|
|
et bien plus dur à mesurer (il faut réussir à faire des mesures fiables,
|
|
dans un environnement proche de celui de la prod, avec les données de la prod).
|
|
|
|
Nous considérerons donc que dans la quasi-totalité des cas,
|
|
le problème vient du nombre de requêtes, pas du temps d'exécution
|
|
d'une requête en particulier.
|
|
Partez du principe que moins vous faites de requêtes, mieux c'est,
|
|
sans prêter attention au temps d'exécution des requêtes.
|
|
|
|
Pour quantifier de manière fiables les requêtes effectuées,
|
|
il y a quelques outils.
|
|
|
|
### `django-debug-toolbar`
|
|
|
|
La `django-debug-toolbar` est une interface disponible
|
|
sur toutes les pages quand vous êtes en mode debug.
|
|
Elle s'affiche à droite et vous permet de voir toutes sortes
|
|
d'informations, parmi lesquelles le nombre de requêtes effectuées.
|
|
|
|
Cette interface est très pratique, puisqu'elle va plus loin
|
|
que simplement compter les requêtes,
|
|
elle vous donne également le SQL qui a été utilisé,
|
|
l'endroit du code, avec fichier et numéro de ligne,
|
|
où cette requête a été faite et, encore mieux,
|
|
elle vous indique quelles requêtes semblent dupliquées.
|
|
|
|
Quand `django-debug-toolbar` vous indique qu'une requête
|
|
a été dupliquée quatre fois, cinq fois, ou même deux cent fois
|
|
(le chiffre peut sembler énorme, mais c'est déjà arrivé),
|
|
vous pouvez être sûr qu'il y a là quelque chose à optimiser.
|
|
|
|
!!!warning
|
|
|
|
Le widget de `django-debug-toolbar` ne s'affiche
|
|
que sur les pages html.
|
|
Si vous voulez étudier autre chose,
|
|
comme une simple fonction,
|
|
ou bien comme une vue retournant du JSON,
|
|
vous n'aurez donc pas `django-debug-toolbar`.
|
|
|
|
|
|
### `connection.queries`
|
|
|
|
Quand vous voulez examiner les requêtes d'un bout de code
|
|
en particulier, Django met à disposition un mécanisme
|
|
permettant d'examiner toutes les requêtes qui sont faites :
|
|
`connection.queries`
|
|
|
|
C'est un historique de toutes les requêtes effectuées,
|
|
qui est assez simple à utiliser :
|
|
|
|
```python
|
|
from django.db import connection
|
|
from core.models import User
|
|
|
|
print(len(connection.queries)) # 0
|
|
|
|
nb_users = User.objects.count()
|
|
|
|
print(len(connection.queries)) # 1
|
|
print(connection.queries) # affiche toutes les requêtes effectuées
|
|
```
|
|
|
|
### `assertNumQueries`
|
|
|
|
Quand on a mis en place une fonctionnalité,
|
|
ou qu'on en a amélioré les performances,
|
|
on veut absolument éviter la régression.
|
|
|
|
Or, une régression ne se manifeste pas forcément
|
|
dans l'apparition d'un bug : ça peut aussi
|
|
être une augmentation du temps d'exécution, possiblement
|
|
causé par une augmentation du nombre de requêtes.
|
|
|
|
C'est pour ça que django met à disposition un moyen
|
|
de tester automatiquement le nombre de requêtes :
|
|
`assertNumQueries`.
|
|
|
|
Il s'agit d'un gestionnaire de contexte accessible
|
|
dans les tests, qui teste le nombre de requêtes
|
|
effectuées en son sein.
|
|
|
|
Par exemple :
|
|
|
|
```python
|
|
from django.test import TestCase
|
|
from django.shortcuts import reverse
|
|
|
|
|
|
class FooTest(TestCase):
|
|
def test_nb_queries(self):
|
|
"""Test that the number of db queries is stable."""
|
|
with self.assertNumQueries(6):
|
|
self.client.get(reverse("foo:bar"))
|
|
```
|
|
|
|
Si l'exécution de la route nécessite plus ou moins de six requêtes,
|
|
alors le test échoue.
|
|
S'il y a eu moins que le nombre de requête attendu, alors tant
|
|
mieux, modifiez le test pour coller au nouveau nombre
|
|
(sous réserve que tous les autres tests passent, bien sûr).
|
|
Si par contre il y a eu plus, alors désolé, vous avez sans doute
|
|
introduit une régression.
|
|
|
|
|