This post will explain how to setup automatic deployments to a Django Web-App hosted on PythonAnywhere.
What people normally do:
1. Everyone does this:
Local -> GitHub
2. People with PAID account do this(using ssh access):
Local -> PythonAnywhere
3. What I do(with a free or paid account):
Local -> GitHub -> PythonAnywhere
Let me first explain why I do this and then I will explain how to do this stuff.
Note: You are free to Jump to the Implementation section
Putting it simply GitHub is great place to collaborate and they have a better UI than PythonAnywhere to view code. Setting up PythonAnywhere's ssh access for everyone who will be working on the project is just too time taking for me and then again explain them what I did is another difficult task and that too if I have a paid account.
So, I thought it would be amazing if we can just push code onto GitHub and GitHub automatically updates the code on PythonAnywhere and not to mention the default features that are provided by GitHub like Pull Requests, Issues, Actions(haven't used them till now still a feature) and everything else that GitHub offers.
We will be using something known as Deploy Keys and Webhooks provided by GitHub to tell our Django Web-App tha it has been updated so it pulls the latest code.
Install the Python package named GitPython by using the following command:
$ pip install GitPython
Write the view function that will receive the update and update the code on the server.
Go to your views.py and add the following code:
import git from django.shortcuts import render from django.http import HttpResponse from django.views.decorators.csrf import csrf_exempt @csrf_exempt def update(request): if request.method == "POST": ''' pass the path of the diectory where your project will be stored on PythonAnywhere in the git.Repo() as parameter. Here the name of my directory is "test.pythonanywhere.com" ''' repo = git.Repo("test.pythonanywhere.com/") origin = repo.remotes.origin origin.pull() return HttpResponse("Updated code on PythonAnywhere") else: return HttpResponse("Couldn't update the code on PythonAnywhere")
You can also send response codes with them by passing the parameter "status" within the "HttpResponse" that will be returned.
Set the URL where the Payload from GitHub will be sent.
To add the payload URL add the following lines into urls.py file of your Django Project:
from django.urls import path ''' here the name of my app is updater so I add the import my view from there replace updater with your app-name where you have your views.py ''' from updater import views urlpatterns = [ path("update_server/", views.update, name="update"), ]
Now push this code to GitHub and login to our account on PythonAnywhere.
Now follow the following steps:
Step 1: Open a bash terminal on PythonAnywhere.
Step 2: Issue the following command:
$ ssh-keygen -t rsa -b 4096 -C "email@example.com"
after this command you will be shown the following prompt:
> Generating public/private rsa key pair.
Step 3: When you're prompted to "Enter a file in which to save the key," press Enter. This accepts the default file location.
> Enter a file in which to save the key (/home/you/.ssh/id_rsa): [Press enter]
Step 4: At the prompt, type a secure passphrase.
> Enter passphrase (empty for no passphrase): [Type a passphrase] > Enter same passphrase again: [Type passphrase again]
Step 5: Issue the following command on the bash terminal of PythonAnywhere to get your key.
$ cat ~/.ssh/id_rsa.pub
then press CTRL+C to copy the key.
Go to your GitHub Account and then Go to the Repository in which you have your Django Project.
Then follow the following steps:
Step 1: From your repository, click Settings as shown below.
Step 2: In the sidebar, click Deploy Keys, then click Add deploy key.
Step 3: Provide a title, paste in your public key.
Step 4: Select Allow write access if you want this key to have write access to the repository. A deploy key with write access lets a deployment push to the repository.
Step 5: Click Add key.
Go to your PythonAnywhere account and go to the bash terminal and setup your Web-App but use the SSH instead of HTTPS for cloning your Repository from GitHub.
Note: I would suggest to use the automated script pa_autoconfigure_django.py available in pythonanywhere package which can be installed via pip to automatically setup my project on PythonAnywhere.
If you already have a project setup on PythonAnywhere using HTTPS then open a bash terminal and edit the remote URL to set SSH by using the following command :
$ git remote set-url origin firstname.lastname@example.org:USERNAME/REPOSITORY.git
Go back to the settings tab of your Repository on GitHub and follow the following steps :
Step 1: In the sidebar, click Webhooks, then click Add webhook.
Step 2: Fill in the Payload URL and set the "Content type" to "application/json" as shown below.
Step 3: click "Add webhook".
You will see the following when it is success
If it show some other icon in place of a green tick then click on edit and at the bottom you can see the "Recent Deliveries" here you can click on the button shown below to see the details of that delivery.
After clicking you will see two tabs "Request" and "Response" to see the details of failure you can click on response and you will get the details as shown below
If you get any response code other than 200 then go to the error log on your PythonAnywhere account and check what's the problem.
The most common mistake that I make is when I return something else instead of HttpResponse it will throw some Attribute error.
If you get stuck somewhere then feel free to contact me on telegram using the link https://t.me/soumya_r .