This is a short story about how I attempted to deploy my own simple Django API with Traefik as a reverse proxy and proxied through Cloudflare, plus the frontend deployed on Cloudflare Pages. While a fairly straightforward process on paper, I gave up once or twice only to come back a week or two later.

You can check out this deployed web app at democracy.ryanfleck.ca

PaaS systems like Heroku spoil young developers by abstracting all of the networking, installation, and general pain associated with the deployment of web applications. While it’s fairly easy to see quick results, I’ve found ensuring you have a secure and durable deployment from a home server (aka for cheep) is rough. It took more than a couple days of reading docs and headbanging to figure out where and why bits and pieces of the solution were failing.

I know that writing this article dramatically compromises the security of my home box, but I’m fairly certain at this point that only what is absolutely necessary is on and enabled. Please reach out to [email protected] if you’d like to alert me to any fatal mistakes.

Alpine Linux and Docker

I chose to use Alpine Linux and Docker containers to build the foundations of my home server. After installing Alpine, adding Docker is as simple as:

apk add docker docker-compose
addgroup <username> docker
rc-update add docker boot
service docker start

Traefik and Portainer

Here’s my traefik.toml - the important part of this configuration is the providers.docker section that allows us to list a network that will be watched for new applications.

debug = false 
checkNewVersion = true

[entryPoints]
  [entryPoints.web]
    address = ":80"
    [entryPoints.web.http.redirections.entryPoint]
      to = "websecure"
      scheme = "https"

  [entryPoints.websecure]
    address = ":443"

[api]
  dashboard = true

[certificatesResolvers.lets-encrypt.acme]
  email = "****************"
  storage = "/letsencrypt/acme.json"

[certificatesResolvers.lets-encrypt.acme.dnschallenge]
  provider = "cloudflare"

[providers.docker]
  watch = true
  network = "web"
  exposedbydefault = false

[acme]
  entryPoint="https"

[acme.httpChallenge]
  entryPoint="http"

In addition to Traefik I wanted dynamic DNS so my server’s public IP is always updated in Cloudflare’s nameservers when my ISP inevitably changes it. To do this I use a cloudflare-ddns container, which is easy to set up.

version: '3'

networks:
  web:
    external: true
  internal:
    external: false

services:

  cloudflare-ddns:
    image: oznu/cloudflare-ddns:latest
    restart: always
    environment:
      - API_KEY=**************************
      - ZONE=********************
      - SUBDOMAIN=server
      - PROXIED=false

  # https://lefthandbrain.com/traefik-wildcard-lets-encrypt/
  traefik:
    image: traefik:latest
    restart: always
    volumes:
      - /var/run/docker.sock:/var/run/docker.sock
      - ./traefik.toml:/traefik.toml
      - ./traefik_dynamic.toml:/traefik_dynamic.toml
      - letsencrypt:/letsencrypt
    ports:
      - 80:80
      - 443:443
    networks:
      - internal
      - web
    environment:
      - CLOUDFLARE_EMAIL=***************************
      - CLOUDFLARE_DNS_API_TOKEN=**************************

volumes:
  letsencrypt:

Finally, a nice UI to visualize all my docker containers would be nice. I chose Portainer and highly recommend it, though don’t like deploying things with its UI - I stick to docker-compose. These lines were added to my “administrative” docker-compose.yml file listed above.

  portainer:
    image: portainer/portainer-ce:latest
    restart: always
    ports:
      - 9000:9000
    volumes:
      - /var/run/docker.sock:/var/run/docker.sock
      - ./data:/portainer-data
    environment:
      - TZ="********************"
    networks:
      - internal
      - web
    labels:
      - "traefik.enable=true"
      - "traefik.http.routers.portainer.rule=Host(`portainer.ryanfleck.ca`)"
      - "traefik.http.routers.portainer.entrypoints=websecure"
      - "traefik.http.services.portainer.loadbalancer.server.port=9000"
      - "traefik.http.routers.portainer.service=portainer"
      - "traefik.http.routers.portainer.tls.certresolver=lets-encrypt"

