Skip to main content

A library for writing reliable data invariants in Django.

Project description

Django Queryset Constraint

Build Status Release Status License Python Versions Django Versions

This library enables one to write reliable data invariants, by compiling Django Querysets to database insert/update triggers, via. the migration system.

Motivation

Django has a built-in signal system, which emits signals on various events, for instance model creations, updates and deletions. However these signals are not emmited for queryset operations, and as such cannot be used to maintain data invariants.

An attempt to ratify this was made with the Django Queryset Signals library. While this library comes closer to a reliable solution, it does not succeed, as it is stil possible to break the data invariants by accessing the database directly.

Database Constraint Triggers will effectively protect against all scenarios.

Installation

pip install django_queryset_constraint

Usage

  • Add the django_queryset_constraint app to INSTALLED_APPS:
# settings.py
INSTALLED_APPS = {
    'django_queryset_constraint',
    ...
}

Note: This should be done before any apps that will be checked

  • Add QuerysetConstraint to constraints to the Meta class of the models to checked:
# models.py
from django.db import models
from django.db.models import Count
from django_queryset_constraint import M, QuerysetConstraint


class Topping(models.Model):
    name = models.CharField(max_length=30)


class PizzaTopping(models.Model):
    class Meta:
        unique_together = ("pizza", "topping")
        constraints = [
            # A pizza with more than 5 toppings gets soggy
            QuerysetConstraint(
                name='At most 5 toppings',
                queryset=M().objects.values(
                    'pizza'
                ).annotate(
                    num_toppings=Count('topping')
                ).filter(
                    num_toppings__gt=5
                ),
            ),
            # This constraint should be self-explanatory for civilized people
            QuerysetConstraint(
                name='No pineapple',
                queryset=M().objects.filter(
                    topping__name="Pineapple"
                )
            ),
        ]

    pizza = models.ForeignKey('Pizza', on_delete=models.CASCADE)
    topping = models.ForeignKey(Topping, on_delete=models.CASCADE)

class Pizza(models.Model):
    name = models.CharField(max_length=30)
    toppings = models.ManyToManyField(Topping, through=PizzaTopping)
  • Make migrations: python manage.py makemigrations
  • Run migrations: python manage.py migrate

Note: Complex triggers introduce performance overhead.

Support Matrix

This app supports the following combinations of Django and Python:

Django Python
2.2 3.6, 3.7

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

django_queryset_constraint-1.0.6.tar.gz (14.3 kB view details)

Uploaded Source

File details

Details for the file django_queryset_constraint-1.0.6.tar.gz.

File metadata

  • Download URL: django_queryset_constraint-1.0.6.tar.gz
  • Upload date:
  • Size: 14.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.22.0 setuptools/41.4.0 requests-toolbelt/0.9.1 tqdm/4.40.2 CPython/3.6.8

File hashes

Hashes for django_queryset_constraint-1.0.6.tar.gz
Algorithm Hash digest
SHA256 3f21a6892890f25396941043096b1bb1a8b0d592aa36789587fb5df1613336a7
MD5 df573f69d3d170b333a07e93bdd9fec6
BLAKE2b-256 fc9f56b1cc28a687b3f416b1c10c669fa8e859385a93de5b9dcebe94c403e8a2

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page