Tutorial

How To Use Postgresql with your Django Application on Debian 8

How To Use Postgresql with your Django Application on Debian 8
Not using Debian 8?Choose a different version or distribution.
Debian 8

Introduction

Django is a flexible framework for quickly creating Python applications. By default, Django applications are configured to store data into a lightweight SQLite database file. While this works well under some loads, a more traditional DBMS can improve performance in production.

In this guide, we’ll demonstrate how to install and configure PostgreSQL to use with your Django applications. We will install the necessary software, create database credentials for our application, and then start and configure a new Django project to use this backend.

Prerequisites

To get started, you will need a clean Debian 8 server instance with a non-root user set up. The non-root user must be configured with sudo privileges. Learn how to set this up by following our initial server setup guide.

When you are ready to continue, log in as your sudo user and read on.

Install the Components from the Debian Repositories

Our first step will be to install all of the pieces that we need from the repositories. We will install pip, the Python package manager, in order to install and manage our Python components. We will also install the database software and the associated libraries required to interact with them.

Python 2 and Python 3 require slightly different packages, so choose the commands below that match the Python version of your project.

If you are using Python 2, type:

  1. sudo apt-get update
  2. sudo apt-get install python-pip python-dev libpq-dev postgresql postgresql-contrib

If you are using Python 3, type:

  1. sudo apt-get update
  2. sudo apt-get install python3-pip python3-dev libpq-dev postgresql postgresql-contrib

With the installation out of the way, we can move on to create our database and database user.

Create a Database and Database User

By default, Postgres uses an authentication scheme called “peer authentication” for local connections. Basically, this means that if the user’s operating system username matches a valid Postgres username, that user can login with no further authentication.

During the Postgres installation, an operating system user named postgres was created to correspond to the postgres PostgreSQL administrative user. We need to use this user to perform administrative tasks. We can use sudo and pass in the username with the -u option.

Log into an interactive Postgres session by typing:

  1. sudo -u postgres psql

First, we will create a database for our Django project. Each project should have its own isolated database for security reasons. We will call our database myproject in this guide, but it’s always better to select something more descriptive for real projects:

Note: Remember to end all commands at an SQL prompt with a semicolon.

  1. CREATE DATABASE myproject;
Output
CREAT DATABASE

Next, we will create a database user which we will use to connect to and interact with the database. Set the password to something strong and secure:

  1. CREATE USER myprojectuser WITH PASSWORD 'password';
Output
CREATE ROLE

Next, we’ll modify a few of the connection parameters for the user we just created. This will speed up database operations since the correct values will not have to be queried and set each time a connection is established.

We are setting the default encoding to UTF-8, which is the format that Django expects. We are also setting the default transaction isolation scheme to “read committed”, which blocks reads from uncommitted transactions. Lastly, we are setting the timezone. By default, our Django projects will be set to use UTC. These are all recommendations from the Django project itself.

  1. ALTER ROLE myprojectuser SET client_encoding TO 'utf8';
  2. ALTER ROLE myprojectuser SET default_transaction_isolation TO 'read committed';
  3. ALTER ROLE myprojectuser SET timezone TO 'UTC';
Output
ALTER ROLE ALTER ROLE ALTER ROLE

Now, all we need to do is give our database user access rights to the database we created:

  1. GRANT ALL PRIVILEGES ON DATABASE myproject TO myprojectuser;
Output
GRANT

Exit the SQL prompt when you’re finished.

  1. \q

You should now be taken back to your previous shell session.

Install Django within a Virtual Environment

Now that our database is set up, we can install Django. For better flexibility, we will install Django and all of its dependencies within a Python virtual environment. The virtualenv package allows you to create these environments easily.

If you are using Python 2, you can install the correct package by typing:

  1. sudo pip install virtualenv

If you are using Python 3, you can install the correct package by typing:

  1. sudo pip3 install virtualenv

Make and move into a directory to hold your Django project:

  1. mkdir ~/myproject
  2. cd ~/myproject

