Integrate a Brunch workflow with Django.
This replaces the runserver
management command with a version that fires up a brunch watch
process alongside the
Django development server to automatically recompile css and javascript. The brunch process is not interrupted when
the Django server reloads, but it will die when you shut down the Django server.
- Run the Django development web server and brunch watch with one command in one window
- Leaves Brunch completely decoupled from Django, so there is no interdependency between the two. This is simply a convenience module.
- Can optionally enable the brunch server so that brunch plugins like
auto-reload-brunch
will do live reloading of app assets such as CSS and Javascript upon saving. Set up brunch to also watch your templates and/or views to get live-reloading on modification of anything.
-
Initialize brunch in your Django project directory. This skeleton will create a brunch-config.js and package.json file in the root of your project directory.
cd myproject brunch new -s https://github.com/nshafer/django-brunch-skeleton
-
Put your CSS and JS files in
static/
, images and other files instatic/assets/
-
Test brunch. This will output to
brunch/
brunch build
-
Add brunch's output directory
brunch/
to the STATICFILES_DIRS list so that Django will find the files that Brunch will create. Remove any existing reference tostatic/
since we now only want to include assets handled by brunch.STATICFILES_DIRS = [ os.path.join(BASE_DIR, "brunch"), ]
-
Link the brunch output in your templates
<link rel="stylesheet" href="{% static "app.css" %}"/> <script src="{% static "app.js" %}"></script>
-
Install 'django-brunch' from PyPI
pip install django-brunch
-
Add
brunch
to INSTALLED_APPS, but BEFOREdjango.contrib.staticfiles
:INSTALLED_APPS = [ ... 'brunch', ... 'django.contrib.staticfiles' ]
-
Configure django-brunch in your settings.py file. At a minimum,
BRUNCH_DIR
must be defined. This needs to be an absolute path to the directory where yourbrunch-config.js
orbrunch-config.coffee
file lives.BRUNCH_DIR = BASE_DIR
-
Run
runserver
like you normally would. A brunch process will be started alongside the Django development web server in the directory you provided with BRUNCH_DIR. One terminal, both processes!./manage.py runserver
By default, the 'brunch-config.js' that is installed from
django-brunch-skeleton
will enable auto-reloading upon modification of files in static/
and template/
as long as you enable
BRUNCH_SERVER
in your Django settings.
Add your individual app static and template directories to paths.watched
to get the same for all of your apps.
Building for production will be much the same, with just one extra step to do before you run collectstatic
. You need
to have brunch build a final production version of your assets.
brunch build --production
./manage.py collectstatic
This should be an absolute path to the location of your brunch directory; where your brunch-config.js
or
brunch-config.coffee
file lives.
Example:
BRUNCH_DIR = os.path.join(BASE_DIR, "brunch")
Default:
BRUNCH_CMD = ("/usr/bin/env", "brunch")
This is the location of your brunch
CLI command. It is passed to
subprocess.Popen so it needs to either be a
string or a sequence type (list, tuple). It is recommended that this is a tuple or list of arguments so that
we can spawn it without involving shell parsing. If it is a string AND the string has a space or other shell characters,
then you will need to set BRUNCH_SHELL to True as well, so that Popen knows to spawn a shell to parse the command.
If you install brunch as a local dependency in the local node_modules
directory, then you'll probably want something
like this:
BRUNCH_CMD = os.path.join(BASE_DIR, "brunch", "node_modules", "brunch", "bin", "brunch")
Default:
BRUNCH_SHELL = False
If your BRUNCH_CMD requires a shell to parse, then set this to True. This is passed directly to
subprocess.Popen's shell
argument. For
example, this needs to be True if your BRUNCH_CMD has a space in it or any other kind of special shell syntax.
Default:
BRUNCH_SERVER = False
If this is enabled, then the brunch watch
command will be started with the --server
argument. This will instruct
Brunch to start up its internal server alongside Django's. This shouldn't be needed in a lot of cases, however one very
useful time for this is if you're using the "auto-reload-brunch" plugin in brunch. If you are, then enabling the brunch
server will allow it to inform the browser of changes to your assets and do live reloads.
To run the example Django project, follow these steps
-
Create a new virtualenv (assuming virtualenvwrapper)
mkvirtualenv django-brunch-example
-
Clone the github repository and install requirements
git clone https://github.com/nshafer/django-brunch.git cd django-brunch/example pip install -r requirements.txt npm install
-
Migrate if needed
./manage.py migrate
-
Run the server and brunch
./manage.py runserver
-
Log in to http://localhost:8000 and see them working together. Modify the stylesheet in
static/style/site.css
and save it to live reloading in action. If you need to get in to the admin, the username and password are 'admin'.
Inspiration for how to integrate Brunch with Django comes from how the Phoenix Framework
does it by default in a new Phoenix project. I used their example as a model for the highly customized brunch-config.js
that departs from brunch conventions a bit.
- Add hook in
collectstatic
to automatically runbrunch build --production
before collecting files - Add command-line
--disable-brunch
to therunserver
command in cases where you might not want to fire it up. Right now you have to disablebrunch
inINSTALLED_APPS
in your setup.py to turn it off. - Add a
initbrunch
management command that clones down django-brunch-skeleton and copies files over safely into an existing Django project. Perhaps pre-initialize thebrunch-config.js
based on existing Django static configuration. - Add a bit of javascript to
brunch-config.js
to allow globs in thepaths.watched
array, so you can just specify something like**/templates
and**/static
to get all app assets automatically without having to automatically add them.