This post will mostly mirror Hugo’s quickstart guide. With a few differences: the theme and customization will be more specific, and will include the steps to get Hugo into an production environment, including CI/CD and Let’s Encrypt certificates.

Step 0 - Installing Hugo

To start, install Hugo from here at gohugo.io.

Then, create a new project and move into the directory it creates.

hugo new site cite-name
cd cite-name

Then create a post.

hugo new posts/hello-world.md
echo Hello World! >> content/posts/hello-world.md

And start the development server with drafts enabled by running hugo server -D.

❯ hugo server -D
Start building sites …
hugo v0.89.4+extended linux/amd64 BuildDate=unknown

                   | EN
-------------------+-----
  Pages            | 12
  Paginator pages  |  0
  Non-page files   |  0
  Static files     |  2
  Processed images |  0
  Aliases          |  3
  Sitemaps         |  1
  Cleaned          |  0

Built in 17 ms
Watching for changes in /home/kgb33/Code/blog.kgb33.dev/{archetypes,content,data,layouts,static,themes}
Watching for config changes in /home/kgb33/Code/blog.kgb33.dev/config.toml
Environment: "development"
Serving pages from memory
Running in Fast Render Mode. For full rebuilds on change: hugo server --disableFastRender
Web Server is available at http://localhost:1313/ (bind address 127.0.0.1)
Press Ctrl+C to stop

By default it should run on http://localhost:1313, and hot-reload. Open a new terminal and navigate back to the cite-name folder to set up git.

git init

cat <<EOF > .gitignore
# Hugo
public/
*.lock
_gen/
hugo_stats.json

# Node
node_modules/
EOF

Hugo creates a bunch of stuff, and its hard to tell what is needed in source control and what can (and will) be recreated. Be aware when adding files to git, and update .gitignore accordingly.

Step 1 - Setting Up the Theme

Theme Installation

From the cite-name directory created in the previous step, clone the schnerring/hugo-theme-gruvbox theme into a sub-module. Make sure to get the themes/gruvbox directory on the end of the command.

git submodule add git@github.com:schnerring/hugo-theme-gruvbox.git themes/gruvbox

Next, initialize a hugo module. This piggybacks of go modules and will create a go.mod and go.sum.

hugo mod init cite-name

Then, open config.toml.

First, change the default baseURL, languageCode and, title. Then add a new line defining the theme.

baseURL = 'https://blog.kgb33.dev'
languageCode = 'en-us'
title = ""
theme = "gruvbox"

Add the following so the theme can resolve directories correctly.

Check https://github.com/schnerring/hugo-theme-gruvbox/issues/16 to see if a better module specification has been implemented first!

[markup]
  # (Optional) To be able to use all Prism plugins, the theme enables unsafe
  # rendering by default
  #_merge = "deep"

[build]
  # The theme enables writeStats which is required for PurgeCSS
  _merge = "deep"

# This hopefully will be simpler in the future.
# See: https://github.com/schnerring/hugo-theme-gruvbox/issues/16
[module]
  [[module.imports]]
    path = "github.com/schnerring/hugo-mod-github-readme-stats"
  [[module.imports]]
    path = "github.com/schnerring/hugo-mod-json-resume"
    [[module.imports.mounts]]
      source = "data"
      target = "data"
    [[module.imports.mounts]]
      source = "layouts"
      target = "layouts"
    [[module.imports.mounts]]
      source = "assets/css/json-resume.css"
      target = "assets/css/critical/44-json-resume.css"
  [[module.mounts]]
    # required by hugo-mod-json-resume
    source = "node_modules/simple-icons/icons"
    target = "assets/simple-icons"
  [[module.mounts]]
    source = "assets"
    target = "assets"
  [[module.mounts]]
    source = "layouts"
    target = "layouts"
  [[module.mounts]]
    source = "static"
    target = "static"
  [[module.mounts]]
    source = "node_modules/prismjs"
    target = "assets/prismjs"
  [[module.mounts]]
    source = "node_modules/prism-themes/themes"
    target = "assets/prism-themes"
  [[module.mounts]]
    source = "node_modules/typeface-fira-code/files"
    target = "static/fonts"
  [[module.mounts]]
    source = "node_modules/typeface-roboto-slab/files"
    target = "static/fonts"
  [[module.mounts]]
    source = "node_modules/@tabler/icons/icons"
    target = "assets/tabler-icons"

Finally, run the following commands to initialize the theme, then restart the server.

hugo mod get
hugo mod npm pack
npm install

hugo server -D

Json Resume

This theme uses schnerring/hugo-mod-json-resume for a bunch of things, including populating the sidebar. To personalize the theme create a new file data/json_resume/en.json

This file must match the spec defined by JSON Resume.

{
  "$schema": "https://raw.githubusercontent.com/jsonresume/resume-schema/v1.0.0/schema.json",
  "basics": {
    "name": "Kelton Bassingthwaite",
    "image": "https://avatars.githubusercontent.com/u/17833726?v=4",
    "email": "KeltonBassingthwaite@gmail.com",
    "url": "https://blog.kgb33.dev",
    "profiles": [
      {
        "network": "Github",
        "username": "KGB33",
        "url": "https://github.com/KGB33"
      }
    ]
  }
}

