Skip to main content

Detect N+1s in your Django app

Project description

django-zeal

Catch N+1 queries in your Django project.

Static Badge PyPI - Version

🔥 Battle-tested at Cinder

Features

  • Detects N+1s from missing prefetches and from use of .defer()/.only()/.get()
  • Friendly error messages like N+1 detected on social.User.followers at myapp/views.py:25 in get_user
  • Configurable thresholds
  • Allow-list
  • Well-tested
  • No dependencies
  • Fast; adds very little overhead to your test suite

Acknowledgements

This library draws heavily from jmcarp's nplusone. It's not a fork, but a lot of the central concepts and initial code came from nplusone.

Installation

First:

uv add django-zeal       # if using uv
pip install django-zeal  # if using pip

Then, add zeal to your INSTALLED_APPS and MIDDLEWARE.

if DEBUG:
    INSTALLED_APPS.append("zeal")
    MIDDLEWARE.append("zeal.middleware.zeal_middleware")

This will detect N+1s that happen in web requests. To catch N+1s in more places, read on!

[!WARNING] You probably don't want to run zeal in production: there is a slight overhead to detecting N+1s. Benchmarks show ~3-5% overhead on a typical workload.

Celery

If you use Celery, you can configure this using signals:

from celery.signals import task_prerun, task_postrun
from zeal import setup, teardown
from django.conf import settings

@task_prerun.connect()
def setup_zeal(*args, **kwargs):
    setup()

@task_postrun.connect()
def teardown_zeal(*args, **kwargs):
    teardown()

Tests

Django runs tests with DEBUG=False, so to run zeal in your tests, you'll first need to ensure it's added to your INSTALLED_APPS and MIDDLEWARE. You could do something like:

import sys

TEST = "test" in sys.argv
if DEBUG or TEST:
    INSTALLED_APPS.append("zeal")
    MIDDLEWARE.append("zeal.middleware.zeal_middleware")

This will enable zeal in any tests that go through your middleware. If you want to enable it in all tests, you need to do a bit more work.

If you use pytest, use a fixture in your conftest.py:

import pytest
from zeal import zeal_context

@pytest.fixture(scope="function", autouse=True)
def use_zeal():
    with zeal_context():
        yield

If you use unittest, add custom test cases and inherit from these rather than directly from Django's test cases:

# In e.g. `myapp/testing/test_cases.py`
from zeal import setup as zeal_setup, teardown as zeal_teardown
import unittest
from django.test import SimpleTestCase, TestCase, TransactionTestCase

class ZealTestMixin(unittest.TestCase):
    def setUp(self, test):
        zeal_setup()
        super().setUp()

    def teardown(self) -> None:
        zeal_teardown()
        return super().teardown(test, err)

class CustomSimpleTestCase(ZealTestMixin, SimpleTestCase):
    pass

class CustomTestCase(ZealTestMixin, TestCase):
    pass

class CustomTransactionTestCase(ZealTestMixin, TransactionTestCase):
    pass

Generic setup

If you also want to detect N+1s in other places not covered here, you can use the setup and teardown functions, or the zeal_context context manager:

from zeal import setup, teardown, zeal_context


def foo():
    setup()
    try:
        # your code goes here
    finally:
        teardown()


@zeal_context()
def bar():
    # your code goes here


def baz():
    with zeal_context():
        # your code goes here

Configuration

By default, any issues detected by zeal will raise a ZealError. If you'd rather log any detected N+1s as warnings, you can set:

ZEAL_RAISE = False

N+1s will be reported when the same query is executed twice. To configure this threshold, set the following in your Django settings.

ZEAL_NPLUSONE_THRESHOLD = 3

To handle false positives, you can temporarily disable zeal in parts of your code using a context manager:

from zeal import zeal_ignore

with zeal_ignore():
    # code in this block will not log/raise zeal errors

If you only want to ignore a specific N+1, you can pass in a list of models/fields to ignore:

with zeal_ignore([{"model": "polls.Question", "field": "options"}]):
    # code in this block will ignore N+1s on Question.options

If you want to listen to N+1 exceptions globally and do something with them, you can listen to the Django signal that zeal emits:

from zeal.signals import nplusone_detected
from django.dispatch import receiver

@receiver(nplusone_detected)
def handle_nplusone(sender, exception):
    # do something

Finally, if you want to ignore N+1 alerts from a specific model/field globally, you can add it to your settings:

ZEAL_ALLOWLIST = [
    {"model": "polls.Question", "field": "options"},

    # you can use fnmatch syntax in the model/field, too
    {"model": "polls.*", "field": "options"},

    # if you don't pass in a field, all N+1s arising from the model will be ignored
    {"model": "polls.Question"},
]

Debugging N+1s

By default, zeal's alerts will tell you the line of your code that executed the same query multiple times. If you'd like to see the full call stack from each time the query was executed, you can set:

ZEAL_SHOW_ALL_CALLERS = True

in your settings. This will give you the full call stack from each time the query was executed.

Comparison to nplusone

zeal borrows heavily from nplusone, but has some differences:

  • zeal also detects N+1 caused by using .only() and .defer()
  • it lets you configure your own threshold for what constitutes an N+1
  • it has slightly more helpful error messages that tell you where the N+1 occurred
  • nplusone patches the Django ORM even in production when it's not enabled. zeal does not!
  • nplusone appears to be abandoned at this point.
  • however, zeal only works with Django, whereas nplusone can also be used with SQLAlchemy.
  • zeal does not (yet) detect unused prefetches, but nplusone does.

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_zeal-2.2.1.tar.gz (19.8 kB view details)

Uploaded Source

Built Distribution

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

django_zeal-2.2.1-py3-none-any.whl (14.8 kB view details)

Uploaded Python 3

File details

Details for the file django_zeal-2.2.1.tar.gz.

File metadata

  • Download URL: django_zeal-2.2.1.tar.gz
  • Upload date:
  • Size: 19.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.13

File hashes

Hashes for django_zeal-2.2.1.tar.gz
Algorithm Hash digest
SHA256 082afc3d0930a85c9ae5544447cd9a9839a71ea265df75cd1a7ace8a97b946df
MD5 d4da71167f2b3d5ed118af9a8dcda823
BLAKE2b-256 776888ee88744130b2a3355fdd92f349053d216df1e5e5b4a55b2759c0fb764c

See more details on using hashes here.

Provenance

The following attestation bundles were made for django_zeal-2.2.1.tar.gz:

Publisher: release.yaml on taobojlen/django-zeal

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file django_zeal-2.2.1-py3-none-any.whl.

File metadata

  • Download URL: django_zeal-2.2.1-py3-none-any.whl
  • Upload date:
  • Size: 14.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.13

File hashes

Hashes for django_zeal-2.2.1-py3-none-any.whl
Algorithm Hash digest
SHA256 bc1a365e4edf167c8aaeb61fbf14d241d5d4ade0dde80b7e517cc91716ccdedf
MD5 c1b2dc4527617433cc8a63319f23aac5
BLAKE2b-256 e2a44e85f5591d31b4e6094a28ef42c4d46c33e32e0b1dceecc5cddcaca245bb

See more details on using hashes here.

Provenance

The following attestation bundles were made for django_zeal-2.2.1-py3-none-any.whl:

Publisher: release.yaml on taobojlen/django-zeal

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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