A dynamic traefik to traefik discovery agent.
"Kobling" means "coupling" or "linking" in Norwegian. Traefik Kobling lets homelab users link Traefik instances on different hosts to a main, public-facing Traefik instance without using Docker Swarm or publishing ports.
Going forward, we will refer to the main, public-facing Traefik instance as the main instance and the others will be called local instances.
This is accomplished by using Traefik's API on the local instances to find out which rules are registered and then publish them to redis, so the main instance can read and use them.
For the main traefik instance, you have to configure it to use the redis provider in traefik.yml
:
providers:
redis:
endpoints:
- "localhost:6379" # change the address if the redis instance is not available in local host
# other providers
Also, if you want to use HTTPS, it should be done in this instance.
Configure all your local traefik instances to have API access enabled in their traefik.yml
:
- If the local traefik instances are located within your own local network, then you can allow insecure access and connect to it over port 8080:
api:
insecure: true
- if your local traefik instances can be accessed through the internet, then you should not be allowing insecure access and you should set up some sort of authentication. Traefik Kobling only supports Basic Auth and here's a guide on how to set it up in your instance.
These instances do not (and probably should not) have HTTPS enabled.
We can write our config file config.yml
:
servers:
- name: "other-host"
apiAddress: http://192.168.0.10:8080
destinationAddress: http://192.168.0.10
entryPoints:
web: web
websecure: web
And then finally, we can set up Traefik Kobling on your main instance:
services:
traefik-kobling:
image: ghcr.io/ldellisola/traefik-kobling
volumes:
- ./config.yml:/config.yml
environment:
REDIS_URL: "localhost:6379"
There are two places where you can configure this application. First there are some environment variables:
REDIS_URL
: Specify the URL of the redis database. The default isredis:6379
.CONFIG_PATH
: It let's the user change the location of theconfig.yml
file. The default location is/config.yml
.RUN_EVERY
: It specifies how many seconds to wait before checking the local instances for changes. The default value is 60 seconds.
You must also create a config.yml
. This file contains a list of servers with information
about what the address of the local traefik instances are and where traffic should
be redirected.
servers:
- name: "host-1"
apiAddress: http://192.168.0.10:8080
destinationAddress: http://192.168.0.10
entryPoints:
web: web
websecure: web
- name: "host-2"
apiAddress: http://192.168.0.11:8080
destinationAddress: http://192.168.0.11
entryPoints:
web-tcp: local-tcp
- name: "host-3"
apiAddress: http://192.168.0.12:8080
destinationAddress: http://192.168.0.12
The entryPoints
mapping works in the following way:
The entrypoints of your main instance are on the left and the entrypoint of the local instance are on the right, where ultimately traffic will be forwarded.
This approach means we do not have to register more routers than necessary and it helps keep our main dashboard clean.
If no entrypoints are provided in the configuration, the default value http
is used for both the main instance as well as local instances.
If your local instance has basic auth enabled, then you have to specify it in the Kobling config:
servers:
- name: "host-1"
apiAddress: http://username:[email protected]
apiHost: traefik.domain.tld
destinationAddress: http://192.168.0.10
entryPoints:
web: web
websecure: web
The address in apiAddress
should include the username and password to access the api and apiHost
should be the host name for that service.
So what does this mean?
Let's say we have 2 machines in our home network:
- Machine A is exposing port 80 and 443 to the internet and it is running the following containers:
traefik
: this is our main instance and has the dashboard exposed in the FQDNmain.domain.tld
redis
: it's the storage for the main instance to use as a provider.traefik koblink
: it will read data from the traefik instance in machine B and provide redirect data for the main traefik instance.
For this machine we will deploy the following docker-compose.yml
file:
networks:
default:
name: "web"
services:
traefik:
image: traefik:latest
ports:
- 80:80
- 443:443
environment:
CF_API_EMAIL: ${CF_API_EMAIL}
CF_API_KEY: ${CF_API_KEY}
volumes:
- /var/run/docker.sock:/var/run/docker.sock
- ./traefik.yml:/traefik.yml
- ./acme.json:/acme.json
redis:
image: redis:alpine
traefik-kobling:
image: ghcr.io/ldellisola/traefik-kobling
volumes:
- ./config.yml:/config.yml
environment:
REDIS_URL: "redis:6379"
The traefik.yml
looks like this:
api:
dashboard: true
entryPoints:
web:
address: ":80"
http:
redirections:
entrypoint:
to: web-secure
scheme: https
web-secure:
address: ":433"
http:
tls:
certResolver: "cloudflare"
domains:
- main: "domain.tld"
sans:
- "*.domain.tld"
serversTransport:
insecureSkipVerify: true
providers:
redis:
endpoints:
- "redis:6379"
certificatesResolvers:
cloudflare:
acme:
email: [email protected]
storage: /acme.json
dnsChallenge:
provider: cloudflare
resolvers:
- "1.1.1.1:53"
- "1.0.0.1:53"
The config.yml
looks like this:
servers:
- name: "machine-b"
apiAddress: http://192.168.0.10:8080
destinationAddress: http://192.168.0.10
entryPoints:
web: local
web-secure: local
- Machine B is on IP 192.168.0.10 and it runs:
traefik
: this is a local instance and has the FQDNlocal.domain.tld
Service B
: another random service hosted in this server, with the FQDNserviceB.domain.tld
It is deployed with the following docker-compose.yml
:
networks:
web:
name: "web"
services:
traefik:
image: traefik:latest
ports:
- 80:80
- 8080:8080
networks:
- web
volumes:
- /var/run/docker.sock:/var/run/docker.sock
- ./traefik.yml:/traefik.yml
labels:
traefik.enable: "true"
traefik.http.routers.traefik.rule: "Host(`local.domain.tld`)
traefik.http.routers.traefik.service: "api@internal"
traefik.http.services.traefik.loadbalancer.server.port: "8080"
service-b:
image: service-b:latest
networks:
- web
labels:
traefik.enable: "true"
traefik.http.routers.service-b.rule: "Host(`serviceB.domain.tld`)
traefik.http.services.service-b.loadbalancer.server.port: "8080"
And the traefik.yml
is:
api:
insecure: true
entryPoints:
local:
address: ":80"
providers:
docker:
endpoint: "unix:///var/run/docker.sock"
exposedByDefault: false
network: web
The main traefik instance is set up with https but the local ones do not have to, and both instances are set up to redirect trafic to the services within the machine according to their domain name.
So, if we want to access serviceB.domain.tld
, the request should be redirected like:
[internet] -- serviceB.domain.tld --> [main traefik]
[main traefik] -- serviceB.domain.tld --> [local traefik] on 192.168.0.10
[local traefik] --> [Service B]
- Traefik Kobling: MIT, (c) 2022, Pixelcop Research, Inc.
- traefik: MIT, Copyright (c) 2016-2020 Containous SAS; 2020-2023 Traefik Labs