Skip to main content

Postgres trigger support integrated with Django models.

Project description

django-pgtrigger

django-pgtrigger helps you write Postgres triggers for your Django models.

Why should I use triggers?

Triggers can solve a variety of complex problems more reliably, performantly, and succinctly than application code. For example,

  • Protecting operations on rows or columns (pgtrigger.Protect).
  • Making read-only models or fields (pgtrigger.ReadOnly).
  • Soft-deleting models (pgtrigger.SoftDelete).
  • Snapshotting and tracking model changes (django-pghistory).
  • Enforcing field transitions (pgtrigger.FSM).
  • Keeping a search vector updated for full-text search (pgtrigger.UpdateSearchVector).
  • Building official interfaces (e.g. enforcing use of User.objects.create_user and not User.objects.create).
  • Versioning models, mirroring fields, computing unique model hashes, and the list goes on...

All of these examples require no overridden methods, no base models, and no signal handling.

Quick start

Install django-pgtrigger with pip3 install django-pgtrigger and add pgtrigger to settings.INSTALLED_APPS.

pgtrigger.Trigger objects are added to triggers in model Meta. django-pgtrigger comes with several trigger classes, such as pgtrigger.Protect. In the following, we're protecting the model from being deleted:

import pgtrigger

class ProtectedModel(models.Model):
    """This model cannot be deleted!"""

    class Meta:
        triggers = [
            pgtrigger.Protect(name="protect_deletes", operation=pgtrigger.Delete)
        ]

When migrations are created and executed, ProtectedModel will raise an exception anytime a deletion is attempted.

Let's extend this example further and only protect deletions on inactive objects. In this example, the trigger conditionally runs when the row being deleted (the OLD row in trigger terminology) is still active:

import pgtrigger

class ProtectedModel(models.Model):
    """Active object cannot be deleted!"""
    is_active = models.BooleanField(default=True)

    class Meta:
        triggers = [
            pgtrigger.Protect(
                name="protect_deletes",
                operation=pgtrigger.Delete,
                condition=pgtrigger.Q(old__is_active=True)
            )
        ]

django-pgtrigger uses pgtrigger.Q and pgtrigger.F objects to conditionally execute triggers based on the OLD and NEW rows. Combining these Django idioms with pgtrigger.Trigger objects can solve a wide variety of problems without ever writing SQL. Users, however, can still use raw SQL for complex cases.

Triggers are installed like other database objects. Run python manage.py makemigrations and python manage.py migrate to install triggers.

If triggers are new to you, don't worry. The pgtrigger docs cover triggers in more detail and provide many examples.

Compatibility

django-pgtrigger is compatible with Python 3.10 - 3.14, Django 4.2 - 6.0, Psycopg 2 - 3, and Postgres 14 - 18.

Documentation

View the django-pgtrigger docs here to learn more about:

  • Trigger basics and motivation for using triggers.
  • How to use the built-in triggers and how to build custom ones.
  • Installing triggers on third-party models, many-to-many fields, and other advanced scenarios.
  • Writing conditional triggers.
  • Ignoring triggers dynamically and deferring trigger execution.
  • Multiple database, schema, and partitioning support.
  • Frequently asked questions, common issues, and upgrading.
  • The commands, settings, and module.

Installation

Install django-pgtrigger with:

pip3 install django-pgtrigger

After this, add pgtrigger to the INSTALLED_APPS setting of your Django project.

Other Material

After you've read the docs, check out this tutorial with interactive examples from a Django meetup talk.

The DjangoCon 2021 talk also breaks down triggers and shows several examples.

Contributing Guide

For information on setting up django-pgtrigger for development and contributing changes, view CONTRIBUTING.md.

Creators

Other Contributors

  • @jzmiller1
  • @rrauenza
  • @ralokt
  • @adamchainz
  • @danifus
  • @kekekekule
  • @peterthomassen
  • @pfouque

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_pgtrigger-4.17.0.tar.gz (32.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

django_pgtrigger-4.17.0-py3-none-any.whl (36.3 kB view details)

Uploaded Python 3

File details

Details for the file django_pgtrigger-4.17.0.tar.gz.

File metadata

  • Download URL: django_pgtrigger-4.17.0.tar.gz
  • Upload date:
  • Size: 32.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.2.1 CPython/3.14.0 Linux/6.8.0-1040-aws

File hashes

Hashes for django_pgtrigger-4.17.0.tar.gz
Algorithm Hash digest
SHA256 55d297a756b5b6a8e0c533d3323fda1589c699e7d14aa3fff7fa7d774c46363c
MD5 074fa9cf23e535bc0f6967bdeab9e5c4
BLAKE2b-256 453dea55015e2e2a8e2230c6a056768512a9660573f8926f87671f166d455936

See more details on using hashes here.

File details

Details for the file django_pgtrigger-4.17.0-py3-none-any.whl.

File metadata

  • Download URL: django_pgtrigger-4.17.0-py3-none-any.whl
  • Upload date:
  • Size: 36.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.2.1 CPython/3.14.0 Linux/6.8.0-1040-aws

File hashes

Hashes for django_pgtrigger-4.17.0-py3-none-any.whl
Algorithm Hash digest
SHA256 06048cdeb57e20987fd0bf7dd673512a1413a08868eea248d70dbb970e967175
MD5 00f142b50e410d8bf322005d08015a85
BLAKE2b-256 914713c0c23c134cd0f0b0addfbc3ef1477f90d891184d21689f9074b794b00c

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