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:
- Create a dedicated Git branch for deployment
- Create a new unprivileged user
- Create an SSH key for the new user
- Restrict the new SSH key to only be able to call
rrsync
(yes, that’s a double “r”) - Add secrets to the GitHub repository
- Webserver Hostname
- New user’s username
- SSH Key
- Add a GitHub action which does the following:
- Trigger on push to the deploy branch
- Checkout the deploy branch
- Build the application
- Create an
~/.ssh/id_rsa
file - Write to the SSH Key secret to the
id_rsa
file - Add the host’s information to
~/.ssh/known_hosts
- Start the SSH agent with the new key
rsync
the generated files to the server
- Create a shell script which
- Changes the ownership of all files in the directory
where the files are pushed to. The owner should be the webserver’s user (ex:
www-data
on Ubuntu) rsync
s the files to the directory being served
- Changes the ownership of all files in the directory
where the files are pushed to. The owner should be the webserver’s user (ex:
- Create a
systemd
timer and service- The service should call the shell script created above
- The timer should run on a schedule (think
cron
, but make itsystemd
)
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