Skip to main content

Automatically reload your browser in development.

Project description

https://img.shields.io/github/actions/workflow/status/adamchainz/django-browser-reload/main.yml?branch=main&style=for-the-badge https://img.shields.io/badge/Coverage-100%25-success?style=for-the-badge https://img.shields.io/pypi/v/django-browser-reload.svg?style=for-the-badge https://img.shields.io/badge/code%20style-black-000000.svg?style=for-the-badge pre-commit

Automatically reload your browser in development.

Requirements

Python 3.7 to 3.11 supported.

Django 3.2 to 4.2 supported.

Your browser needs to support:


Want to work smarter and faster? Check out my book Boost Your Django DX which covers django-browser-reload and many other tools. I wrote django-browser-reload whilst working on the book!


Installation

  1. Install with pip:

    python -m pip install django-browser-reload
  2. Ensure you have "django.contrib.staticfiles" in your INSTALLED_APPS.

  3. Add django-browser-reload to your INSTALLED_APPS:

    INSTALLED_APPS = [
        ...,
        "django_browser_reload",
        ...,
    ]
  4. Include the app URL’s in your root URLconf(s):

    from django.urls import include, path
    
    urlpatterns = [
        ...,
        path("__reload__/", include("django_browser_reload.urls")),
    ]

    You can use another prefix if required.

  5. Add the middleware:

    MIDDLEWARE = [
        # ...
        "django_browser_reload.middleware.BrowserReloadMiddleware",
        # ...
    ]

    The middleware should be listed after any that encode the response, such as Django’s GZipMiddleware.

    The middleware automatically inserts the required script tag on HTML responses before </body> when DEBUG is True. It does so to every HTML response, meaning it will be included on Django’s debug pages, admin pages, etc. If you want more control, you can instead insert the script tag in your templates—see below.

All done! 📯

For faster and more efficient reloading, also set up Django’s built-in Watchman support.

What It Does

When DEBUG is True, the template tag includes a small script. This script connects back to the development server and will automatically reload when static assets or templates are modified, or after runserver restarts. (Detecting modification of Django templates requires Django 3.2+.) The reload only happens in the most recently opened tab.

Example Project

See the example project in the example/ directory of the GitHub repository. Start it up and modify its files to see the reloading in action.

Template Tag

If the middleware doesn’t work for you, you can also use a template tag to insert the script on relevant pages. The template tag has both Django templates and Jinja versions, and only outputs the script tag when DEBUG is True.

For Django Templates, load the tag and use it in your base template. The tag can go anywhere, but it’s best just before </body>:

{% load django_browser_reload %}

...

    {% django_browser_reload_script %}
  </body>
</html>

To add django-browser-reload to Django’s admin, do so in a template called admin/base_site.html:

{% extends "admin/base_site.html" %}

{% load django_browser_reload %}

{% block extrahead %}
    {{ block.super }}
    {% django_browser_reload_script %}
{% endblock %}

This follows Django’s documentation on extending an overriden template.

For Jinja Templates, you need to perform two steps. First, load the tag function into the globals of your custom environment:

# myproject/jinja2.py
from jinja2 import Environment
from django_browser_reload.jinja import django_browser_reload_script


def environment(**options):
    env = Environment(**options)
    env.globals.update(
        {
            # ...
            "django_browser_reload_script": django_browser_reload_script,
        }
    )
    return env

Second, render the tag in your base template. It can go anywhere, but it’s best just before </body>:

...
    {{ django_browser_reload_script() }}
  </body>
</html>

Ta-da!

How It Works

Here’s a diagram:

                                     Browser

                             Tab 1    Tab 2     Tab N
                           listener  listener  listener
                                \       |       /
  Django                         \      |      /
                                  \     |     /
Events View --------------------> Shared worker

The template tag includes a listener script on each page. This listener script starts or connects to a SharedWorker, running a worker script. The worker script then connects to the events view in Django, using an EventSource to receive server-sent events.

This event source uses StreamingHttpResponse to send events to the worker. The view continues streaming events indefinitely, until disconnected. (This requires a thread and will not work if you use runserver’s --nothreading option.)

On a relevant event, the worker will reload the most recently connected tab. (It avoids reloading all tabs since that could be expensive.)

To reload when a template changes, django-browser-reload piggybacks on Django’s autoreloading infrastructure. An internal Django signal indicates when a template file has changed. The events view receives this signal and sends an event to the worker, which triggers a reload. There is no smart filtering - if any template file changes, the view is reloaded.

To reload when the server restarts, django-browser-reload uses a version ID. This ID is randomly generated when the view module is imported, so it will be different every time the server starts. When the server restarts, the worker’s EventSource reconnects with minimal delay. On connection, the events view sends the version ID, which the worker sees as different, so it triggers a reload.

The events view also sends the version ID every second to keep the connection alive.

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_browser_reload-1.8.0.tar.gz (16.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_browser_reload-1.8.0-py3-none-any.whl (12.1 kB view details)

Uploaded Python 3

File details

Details for the file django_browser_reload-1.8.0.tar.gz.

File metadata

  • Download URL: django_browser_reload-1.8.0.tar.gz
  • Upload date:
  • Size: 16.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.11.2

File hashes

Hashes for django_browser_reload-1.8.0.tar.gz
Algorithm Hash digest
SHA256 054ef5ce8b80f46e0beda1a3c847becdff51272cad6f2bda607e201a7afe7425
MD5 bd01aeb057b957011027a5d9d8a48c13
BLAKE2b-256 429d55187254d597b1d4a37c218b625cc1ad208ed92dd65062230a6d1d3cb00f

See more details on using hashes here.

File details

Details for the file django_browser_reload-1.8.0-py3-none-any.whl.

File metadata

File hashes

Hashes for django_browser_reload-1.8.0-py3-none-any.whl
Algorithm Hash digest
SHA256 6db7e6c5807fff87e421b53926c50cfa3de0115e438f3c2de6103d3069557adb
MD5 d95c13b9d7dff5bbc350f4987a9bab7f
BLAKE2b-256 8d043f9b2c73ffd6959000e1f491f3fa4b1f93aace1d4d5f1195d107cc8e4def

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