Meta
des modèles¶Ce document explique toutes les options de métadonnées qu’il est possible d’attribuer aux modèles dans leur classe Meta
interne.
Meta
disponibles¶abstract
¶Options.
abstract
¶Si abstract = True
, ce modèle sera une classe de base abstraite.
app_label
¶Options.
app_label
¶Si un modèle est défini en dehors d’une application dans INSTALLED_APPS
, il doit déclarer à quelle application il appartient :
app_label = 'myapp'
Si vous souhaitez représenter un modèle avec le format étiquette_app.nom_objet
ou étiquette_app.nom_modèle
, vous pouvez respectivement utiliser model._meta.label
ou model._meta.label_lower
.
base_manager_name
¶Options.
base_manager_name
¶Le nom d’attribut du gestionnaire à utiliser pour l’attribut _base_manager
du modèle, par exemple 'objects'
.
db_table
¶Options.
db_table
¶Le nom de la table de base de données à utiliser pour le modèle :
db_table = 'music_album'
Pour vous faire gagner du temps, Django dérive automatiquement le nom de la table de base de données du nom de la classe de modèle et de l’application dans laquelle elle se trouve. Un nom de table de base de données d’un modèle est construit en adjoignant le « label » d’application du modèle (le nom utilisé lors de la commande manage.py startapp
) au nom de classe du modèle, combinés par un soulignement.
Par exemple, pour une application bibliotheque
(qui serait créée avec manage.py startapp bibliotheque
) et un modèle défini par class Livre
, le nom de la table de base de données serait bibliotheque_livre
.
Pour surcharger le nom de la table de base de données, utilisez l’attribut db_table
de la classe Meta
.
Si votre nom de table de base de données est un mot SQL réservé ou s’il contient des caractères non autorisés dans des noms de variables Python (en particulier le trait d’union), il n’y a pas de souci à se faire. Django place entre guillemets les noms de colonnes et de tables en arrière-plan.
Noms de tables en minuscules pour MariaDB et MySQL
Il est fortement conseillé d’écrire les noms de tables en minuscules lors de leur surcharge avec db_table
, particulièrement lorsqu’on utilise le moteur MySQL. Voir les notes pour MySQL pour plus de détails.
Noms de tables entre guillemets pour Oracle
Afin de respecter la limite de 30 caractères qu’Oracle impose aux noms de tables et pour suivre les conventions habituelles des bases de données Oracle, Django peut raccourcir les noms de tables et les transformer en majuscules. Pour empêcher ces transformations, écrivez des valeurs entre guillemets dans db_table
:
db_table = '"name_left_in_lowercase"'
Il est aussi possible d’utiliser de tels noms entre guillemets avec les autres moteurs de base de données pris en charge par Django ; cependant, contrairement à Oracle, les guillemets n’ont pas d’effet. Voir les notes sur Oracle pour plus de détails.
db_tablespace
¶Options.
db_tablespace
¶Le nom de l”espace de tables de base de données à utiliser pour ce modèle. La valeur par défaut est le réglage DEFAULT_TABLESPACE
du projet, s’il est défini. Si le moteur ne prend pas en charge les espaces de tables, cette option est ignorée.
default_manager_name
¶Options.
default_manager_name
¶Le nom du gestionnaire à utiliser pour l’attribut _default_manager
du modèle.
get_latest_by
¶Options.
get_latest_by
¶Le nom d’un champ ou une liste de noms de champs du modèle, typiquement un champ DateField
, DateTimeField
ou IntegerField
. Cette option définit le ou les champs par défaut à utiliser dans les méthodes latest()
et earliest()
du gestionnaire (Manager
) du modèle.
Exemple :
# Latest by ascending order_date.
get_latest_by = "order_date"
# Latest by priority descending, order_date ascending.
get_latest_by = ['-priority', 'order_date']
Voir la documentation de latest()
pour plus de détails.
managed
¶Options.
managed
¶La valeur par défaut est True
, signifiant que Django crée les tables de base de données adéquates dans migrate
ou dans le cadre des migrations et les efface dans le contexte de la commande d’administration flush
. C’est-à-dire que Django pilote le cycle de vie des tables de la base de données.
Si la valeur est False
, aucune opération de création, de modification ou de suppression de table de base de données n’est effectuée pour ce modèle. Utile si le modèle représente une table existante ou une vue de base de données qui a été créée par un autre moyen. C’est l’unique différence lorsque managed=False
. Tous les autres aspects de la gestion des modèles sont traités comme pour un modèle normal. Cela inclut :
L’ajout automatique d’un champ de clé primaire au modèle si aucun n’est déclaré. Pour enlever toute confusion possible aux lecteurs de votre code, il est recommandé d’indiquer toutes les colonnes de la table de base de données que vous modélisez en utilisant des modèles non pilotés.
Si un modèle avec managed=False
contient un champ ManyToManyField
qui pointe vers un autre modèle non piloté, la table intermédiaire de la relation plusieurs-à-plusieurs ne sera pas non plus créée. Cependant, la table intermédiaire entre un modèle piloté et un modèle non piloté sera bel et bien créée.
Si vous aimeriez changer ce comportement par défaut, créez explicitement le modèle de la table intermédiaire (avec l’option managed
appropriée) et utilisez l’attribut ManyToManyField.through
pour que la relation utilise le modèle ainsi créé.
En ce qui concerne les tests impliquant des modèles avec managed=False
, c’est à vous de vous assurer que les tables correctes soient créées dans le cadre de la mise en place des tests.
Si vous êtes intéressé à modifier le comportement Python d’une classe de modèle, vous pourriez utiliser managed=False
et créer une copie d’un modèle existant. Cependant, il existe une meilleure approche pour ce genre de situation : Modèles mandataires.
order_with_respect_to
¶Options.
order_with_respect_to
¶Rend cet objet « triable » en fonction du champ donné, presque toujours un champ ForeignKey
. Cela peut être utilisé pour permettre à des objets liés d’être triés en fonction d’un objet parent. Par exemple, si un objet Answer
(réponse) est lié à un objet Question
et qu’une question possède plus d’une réponse et que l’ordre des réponses est signifiant, voici comment cela serait défini :
from django.db import models
class Question(models.Model):
text = models.TextField()
# ...
class Answer(models.Model):
question = models.ForeignKey(Question, on_delete=models.CASCADE)
# ...
class Meta:
order_with_respect_to = 'question'
Lorsque order_with_respect_to
est défini, deux méthodes complémentaires sont fournies pour récupérer et définir l’ordre des objets liés : get_RELATED_order()
et set_RELATED_order()
, où RELATED
est le nom du modèle en minuscules. Par exemple, en supposant qu’un objet Question
possède plusieurs objets Answer
liés, la liste renvoyée contient les clés primaires des objets Answer
liés :
>>> question = Question.objects.get(id=1)
>>> question.get_answer_order()
[1, 2, 3]
L’ordre des objets liés Answer
d’un objet Question
peut être défini en passant une liste de clés primaires d’objets Answer
:
>>> question.set_answer_order([3, 1, 2])
Les objets liés reçoivent aussi deux méthodes, get_next_in_order()
et get_previous_in_order()
, qui peuvent être utilisées pour accéder à ces objets dans l’ordre correct. En supposant que les objets Answer
sont triés par id
:
>>> answer = Answer.objects.get(id=2)
>>> answer.get_next_in_order()
<Answer: 3>
>>> answer.get_previous_in_order()
<Answer: 1>
order_with_respect_to
définit implicitement l’option ordering
.
En interne, order_with_respect_to
ajoute un champ (colonne de base de données) supplémentaire nommé _order
et définit l’option ordering
du modèle avec ce champ. Par conséquent, order_with_respect_to
et ordering
ne peuvent pas être utilisés conjointement, et l’ordre de tri ajouté par order_with_respect_to
s’applique chaque fois que vous obtenez une liste d’objets pour ce modèle.
Modification de order_with_respect_to
Comme order_with_respect_to
ajoute une colonne de base de données supplémentaire, il vous revient donc de créer et d’appliquer les migrations appropriées si vous ajoutez ou modifiez order_with_respect_to
après la création initiale opérée par migrate
.
ordering
¶Options.
ordering
¶Le tri par défaut des objets, lors de l’obtention de listes d’objets :
ordering = ['-order_date']
Il s’agit d’un tuple ou d’une liste de chaînes, ou d’expressions de requête. Chaque chaîne correspond à un nom de champ, facultativement préfixée par « - » indiquant un tri descendant. Les champs sans préfixe « - » sont triés dans l’ordre ascendant. Indiquez la chaîne « ? » pour trier de façon aléatoire.
Par exemple, pour trier sur un champ pub_date
dans l’ordre ascendant, indiquez :
ordering = ['pub_date']
Pour trier par pub_date
dans l’ordre descendant, indiquez :
ordering = ['-pub_date']
Pour trier par pub_date
dans l’ordre descendant, puis par author
dans l’ordre ascendant, indiquez :
ordering = ['-pub_date', 'author']
Il est aussi possible d’utiliser des expressions de requête. Pour trier par author
dans l’ordre croissant et pour que les valeurs nulles apparaissent en dernier, écrivez ceci
from django.db.models import F
ordering = [F('author').asc(nulls_last=True)]
Avertissement
Le tri n’est pas une opération anodine. Chaque champ ajouté dans les critères de tri implique un coût au niveau de la base de données. Chaque clé étrangère ajoutée inclut aussi implicitement tous ses champs de tri par défaut.
Si une requête ne possède pas d’ordre de tri, les résultats sont renvoyés de la base de données dans un ordre non défini. Un ordre particulier ne peut être garanti que si l’ordre se base sur un ou des champs qui identifient chaque objet du résultat de manière unique. Par exemple, si un champ nom
n’est pas unique, le tri par ce champ ne garantit pas que les objets de même nom apparaissent toujours dans le même ordre.
permissions
¶Options.
permissions
¶Permissions supplémentaires à intégrer dans la table des permissions lors de la création de cet objet. Les permissions d’ajout, de modification, de suppression et d’affichage sont automatiquement créées pour tous les modèles. Cet exemple ajoute une permission supplémentaire, can_deliver_pizzas
:
permissions = [('can_deliver_pizzas', 'Can deliver pizzas')]
Il s’agit d’une liste ou d’un tuple de tuples binaires dans le format (code_permission, nom_verbeux_permission)
.
default_permissions
¶Options.
default_permissions
¶Contient ('add', 'change', 'delete', 'view')
par défaut. Vous pouvez ajuster cette liste, par exemple en définissant une liste vide si votre application n’a besoin d’aucune permission par défaut. Les permissions doivent être définies pour le modèle avant que celui-ci ne soit créé par migrate
afin d’éviter la création de permissions non utilisées.
proxy
¶Options.
proxy
¶Si proxy = True
, un modèle qui hérite d’un autre modèle sera considéré comme un modèle mandataire.
required_db_features
¶Options.
required_db_features
¶Liste de fonctionnalités de base de données que la connexion en cours devrait avoir pour que le modèle soit considéré pendant la phase de migration. Par exemple, si cette liste contient ['gis_enabled']
, le modèle ne sera synchronisé qu’avec les bases de données possédant des fonctionnalités SIG. Ceci est également utile pour omettre certains modèles lors de tests avec plusieurs moteurs de base de données différents. Évitez les relations avec ces modèles qui sont créés conditionnellement car l’ORM ne sait pas comment traiter ces cas.
required_db_vendor
¶Options.
required_db_vendor
¶Nom d’un fournisseur de base de données pour lequel ce modèle est spécifique. Les noms des fournisseurs intégrés actuels sont : sqlite
, postgresql
, mysql
et oracle
. Si cet attribut n’est pas vide et que le fournisseur de la connexion actuelle ne correspond pas, le modèle n’est pas synchronisé.
select_on_save
¶Options.
select_on_save
¶Détermine si Django utilise l’algorithme django.db.models.Model.save()
d’avant la version 1.6. L’ancien algorithme utilise SELECT
pour savoir s’il y a une ligne existant à mettre à jour. Le nouvel algorithme essaie directement un UPDATE
. Dans de rares cas, la mise à jour par UPDATE
d’une ligne existante n’est pas visible par Django. Un des exemples est le déclencheur ON UPDATE
de PostgreSQL qui renvoie NULL
. Dans de telles situations, le nouvel algorithme effectue finalement un INSERT
même quand une ligne existe déjà dans la base de données.
Il n’y a habituellement pas besoin de définir cet attribut. La valeur par défaut est False
.
Voir django.db.models.Model.save()
pour plus de détails sur l’ancien et le nouvel algorithme d’enregistrement.
indexes
¶Options.
indexes
¶Une liste des index que vous souhaitez définir pour le modèle :
from django.db import models
class Customer(models.Model):
first_name = models.CharField(max_length=100)
last_name = models.CharField(max_length=100)
class Meta:
indexes = [
models.Index(fields=['last_name', 'first_name']),
models.Index(fields=['first_name'], name='first_name_idx'),
]
unique_together
¶Options.
unique_together
¶Noms de champs qui forment ensemble des index uniques :
unique_together = [['driver', 'restaurant']]
Il s’agit d’une liste de listes dont les valeurs combinées doivent être uniques. Utilisé dans l’administration de Django et appliqué au niveau base de données (c’est-à-dire que les commandes UNIQUE
appropriées sont inclues dans les commandes CREATE TABLE
).
Par commodité, unique_together
peut être formé d’une seule liste lorsqu’un seul groupe de champs est défini :
unique_together = ['driver', 'restaurant']
Il n’est pas possible d’inclure un champ ManyToManyField
dans unique_together
(il n’est même pas évident de savoir ce que ça signifierait). Si vous avez besoin de valider de l’unicité au niveau d’un champ ManyToManyField
, essayez d’utiliser un signal ou un modèle through
explicite.
L’exception ValidationError
générée durant la validation du modèle lorsque la contrainte est violée possède le code d’erreur unique_together
.
index_together
¶Options.
index_together
¶Noms de champs qui forment ensemble des index :
index_together = [
["pub_date", "deadline"],
]
Cette liste de champs formera un index combiné (c’est-à-dire que la commande CREATE INDEX
appropriée sera produite).
Par commodité, index_together
peut être formé d’une seule liste lorsqu’un seul groupe de champs est défini :
index_together = ["pub_date", "deadline"]
constraints
¶Options.
constraints
¶Une liste des contraintes que vous souhaitez définir pour le modèle :
from django.db import models
class Customer(models.Model):
age = models.IntegerField()
class Meta:
constraints = [
models.CheckConstraint(check=models.Q(age__gte=18), name='age_gte_18'),
]
verbose_name
¶Options.
verbose_name
¶Un nom verbeux pour l’objet, au singulier :
verbose_name = "pizza"
Si cette option est absente, Django manipule le nom de la classe : CamelCase
devient camel case
.
verbose_name_plural
¶Options.
verbose_name_plural
¶Le nom verbeux pluriel pour l’objet :
verbose_name_plural = "stories"
Si cette option est absente, Django utilise verbose_name
et y ajoute un « s ».
déc. 07, 2021