Logo kepa.eu.org

setup static page with hugo, github pages, actions and custom domain

Nov 1, 2024 - 9 minute read

preface


In this series of brief tutorials, I will show you how to generate a simple static website, host it on GitHub Pages with a custom domain, and do it all for free.

I’ve noticed many people using Medium to share their ideas, and it’s a fantastic platform. The concentration of brilliant techies posting their thoughts and solutions is truly impressive. However, what about those who want to build their own brand, promote their own domain independently, and avoid getting lost in the crowd? While it’s important to promote our site through various channels to achieve the best rankings, we can still host our small projects on our own domain where we have near-complete control.

steps

install hugo


To start creating your page, we first need to install the Hugo environment. There are various static page generators like Jekyll, Next.js, and Gatsby.

My goal was to be quite versatile. The benefits and ease of use when creating a website comes down to creating the proper Markdown files, which is both easy and awesome! I chose Hugo after digging into multiple sources and receiving good feedback, as well as the extensive marketplace of themes.

Hugo is blazing fast and is supported on many platforms and through multiple package managers. I personally use Ubuntu Linux as my base host, which is supported out of the box.

Whole setup details, including repositories, can be found under this link.

Keep in mind that Hugo requires Golang version 1.22 or higher and sometimes the dart-sass package for proper CSS transpilation.

create a simple website


The documentation steps can be found under this link.

After installing Hugo, you can check if it’s ready to go by following these steps:

  • Create a website with YAML as the starting config format (TOML is the default).
  • Navigate to the folder of your website and init repo.
  • Add the initial theme through a git submodule.
  • Add the theme name to the base config.
  • Run and check the output.
hugo new site demo --format yaml
cd demo && git init
git submodule add https://github.com/theNewDynamic/gohugo-theme-ananke.git themes/ananke
echo "theme: ananke" >> hugo.yaml
hugo serve

Navigate to the link from the CLI output to check the results, which by default is http://localhost:1313/ If the port is already taken, you will get a random port from the high port range (30000+).

Your whole statically generated page is in the ./public folder within the demo directory. From here, you can take it and publish it wherever you like.

Keep in mind that if you did not stop the hugo serve command, all changes will be reflected immediately after saving any config or site files.

install theme


⚠️ This part can be tricky since each template works differently! Some templates work directly via a git submodule add or the hugo mod get command. Others need to be copied or cloned directly from the template creator’s Git repository.

First, choose your favorite theme from the official site https://themes.gohugo.io/. Then, navigate to the details of the theme and follow the instructions provided in the description!

After you have found it and properly installed and configured the details, give it a try and run the hugo serve command!

own domain


If you already have your domain, you can skip this point.

I have the privilege of having my domain registered with nic.eu.org, which is completely free of charge and leased for 100 years.

To register your own domain, please follow the instructions from this link.

Keep in mind that since this is an unpaid service, any kind of help is welcome, and the request can take some time. I waited around two years for mine, but that was more than ten years ago. It’s still worth it!

domain parking

To complement the open-source project of domain registration, I use the freeDNS 42 service https://freedns.42.pl/ where I parked my domain. It works brilliantly, and I can highly recommend it!

point your origin to GitHubPages


The entire process is detailed in this official GitHub Pages documentation.

I am describing it prior to repository deployment because DNS changes can take up to 24 hours to propagate!

If you have your own domain registered already, inside your DNS settings point the A/AAAA or an ALIAS record of the origin @ to the following IP addresses:

# IPv4 (A)
185.199.108.153
185.199.109.153
185.199.110.153
185.199.111.153

# IPv6 (AAAA)
2606:50c0:8000::153
2606:50c0:8001::153
2606:50c0:8002::153
2606:50c0:8003::153

An example dig anwser

# get A records
$ dig +nocmd +noall +answer www.kepa.eu.org A
www.kepa.eu.org.	6883	IN	CNAME	kepa.eu.org.
kepa.eu.org.		6883	IN	A	185.199.110.153
kepa.eu.org.		6883	IN	A	185.199.111.153
kepa.eu.org.		6883	IN	A	185.199.108.153
kepa.eu.org.		6883	IN	A	185.199.109.153

# get AAAA records
$ dig +nocmd +noall +answer kepa.eu.org AAAA
kepa.eu.org.		6917	IN	AAAA	2606:50c0:8001::153
kepa.eu.org.		6917	IN	AAAA	2606:50c0:8003::153
kepa.eu.org.		6917	IN	AAAA	2606:50c0:8000::153
kepa.eu.org.		6917	IN	AAAA	2606:50c0:8002::153

At a later step, when registering the custom domain in your GitHub repository, you will be asked to add a TXT record to verify that you are the owner of the DNS origin. This is a common process for many providers, so there’s no need to worry.

redirection with www+yourTLD

Please don’t forget to create a CNAME record inside your DNS Zone for your custom domain with www in front!

This will allow you to reach your website not only with the domain (TLD) but also with www.

For example, you can reach my website at kepa.eu.org as well as www.kepa.eu.org. Many forget about this, but it’s quite handy and eliminates potential issues with link forwarding!

Example answer from dig

# get CNAME records for your www+tld domain
$ dig +nocmd +noall +answer www.kepa.eu.org CNAME
www.kepa.eu.org.	7119	IN	CNAME	kepa.eu.org.

