Skip to content

Automating Static Site Deployment

Posted on:August 29, 2023

I wanted to find a way to securely automate deployment of my shiny new statically generated site. I’ve managed to do it and sharing it here to help myself and others in the future.

Table of contents

Open Table of contents

The problem

You just made yourself a pretty new statically generated site using something like Astro and you want to deploy it automatically after pushing it to Git (you do have it in source control riiiiight?).

You could easily set up a GitHub action to SSH into your server and deploy it, but being security minded, the principal of Least Privilege means that your webserver is locked down and nobody but the webserver’s user can write to the directory.

This is exactly the situation I found myself in. I managed to solve it in a (IMHO) creative way without compromising security or efficiency.

The solution

High level solution

THe solution I cobbled together is as follows:

With all that done, everything is now set up and a push to the ‘deploy’ branch results in my website being updated within 10 minutes.

Create a deploy branch

In the root of your git project, simply create a new branch:

user$ git checkout -b deploy
user$ git push

Create a new unprivileged user

Depending on your OS, the method will differ. The instructions below are for Ubuntu.

root# useradd --create-home github-deploy

Create an SSH key for the new user

Creating an SSH key using the usual method, then copy the public key to authorized_keys.

github-deploy$ ssh-keygen -m PEM -t rsa -b 4096
github-deploy$ cp id_rsa.pub authorized_keys

Once that is done, copy the private key (located at /home/github-deploy/.ssh/id_rsa) as it will be used later.

Restrict the SSH key to allow a single command

For security purposes, I wanted to restrict the access the github-deploy user has. Do to this I used the mechanism built in to OpenSSH which allows restricting specific authorized keys to a single command.

To accomplish this, add the following at the beginning of the relevant key in /home/github-deploy/.ssh/authorized_keys file.

command="/usr/bin/rrsync -wo /home/github-deploy/deploy/",restrict ssh-rsa AAAAB3NzaC....

With that done, the github-deploy user can only use rrsync (yes, that’s a double ‘r’ for “remote rsync”).

Configure GitHub

Create a script to deploy pushed files

#!/bin/bash

SOURCE_DIR=/home/github-deploy/deploy/
TEMP_WORKING_DIR=/tmp/WEBSITE.tmp/
DEST_DIR=/var/www/

mkdir -p $TEMP_WORKING_DIR

rsync -a $SOURCE_DIR $TEMP_WORKING_DIR
chown -R www-data:www-data $TEMP_WORKING_DIR
rsync -a --stats $TEMP_WORKING_DIR $DEST_DIR

rm -r $TEMP_WORKING_DIR