Sith/docs/howto/querysets.md
2024-07-21 00:57:15 +02:00

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.