Options 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.
Options 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_objetouétiquette_app.nom_modèle, vous pouvez respectivement utilisermodel._meta.labeloumodel._meta.label_lower.
base_manager_name¶
-
Options.base_manager_name¶ Le nom du gestionnaire à utiliser pour l’attribut
_base_managerdu modèle.
db_table¶
-
Options.db_table¶ Le nom de la table de base de données à utiliser pour le modèle :
db_table = 'music_album'
Noms de tables¶
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 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_TABLESPACEdu 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_managerdu 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,DateTimeFieldouIntegerField. Cette option définit le ou les champs par défaut à utiliser dans les méthodeslatest()etearliest()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.Changed in Django 2.0:La prise en charge d’une liste de champs a été ajoutée.
managed¶
-
Options.managed¶ La valeur par défaut est
True, signifiant que Django crée les tables de base de données adéquates dansmigrateou dans le cadre des migrations et les efface dans le contexte de la commande d’administrationflush. 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 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 lorsquemanaged=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=Falsecontient un champManyToManyFieldqui 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
managedappropriée) et utilisez l’attributManyToManyField.throughpour 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=Falseet 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 objetAnswer(réponse) est lié à un objetQuestionet 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_toest 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()etset_RELATED_order(), oùRELATEDest le nom du modèle en minuscules. Par exemple, en supposant qu’un objetQuestionpossède plusieurs objetsAnswerliés, la liste renvoyée contient les clés primaires des objetsAnswerliés :>>> question = Question.objects.get(id=1) >>> question.get_answer_order() [1, 2, 3]
L’ordre des objets liés
Answerd’un objetQuestionpeut être défini en passant une liste de clés primaires d’objetsAnswer:>>> question.set_answer_order([3, 1, 2])
Les objets liés reçoivent aussi deux méthodes,
get_next_in_order()etget_previous_in_order(), qui peuvent être utilisées pour accéder à ces objets dans l’ordre correct. En supposant que les objetsAnswersont triés parid:>>> 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_datedans l’ordre ascendant, indiquez :ordering = ['pub_date']
Pour trier par
pub_datedans l’ordre descendant, indiquez :ordering = ['-pub_date']
Pour trier par
pub_datedans l’ordre descendant, puis parauthordans l’ordre ascendant, indiquez :ordering = ['-pub_date', 'author']
Il est aussi possible d’utiliser des expressions de requête. Pour trier par
authordans l’ordre croissant et pour que les valeurs nulles apparaissent en dernier, écrivez cecifrom django.db.models import F ordering = [F('author').asc(nulls_last=True)]
Le tri par défaut affecte également les requêtes d’agrégation.
Changed in Django 2.0:La prise en charge des expressions de requête a été ajoutée.
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éé parmigrateafin d’éviter la création de permissions non utilisées.Changed in Django 2.1:La permission
viewa été ajoutée.
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,mysqletoracle. 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 utiliseSELECTpour savoir s’il y a une ligne existant à mettre à jour. Le nouvel algorithme essaie directement unUPDATE. Dans de rares cas, la mise à jour parUPDATEd’une ligne existante n’est pas visible par Django. Un des exemples est le déclencheurON UPDATEde PostgreSQL qui renvoieNULL. Dans de telles situations, le nouvel algorithme effectue finalement unINSERTmê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’un tuple de tuples 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
UNIQUEappropriées sont inclues dans les commandesCREATE TABLE).Par commodité,
unique_togetherpeut être formé d’un seul tuple lorsqu’un seul groupe de champs est défini :unique_together = ("driver", "restaurant")
Il n’est pas possible d’inclure un champ
ManyToManyFielddansunique_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 champManyToManyField, essayez d’utiliser un signal ou un modèlethroughexplicite.L’exception
ValidationErrorgénérée durant la validation du modèle lorsque la contrainte est violée possède le code d’erreurunique_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 INDEXappropriée sera produite).Par commodité,
index_togetherpeut être formé d’une seule liste lorsqu’un seul groupe de champs est défini :index_together = ["pub_date", "deadline"]
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 :
CamelCasedevientcamel 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_nameet y ajoute un « s ».