Deploying an Existing Django App to Heroku on a Mac

Because the tutorial almost, but didn’t quite, work.

The Django tutorial Heroku provides ALMOST worked. I ironed out a couple crinkles when I went through the process myself so I thought I would share it with the world to save people some heartache. You should read this in conjunction with that tutorial to get your Django app up and running on Heroku.

First Things First

First, if you do not have Postgresql, you should get it because you will need it for Heroku. I was originally using an SQLite backend—this does not work on Heroku and you will need to download Postgresql. If you have Homebrew,

brew install postgresql

If not, install Homebrew and then install Postgresql. Then, in your Django App,

pip install virtualenv

virtualenv venv

source venv/bin/activate

*export CFLAGS=-Qunused-arguments export CPPFLAGS=-Qunused-arguments*

pip install django-toolbelt

Note the extra line. I ran into a terrible error during “pip install django-toolbelt” (mno-fused-madd MADNESS related to XCode Command Line Tools). The export CFlags line will save you from this torture.

Create a Procfile. This should be in our project folder (outer folder) but not within your app folder. In your Procfile,

web: gunicorn INSERTNAMEOFDJANGOPROJECT.wsgi

You should now be able to, in terminal:

foreman start

My foreman start merely said “web: 1: started pid #” rather than the “Starting gunicorn” promised by the tutorial. The default port is 5000 with foreman so you should be able to navigate to localhost:5000 and see your site.

Back to Heroku Config Stuff

As the tutorial states, run:

pip freeze > requirements.txt

This will generate a requirements.txt with all of your dependencies.

Setting up Postgresql database on Heroku

Create a Postgresql database at https://postgres.heroku.com/databases. When I created a database, it automatically generated a Heroku app with a generic name to go along with it. I was able to change the name of the Heroku app later to a more appropriate URL by visiting my apps dashboard. Back on the databases dashboard, click on the database name to view Host, Database Name, Username, Password, etc.

In your settings.py, replace your existing DATABASES field with the fields specified for the Heroku database created. Please note that these are not your Heroku username and password but rather the username and password provided for that specific Heroku generated database:

DATABASES = {
 ‘default’: {
 ‘ENGINE’: ‘django.db.backends.postgresql_psycopg2',
 ‘NAME’: ‘DATABASE_FROM_HEROKU’, 
 ‘USER’: ‘USERNAME_FROM_HEROKU’,
 ‘PASSWORD’: ‘PASSWORD_FROM_HEROKU’,
 ‘HOST’: ‘HOST_FROM_HEROKU’
 }
}

This step is not included in the tutorial Heroku provided but needs to be done to use their Postgresql backend.

Next, follow tutorial instructions by adding the djdatabaseurl lines and static asset configuration lines to your settings.py file.

# Parse database configuration from $DATABASE_URL
import dj_database_url
DATABASES['default'] = dj_database_url.config()

# Honor the 'X-Forwarded-Proto' header for request.is_secure()
SECURE_PROXY_SSL_HEADER = ('HTTP_X_FORWARDED_PROTO', 'https')

# Allow all host headers
ALLOWED_HOSTS = ['*']

# Static asset configuration
import os
BASE_DIR = os.path.dirname(os.path.abspath(__file__))
STATIC_ROOT = 'staticfiles'
STATIC_URL = '/static/'

STATICFILES_DIRS = (
 os.path.join(BASE_DIR, 'static'),
)

Note on Static Files:

Essentially Django pulls the used static files in project(outer)/app(inner)/static/ and collects and copies them to the project(outer)/staticfiles/ directory.

If this is set up correctly, you should be able to run

python manage.py collectstatic —noinput

while in virtualenv and have it collect the static files to the project(outer)/staticfiles/ directory. When visiting the site locally (localhost:8000) with “python manage.py runserver”, you should be able to access your static files at “localhost:8000/static/…”. In your templates, you can access these static files as such

 {% load staticfiles %}

<link href=”{% static “bootstrap/css/bootstrap.min.css” %}” rel=”stylesheet” type=”text/css” />

Back to Heroku Stuff

DO NOT follow the tutorial in adding “Cling” to wsgi.py. This did not work for me and only caused gunicorn to break. Simple leave wsgi.py alone and ensure that “foreman start” still runs as expected.

I did not change my wsgi.py at all—it is as follows:

import os
os.environ.setdefault(“DJANGO_SETTINGS_MODULE”, “APP.settings”)

from django.core.wsgi import get_wsgi_application
application = get_wsgi_application()

Last Mile

Create a .gitignore if you do not already have one:

venv
*.pyc
staticfiles

Hopefully you already have a git. If not,

git init

git add .

git commit -m "message"

heroku git:remote -a NAMEOFHEROKUAPP

Lastly,

git push heroku master

and

heroku ps:scale web=1

heroku run python manage.py syncdb

Set up your database as you see fit and run:

heroku open

Hopefully your Heroku Django app is up and running. If not, feel free to comment or reach out.