CI/CD

Pre-commit

pre-commit is an awesome python cli tool to manage git hooks. I recommend installing it via pipx: pipx install pre-commit.

Then, create a new file .pre-commit-config.yaml.

# See https://pre-commit.com for more information
# See https://pre-commit.com/hooks.html for more hooks
repos:
  - repo: https://github.com/pre-commit/pre-commit-hooks
    rev: v4.1.0
    hooks:
      - id: trailing-whitespace
      - id: end-of-file-fixer
      - id: check-yaml
      - id: check-added-large-files
      - id: check-toml
  - repo: https://github.com/thlorenz/doctoc
    rev: v2.1.0
    hooks:
      - id: doctoc

Install the hooks to run on every (local) git commit via pre-commit install Then, to immediately run the hooks on all files use pre-commit run --all-files. To force a commit when the hooks fail use git commit --no-verify

Github Actions

If pre-commit is for local hooks, Github actions is for remote hooks. Each action is defined by a file .github/workflows/<hook-name>.yaml. Checkout https://docs.github.com/en/actions/quickstart for more info.

gaurav-nelson/github-action-markdown-link-check is exactly what it sounds like. Too add it create a new file, .github/workflows/md-link-check.yaml with the following content.

name: Check Markdown links

on:
  workflow_dispatch:
  schedule:
  # Run every Sunday at midnight
  - cron: "0 0 * * 0"

jobs:
  markdown-link-check:
    runs-on: ubuntu-latest
    steps:
    - uses: actions/checkout@master
    - uses: gaurav-nelson/github-action-markdown-link-check@v1
      with:
        use-quiet-mode: 'yes'
        use-verbose-mode: 'yes'
		config-file: 'mlc_config.json'

Where mlc_config.json contains:

{
  "ignorePatterns": [
    {
      "pattern": "^https?://localhost*"
    }
  ]
}

Once the above two files are committed and pushed to Github, navigate to the Actions tab, click “Check Markdown Links” in the left column, then hit the “Run workflow” drop down menu. The action will also run automatically at midnight every Sunday and Github will send you an email if it fails.

Production

Production Environment

This website is ran off of an Ubuntu 21.10 LXE container managed by Proxmox. Once the container is up and running make sure to preform an update and download git and curl.

apt update && \
apt upgrade -y && \
apt install git curl golang -y

Next, download and install the latest binary release for Hugo (extended) from the releases page.

export HUGO_VER="0.91.2" && \
curl -L https://github.com/gohugoio/hugo/releases/download/v${HUGO_VER}/hugo_extended_${HUGO_VER}_Linux-64bit.deb -o hugo.deb && \
apt install ./hugo.deb

Then do the same for the latest NodeJS version.

export NODE_VER="17.3.0" && \
curl -L https://nodejs.org/dist/v${NODE_VER}/node-v${NODE_VER}-linux-x64.tar.gz -o node.tar.gz && \
tar -C /usr/local --strip-components 1 -xzf node.tar.gz && source ~/.bashrc

Then clone the blog repository and install dependencies.

git clone --recurse-submodules --remote-submodules https://github.com/KGB33/blog.kgb33.dev.git && \
cd blog.kgb33.dev && \
hugo mod get && \
hugo mod npm pack && \
npm install

Note: At least one published post is required for Hugo build successfully with this theme.

Systemd Service

Systemd will be responsible for managing the Hugo server. This will ensure that Hugo will restart after a reboot or crash. First, create a systemd unit file at /etc/systemd/system/hugoserver.service.

[Unit]
Description=Hugo-Server

[Service]
ExecStart=/usr/local/bin/hugo server --bind=0.0.0.0
WorkingDirectory=/root/blog.kgb33.dev/
Restart=always
RestartSec=5

[Install]
WantedBy=multi-user.target

Nginx Reverse Proxy

I have Nginx running as a reverse proxy so I can serve separate sub-domains from the same public IP address.

To serve the blog add the following file /etc/nginx/sites-avalable/blog

server {
        server_name blog.kgb33.dev;

        location / {
            proxy_pass  http://10.0.0.106:1313;
			proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
        	proxy_set_header Host $http_host;
        	proxy_set_header X-Forwarded-Proto https;
        	proxy_redirect off;
        	proxy_http_version 1.1;
        }
}

Then enable the site using a symlink.

sudo ln -s sites-available/blog sites-enabled/blog

Lets Encrypt Auto-certs via Cloudflare DNS

Install and configure certbot using their docs and setup a DNS record for blog.kgb33.dev. Then run the following command to get the certs from Lets Encrypt. Where cloudflare.ini contains the Cloudflare api token.

sudo certbot -i nginx --dns-cloudflare --dns-cloudflare-credentials ~/cloudflare.ini -d blog.kgb33.dev

Lastly, restart nginx.

systemctl restart nginx

Then checkout https://blog.kgb33.dev!