Framework commun
Boîte à outils commune pour des développements autour de Django (https://www.djangoproject.com/) et
Django REST Framework (http://www.django-rest-framework.org/).
Django
Entités
Une entité est un super-modèle Django offrant un nombre important de nouvelles fonctionnalités tels que l'historisation
des données, la réversion, les métadonnées, la sérialisation et la représentation des données sous forme de
dictionnaire.
Pour utiliser ces fonctionnalités, il est nécessaire de faire hériter chaque modèle de common.models.Entity
qui
hérite lui-même de django.db.models.Model
.
from common.models import Entity
class Personne(Entity):
pass
Les modèles entités possèdent par défaut les champs suivants :
creation_date
: date de création de l'entitémodification_date
: date de dernière modification de l'entitécurrent_user
: dernier utilisateur à l'origine de la création/modificationuuid
: identifiant unique de l'entité
L'identifiant unique permet de retrouver n'importe quelle entité depuis la base de données car chacune est
enregistrée dans un référentiel global conjointement à son type. Il est donc possible de récupérer une entité ainsi :
from common.models import Global
entity = Global.objects.from_uuid('4b9abebd-8157-4e49-bcae-1a7e063a9f86')
Attention ! Les entités surchargent les méthodes de persistance par défaut de Django
(save()
, create()
, delete()
).
update()
garde cependant son comportement par défaut car il exécute directement la mise à jour en base de
données, il sera donc impossible de détecter les changements si elle est utilisée.
Une version allégée d'entité est à disposition sans l'historisation ni le référentiel global, il suffit alors
d'hériter les modèles de common.models.CommonModel
à la place de common.models.Entity
.
Entités périssables
Une entité périssable possède une durée de validité dans le temps, elle possède les mêmes fonctionnalités que l'entité
mais possède en prime une date de début (start_date
) et une date de fin potentielle (end_date
).
Pour définir une entité périssable, il suffit de faire hériter le modèle de common.models.PerishableEntity
au lieu
de common.models.Entity
. Il n'existe pas d'alternative simplifiée hors historisation des entités périssables.
Lorsqu'une entité périssable existante en base de données est sauvegardée, le mécanisme suivant s'exécute :
- si une date de fin est fournie, l'entité courante est modifiée et clôturée à cette date
- sans date de fin fournie, l'entité courante est clôturée à cette date et une autre entité avec les modifications est
créée avec une date de début actualisée et sans date de fin programmée
Les entités périssables implémentent une fonction de récupération des données par rapport à une date donnée, si la date
n'est pas fournie la requête récupérera toutes les entités qui sont valides par rapport à la date et heure courante.
adresses = Adresse.objects.filter(personne_id=1).select_valid(date='2016-08-01T00:00:00')
Administration
Afin de garantir les fonctionnalités de l'historisation dans l'interface d'administration, il est nécessaire de faire
hériter les classes d'administration de common.admin.EntityAdmin
qui permet les fonctionnalités suivantes :
- Gère automatiquement l'utilisateur connecté pour les modifications
- Affiche automatiquement la date de création et de modification et fournit les filtres de recherche
- Gère les permissions d'affichage
- Affiche les métadonnées de l'entité
Pour les entités périssables, il convient d'utiliser common.admin.PerishableEntityAdmin
à la place de
common.admin.EntityAdmin
.
Pour les inlines, des équivalents pour les entités sont également à disposition :
common.admin.EntityTabularInline
et common.admin.EntityStackedInline
.
Une version allégée existe pour common.models.CommonModel
qui est common.admin.CommonAdmin
qui implémente
l'ensemble des fonctionnalités en dehors de l'historisation.
Historisation
Toute altération au niveau des données d'un modèle entité est détectée et enregistrée dans un historique,
cela permet un suivi des modifications par utilisation ainsi que la possibilité de revenir à un état antérieur pour
l'entité ou un champ spécifique de l'entité.
A chaque modification, un historique d'entité (common.models.History
) est sauvegardé pour l'entité et un élément
par champ modifié dans l'historique (common.models.HistoryField
) uniquement s'il s'agit d'une modification.
- Les relations de type many-to-many sont traités séparément à l'historisation.
- Si Celery est installé (http://www.celeryproject.org/), l'historisation est exécutée de manière asynchrone.
- L'utilisateur à l'origine de la modification est conservé dans l'historique.
- Il est possible d'ajouter un message et/ou modifier l'utilisateur à l'historique via le code.
personne = Personne(nom='Marc', age=30)
personne.save(_current_user=utilisateur, _reason="Message d'information")
Il est possible sur un historique d'entité ou un historique de champ de demander une restauration des données de
l'historique ciblé sur l'entité concernée grâce à la méthode restore()
, il est possible de lui passer
également un utilisateur et un message d'information.
La restauration provoque elle-même un historique spécifique qui permet de revenir en arrière si nécessaire.
history.restore(current_user=utilisateur, reason="Message d'information", rollback=False)
Métadonnées
Les métadonnées permettent d'ajouter des données tierces sur une entité, ces données peuvent être structurées comme
l'utilisateur le souhaitera et seront stockées sous forme de JSON dans la base de données. Chaque entité possède des
méthodes permettant d'accéder aux métadonnées de l'entité.
Les données stockées dans les métadonnées peuvent être de n'importe quel type Python à condition que ce dernier ainsi
que les données qu'il contient éventuellement soient sérialisables (list, dict, int, float, etc...).
personne.set_metadata('cle', 'valeur')
personne.get_metadata('cle')
>>> 'valeur'
personne.del_metadata('cle')
personne.get_metadata('cle')
>>> None
Les métadonnées d'une entité peuvent être directement utilisées dans des requêtes même si ce n'est pas recommandé pour
des raisons de performances (en fonction du volume de données).
Elles sont représentées sous forme d'un modèle Django common.models.MetaData
.
Personne.objects.filter(metadata__key='cle', metadata__value='valeur').first()
MetaData.objects.search(key='cle', value='valeur', type=Personne)
Sérialisation
Chaque entité ou requête concernant une entité peut être sérialisée en utilisant la méthode
serialize(format='json')
dans l'un des formats suivants:
personne = Personne.objects.first()
resultat = personne.serialize()
resultat
>>> [json] Personne (1)
resultat.data
>>> '[{"nom": "Marc", "age": 30}]'
personne = resultat.deserialize()
personne
>>> Personne: Marc (30 ans)
personnes = Personne.objects.all()
resultat = personnes.serialize()
resultat
>>> [json] Personne (2)
resultat.data
>>> '[{"nom": "Marc", "age": 30}, {"nom": "Eric", "age": 40}]'
personnes = resultat.deserialize()
personnes
>>> [Personne: Marc (30 ans), Personne: Eric (40 ans)]
Représentation en dictionnaire
Une requête ou une entité peut être représentée par un dictionnaire Python avec la méthode to_dict()
.
personne.to_dict()
>>> {"nom": "Marc", "age": 30}
Personne.objects.all().to_dict()
>>> [{"nom": "Marc", "age": 30}, {"nom": "Eric", "age": 40}]
to_dict()
prend plusieurs arguments optionnels en paramètres, la méthode est documentée dans le code.
Type d'entité
Django conserve systématiquement le type des différents modèles en base de données, les entités possèdent un moyen
simple pour récupérer ce type sur chaque entité de manière performante et unique.
Personne.get_model_type()
>>> <ContentType: Personne>
personne.model_type
>>> <ContentType: Personne>
WebHooks
Un webhook est un callback HTTP qui transmet des données à un serveur externe en fonction d'une action
réalisée sur l'application. Le récepteur doit être en mesure de comprendre le message qui est transmis.
Il est possible de configurer un webhook qui réagit à un ou plusieurs actions sur une ou plusieurs entités.
Les actions couvertes sont les suivantes :
- Création
- Modification
- Suppression
- Relations de type many-to-many
Un webhook peut être configuré pour utiliser une authentification sur le serveur externe si nécessaire.
Par défaut, le webhook exécute la méthode to_dict()
sur l'instance concerné avec les paramètres définis dans
NOTIFY_OPTIONS
, mais il est possible de changer ce comportement en définissant une méthode get_webhook_data
au
niveau du modèle.
Les notifications des changements par webhook est désactivée par défaut et peut être activé via NOTIFY_CHANGES
.
Usage de service
L'usage des services permet de compter le nombre de fois où une URL est appelée dans l'application par un même
utilisateur et même de limiter le nombre d'usages.
La fonctionnalité est désactivée par défaut et peut être activée via SERVICE_USAGE
.
Il est également nécessaire d'ajouter 'common.middleware.ServiceUsageMiddleware'
dans MIDDLEWARE_CLASSES
.
Configuration :
SERVICE_USAGE
(False
par défaut) : active ou non la surveillanceSERVICE_USAGE_LOG_DATA
(False
par défaut) : suit également les données transmisses par les servicesSERVICE_USAGE_LIMIT_ONLY
(False
par défaut) : utilise uniquement le système de restriction des services
Métadonnées utilisateurs & groupes
De la même manière que sur les entités, les utilisateurs et les groupes ont la possibilité de conserver de
l'information contextuelle sous forme de métadonnée, cependant le fonctionnement diffère car ces modèles peuvent être
substitués dans le développement.
Il s'agit alors d'une relation de type one-to-one systématiquement présent à la création d'un nouvel utilisateur ou
d'un nouveau groupe et possédant un champ de type JSON pour contenir ces données.
Utilitaires
Un grand nombre de fonctions, décorateurs et classes utilitaires sont à disposition des développeurs pour accélérer la
production de nouvelles fonctionnalités. Ces utilitaires sont regroupés dans common.utils
pour tout ce qui est
relatif à Django et dans common.api.utils
pour tout ce qui est relatif à Django REST Framework.
singleton
: décorateur permettant de transformer une classe en singletonget_current_app
: permet de récupérer l'application Celery actuelle ou un mock si Celery n'est pas installéparsedate
: permet d'évaluer une date dans n'importe quel formattimeit
: décorateur permettant de calculer le temps d'exécution d'une fonctionsynchronized
: décorateur permettant de rendre thread-safe l'exécution d'une fonctiontemporary_upload
: décorateur permettant à une vue de supporter l'import d'un fichier de manière temporairedownload_file
: décorateur permettant à une vue de supporter le téléchargement d'un fichierrender_to
: décorateur permettant de simplifier l'écriture d'une vue avec templateajax_request
: décorateur permettant à une vue de se comporter comme une api_viewevaluate
: permet d'évaluer une expression Python de manière plus sûreexecute
: permet d'exécuter du code Python de manière plus sûrepatch_settings
: context manager permettant d'altérer la configuration le temps de l'exécutionrecursive_dict_product
: permet de faire un produit cartésien des données d'un dictionnaireget_choices_fields
: permet de récupérer les choix des modèles d'une ou plusieurs applicationsget_prefetchs
: permet de récupérer toutes les relations inversées d'un modèleget_related
: permet de récupérer toutes les relations ascendantes d'un modèleprefetch_generics
: permet de récupérer les relations génériques d'un modèlestr_to_bool
: permet de convertir une chaîne de caractères quelconque en booléendecimal
: permet de convertir un élément quelconque en nombre décimaldecimal_to_str
: permet de convertir un nombre décimal en chaîne de caractèresrecursive_get_urls
: permet de récupérer toutes les URLs d'un moduleidict
: dictionnaire donc les clés sont toujours converties dans un format uniformesort_dict
: permet de trier un dictionnaire par ses clésmerge_dict
: permet de fusionner un ou plusieurs dictionnaires imbriqués sur un autrenull
: objet nul absolu retournant toujours une valeur nulle sans erreurto_tuple
: permet de convertir un dictionnaire en un tupleto_object
: permet de convertir un dictionnaire en un objet Pythonget_size
: permet de récupérer la taille en mémoire d'un objet Python quelconquefile_is_text
: permet de vérifier qu'un fichier est au format texteprocess_file
: permet de s'assurer qu'un fichier est bien complet et décompresse les éventuelles archivesbase64_encode
: permet d'encoder une chaîne de caractères en base 64base64_decode
: permet de décoder une chaîne de caractères en base 64short_identifier
: permet de générer un identifiant "unique" courtjson_encode
: permet de sérialiser un objet Python en JSONjson_decode
: permet de désérialiser une chaîne de caractères JSON en objet Pythonget_current_user
: permet de récupérer l'utilisateur actuellement connecté dans la pile d'exécutionget_pk_field
: permet de récupérer le champ de clé primaire d'un modèle en héritage concretcollect_deleted_data
: permet de récupérer les impacts potentiels d'une suppression d'entitésend_mail
: permet d'envoyer un emailmerge_validation_errors
: permet de fusionner plusieurs exceptions de validation en une seuleget_all_models
: récupère tous les modèles enregistrés dans les applicationsget_all_permissions
: récupère toutes les permissions existantes dans les applicationsget_models_from_queryset
: recupère tous les modèles qui ont été traversés par une requêteget_model_permissions
: récupère toutes les permissions liées à un modèle et à un utilisateurget_client_ip
: récupère l'adresse IP du client depuis une requête Djangohash_file
: calcule la somme de contrôle d'un fichier
Autres (common.admin
)
create_admin
: permet de créer automatiquement les classes d'administration d'un modèle
Modèles
Les differents utilitaires de modèles sont définis dans common.models
.
CustomGenericForeignKey
: champ générique qui ne vide pas les propriétés de la clé si l'instance n'existe pasCustomGenericRelation
: relation d'une clé étrangère générique qui force la conversion de l'identifiant en texteMetaData
: modèle des métadonnées associées aux entitésCommonModel
: modèle abstrait commun permettant d'accéder aux multiples utilitaires décrits précédemmentEntity
: modèle commun avec historisation des modificationsPerishableEntity
: même chose qu'Entity
mais se duplicant en cas de modificationHistory
: modèle d'historique d'une entitéHistoryField
: modèle d'historique d'un champ d'une entitéGlobal
: modèle point d'entrée global des entitésWebhook
: modèle de configuration d'un webhookServiceUsage
: modèle d'un historique d'accès à une ressource du site (avec ou sans limitation)from_dict
: construit une instance d'un modèle à partir d'un dictionnaireto_dict
: appel générique transformant une instance de modèle en dictionnairemodel_to_dict
: transforme récursivement une instance de modèle en dictionnaire
Champs de modèles
Les champs de modèle sont définis dans common.fields
.
CustomDecimalField
: champ décimal pour éviter la représentation scientifique des nombresPickleField
: champ binaire pour contenir de la donnée Python bruteJsonField
: champ pour représenter des données JSON (compatible avec les autres SGBD)
Formulaires
Les utilitaires autour des formulaires sont définis dans common.forms
. Chaque classe utilitaire pour les
formulaires possède une interface de base, préfixée généralement Base
pour d'autres implémentations.
CommonForm
: classe de base pour les formulaires avec gestion des historiquesCommonFormSet
: classe de base pour les ensembles de formulaires avec gestion des historiquesCommonModelForm
: classe de base pour représenter les formulaires issus d'entitésCommonModelFormSet
: classe de base pour représenter les ensembles de formulaires issus d'entitésJsonField
: champ de formulaire pour représenter les données JSONBaseFilterForm
: classe de base pour faciliter la création de formulaires de rechercheget_model_form
: fonction permettant de créer un formulaire d'entité avec des imbrications
Django REST Framework
Usage
La boîte à outils vous permet de générer rapidement des APIs RESTful pour vos modèles de base de données en leur
fournissant au passage le moyen de faire des requêtes plus évoluées via l'URL. Cette génération s'adresse
principalement aux développeurs qui ont besoin d'accéder à toute la richesse de leurs modèles et de l'ORM Django
via les APIs le plus rapidement possible sans avoir à configurer finement chaque ressource.
Configuration rapide
Dans votre application Django, créez un nouveau module et ajoutez ces quelques lignes :
from common.api.utils import create_api
from myapp.models import ModelA, ModelB
namespace = "myapp-api"
app_name = "myapp-api"
router, all_serializers, all_viewsets = create_api(ModelA, ModelB)
urlpatterns = [
] + router.urls
urls = (urlpatterns, namespace, app_name)
Cela a pour conséquence de créer automatiquement les APIs pour les modèles que vous souhaitez ainsi que la table de
routage nécessaire pour accéder à ces ressources.
De nombreuses options de configuration existent pour vos APIs, vous pouvez par exemple définir précisément le
serialiseur pour chaque modèle, configurer la récupération automatique des entités liées par clé étrangère (related) ou
des relations de type plusieurs-à-plusieurs (prefetch) ou personnaliser les requêtes.
Pour configurer globalement et de manière homogène vos modèles, modifiez les paramètres dans common.api.base
.
Guide d'utilisation rapide
Utilitaires
to_model_serializer
: décorateur permettant de convertir un sérialiseur classique en sérialiseur de modèleto_model_viewset
: décorateur permettant d'associer un modèle et à un sérialiseur à une vuecreate_model_serializer_and_viewset
: permet de créer en une fois le sérialiseur et la vue pour un modèleperishable_view
: permet de gérer les données périssables d'une vue à travers l'URLapi_view_with_serializer
: décorateur permettant de créer une vue assujettie à un sérialiseurcreate_model_serializer
: permet de créer un sérialiseur de modèleauto_view
: décorateur permettant de créer une vue à partir d'un QuerySetapi_paginate
: permet d'ajouter une pagination sur le résultat d'une requête pour une vuecreate_api
: permet de créer les APIs standards (RESTful) pour un ou plusieurs modèlesdisable_relation_fields
: permet de désactiver les listes déroulantes pour les relations des APIs
Sérialiseurs (common.api.serializers
)
CommonModelSerializer
: sérialiseur commun pour représenter les entitésGenericFormSerializer
: sérialiseur permettant l'imbrication d'entités pour les formulaires
Champs (common.api.fields
)
JsonField
: champ permettant la gestion des données JSONAsymetricRelatedField
: champ permettant l'affichage de l'ensemble des données des entités de clés étrangères
mais accepte néanmoins un simple identifiant à la création/modificationCustomHyperlinkedField
: champ permettant de gérer les liens vers d'autres APIs en permettant de croiser
les différents namespaces (utilisé par défaut par les utilitaires)
CustomPageNumberPagination
: pagination améliorée pour les APIs
(doit être défini dans DEFAULT_PAGINATION_CLASS
de REST_FRAMEWORK
)
Rendu (common.api.renderers
)
CustomCSVRenderer
: rendu CSV amélioré avec téléchargement (uniquement si django-rest-framework-csv est installé,
doit être défini dans DEFAULT_RENDERER_CLASSES
de REST_FRAMEWORK
)
Tests (common.tests
)
BaseApiTestCase
: classe de base pour les tests unitaires des APIsAuthenticatedBaseApiTestCase
: classe de base pour les tests unitaires des APIs avec authentificationcreate_api_test_class
: fonction pour générer tous les tests d'une API standard (RESTful)