Running multiple Docker (microservice) containers on the same local machine can introduce some challenges: only one
container can bind to port 80 at any given time, and assuming that every microservice has their
own docker-compose.yaml
, it can be difficult to make multiple microservices communicate with each other.
In production, you might use a container orchestration tool (such as Kubernetes) which takes care of these issues, but
running Kubernetes on your local machine is usually overkill.
This repository contains a simple docker-compose.yaml
file that runs an instance of
Traefik Proxy for local development purposes. This allows you to run multiple web
services locally, without port conflicts, each with their own domain name.
The docker-compose.yaml
in this repository creates a new Docker network and starts an instance of Traefik, a reverse
proxy. Port 80 of the Traefik container is mapped to port 80 of the host. Other applications (with their
own docker-compose.yaml
) can connect to the shared network and will be proxied by Traefik instead of publishing their
ports on the host machine.
Traefik can be configured using Docker labels, so exposing a new application is a matter of adding a network and some
labels to the application's docker-compose.yaml
.
git clone git@github.com:nicwortel/local-development-proxy.git
cd local-development-proxy
docker compose up -d
After doing this once, the proxy container will start up automatically when Docker starts (for example after rebooting
your machine), thanks to the line
restart: always
.
In order to connect with the reverse proxy and expose your application, add this to the docker-composer.yaml
of your
project:
services:
web:
image: my/image
+ # Connect to this project's network (default) AND the local development proxy network (web)
+ networks:
+ - default
+ - web
+ labels:
+ traefik.enable: 'true'
+ traefik.http.routers.myproject.rule: 'Host(`some-application.localhost`)'
+ traefik.http.routers.myproject.entrypoints: web
+ traefik.docker.network: web
db:
image: mysql
+networks:
+ web:
+ external: true
Traefik will use the Host
HTTP header to route requests to the right container, but in order for your request to be
handled by Traefik you need to ensure that your host machine can resolve the application's domain name to 127.0.0.1.
There are a few ways to achieve this:
- There is a draft RFC to have
any *.localhost domain resolve to 127.0.0.1.
systemd-resolved (used by Ubuntu
Linux) and certain browsers (Chrome) already implement this and resolve all domain names ending with
.localhost
to 127.0.0.1. - You can use dnsmasq and route all domain names ending with
.localhost
to 127.0.0.1. - You can manually add the domain name(s) to your hosts file.