add proper .gitignore

When we run the hugo command, the engine generates several additional files and folders where we can see the result of our generated website. These files change quite frequently, so we might consider excluding them from the target repository, unless we intend to serve the build via GitHub Actions.

Keep in mind that after you run Hugo in your CLI shell, the ./public folder is all you need to run your website. This folder contains everything necessary, such as HTML files, SCSS, JS, and images, and you’re ready to go.

Below an example

# Hugo stuff
.hugo_build.lock
resources/
public/

deploy to repo and activate pages

Before or after creating your repository, ensure that it is publicly available to host it properly.

pic

We will now push our changes to the freshly created repository and activate GitHub Pages.

At this moment I do expect that you are publishing your changes to GitHub repo 🤖 and if you don’t know how please support yourself with this link!.

P.S. Use this command to generate a strong SSH key for communicating with your resources on GitHub:

ssh-keygen -t ed25519 -a 100 -C "yourEmail@domain.com"

After the changes are pushed, navigate to

  • the Settings 🛞 icon of this repository
  • from the left sidebar, under General 🛞
  • go to Pages inside the Code and automation section.

As mentioned before, since you have already generated your page, all you need to do is publish the content of the * ./public* folder to your repository. You can add it either to the root directory /(root) or the /docs folder to publish your website. That’s all you need to do for a classical setup.

pic

However, if you want to build everything with GitHub Actions, please add the .gitignore file as mentioned earlier and publish the entire content of your current Hugo website folder.

Custom Domain

CNAME and certificate

After saving your changes when deploying from a branch (GitHub Actions will follow), please add your custom domain name as described in step 2 (as shown in the picture). For more details, follow this link.

Additionally, make sure to enforce the HTTPS checkbox, as indicated in step 3.

The domain of your choice should also be saved as a file named CNAME, containing the correct subdomain name, inside the folder of your website. Otherwise, proper HTTPS and domain redirection will not work. For more details, follow this link

pic

setup proper DNS TXT verification

After you create and activate your custom domain, you will be prompted to add TXT challenge to your DNS configuration to verify ownership. For more details, follow this link.

deploy with GH Actions

proper file

Original documentation link.

Please find here an example file which I am using to provision my GitHub Action. Remember to place your yaml file inside following folder structure inside your repo .github/workflows

The most crucial parts are the following:

Code Line numberDescription
33the same OS I am currently testing my changes
35the same cli version of hugo I am using locally
60use this to properly translate links for your internal subpages
64all published inside ./public folder will be visible under your apex domain
# Sample workflow for building and deploying a Hugo site to GitHub Pages
name: Deploy Hugo site to Pages

on:
  # Runs on pushes targeting the default branch
  push:
    branches:
      - main

  # Allows you to run this workflow manually from the Actions tab
  workflow_dispatch:

# Sets permissions of the GITHUB_TOKEN to allow deployment to GitHub Pages
permissions:
  contents: read
  pages: write
  id-token: write

# Allow only one concurrent deployment, skipping runs queued between the run in-progress and latest queued.
# However, do NOT cancel in-progress runs as we want to allow these production deployments to complete.
concurrency:
  group: "pages"
  cancel-in-progress: false

# Default to bash
defaults:
  run:
    shell: bash

jobs:
  # Build job
  build:
    runs-on: ubuntu-22.04
    env:
      HUGO_VERSION: 0.137.1
    steps:
      - name: Install Hugo CLI
        run: |
          wget -O ${{ runner.temp }}/hugo.deb https://github.com/gohugoio/hugo/releases/download/v${HUGO_VERSION}/hugo_extended_${HUGO_VERSION}_linux-amd64.deb \
          && sudo dpkg -i ${{ runner.temp }}/hugo.deb          
      - name: Install Dart Sass
        run: sudo snap install dart-sass
      - name: Checkout
        uses: actions/checkout@v4
        with:
          submodules: recursive
          fetch-depth: 0
      - name: Setup Pages
        id: pages
        uses: actions/configure-pages@v5
      - name: Install Node.js dependencies
        run: "[[ -f package-lock.json || -f npm-shrinkwrap.json ]] && npm ci || true"
      - name: Build with Hugo
        env:
          HUGO_CACHEDIR: ${{ runner.temp }}/hugo_cache
          HUGO_ENVIRONMENT: production
          TZ: Etc/UTC
        run: |
          hugo \
            --environment prod          
      - name: Upload artifact
        uses: actions/upload-pages-artifact@v3
        with:
          path: ./public

  # Deployment job
  deploy:
    environment:
      name: github-pages
      url: ${{ steps.deployment.outputs.page_url }}
    runs-on: ubuntu-22.04
    needs: build
    steps:
      - name: Deploy to GitHub Pages
        id: deployment
        uses: actions/deploy-pages@v4

proper setting inside repository settings

Once again navigate to

  • the Settings 🛞 icon of this repository
  • from the left sidebar, under General 🛞
  • go to Pages inside the Code and automation section.

This time change Source to GitHub Actions and select proper file name Deploy Hugo site to Pages like in my case.

pic

After this is done, each commit you push to master or main branch will result an action like in the picture below

pic

That’s it!

Have fun with troubleshooting 😄


In my next post, I’ll show you how to properly set up a free and secure email box with custom domain which works like a charm.

  • create custom mail with zoho
  • setup spf + dkim