Nisanth Chunduru

Nisanth Chunduru

Moving from kamal to Docker. Host your blog using Docker.

2025-02-28T00:00:00Z

Since I started this blog, I hosted it on a Cloud VM using Basecamp’s kamal. Last year, Basecamp released kamal 2 which is slightly backwards incompatible with kamal 1. When a kamal command I ran failed, I decided to just upgrade to kamal 2 (instead of investing time to investigate a failure in an older and possibly unmaintained version).

After modifying my kamal configuration file (so it became compatible with kamal 2) and creating the .kamal/secrets file, I tried upgrading to kamal 2.

I first ran the kamal upgrade command which ƒailed because my blog’s Docker container failed kamal’s health check https://gist.github.com/nisanthchunduru/dc908ac676454c18adb64837bf204aeb I resolved the problem by adding a HEALTHCHECK directive to my blog’s Dockerfile. Unfortunately, while doing that caused kamal upgrade to succeed, kamal deploy started to fail with an error message

...
INFO [f6b4836f] Running /usr/bin/env git -C /var/folders/q3/0jl1k3mx143fw8xbs4m8qp0w0000gq/T/kamal-clones/blog-f5d00af52b8b7/blog/ status --porcelain as chunisan@localhost
INFO [f6b4836f] Finished in 0.038 seconds with exit status 0 (successful).
INFO [98e6f00b] Running /usr/bin/env git -C /var/folders/q3/0jl1k3mx143fw8xbs4m8qp0w0000gq/T/kamal-clones/blog-f5d00af52b8b7/blog/ rev-parse HEAD as chunisan@localhost        
INFO [98e6f00b] Finished in 0.031 seconds with exit status 0 (successful).        
Finished all in 8.5 second
ERROR (NoMethodError): undefined method `gsub' for true

gsub is a method that a Ruby string has. A variable that kamal expected to a string was instead a boolean and had the value true.

While I’ve extensive experience with Ruby and will likely be able to resolve this problem, I wondered if it may far easier to just start Docker container/containers myself (especially given the application I was trying to host is just a blog).

To my pleasant surprise, I found that it was very easy to start a Docker container. I ran the command below

mkdir -p /opt/blog-content/
docker run -d -p 80:1313 -v /opt/blog-content:/opt/blog/content --restart unless-stopped nisanth074/blog:latest

and just that single command brought my blog online and made it publicly on http://

HTTPS

To make my blog available on https://, I turned on Cloudflare’s Proxy feature for my blog domain’s DNS A record in Cloudflare’s dashboard. Doing that causes Cloudflare to issue an SSL certificate and serve my blog over https. While the traffic between Cloudflare and my blog is still unencrypted, that’s an acceptable tradeoff for me.

Untitled If you don’t use Cloudflare or dislike the approach above, you can easily install and configure Caddy. Caddy will automatically obtain an SSL certificate for your blog’s domain and serve it over https://

Ancillary Docker containers

As I write my blog posts in Notion, I also started hugo-notion (a CLI tool that I developed in Go to sync blog posts from Notion to Hugo’s content dir) in a different Docker container

mkdir -p /opt/blog

touch /opt/blog/.env
echo "CONTENT_NOTION_URL=https://www.notion.so/blog_content-0f1b55769779411a95df1ee9b4b070c9" >> /opt/blog/.env
echo "NOTION_TOKEN=my_notion_access_token" >> /opt/blog/.env
echo "GOPROXY=direct" >> /opt/blog/.env

docker run -d -v /opt/blog-content:/opt/blog/content --env-file /opt/blog/.env --restart unless-stopped nisanth074/hugo-notion 

Additionally, to restart/update my blog and hugo-notion with more ease, I added a docker-compose.yml file

services:
  sinatra:
    image: nisanth074/blog:latest
    volumes:
      - /opt/blog-content:/opt/blog/content/
    ports:
      - "80:4567"
    restart: unless-stopped
  hugo-notion:
    image: nisanth074/hugo-notion
    volumes:
      - /opt/blog-content:/opt/blog/content
    env_file: /opt/blog/.env
    working_dir: /opt/blog

and a bash script to copy the docker compose file to my CloudVM and to restart Docker containers

npm run docker:push:x86
scp .env [email protected]:/opt/blog/.env
scp docker-compose.production.yml [email protected]:/opt/blog/docker-compose.yml
ssh [email protected] "cd /opt/blog && docker compose down"
ssh [email protected] "cd /opt/blog && docker compose up -d --pull always"

This move re-inforced my opinion that every abstraction introduced to a project (kamal in this scenario) should bring value that significantly surpasses the learning curve and complexity the abstraction would bring about. If not, that abstraction should be discarded.

I hope this post helps you quickly host your hobby projects using Docker.

Built upon Notion & Sinatra. Source Code.