GOAL: In this guide, you'll learn how to set up portainer with valid SSL and Host-based routing privately on your LAN.
Turn this:
http://myserver.lan:8080
Into this:
https://portainer.mydomain.com
...all while keepoing your private services off the internet.
When running services in traefik, you'll likely want to expose some to the internet (like plex) and keep others accessible only from your local network (like portainer). This document is mostly about IP whitelisting, but I want to first talk about SSL and security.
There are 2 main ways of creating routing rules for apps: Host rules and PathPrefix rules.
- Host rules route based on the hostname of the destination, like
foo.mydomain.com
orhttp://192.168.0.10
. - PathPrefix rules route based on some prefix substring, like
/plex
or/portainer
.
You may think that, without rolling out some robust DNS on your home network, you're stuck with http://myserver.lan/portainer
as your best option for routing.
This has drawbacks:
- PathPrefix rules are an enormous pain in the ass because nobody understands how they should work.
- Without a legitimate domain name, you're stuck with self-signed certificates.
Instead, I like to use my real domain even for local routes.
- Create a CNAME or A record for
portainer.mydomain.com
to point to either your server's private IP or even your network's public IP. NOTE: if you're using unbound DNS or a DNS resolver that blocks resolution for private addresses, you'll have to use your public IP and set up port forwarding even if you block all public addresses. Might be possible to do some NAT magic to avoid this. - Set up Wildcard SSL for your domain
Now, you'd be ready to set up a publicly accessible service, except we're going to restrict access.
With the IPWhitelist middleware, we're going to restrict access to your LAN subnet. You can run traefik exactly the same as in the wildcard SSL tutorial.
- If you choose to use your Public IP, you still need to set up port forwarding on your router for this to work correctly. This is because requests to
portainer.mydomain.com
will resolve to your public IP, get routed to your public interface, then directed back into your LAN, so the actual external filtering will happen on your traefik host. - If you choose to use your Private IP in your DNS records, you don't need port forwarding becasue the request will never hit your router. Beware that certain configurations of Unbound DNS will block DNS resolution to private IPs (if you don't know what this is, you're probably not affected).
This part will also be almost the same as the wildcard tutorial, with the addition of 1 middleware. Refer to CIDR notation if you don't know how to represent your subnet as a CIDR block.
export MY_APP_DOMAIN=mydomain.com
export SUBNET="192.168.0.0/24"
docker run --rm --name portainer \
--label traefik.enable=true \
--label traefik.http.services.my-service.loadbalancer.server.port="9000" \
--label traefik.http.middlewares.middleware-redirect-https.redirectscheme.scheme="https" \
--label traefik.http.routers.my-route.entrypoints=web \
--label traefik.http.routers.my-route.rule="Host(`portainer.${MY_APP_DOMAIN}`)" \
--label traefik.http.routers.my-route.middlewares="middleware-redirect-https@docker" \
--label traefik.http.routers.my-route-secure.entrypoints=websecure \
--label traefik.http.routers.my-route-secure.rule="Host(`portainer.${MY_APP_DOMAIN}`)" \
--label traefik.http.routers.my-route-secure.tls.domains[0].main="*.${MY_APP_DOMAIN}" \
--label traefik.http.routers.my-route-secure.tls.certresolver="myresolver" \
--label traefik.http.routers.my-route-secure.tls=true \
--label traefik.http.middlewares.middleware-ipwhitelist.ipwhitelist.sourcerange="127.0.0.1/32,${SUBNET}" \
--label traefik.http.routers.%N-secure.middlewares="middleware-ipwhitelist@docker" \
--volume /tmp/portainer/data/:/data \
--volume /var/run/docker.sock:/var/run/docker.sock \
portainer/portainer
Visit https://portainer.mydomain.com
It doesn't matter whether your DNS record for portainer.mydomain.com
points at your network's Public IP or the traefik server's private IP. When the request hits your Firewall or router, you'll get redirected internally and traefik will examine the origin of the request, which will be your host's private IP.
To verify this, you can:
- Try the request from a host on the subnet. It will succeed.
- Try the request from a host with a properly configured active VPN running. It will STILL succeed.
- Try the request from a mobile phone with wifi off. u'll get a 401 Unauthorized response!
- Try whatever else you can think of to trick traefik. It won't work :)
NOTE: If you run traefik behind another proxy that uses X-Forwarded-For header, you may have to configure other settings to pick the right IP