We can create a virtual environment to store our Django project’s Python requirements by typing:

  1. virtualenv venv

This will install a local copy of Python and a local pip command into a directory called venv within your project directory.

Before we install applications within the virtual environment, we need to activate it. You can do so by typing:

  1. source venv/bin/activate

Your prompt will change to indicate that you are now operating within the virtual environment. It will look something like this (venv)user@host:~/myproject$.

Once your virtual environment is active, you can install Django with pip. We will also install the psycopg2 package that will allow us to use the database we configured:

Note

Regardless of which version of Python you are using, when the virtual environment is activated, you should use the pip command (not pip3).

  1. pip install django psycopg2

We can now start a Django project within our myproject directory. This will create a child directory of the same name to hold the code itself, and will create a management script within the current directory:

Note: Make sure to add the dot at the end of the command so that this is set up correctly. Since we already created a parent project directory to hold our virtual environment directory, we do not want the extra directory level that will be created if we leave off the dot.

  1. django-admin.py startproject myproject .

Your current directory structure should look something like this:

.
└── ./myproject/
    ├── manage.py
    ├── myproject/
    │   ├── __init__.py
    │   ├── settings.py
    │   ├── urls.py
    │   └── wsgi.py
    └── venv/
        └── . . .

As you can see, we have a parent project directory that holds a manage.py script, an inner project directory, and the venv virtual environment directory we created earlier.

Configure the Django Database Settings

Now that we have a project, we need to configure it to use the database we created.

Open the main Django project settings file located within the child project directory:

  1. nano ~/myproject/myproject/settings.py

Before we set up the database, you may also need to adjust the ALLOWED_HOSTS directive. This defines a whitelist of addresses or domain names that may be used to connect to the Django instance. Any incoming requests with a Host header that is not in this list will raise an exception. Django requires that you set this to prevent a certain class of security vulnerability.

In the square brackets, list the IP addresses or domain names that are associated with your Django server. Each item should be listed in quotations with entries separated by a comma. If you wish to respond to requests for a domain and any subdomains, prepend a period to the beginning of the entry. In the snippet below, there are a few commented out examples used to demonstrate the correct way to format entries:

~/myproject/myproject/settings.py
. . .
# The simplest case: just add the domain name(s) and IP addresses of your Django server
# ALLOWED_HOSTS = [ 'example.com', '203.0.113.5']
# To respond to 'example.com' and any subdomains, start the domain with a dot
# ALLOWED_HOSTS = ['.example.com', '203.0.113.5']
ALLOWED_HOSTS = ['your_server_domain_or_IP', 'second_domain_or_IP', . . .]

Next, find the DATABASES section that looks like this:

~/myproject/myproject/settings.py
. . .

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.sqlite3',
        'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
    }
}

. . .

This is currently configured to use SQLite as a database. We need to change this so that our PostgreSQL database is used instead.

First, change the engine so that it uses the postgresql_psycopg2 adaptor instead of the sqlite3 adaptor. For the NAME, use the name of your database (myproject in our example). We also need to add login credentials. We need the username, password, and host to connect to. We’ll add and leave blank the port option so that the default is selected:

~/myproject/myproject/settings.py
. . .

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.postgresql_psycopg2',
        'NAME': 'myproject',
        'USER': 'myprojectuser',
        'PASSWORD': 'password',
        'HOST': 'localhost',
        'PORT': '',
    }
}

. . .

When you are finished, save and close the file.

Migrate the Database and Test your Project

Now that the Django settings are configured, we can migrate our data structures to our database and test out the server.

We can begin by creating and applying migrations to our database. Since we don’t have any actual data yet, this will simply set up the initial database structure:

  1. cd ~/myproject
  2. ./manage.py makemigrations
  3. ./manage.py migrate

After creating the database structure, we can create an administrative account by typing:

  1. ./manage.py createsuperuser

