Requirements#

The source code must be stored on a public Github repository, as the ability to use Github Pages with private repositories is limited to Enterprise users.

Deploy the website#

Before having the option of configuring a custom domain name, one must first deploy their website through Github Pages. Depending on the way the website has been built, these instructions differ. For instance, the website you are currently on has been generated using Hugo, and the following Github workflow is used to automatically build the website.

.github/workflows/hugo.yaml
# 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:
      - master

  # 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-latest
    env:
      HUGO_VERSION: 0.145.0
      HUGO_ENVIRONMENT: production
      TZ: Europe/Zurich
    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: Cache Restore
        id: cache-restore
        uses: actions/cache/restore@v4
        with:
          path: |
            ${{ runner.temp }}/hugo_cache
          key: hugo-${{ github.run_id }}
          restore-keys:
            hugo-
      - name: Build with Hugo
        run: |
          hugo \
            --gc \
            --minify \
            --baseURL "${{ steps.pages.outputs.base_url }}/" \
            --cacheDir "${{ runner.temp }}/hugo_cache"
      - name: Cache Save
        id: cache-save
        uses: actions/cache/save@v4
        with:
          path: |
            ${{ runner.temp }}/hugo_cache
          key: ${{ steps.cache-restore.outputs.cache-primary-key }}
      - 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-latest
    needs: build
    steps:
      - name: Deploy to GitHub Pages
        id: deployment
        uses: actions/deploy-pages@v4

This configuration file is based on the one provided by the Hugo documentation1.

DNS configuration#

In order to explain to the DNS servers how to resolve the custom domain name to the existing domain automatically generated by Github, a new CNAME record must be created. For instance, in my case I added the following line to “link” the blog subdomain:

Enregistrement CNAME
blog 300 IN CNAME wolfiiy.github.io.

or, using my registrar’s GUI:

Adding a CNAME record for the blog subdomain

Adding a CNAME record for the blog subdomain.

Verification#

You can verify the newly created CNAME record using the command dig <subdomain>.<domain>.<tld>. Upon execution, the default domain name and the Github Pages IP addresses should be listed2:

  • 185.199.108.153
  • 185.199.109.153
  • 185.199.110.153
  • 185.199.111.153

For instance, once the command dig blog.wolfiy.ch has been executed, the following output can be seen:

Output de la commande dig
; <<>> DiG 9.20.7 <<>> blog.wolfiy.ch
;; global options: +cmd
;; Got answer:
;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 65404
;; flags: qr rd ra; QUERY: 1, ANSWER: 5, AUTHORITY: 0, ADDITIONAL: 1

;; OPT PSEUDOSECTION:
; EDNS: version: 0, flags:; udp: 1232
;; QUESTION SECTION:
;blog.wolfiy.ch.			IN	A

;; ANSWER SECTION:
blog.wolfiy.ch.		269	IN	CNAME	wolfiiy.github.io.
wolfiiy.github.io.	3569	IN	A	185.199.108.153
wolfiiy.github.io.	3569	IN	A	185.199.109.153
wolfiiy.github.io.	3569	IN	A	185.199.110.153
wolfiiy.github.io.	3569	IN	A	185.199.111.153

;; Query time: 3 msec
;; SERVER: 192.168.1.1#53(192.168.1.1) (UDP)
;; WHEN: Tue Apr 01 20:22:07 CEST 2025
;; MSG SIZE  rcvd: 138

Add subdomain#

Now that the DNS has been configured, open the repository’s Github page and, under Settings > Pages > Custom domain, enter your subdomain and click Save.

DNS check in progress

DNS check in progress.

Once saved, a DNS check is done. This step can last a few minutes and, once over, the DNS Check successful message appears just below your subdomain.

DNS check successful

DNS check successful.

HTTPS#

HTTPS can be enforced on the website by clicking the checkbox under the custom domain field. Note that this option might be temporarily unavailable if the domain was recently setup since a new certificate has to be generated.

First step of the certificate generation process

First step of the certificate generation process.

Second step of the certificate generation process

Second step of the certificate generation process.

Last step of the certificate generation process

Last step of the certificate generation process.

In theory, the process can take up to 24 hours. In parctice, a dozen minutes is enough.

Finally, the Enforce HTTPS checkbox can be toggled.

Enforcing HTTPS

Enforcing HTTPS.

Potential issues#

If your website was built using Hugo and no styles nor scripts seem to load on Chromium based browsers, there might be an issue with the internal references between files. Verify that the baseUrl has been correctly set and unpublish/republish the website to clear the cache3.