…oh, but what are those labels at the bottom? Now that I’ve set up Traefik, any service I run can be selectively exposed to the internet as long as it exists on the web network and has this set of labels to selectively enable networking features.

Forward port 443 from your router to your server after executing the following command to get your server online, and don’t forget to add subdomain CNAME records pointing towards your server’s A record.

docker-compose up -d

Troubleshooting the Networking Configs

Spinning up the Django app was the easy part. Cookie-Cutter Django provides a flexible template with reasonable defaults, and with everything running on one box, requests between Django, Postgres, and Redis should be lightning fast.

By far, the most difficult part was figuring out if Traefik, Cloudflare, or Django itself was bungling my Access-Control-Allow-Origin headers. After quite a long time fiddling with Django’s CORS settings and searching for Cloudflare settings, I figured Traefik was the part of the networking spiderweb that was being restrictive, and for good reason.

Adding some Traefik middleware saved the day.

version: '3'

volumes:
  production_postgres_data: {}
  production_postgres_data_backups: {}

networks:
  web:
    external: true
  internal:
    external: false

services:
  democracy-be-prod:
    image: democracy_production_django
    restart: always
    build:
      context: .
      dockerfile: ./compose/production/django/Dockerfile
    depends_on:
      - postgres
      - redis
    env_file:
      - ./.envs/.production/.django
      - ./.envs/.production/.postgres
      - ./.env
    command: /start
    ports:
      - "5000:5000"
    labels:
      - "traefik.enable=true"
      - "traefik.http.routers.democracy-be-prod.rule=Host(`democracy-api.ryanfleck.ca`)"
      - "traefik.http.routers.democracy-be-prod.entrypoints=websecure"
      - "traefik.http.services.democracy-be-prod.loadbalancer.server.port=5000"
      - "traefik.http.routers.democracy-be-prod.service=democracy-be-prod"
      - "traefik.http.routers.democracy-be-prod.tls.certresolver=lets-encrypt"
      - "traefik.http.middlewares.corsHeader.headers.accesscontrolallowmethods=GET,POST,OPTIONS,DELETE,PUT,PATCH"
      - "traefik.http.middlewares.corsHeader.headers.accesscontrolalloworiginlist=https://democracy.ryanfleck.ca,https://democracy-api.ryanfleck.ca"
      - "traefik.http.middlewares.corsHeader.headers.accesscontrolmaxage=100"
      - "traefik.http.middlewares.corsHeader.headers.customresponseheaders.X-RCF-Test-Header=Bitchin"
      - "traefik.http.middlewares.corsHeader.headers.accessControlAllowHeaders=*"
      - "traefik.http.middlewares.corsHeader.headers.addvaryheader=true"
      - "traefik.http.routers.democracy-be-prod.middlewares=corsHeader@docker"
    networks:
      - internal
      - web

  postgres:
    image: democracy_production_postgres
    restart: always
    build:
      context: .
      dockerfile: ./compose/production/postgres/Dockerfile
    volumes:
      - production_postgres_data:/var/lib/postgresql/data:Z
      - production_postgres_data_backups:/backups:z
    env_file:
      - ./.envs/.production/.django
      - ./.envs/.production/.postgres
      - ./.env
    networks:
      - internal

  redis:
    image: redis:5.0
    restart: always
    env_file:
      - ./.envs/.production/.django
      - ./.envs/.production/.postgres
      - ./.env
    networks:
      - internal

The system was repeatedly brought down and back up with:

docker-compose -f production.yml --env-file .env up --force-recreate -d

So I tried to create an account, and… crap, I hadn’t run the migrations. Thought it’d do that for me. Oh well, that’s one step away.

It turns out that opening the docker continer directly with docker exec -it <container> /bin/bash doesn’t load all the container’s environment variables. My next instinct was to run the command through docker-compose like this:

docker-compose -f production.yml run --rm django python manage.py migrate

Bam, it works. The whole thing. Got the registration email in my inbox. Time for some user testing! Maybe I’ll deploy a few more apps.


…so, what did I learn?

  • Restrictive networking by default is good.
  • The above is painful until you know where the defenses are raised.
  • Docker is a blessing.