You will be asked to select a username, provide an email address, and choose and confirm a password for the account.

Note: Before you try the development server, make sure you open the port in your firewall.

If you happen to be running a ufw firewall, you can open the appropriate port by typing:

  1. sudo ufw allow 8000

If you are running an iptables firewall, the exact command you need depends on your current firewall configuration. For most configurations, this command should work:

  1. sudo iptables -I INPUT -p tcp --dport 8000 -j ACCEPT

Next, you can test that your database is performing correctly by starting up the Django development server:

  1. ./manage.py runserver 0.0.0.0:8000

In your web browser, visit your server’s domain name or IP address followed by :8000 to reach default Django root page:

http://server_domain_or_IP:8000

You should see the default index page:

Django index

Append /admin to the end of the URL and you should be able to access the login screen to the admin interface:

Django admin login

Enter the username and password you just created using the createsuperuser command. You will then be taken to the admin interface:

Django admin interface

When you’re done investigating, you can stop the development server by hitting CTRL-C in your terminal window.

By accessing the admin interface, we have confirmed that our database has stored our user account information and that it can be appropriately accessed.

We can validate this further by querying the Postgres database itself using the psql client. For instance, we can connect to our project database (myproject) with our project’s user (myprojectuser) and print out all the available tables by typing:

  1. psql -W myproject myprojectuser -h 127.0.0.1 -f <(echo '\dt')

The -W flag makes psql prompt you for appropriate password. We have to explicitly use the -h flag to connect to the localhost over the network to indicate that we want to use password authentication instead of peer authentication. We’re using the -f flag to pass in the psql meta-command we want to execute, \dt, which lists all of the tables in the database:

Output
List of relations Schema | Name | Type | Owner --------+----------------------------+-------+--------------- public | auth_group | table | myprojectuser public | auth_group_permissions | table | myprojectuser public | auth_permission | table | myprojectuser public | auth_user | table | myprojectuser public | auth_user_groups | table | myprojectuser public | auth_user_user_permissions | table | myprojectuser public | django_admin_log | table | myprojectuser public | django_content_type | table | myprojectuser public | django_migrations | table | myprojectuser public | django_session | table | myprojectuser (10 rows)

As you can see, Django has created some tables within our database which confirms that our settings are valid.

Conclusion

In this guide, we’ve demonstrated how to install and configure PostgreSQL as the backend database for a Django project. While SQLite can easily handle the load during development and light production use, most projects benefit from implementing a more full-featured DBMS.

Thanks for learning with the DigitalOcean Community. Check out our offerings for compute, storage, networking, and managed databases.

Learn more about our products

About the authors

Still looking for an answer?

Ask a questionSearch for more help

Was this helpful?
 
1 Comments


This textbox defaults to using Markdown to format your answer.

You can type !ref in this text area to quickly search our full set of tutorials, documentation & marketplace offerings and insert the link!

The django.db.backends.postgresql_psycopg2 backend has been renamed to django.db.backends.postgresql in Django 1.9. (The psycopg2 name can still be used for backwards compatibility.)

Essentially, for Django ≥1.9, use django.db.backends.postgresql

From this stackoverflow answer: Which Postgres value should I use in Django’s DATABASE_ENGINE?

Try DigitalOcean for free

Click below to sign up and get $200 of credit to try our products over 60 days!

Sign up

Join the Tech Talk
Success! Thank you! Please check your email for further details.

Please complete your information!

Featured on Community

Get our biweekly newsletter

Sign up for Infrastructure as a Newsletter.

Hollie's Hub for Good

Working on improving health and education, reducing inequality, and spurring economic growth? We'd like to help.

Become a contributor

Get paid to write technical tutorials and select a tech-focused charity to receive a matching donation.

Welcome to the developer cloud

DigitalOcean makes it simple to launch in the cloud and scale up as you grow — whether you're running one virtual machine or ten thousand.

Learn more
Animation showing a Droplet being created in the DigitalOcean Cloud console