Here are the steps that I took to deploy a Django Web Application to AWS Elastic Beanstalk using an RDS managed database running PostgreSQL. I last used this in a successful deployment on March 13th, 2023.
Pre-requisites: Python, PostgreSQL, pip, virtualenv, awsebcli (with aws credentials set up)
Here are some links to help if you need to install any of these. You will also need to make sure you have an AWS account and your credentials set up for the AWS CLI. Instructions for that can be found here: AWS CLI Setup
Python Install Instructions pip Install Instructions virtualenv Install Instructions awsebcli Install Instructions PostgreSQL Install Instructions
Create a new directory to house the project
mkdir djangoproject
Create a virtual environment with:
virtualenv venv
Activate the virtual environment. Once it is activated, you will see the name of the virtual environment in parentheses at the beginning of your command line, like this: (venv)~$
source venv/bin/activate
pip install django
*You may get issues depending on which version of Django you use, if you are running into issues, look at installing a previous version. The latest install worked for me (January ‘23)
Start a new django project (Don’t forget to add a ‘.’ at the end of the command to ensure proper files are loaded)
django-admin startproject ebdjango .
Now, before doing anything else, we need to set up the local PostgreSQL database for local development. This will be different from the RDS database that we will set up for the live environment, This local database is used to test and develop your Django Web Application before deployment to Elastic Beanstalk.
Install gunicorn
pip install gunicorn
Install pyscopg2-binary for configuring your local database. (This seems to be the best solution, as I experienced a lot of issues trying to work with the non-binary install of psycopg2.)
pip install psycopg2-binary
Verify that all of your packages are installed with pip freeze
pip freeze
Elastic Beanstalk will automatically set up the server with all the required packages, but we need to have a file called requirements.txt to communicate what packages to install. Create the requirements.txt file
pip freeze > requirements.txt
The first step is to install and set up Postgresql on your machine, check the setup for PostgreSQL for your setup here: PostgreSQL
- Create a new database (don’t forget to always have a ‘;’ at the end of each command!). Make sure to remember what name you give your database, as we will need it soon.
CREATE DATABASE databasename;
- Create a new user with a new password
CREATE USER user WITH PASSWORD ‘password’;
- Grant the user privileges on the database
GRANT ALL PRIVILEGES ON DATABASE databasename TO user;
- Exit with \q
\q
Make sure you have written down your database name, user and password as we need it in the next steps.
You will then need to go into your main project folder and edit the settings.py file to incorporate the database information that you just setup.
Under the DATABASES section in your settings.py, edit it to the following making sure to add the information that you used in the earlier step. If you want a copy/paste option, check out the samples contained in this repository.
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.postgresql_psycopg2',
'NAME': 'dbcreatedinpsql',
'USER': 'createuser',
'PASSWORD': 'createpassword',
'HOST': 'localhost',
'PORT': '5432',
}
}
You may run into some issues with the database not being able to connect. If you are having trouble running the server, such as not being able to connect to the local host, you may need to edit the pg_hba.conf file. This is how you can do that with an Amazon Linux 2 machine:
Login as the postgres user (you need permission to edit the file)
sudo su postgres
Edit the pg_hba.conf file with a text editor, such as nano
nano /var/lib/psql/data/pg_hba.conf
Edit the file so it looks like the following. You want to change your IPV4 settings to ‘trust’ by replacing the ‘ident’
(Screen shot here for the pg_hba.conf file)
Once all of this is done,
sudo systemctl restart postgresql
You can then head back into your project file and activate your virtual environment and migrate your database
python manage.py migrate
Once you have set up your local database, you can create a superuser in order to administrate your application. This will be your local superuser, when we set up the RDS for the live environment, we will set up the live superuser as well as automate that process.
- Migrate the Database if you haven't already
python manage.py migrate
- Create superuser
python manage.py createsuperuser
- Run your Django site locally
Run the Django site locally to see if the install worked:
python manage.py runserver
-
Check the site in a browser at 127.0.0.1:8000
-
Login to the admin site with your superuser account credentials that you set up at 127.0.0.1:8000/admin
If you made it here, you have successfully installed django and the basic requirements for deploying it to AWS Elastic Beanstalk!
Now that you have set up your project on the local server, you can set up your configuration for deployment to AWS Elastic Beanstalk. Make sure you have your AWS credentials set up and configured as well as the awsebcli installed and set up.
Make sure the virtual environment is activated
Now, we need to specify more settings for the Elastic Beanstalk Environment. For that, we need to create a directory and call it: .ebextensions:
mkdir .ebextensions
Now, you want to create a file within that .ebextensions directory and call it django.config. (See sample files for an example)
Open that file in a text editor and add the following text. Be mindful of your spacing, check the sample documents contained in this repo if you want a copy/paste option that has proper spacing:
option_settings:
aws:elasticbeanstalk:container:python:
WSGIPath: ebdjango.wsgi:application
Install ebhealthcheck (optional) - If you are running an application load balanced environment, this will ensure that your health checks will work properly. Without it, you will run into health check issues in your environment.
pip install django-ebhealthcheck
Add ebhealthcheck to settings.py INSTALLED APPS
INSTALLED_APPS = [
...
'ebhealthcheck.apps.EBHealthCheckConfig',
...
]
Next, we are going to deploy our application to Elastic Beanstalk. There is still more to go after this step though, as we are going to have to set up a live RDS Postgres Database so that our application will be able to work.
Initialize the repository with eb init command:
eb init
Answer the question prompts to make sure you are setting it up in the correct region and with an SSH key (optional).
Create an environment and deploy the application to it with eb create. This will ask you about the name, etc. This will ask about specifying the type of load balancer. For this project, select Application Load Balancer.
eb create
Once you have completed that, it will go through and create the resources (may take a couple of minutes)
Use eb status after everything is created to check the status.
You will get an error on this deployment, as you need to have the below added in so that it will work.
eb status
Copy the CNAME property (this is the domain name of where the application is located)
Open the settings.py file and locate the ALLOWED_HOSTS setting and add the CNAME into that section. If you don't add this to the allowed hosts, you will get errors and your site will not be available.
Save that file and run eb deploy to update that: eb deploy
Verify that you have imported os in your settings.py file.
import os
First, go to the elastic beanstalk page in the console
Click on configuration link
Create a new RDS database
Change the DB engine to postgresql
Add a username and password
Save the changes
This will take a while, so in the meantime, head into the settings.py and update the DATABASE section to include the settings for utilizing the RDS
Then, you need to update your settings.py to look like the following:
if 'RDS_DB_NAME' in os.environ:
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.postgresql_psycopg2',
'NAME': os.environ['RDS_DB_NAME'],
'USER': os.environ['RDS_USERNAME'],
'PASSWORD': os.environ['RDS_PASSWORD'],
'HOST': os.environ['RDS_HOSTNAME'],
'PORT': os.environ['RDS_PORT'],
}
}
else:
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.postgresql_psycopg2',
'NAME': 'dbcreatedinpsql',
'USER': 'createuser',
'PASSWORD': 'createpassword',
'HOST': 'localhost',
'PORT': '5432',
}
}
You can find this info at realpython.com/deploying-a-django-app-to-aws-elastic-beanstalk/#configuring-a-database , which was very helpful to me in getting this set up!
Once you have your live site set up, you don’t want to have to login and do all of the migrations manually, and so for that we can add another configuration file which will tell elastic beanstalk to make the necessary changes.
Add a configuation file in .ebextensions called eb-commands.config (see instructions here https://docs.aws.amazon.com/elasticbeanstalk/latest/dg/create-deploy-python-django.html
Deploy with eb deploy after everything is updated
Collecting Static Files and Automating the database migrations process by adding the following to your eb-commands.config file
- Create a new config file in your .ebextensions directory called: eb-commands.config
touch eb-commands.config
Add the following to the file and make sure to change the ebdjango.settings to reflect your project name
01_collectstatic:
command: "source /var/app/venv/*/bin/activate && python3 manage.py collectstatic --noinput"
leader_only: true
02_migrate:
command: "source /var/app/venv/*/bin/activate && python3 manage.py migrate --noinput"
leader_only: true
option_settings:
aws:elasticbeanstalk:application:environment:
DJANGO_SETTINGS_MODULE: ebjango.settings
PYTHONPATH: "/var/app/current:$PYTHONPATH"
aws:elasticbeanstalk:environment:proxy:staticfiles:
/static: static
Make sure to add a STATIC_ROOT to your settings.py as well:
Static files (CSS, JavaScript, Images)
https://docs.djangoproject.com/en/2.2/howto/static-files/
STATIC_URL = '/static/'
STATIC_ROOT = 'static'
The easiest (and simplest) way to create a superuser for your live database is to put it into your eb-commands.config file. In order to do that, you need to set up some Python code to run. In order to do that, we can create a new app to hold our code.
Create a new app with django admin, you can call it whatever you like to help you stay organized. For this example, I will use 'accounts'.
django-admin startapp accounts
Change directory into accounts/ and create a new directory within it called management
mkdir management
cd into management directory and create init.py file within management directory
touch __init__.py
Create a directory within management directory called commands
mkdir commands
cd into commands directory and create init.py file within commands directory
touch __init__.py
Create a new file called makesuper.py
touch makesuper.py
Add the following code to the makesuper.py file. Check the samples in this repository for a sample file.
from django.contrib.auth import get_user_model
from django.core.management.base import BaseCommand
class Command(BaseCommand):
def handle(self, *args, **options):
User = get_user_model()
if not User.objects.filter(username="yourusername").exists():
User.objects.create_superuser("yourusername", "emailaddress@email.com", "passwordforuser")
Once you have added the code above, head back to your eb-commands.config file and add the following code to your configuration file:
03_createsuper:
command: "source /var/app/venv/*/bin/activate && python3 manage.py createsuper"
leader_only: true
Make sure to add the 'accounts' app to your settings.py INSTALLED APPS:
INSTALLED_APPS = [
...
'accounts',
'ebhealthcheck.apps.EBHealthCheckConfig',
...
]
Now, you will have a superuser created when you deploy your site. No need to get rid of the code, unless you want to be really strict with security, as the code will only create the superuser if there is not already a superuser with the account name.
Set up your version control
It’s vital to make sure you have some sort of version control set up, whether it is through AWS CodeCommit or Github but that is beyond the scope of this tutorial. Consider this a friendly reminder 😊
Bonus:
Create a .gitignore file for your django project. There is a great starter template for your django .gitignore file, which is really helpful for setting up git with django. Link to .gitignore file.