Skip to content

Using Git Hooks To Automate Development and Deployment

mkdocs example

Create your directory structure(s)

Local development and testing

mkdir ~/projects/sites/www.universalcake.local

For a production site

mkdir -p ~/projects/sites/www.universalcake.com/mkdocs
cd ~/projects/sites/www.universalcake.com/mkdocs
git init

output exanmple:

Initialized empty Git repository in /home/cjs/projects/sites/www.universalcake.com/mkdocs/.git/

hooks

Lets take a look at the hook samples

cd .git/hooks
ls -l

Output:

total 52
-rwxr-xr-x 1 cjs cjs  478 Feb 26 15:54 applypatch-msg.sample
-rwxr-xr-x 1 cjs cjs  896 Feb 26 15:54 commit-msg.sample
-rwxr-xr-x 1 cjs cjs 3079 Feb 26 15:54 fsmonitor-watchman.sample
-rwxr-xr-x 1 cjs cjs  189 Feb 26 15:54 post-update.sample
-rwxr-xr-x 1 cjs cjs  424 Feb 26 15:54 pre-applypatch.sample
-rwxr-xr-x 1 cjs cjs 1638 Feb 26 15:54 pre-commit.sample
-rwxr-xr-x 1 cjs cjs  416 Feb 26 15:54 pre-merge-commit.sample
-rwxr-xr-x 1 cjs cjs 1492 Feb 26 15:54 prepare-commit-msg.sample
-rwxr-xr-x 1 cjs cjs 1348 Feb 26 15:54 pre-push.sample
-rwxr-xr-x 1 cjs cjs 4898 Feb 26 15:54 pre-rebase.sample
-rwxr-xr-x 1 cjs cjs  544 Feb 26 15:54 pre-receive.sample
-rwxr-xr-x 1 cjs cjs 3610 Feb 26 15:54 update.sample

Notice that each of these files are marked executable and that each file ends in .sample.

These scripts are called (executed) by name so the first line must be a shebang reference to call the correct script interpreter:

#! /bin/sh – Execute the file using the Bourne or compatible shell in the /bin directory
#! /bin/bash – Execute using the Bash shell
#! /usr/bin/pwsh – Execute using PowerShell
#! /usr/bin/env python3 – Execute with a Python interpreter, using the env program search path to find it
#! /bin/false - Do nothing, but return a non-zero exit status, indicating failure

Other script languages can be use, three common ones include bash, perl and python

files that end in .sample come with git but are not executed as git simply looks at the filename (not the extension) when trying to find hook files to execute.

So rename a script something else like update.test "disables" the script. If you wanted to enable any of the scripts in this directory, you would have to remove the .sample suffix.

Deploying to a local web server

Here we will use a post-commit hook to deploy to a local server

ExampleRequirements

sudo apt-get update
sudo apt-get install apache2

Deluxe File Layout example

This directory structure would allow for the hosting of multiple websites. Here we are going to focus on public_html directory for universalcake.com

/var/www/
├── www.universalcake.com/
│   ├── public_html/       # Web root for the site (DocumentRoot)   ├── logs/              # Access and error logs (optional)   ├── backups/           # Site backups (optional)   ├── ssl/               # SSL certificates (if storing locally)   ├── config/            # Site-specific configs (optional)   ├── tmp/               # Temporary files (optional)   ├── sessions/          # PHP session files (optional)   └── private/           # Private files outside web root
│
└── shared/                # Optional shared assets (e.g., images, scripts, static files)

Create directory(ise)

sudo mkdir -p /var/www/universalcake.com/public_html

Setting up permissions:

sudo chown -R `whoami`:`id -gn` /var/www/universalcake.com/public_html
sudo chown -R `whoami`:`id -gn` /var/www/www.universalcake.com/public_html

Confirmation

ls -al /var/www/www.universalcake.com/public_html

Site project directory

you can manually add an index.html to your website projects root directory. In this case we are going to use mkdocs to do this for us.

cd ~/projects/sites/www.universalcake.com/mkdocs

we will be using an venv so we will start this in order to use mkdocs:

mkdocs-static-i18n-1.2.3 

now we will initialize our site in our sites mkdocs directory

mkdocs new .

Next we build the default site so that we have files in our projects git repository

mkdocs build

Add the new file to tell git to track the file:

git add .

Running git status

git status

Output:

On branch master

No commits yet

Changes to be committed:
  (use "git rm --cached <file>..." to unstage)
    new file:   docs/index.md
    new file:   mkdocs.yml
    new file:   site/404.html
...

Before we commit we are going to create a post-commit hook for the repository. Create this file within the .git/hooks directory for the project:

nano .git/hooks/post-commit 

content example

#!/bin/bash
source ~/.venv/mkdocs-static-i18n-1.2.3/bin/mkdocs
mkdocs build
unset GIT_INDEX_FILE
git --work-tree=/var/www/www.universalcake.com/public_html --git-dir=$HOME/projects/sites/www.universalcake.com/mkdocs/site checkout -f

Notes:

environmental variables that are set each time the post-commit hook is called. In particular, GIT_INDEX_FILE is set to .git/index. This path is in relation to the working directory, which in this case is /var/www/www.universalcake.com/public_html.

Since the git index does not exist at this location, the script will fail if you leave it as-is. To avoid this situation, you can manually unset the variable.

Then we use git itself to unpack the newest version of the repository after the commit, into your web directory.

You will want to force this transaction to make sure this is successful each time.

Make your git hook executable

chmod +x .git/hooks/post-commit

Test Commit

git commit -m 'initial commit test'

Output

import-im6.q16: attempt to perform an operation not allowed by the security policy `PS' @ error/constitute.c/IsCoderAuthorized/413.
import-im6.q16: attempt to perform an operation not allowed by the security policy `PS' @ error/constitute.c/IsCoderAuthorized/413.
from: can't read /var/mail/mkdocs.__main__
/home/cjs/.venv/mkdocs-static-i18n-1.2.3/bin/mkdocs: line 7: syntax error near unexpected token `('
/home/cjs/.venv/mkdocs-static-i18n-1.2.3/bin/mkdocs: line 7: `    sys.argv[0] = re.sub(r'(-script\.pyw|\.exe)?$', '', sys.argv[0])'
INFO    -  Cleaning site directory
INFO    -  Building documentation to directory: /home/cjs/projects/sites/www.universalcake.com/mkdocs/site
INFO    -  Documentation built in 0.06 seconds
fatal: not a git repository: '/home/cjs/projects/sites/www.universalcake.com/mkdocs/site'
[master (root-commit) 142f356] initial test commit
 31 files changed, 4806 insertions(+)
 create mode 100644 docs/index.md
 create mode 100644 mkdocs.yml
 create mode 100644 site/404.html
 create mode 100644 site/css/base.css
...
 create mode 100644 site/webfonts/fa-v4compatibility.woff2

testing

Resources