2019-08-31 09:28:04 +02:00
# Migration Guide: From v1 to v2
How to Migrate from Traefik v1 to Traefik v2.
{: .subtitle }
2019-10-14 17:26:05 +02:00
The version 2 of Traefik introduces a number of breaking changes,
2019-08-31 09:28:04 +02:00
which require one to update their configuration when they migrate from v1 to v2.
2019-10-14 17:26:05 +02:00
The goal of this page is to recapitulate all of these changes, and in particular to give examples,
2019-08-31 09:28:04 +02:00
feature by feature, of how the configuration looked like in v1, and how it now looks like in v2.
2019-09-23 14:32:04 +02:00
!!! info "Migration Helper"
2019-10-14 17:26:05 +02:00
2019-09-09 10:36:08 +02:00
We created a tool to help during the migration: [traefik-migration-tool ](https://github.com/containous/traefik-migration-tool )
This tool allows to:
- convert `Ingress` to Traefik `IngressRoute` resources.
- convert `acme.json` file from v1 to v2 format.
2019-09-23 14:32:04 +02:00
- migrate the static configuration contained in the file `traefik.toml` to a Traefik v2 file.
2019-09-09 10:36:08 +02:00
2019-08-31 09:28:04 +02:00
## Frontends and Backends Are Dead... <br/>... Long Live Routers, Middlewares, and Services
During the transition from v1 to v2, a number of internal pieces and components of Traefik were rewritten and reorganized.
2019-09-18 08:38:05 +02:00
As such, the combination of core notions such as frontends and backends has been replaced with the combination of [routers ](../routing/routers/index.md ), [services ](../routing/services/index.md ), and [middlewares ](../middlewares/overview.md ).
2019-08-31 09:28:04 +02:00
Typically, a router replaces a frontend, and a service assumes the role of a backend, with each router referring to a service.
However, even though a backend was in charge of applying any desired modification on the fly to the incoming request,
the router defers that responsibility to another component.
Instead, a dedicated middleware is now defined for each kind of such modification.
Then any router can refer to an instance of the wanted middleware.
!!! example "One frontend with basic auth and one backend, become one router, one service, and one basic auth middleware."
2019-09-18 08:38:05 +02:00
!!! info "v1"
2019-10-14 17:26:05 +02:00
2019-08-31 09:28:04 +02:00
```yaml tab="Docker"
labels:
- "traefik.frontend.rule=Host:test.localhost;PathPrefix:/test"
2019-09-30 18:32:04 +02:00
- "traefik.frontend.auth.basic.users=test:$$apr1$$H6uskkkW$$IgXLP6ewTrSuBkTrqE8wj/,test2:$$apr1$$d9hr9HBB$$4HxwgUir3HP4EsggP/QNo0"
2019-08-31 09:28:04 +02:00
```
```yaml tab="K8s Ingress"
apiVersion: extensions/v1beta1
kind: Ingress
metadata:
name: traefik
namespace: kube-system
annotations:
kubernetes.io/ingress.class: traefik
traefik.ingress.kubernetes.io/rule-type: PathPrefix
spec:
rules:
- host: test.locahost
http:
paths:
- path: /test
backend:
serviceName: server0
servicePort: 80
- path: /test
backend:
serviceName: server1
servicePort: 80
```
```toml tab="File (TOML)"
[frontends]
[frontends.frontend1]
entryPoints = ["http"]
backend = "backend1"
[frontends.frontend1.routes]
[frontends.frontend1.routes.route0]
rule = "Host:test.localhost"
[frontends.frontend1.routes.route0]
rule = "PathPrefix:/test"
[frontends.frontend1.auth]
[frontends.frontend1.auth.basic]
users = [
"test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/",
"test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0",
]
[backends]
[backends.backend1]
[backends.backend1.servers.server0]
url = "http://10.10.10.1:80"
[backends.backend1.servers.server1]
url = "http://10.10.10.2:80"
[backends.backend1.loadBalancer]
method = "wrr"
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
!!! info "v2"
2019-10-14 17:26:05 +02:00
2019-08-31 09:28:04 +02:00
```yaml tab="Docker"
labels:
- "traefik.http.routers.router0.rule=Host(`bar.com` ) && PathPrefix(`/test` )"
- "traefik.http.routers.router0.middlewares=auth"
2019-09-30 18:32:04 +02:00
- "traefik.http.middlewares.auth.basicauth.users=test:$$apr1$$H6uskkkW$$IgXLP6ewTrSuBkTrqE8wj/,test2:$$apr1$$d9hr9HBB$$4HxwgUir3HP4EsggP/QNo0"
2019-08-31 09:28:04 +02:00
```
```yaml tab="K8s IngressRoute"
2019-10-14 17:26:05 +02:00
# The definitions below require the definitions for the Middleware and IngressRoute kinds.
2020-01-07 11:26:05 +01:00
# https://docs.traefik.io/v2.1/reference/dynamic-configuration/kubernetes-crd/#definitions
2019-08-31 09:28:04 +02:00
apiVersion: traefik.containo.us/v1alpha1
kind: Middleware
metadata:
name: basicauth
namespace: foo
spec:
basicAuth:
users:
- test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/
- test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0
---
apiVersion: traefik.containo.us/v1alpha1
kind: IngressRoute
metadata:
name: ingressroutebar
spec:
entryPoints:
- http
routes:
- match: Host(`test.localhost` ) && PathPrefix(`/test` )
kind: Rule
services:
- name: server0
port: 80
- name: server1
port: 80
middlewares:
- name: basicauth
namespace: foo
```
```toml tab="File (TOML)"
[http.routers]
[http.routers.router0]
rule = "Host(`test.localhost` ) && PathPrefix(`/test` )"
middlewares = ["auth"]
service = "my-service"
[http.services]
[[http.services.my-service.loadBalancer.servers]]
url = "http://10.10.10.1:80"
[[http.services.my-service.loadBalancer.servers]]
url = "http://10.10.10.2:80"
[http.middlewares]
[http.middlewares.auth.basicAuth]
users = [
2019-10-14 17:26:05 +02:00
"test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/",
2019-08-31 09:28:04 +02:00
"test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0",
]
```
```yaml tab="File (YAML)"
http:
routers:
router0:
rule: "Host(`test.localhost` ) && PathPrefix(`/test` )"
service: my-service
middlewares:
2019-09-23 17:00:06 +02:00
- auth
2019-08-31 09:28:04 +02:00
services:
my-service:
loadBalancer:
servers:
2019-09-23 17:00:06 +02:00
- url: http://10.10.10.1:80
- url: http://10.10.10.2:80
2019-08-31 09:28:04 +02:00
middlewares:
auth:
basicAuth:
users:
2019-09-23 17:00:06 +02:00
- "test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/"
- "test2:$apr1$d9hr9HBB$4HxwgUir3HP4EsggP/QNo0"
2019-08-31 09:28:04 +02:00
```
2020-01-20 15:04:09 +01:00
## TLS Configuration is Now Dynamic, per Router.
2019-08-31 09:28:04 +02:00
TLS parameters used to be specified in the static configuration, as an entryPoint field.
With Traefik v2, a new dynamic TLS section at the root contains all the desired TLS configurations.
2019-09-18 08:38:05 +02:00
Then, a [router's TLS field ](../routing/routers/index.md#tls ) can refer to one of the [TLS configurations ](../https/tls.md ) defined at the root, hence defining the [TLS configuration ](../https/tls.md ) for that router.
2019-08-31 09:28:04 +02:00
2020-01-23 16:36:08 +01:00
!!! example "TLS on websecure entryPoint becomes TLS option on Router-1"
2019-08-31 09:28:04 +02:00
2019-09-18 08:38:05 +02:00
!!! info "v1"
2019-08-31 09:28:04 +02:00
```toml tab="File (TOML)"
# static configuration
[entryPoints]
2020-01-23 16:36:08 +01:00
[entryPoints.websecure]
2019-08-31 09:28:04 +02:00
address = ":443"
2020-01-23 16:36:08 +01:00
[entryPoints.websecure.tls]
2019-08-31 09:28:04 +02:00
minVersion = "VersionTLS12"
cipherSuites = [
"TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256",
2019-10-08 14:38:04 +02:00
"TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384",
"TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305",
"TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305",
"TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256",
"TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256",
2020-01-20 15:04:09 +01:00
]
2020-01-23 16:36:08 +01:00
[[entryPoints.websecure.tls.certificates]]
2019-08-31 09:28:04 +02:00
certFile = "path/to/my.cert"
keyFile = "path/to/my.key"
```
2019-10-14 17:26:05 +02:00
2019-08-31 09:28:04 +02:00
```bash tab="CLI"
2020-01-23 16:36:08 +01:00
--entryPoints='Name:websecure Address::443 TLS:path/to/my.cert,path/to/my.key TLS.MinVersion:VersionTLS12 TLS.CipherSuites:TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256,TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384,TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305,TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305,TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256,TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256'
2019-08-31 09:28:04 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
!!! info "v2"
2019-10-14 17:26:05 +02:00
2019-08-31 09:28:04 +02:00
```toml tab="File (TOML)"
# dynamic configuration
[http.routers]
[http.routers.Router-1]
rule = "Host(`bar.com` )"
service = "service-id"
# will terminate the TLS request
[http.routers.Router-1.tls]
options = "myTLSOptions"
[[tls.certificates]]
certFile = "/path/to/domain.cert"
keyFile = "/path/to/domain.key"
[tls.options]
[tls.options.myTLSOptions]
2020-02-17 17:38:05 +01:00
minVersion = "VersionTLS12"
2019-08-31 09:28:04 +02:00
cipherSuites = [
2020-01-20 15:04:09 +01:00
"TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256",
"TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384",
"TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305",
"TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305",
"TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256",
"TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256",
]
2019-08-31 09:28:04 +02:00
```
2019-10-14 17:26:05 +02:00
2019-08-31 09:28:04 +02:00
```yaml tab="File (YAML)"
http:
routers:
Router-1:
rule: "Host(`bar.com` )"
service: service-id
# will terminate the TLS request
tls:
options: myTLSOptions
tls:
certificates:
- certFile: /path/to/domain.cert
keyFile: /path/to/domain.key
options:
myTLSOptions:
2020-02-17 17:38:05 +01:00
minVersion: VersionTLS12
2019-08-31 09:28:04 +02:00
cipherSuites:
2019-10-08 14:38:04 +02:00
- TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384
- TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305
- TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305
- TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256
- TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256
2019-08-31 09:28:04 +02:00
```
2019-10-14 17:26:05 +02:00
2019-08-31 09:28:04 +02:00
```yaml tab="K8s IngressRoute"
2019-10-14 17:26:05 +02:00
# The definitions below require the definitions for the TLSOption and IngressRoute kinds.
2020-01-07 11:26:05 +01:00
# https://docs.traefik.io/v2.1/reference/dynamic-configuration/kubernetes-crd/#definitions
2019-08-31 09:28:04 +02:00
apiVersion: traefik.containo.us/v1alpha1
kind: TLSOption
metadata:
name: mytlsoption
namespace: default
spec:
2020-02-17 17:38:05 +01:00
minVersion: VersionTLS12
2019-08-31 09:28:04 +02:00
cipherSuites:
2019-10-08 14:38:04 +02:00
- TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384
- TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305
- TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305
- TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256
- TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256
2019-08-31 09:28:04 +02:00
---
apiVersion: traefik.containo.us/v1alpha1
kind: IngressRoute
metadata:
name: ingressroutebar
spec:
entryPoints:
- web
routes:
2019-09-23 17:00:06 +02:00
- match: Host(`bar.com` )
kind: Rule
services:
- name: whoami
port: 80
2019-08-31 09:28:04 +02:00
tls:
2019-10-14 17:26:05 +02:00
options:
2019-08-31 09:28:04 +02:00
name: mytlsoption
namespace: default
```
2019-10-14 17:26:05 +02:00
2019-08-31 09:28:04 +02:00
```yaml tab="Docker"
labels:
# myTLSOptions must be defined by another provider, in this instance in the File Provider.
# see the cross provider section
- "traefik.http.routers.router0.tls.options=myTLSOptions@file "
```
2020-01-20 15:04:09 +01:00
## HTTP to HTTPS Redirection is Now Configured on Routers
2019-08-31 09:28:04 +02:00
2019-09-18 08:38:05 +02:00
Previously on Traefik v1, the redirection was applied on an entry point or on a frontend.
With Traefik v2 it is applied on a [Router ](../routing/routers/index.md ).
2019-08-31 09:28:04 +02:00
2019-09-18 08:38:05 +02:00
To apply a redirection, one of the redirect middlewares, [RedirectRegex ](../middlewares/redirectregex.md ) or [RedirectScheme ](../middlewares/redirectscheme.md ), has to be configured and added to the router middlewares list.
2019-08-31 09:28:04 +02:00
2020-01-20 15:04:09 +01:00
!!! example "Global HTTP to HTTPS redirection"
2019-09-18 08:38:05 +02:00
!!! info "v1"
```toml tab="File (TOML)"
# static configuration
2020-01-20 15:04:09 +01:00
defaultEntryPoints = ["web", "websecure"]
2019-09-18 08:38:05 +02:00
[entryPoints]
2020-01-20 15:04:09 +01:00
[entryPoints.web]
2019-09-18 08:38:05 +02:00
address = ":80"
2020-01-20 15:04:09 +01:00
[entryPoints.web.redirect]
entryPoint = "websecure"
2019-09-18 08:38:05 +02:00
2020-01-20 15:04:09 +01:00
[entryPoints.websecure]
2019-09-18 08:38:05 +02:00
address = ":443"
2020-01-20 15:04:09 +01:00
[entryPoints.websecure.tls]
```
```bash tab="CLI"
--entrypoints=Name:web Address::80 Redirect.EntryPoint:websecure
--entryPoints='Name:websecure Address::443 TLS'
```
!!! info "v2"
```yaml tab="Docker"
# ...
traefik:
image: traefik:v2.1
command:
- --entrypoints.web.address=:80
- --entrypoints.websecure.address=:443
- --providers.docker=true
ports:
- 80:80
- 443:443
labels:
traefik.http.routers.http_catchall.rule: HostRegexp(`{any:.+}` )
traefik.http.routers.http_catchall.entrypoints: web
traefik.http.routers.http_catchall.middlewares: https_redirect
traefik.http.middlewares.https_redirect.redirectscheme.scheme: https
traefik.http.middlewares.https_redirect.redirectscheme.permanent: true
volumes:
- /var/run/docker.sock:/var/run/docker.sock
```
```yaml tab="K8s IngressRoute"
# The entry points web (port 80) and websecure (port 443) must be defined the static configuration.
apiVersion: traefik.containo.us/v1alpha1
kind: IngressRoute
metadata:
name: http_catchall
namespace: traefik
spec:
entryPoints:
- web
routes:
- match: HostRegexp(`{any:.+}` )
kind: Rule
services:
# any service in the namespace
# the service will be never called
- name: noop
port: 80
middlewares:
- name: https_redirect
# if the Middleware has distinct namespace
namespace: traefik
---
apiVersion: traefik.containo.us/v1alpha1
kind: Middleware
metadata:
name: https_redirect
namespace: traefik
spec:
redirectScheme:
scheme: https
permanent: true
```
```toml tab="File (TOML)"
# traefik.toml
## static configuration
[entryPoints]
[entryPoints.web]
address = 80
[entryPoints.websecure]
address = 443
[providers.file]
directory = "/dynamic/"
##--------------------##
# /dynamic/redirect.toml
## dynamic configuration
[http.routers]
[http.routers.http_catchall]
entryPoints = ["web"]
middlewares = ["https_redirect"]
rule = "HostRegexp(`{any:.+}` )"
service = "noop"
[http.services]
# noop service, the URL will be never called
[http.services.noop.loadBalancer]
[[http.services.noop.loadBalancer.servers]]
url = "http://192.168.0.1:1337"
[http.middlewares]
[http.middlewares.https_redirect.redirectScheme]
scheme = "https"
permanent = true
```
```yaml tab="File (YAML)"
# traefik.yaml
## static configuration
entryPoints:
web:
address: 80
websecure:
address: 443
providers:
file:
directory: /dynamic/
##--------------------##
# /dynamic/redirect.yml
## dynamic configuration
http:
routers:
http_catchall:
entryPoints:
- web
middlewares:
- https_redirect
rule: "HostRegexp(`{any:.+}` )"
service: noop
services:
# noop service, the URL will be never called
noop:
loadBalancer:
servers:
- url: http://192.168.0.1:1337
middlewares:
https_redirect:
redirectScheme:
scheme: https
permanent: true
```
!!! example "HTTP to HTTPS redirection per domain"
!!! info "v1"
```toml tab="File (TOML)"
# static configuration
defaultEntryPoints = ["web", "websecure"]
[entryPoints]
[entryPoints.web]
address = ":80"
[entryPoints.web.redirect]
entryPoint = "websecure"
[entryPoints.websecure]
address = ":443"
[entryPoints.websecure.tls]
[[entryPoints.websecure.tls.certificates]]
2019-09-18 08:38:05 +02:00
certFile = "examples/traefik.crt"
keyFile = "examples/traefik.key"
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
2020-01-20 15:04:09 +01:00
--entrypoints=Name:web Address::80 Redirect.EntryPoint:websecure
--entryPoints='Name:websecure Address::443 TLS:path/to/my.cert,path/to/my.key'
2019-09-18 08:38:05 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
!!! info "v2"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```yaml tab="Docker"
labels:
2020-01-20 15:04:09 +01:00
traefik.http.routers.app.rule: Host(`foo.com` )
traefik.http.routers.app.entrypoints: web
traefik.http.routers.app.middlewares: https_redirect
traefik.http.routers.appsecured.rule: Host(`foo.com` )
traefik.http.routers.appsecured.entrypoints: websecure
traefik.http.routers.appsecured.tls: true
traefik.http.middlewares.https_redirect.redirectscheme.scheme: https
traefik.http.middlewares.https_redirect.redirectscheme.permanent: true
2019-09-18 08:38:05 +02:00
```
```yaml tab="K8s IngressRoute"
apiVersion: traefik.containo.us/v1alpha1
kind: IngressRoute
metadata:
name: http-redirect-ingressRoute
spec:
entryPoints:
- web
routes:
- match: Host(`foo.com` )
kind: Rule
services:
- name: whoami
port: 80
middlewares:
2020-01-20 15:04:09 +01:00
- name: https_redirect
2019-09-18 08:38:05 +02:00
---
apiVersion: traefik.containo.us/v1alpha1
kind: IngressRoute
metadata:
name: https-ingressRoute
spec:
entryPoints:
2020-01-20 15:04:09 +01:00
- websecure
2019-09-18 08:38:05 +02:00
routes:
- match: Host(`foo` )
kind: Rule
services:
- name: whoami
port: 80
tls: {}
---
apiVersion: traefik.containo.us/v1alpha1
kind: Middleware
metadata:
2020-01-20 15:04:09 +01:00
name: https_redirect
2019-09-18 08:38:05 +02:00
spec:
redirectScheme:
scheme: https
2020-01-20 15:04:09 +01:00
permanent: true
2019-09-18 08:38:05 +02:00
```
```toml tab="File (TOML)"
## static configuration
# traefik.toml
[entryPoints.web]
address = ":80"
2020-01-20 15:04:09 +01:00
[entryPoints.websecure]
2019-09-18 08:38:05 +02:00
address = ":443"
##---------------------##
## dynamic configuration
2019-09-26 12:20:04 -04:00
# dynamic-conf.toml
2019-09-18 08:38:05 +02:00
[http.routers]
[http.routers.router0]
rule = "Host(`foo.com` )"
service = "my-service"
2019-09-23 04:30:04 -04:00
entrypoints = ["web"]
2020-01-20 15:04:09 +01:00
middlewares = ["https_redirect"]
2019-09-18 08:38:05 +02:00
[http.routers.router1]
rule = "Host(`foo.com` )"
service = "my-service"
2020-01-20 15:04:09 +01:00
entrypoints = ["websecure"]
2019-09-18 08:38:05 +02:00
[http.routers.router1.tls]
[http.services]
[[http.services.my-service.loadBalancer.servers]]
url = "http://10.10.10.1:80"
[[http.services.my-service.loadBalancer.servers]]
url = "http://10.10.10.2:80"
[http.middlewares]
2020-01-20 15:04:09 +01:00
[http.middlewares.https_redirect.redirectScheme]
2019-09-18 08:38:05 +02:00
scheme = "https"
2020-01-20 15:04:09 +01:00
permanent = true
2019-09-18 08:38:05 +02:00
[[tls.certificates]]
certFile = "/path/to/domain.cert"
2019-10-14 17:26:05 +02:00
keyFile = "/path/to/domain.key"
2019-09-18 08:38:05 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```yaml tab="File (YAML)"
## static configuration
# traefik.yml
entryPoints:
web:
address: ":80"
2020-01-20 15:04:09 +01:00
websecure:
2019-09-18 08:38:05 +02:00
address: ":443"
##---------------------##
## dynamic configuration
2019-09-26 12:20:04 -04:00
# dynamic-conf.yml
2019-09-18 08:38:05 +02:00
http:
routers:
router0:
2019-09-20 16:44:04 +02:00
rule: "Host(`foo.com` )"
entryPoints:
2019-09-23 17:00:06 +02:00
- web
2019-09-20 16:44:04 +02:00
middlewares:
2020-01-20 15:04:09 +01:00
- https_redirect
2019-09-20 16:44:04 +02:00
service: my-service
2019-09-18 08:38:05 +02:00
router1:
2019-09-20 16:44:04 +02:00
rule: "Host(`foo.com` )"
entryPoints:
2020-01-20 15:04:09 +01:00
- websecure
2019-09-20 16:44:04 +02:00
service: my-service
tls: {}
2019-09-18 08:38:05 +02:00
services:
my-service:
loadBalancer:
servers:
2019-09-23 17:00:06 +02:00
- url: http://10.10.10.1:80
- url: http://10.10.10.2:80
2019-09-18 08:38:05 +02:00
middlewares:
2020-01-20 15:04:09 +01:00
https_redirect:
2019-09-18 08:38:05 +02:00
redirectScheme:
scheme: https
2020-01-20 15:04:09 +01:00
permanent: true
2019-09-18 08:38:05 +02:00
tls:
certificates:
2019-09-23 17:00:06 +02:00
- certFile: /app/certs/server/server.pem
keyFile: /app/certs/server/server.pem
2019-10-14 17:26:05 +02:00
```
## Strip and Rewrite Path Prefixes
With the new core notions of v2 (introduced earlier in the section
["Frontends and Backends Are Dead... Long Live Routers, Middlewares, and Services" ](#frontends-and-backends-are-dead-long-live-routers-middlewares-and-services )),
2020-01-20 15:04:09 +01:00
transforming the URL path prefix of incoming requests is configured with [middlewares ](../middlewares/overview.md ),
2020-02-17 11:04:04 +01:00
after the routing step with [router rule `PathPrefix` ](../routing/routers/index.md#rule ).
2019-10-14 17:26:05 +02:00
Use Case: Incoming requests to `http://company.org/admin` are forwarded to the webapplication "admin",
with the path `/admin` stripped, e.g. to `http://<IP>:<port>/` . In this case, you must:
* First, configure a router named `admin` with a rule matching at least the path prefix with the `PathPrefix` keyword,
2020-01-20 15:04:09 +01:00
* Then, define a middleware of type [`stripprefix` ](../middlewares/stripprefix.md ), which removes the prefix `/admin` , associated to the router `admin` .
2019-10-14 17:26:05 +02:00
!!! example "Strip Path Prefix When Forwarding to Backend"
!!! info "v1"
```yaml tab="Docker"
labels:
- "traefik.frontend.rule=Host:company.org;PathPrefixStrip:/admin"
```
```yaml tab="Kubernetes Ingress"
apiVersion: networking.k8s.io/v1beta1
kind: Ingress
metadata:
name: traefik
annotations:
kubernetes.io/ingress.class: traefik
traefik.ingress.kubernetes.io/rule-type: PathPrefixStrip
spec:
rules:
- host: company.org
http:
paths:
- path: /admin
backend:
serviceName: admin-svc
servicePort: admin
```
```toml tab="File (TOML)"
[frontends.admin]
[frontends.admin.routes.admin_1]
rule = "Host:company.org;PathPrefixStrip:/admin"
```
!!! info "v2"
```yaml tab="Docker"
labels:
- "traefik.http.routers.admin.rule=Host(`company.org` ) && PathPrefix(`/admin` )"
2019-12-06 02:42:04 +03:00
- "traefik.http.routers.admin.middlewares=admin-stripprefix"
2019-10-14 17:26:05 +02:00
- "traefik.http.middlewares.admin-stripprefix.stripprefix.prefixes=/admin"
```
```yaml tab="Kubernetes IngressRoute"
---
apiVersion: traefik.containo.us/v1alpha1
kind: IngressRoute
metadata:
name: http-redirect-ingressRoute
namespace: admin-web
spec:
entryPoints:
- web
routes:
- match: Host(`company.org` ) && PathPrefix(`/admin` )
kind: Rule
services:
- name: admin-svc
port: admin
middlewares:
- name: admin-stripprefix
---
kind: Middleware
metadata:
name: admin-stripprefix
spec:
stripPrefix:
prefixes:
- /admin
```
```toml tab="File (TOML)"
## Dynamic configuration
# dynamic-conf.toml
[http.routers.router1]
rule = "Host(`company.org` ) && PathPrefix(`/admin` )"
service = "admin-svc"
entrypoints = ["web"]
middlewares = ["admin-stripprefix"]
[http.middlewares]
[http.middlewares.admin-stripprefix.stripPrefix]
prefixes = ["/admin"]
# ...
```
```yaml tab="File (YAML)"
## Dynamic Configuration
# dynamic-conf.yml
# As YAML Configuration File
http:
routers:
admin:
service: admin-svc
middlewares:
- "admin-stripprefix"
rule: "Host(`company.org` ) && PathPrefix(`/admin` )"
middlewares:
admin-stripprefix:
2019-10-21 04:40:03 +07:00
stripPrefix:
prefixes:
- "/admin"
2019-10-14 17:26:05 +02:00
# ...
```
??? question "What About Other Path Transformations?"
Instead of removing the path prefix with the [`stripprefix` middleware ](../../middlewares/stripprefix/ ), you can also:
* Add a path prefix with the [`addprefix` middleware ](../../middlewares/addprefix/ )
* Replace the complete path of the request with the [`replacepath` middleware ](../../middlewares/replacepath/ )
* ReplaceRewrite path using Regexp with the [`replacepathregex` middleware ](../../middlewares/replacepathregex/ )
* And a lot more on the [`middlewares` page ](../../middlewares/overview/ )
2019-09-18 08:38:05 +02:00
## ACME (LetsEncrypt)
[ACME ](../https/acme.md ) is now a certificate resolver (under a certificatesResolvers section) but remains in the static configuration.
!!! example "ACME from provider to a specific Certificate Resolver"
!!! info "v1"
```toml tab="File (TOML)"
# static configuration
2020-01-23 16:36:08 +01:00
defaultEntryPoints = ["websecure","web"]
2019-09-18 08:38:05 +02:00
[entryPoints.web]
address = ":80"
[entryPoints.web.redirect]
entryPoint = "webs"
2020-01-23 16:36:08 +01:00
[entryPoints.websecure]
2019-09-18 08:38:05 +02:00
address = ":443"
2020-02-17 11:04:04 +01:00
[entryPoints.websecure.tls]
2019-09-18 08:38:05 +02:00
[acme]
email = "your-email-here@my -awesome-app.org"
storage = "acme.json"
2020-01-23 16:36:08 +01:00
entryPoint = "websecure"
2019-09-18 08:38:05 +02:00
onHostRule = true
[acme.httpChallenge]
entryPoint = "web"
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
2020-01-23 16:36:08 +01:00
--defaultentrypoints=websecure,web
--entryPoints=Name:web Address::80 Redirect.EntryPoint:websecure
--entryPoints=Name:websecure Address::443 TLS
2019-09-18 08:38:05 +02:00
--acme.email=your-email-here@my -awesome-app.org
--acme.storage=acme.json
2020-01-23 16:36:08 +01:00
--acme.entryPoint=websecure
2019-09-18 08:38:05 +02:00
--acme.onHostRule=true
--acme.httpchallenge.entrypoint=http
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
!!! info "v2"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```toml tab="File (TOML)"
# static configuration
[entryPoints]
[entryPoints.web]
address = ":80"
2020-01-23 16:36:08 +01:00
[entryPoints.websecure]
2019-09-18 08:38:05 +02:00
address = ":443"
2020-02-17 11:04:04 +01:00
[certificatesResolvers.myresolver.acme]
2019-09-18 08:38:05 +02:00
email = "your-email@your -domain.org"
storage = "acme.json"
2020-02-17 11:04:04 +01:00
[certificatesResolvers.myresolver.acme.httpChallenge]
2019-09-18 08:38:05 +02:00
# used during the challenge
entryPoint = "web"
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```yaml tab="File (YAML)"
entryPoints:
web:
address: ":80"
2020-01-23 16:36:08 +01:00
websecure:
2019-09-18 08:38:05 +02:00
address: ":443"
certificatesResolvers:
2020-02-17 11:04:04 +01:00
myresolver:
2019-09-18 08:38:05 +02:00
acme:
email: your-email@your -domain.org
storage: acme.json
httpChallenge:
# used during the challenge
entryPoint: web
2019-10-14 17:26:05 +02:00
```
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
2019-11-19 10:18:05 +01:00
--entryPoints.web.address=:80
--entryPoints.websecure.address=:443
2020-02-17 11:04:04 +01:00
--certificatesResolvers.myresolver.acme.email=your-email@your -domain.org
--certificatesResolvers.myresolver.acme.storage=acme.json
--certificatesResolvers.myresolver.acme.httpChallenge.entryPoint=web
2019-09-18 08:38:05 +02:00
```
2019-08-31 09:28:04 +02:00
## Traefik Logs
2019-09-18 08:38:05 +02:00
In the v2, all the [log configuration ](../observability/logs.md ) remains in the static part but are unified under a `log` section.
There is no more log configuration at the root level.
!!! example "Simple log configuration"
!!! info "v1"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```toml tab="File (TOML)"
# static configuration
logLevel = "DEBUG"
[traefikLog]
filePath = "/path/to/traefik.log"
format = "json"
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
2019-11-19 10:18:05 +01:00
--logLevel=DEBUG
--traefikLog.filePath=/path/to/traefik.log
--traefikLog.format=json
2019-09-18 08:38:05 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
!!! info "v2"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```toml tab="File (TOML)"
# static configuration
[log]
level = "DEBUG"
filePath = "/path/to/log-file.log"
format = "json"
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```yaml tab="File (YAML)"
# static configuration
log:
level: DEBUG
filePath: /path/to/log-file.log
format: json
2019-10-14 17:26:05 +02:00
```
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
2019-11-19 10:18:05 +01:00
--log.level=DEBUG
--log.filePath=/path/to/traefik.log
--log.format=json
2019-09-18 08:38:05 +02:00
```
2019-08-31 09:28:04 +02:00
## Tracing
2019-09-18 08:38:05 +02:00
Traefik v2 retains OpenTracing support. The `backend` root option from the v1 is gone, you just have to set your [tracing configuration ](../observability/tracing/overview.md ).
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
!!! example "Simple Jaeger tracing configuration"
!!! info "v1"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```toml tab="File (TOML)"
# static configuration
[tracing]
backend = "jaeger"
servicename = "tracing"
[tracing.jaeger]
samplingParam = 1.0
samplingServerURL = "http://12.0.0.1:5778/sampling"
samplingType = "const"
localAgentHostPort = "12.0.0.1:6831"
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
2019-11-19 10:18:05 +01:00
--tracing.backend=jaeger
--tracing.servicename=tracing
--tracing.jaeger.localagenthostport=12.0.0.1:6831
--tracing.jaeger.samplingparam=1.0
--tracing.jaeger.samplingserverurl=http://12.0.0.1:5778/sampling
--tracing.jaeger.samplingtype=const
2019-09-18 08:38:05 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
!!! info "v2"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```toml tab="File (TOML)"
# static configuration
[tracing]
servicename = "tracing"
[tracing.jaeger]
samplingParam = 1.0
samplingServerURL = "http://12.0.0.1:5778/sampling"
samplingType = "const"
localAgentHostPort = "12.0.0.1:6831"
2019-09-19 11:10:04 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```yaml tab="File (YAML)"
# static configuration
tracing:
servicename: tracing
jaeger:
samplingParam: 1
samplingServerURL: 'http://12.0.0.1:5778/sampling'
samplingType: const
localAgentHostPort: '12.0.0.1:6831'
2019-10-14 17:26:05 +02:00
```
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
2019-11-19 10:18:05 +01:00
--tracing.servicename=tracing
--tracing.jaeger.localagenthostport=12.0.0.1:6831
--tracing.jaeger.samplingparam=1.0
--tracing.jaeger.samplingserverurl=http://12.0.0.1:5778/sampling
--tracing.jaeger.samplingtype=const
2019-09-18 08:38:05 +02:00
```
2019-08-31 09:28:04 +02:00
## Metrics
2019-09-18 08:38:05 +02:00
The v2 retains metrics tools and allows metrics to be configured for the entrypoints and/or services.
2019-10-14 17:26:05 +02:00
For a basic configuration, the [metrics configuration ](../observability/metrics/overview.md ) remains the same.
2019-09-18 08:38:05 +02:00
!!! example "Simple Prometheus metrics configuration"
!!! info "v1"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```toml tab="File (TOML)"
# static configuration
[metrics.prometheus]
2019-09-20 16:44:04 +02:00
buckets = [0.1,0.3,1.2,5.0]
entryPoint = "traefik"
2019-09-18 08:38:05 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
--metrics.prometheus.buckets=[0.1,0.3,1.2,5.0]
2019-11-19 10:18:05 +01:00
--metrics.prometheus.entrypoint=traefik
2019-09-18 08:38:05 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
!!! info "v2"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```toml tab="File (TOML)"
# static configuration
[metrics.prometheus]
2019-09-20 16:44:04 +02:00
buckets = [0.1,0.3,1.2,5.0]
entryPoint = "metrics"
2019-09-18 08:38:05 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```yaml tab="File (YAML)"
# static configuration
metrics:
prometheus:
buckets:
- 0.1
- 0.3
- 1.2
- 5
entryPoint: metrics
2019-10-14 17:26:05 +02:00
```
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
--metrics.prometheus.buckets=[0.1,0.3,1.2,5.0]
2019-11-19 10:18:05 +01:00
--metrics.prometheus.entrypoint=metrics
2019-09-18 08:38:05 +02:00
```
2019-08-31 09:28:04 +02:00
2019-10-14 17:26:05 +02:00
## No More Root Level Key/Values
2019-08-31 09:28:04 +02:00
2019-09-18 08:38:05 +02:00
To avoid any source of confusion, there are no more configuration at the root level.
Each root item has been moved to a related section or removed.
!!! example "From root to dedicated section"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
!!! info "v1"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```toml tab="File (TOML)"
# static configuration
checkNewVersion = false
sendAnonymousUsage = true
logLevel = "DEBUG"
insecureSkipVerify = true
rootCAs = [ "/mycert.cert" ]
maxIdleConnsPerHost = 200
providersThrottleDuration = "2s"
AllowMinWeightZero = true
debug = true
2020-01-23 16:36:08 +01:00
defaultEntryPoints = ["web", "websecure"]
2019-09-18 08:38:05 +02:00
keepTrailingSlash = false
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
--checknewversion=false
--sendanonymoususage=true
2019-11-19 10:18:05 +01:00
--loglevel=DEBUG
2019-09-18 08:38:05 +02:00
--insecureskipverify=true
2019-11-19 10:18:05 +01:00
--rootcas=/mycert.cert
2019-09-18 08:38:05 +02:00
--maxidleconnsperhost=200
2019-11-19 10:18:05 +01:00
--providersthrottleduration=2s
2019-09-18 08:38:05 +02:00
--allowminweightzero=true
--debug=true
2020-01-23 16:36:08 +01:00
--defaultentrypoints=web,websecure
2019-09-18 08:38:05 +02:00
--keeptrailingslash=true
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
!!! info "v2"
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```toml tab="File (TOML)"
# static configuration
[global]
checkNewVersion = true
sendAnonymousUsage = true
[log]
level = "DEBUG"
[serversTransport]
2019-09-20 16:44:04 +02:00
insecureSkipVerify = true
rootCAs = [ "/mycert.cert" ]
maxIdleConnsPerHost = 42
2019-09-18 08:38:05 +02:00
[providers]
2019-10-14 17:26:05 +02:00
providersThrottleDuration = 42
2019-09-18 08:38:05 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-18 08:38:05 +02:00
```yaml tab="File (YAML)"
# static configuration
global:
checkNewVersion: true
sendAnonymousUsage: true
log:
level: DEBUG
serversTransport:
insecureSkipVerify: true
rootCAs:
- /mycert.cert
maxIdleConnsPerHost: 42
providers:
providersThrottleDuration: 42
2019-10-14 17:26:05 +02:00
```
2019-09-18 08:38:05 +02:00
```bash tab="CLI"
--global.checknewversion=true
--global.sendanonymoususage=true
2019-11-19 10:18:05 +01:00
--log.level=DEBUG
2019-09-18 08:38:05 +02:00
--serverstransport.insecureskipverify=true
2019-11-19 10:18:05 +01:00
--serverstransport.rootcas=/mycert.cert
2019-09-18 08:38:05 +02:00
--serverstransport.maxidleconnsperhost=42
--providers.providersthrottleduration=42
```
2019-10-14 17:26:05 +02:00
2019-09-20 16:44:04 +02:00
## Dashboard
2019-10-07 14:38:06 +02:00
You need to activate the API to access the [dashboard ](../operations/dashboard.md ).
2019-09-20 16:44:04 +02:00
As the dashboard access is now secured by default you can either:
* define a [specific router ](../operations/api.md#configuration ) with the `api@internal` service and one authentication middleware like the following example
2019-11-13 00:34:04 +01:00
* or use the [insecure ](../operations/api.md#insecure ) option of the API
2019-09-20 16:44:04 +02:00
2019-09-23 14:32:04 +02:00
!!! info "Dashboard with k8s and dedicated router"
2019-09-20 16:44:04 +02:00
As `api@internal` is not a Kubernetes service, you have to use the file provider or the `insecure` API option.
2019-10-14 17:26:05 +02:00
2019-09-20 16:44:04 +02:00
!!! example "Activate and access the dashboard"
!!! info "v1"
2019-10-14 17:26:05 +02:00
2019-09-20 16:44:04 +02:00
```toml tab="File (TOML)"
## static configuration
# traefik.toml
2020-01-23 16:36:08 +01:00
[entryPoints.websecure]
2019-09-20 16:44:04 +02:00
address = ":443"
2020-01-23 16:36:08 +01:00
[entryPoints.websecure.tls]
[entryPoints.websecure.auth]
[entryPoints.websecure.auth.basic]
2019-09-20 16:44:04 +02:00
users = [
"test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/"
]
[api]
2020-01-23 16:36:08 +01:00
entryPoint = "websecure"
2019-09-20 16:44:04 +02:00
```
2019-10-14 17:26:05 +02:00
2019-09-20 16:44:04 +02:00
```bash tab="CLI"
2020-01-23 16:36:08 +01:00
--entryPoints='Name:websecure Address::443 TLS Auth.Basic.Users:test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/'
2019-09-20 16:44:04 +02:00
--api
```
2019-10-14 17:26:05 +02:00
2019-09-20 16:44:04 +02:00
!!! info "v2"
2019-10-14 17:26:05 +02:00
2019-09-20 16:44:04 +02:00
```yaml tab="Docker"
# dynamic configuration
labels:
- "traefik.http.routers.api.rule=Host(`traefik.docker.localhost` )"
2020-01-23 16:36:08 +01:00
- "traefik.http.routers.api.entrypoints=websecured"
2019-09-20 16:44:04 +02:00
- "traefik.http.routers.api.service=api@internal "
- "traefik.http.routers.api.middlewares=myAuth"
- "traefik.http.routers.api.tls"
2019-09-30 18:32:04 +02:00
- "traefik.http.middlewares.myAuth.basicauth.users=test:$$apr1$$H6uskkkW$$IgXLP6ewTrSuBkTrqE8wj/"
2019-09-20 16:44:04 +02:00
```
2019-08-31 09:28:04 +02:00
2019-09-20 16:44:04 +02:00
```toml tab="File (TOML)"
## static configuration
# traefik.toml
2020-01-23 16:36:08 +01:00
[entryPoints.websecure]
2019-09-20 16:44:04 +02:00
address = ":443"
[api]
[providers.file]
2019-12-09 11:48:05 +01:00
directory = "/path/to/dynamic/config"
2019-09-20 16:44:04 +02:00
##---------------------##
## dynamic configuration
2019-12-09 11:48:05 +01:00
# /path/to/dynamic/config/dynamic-conf.toml
2019-09-20 16:44:04 +02:00
[http.routers.api]
rule = "Host(`traefik.docker.localhost` )"
2020-01-23 16:36:08 +01:00
entrypoints = ["websecure"]
2019-09-20 16:44:04 +02:00
service = "api@internal "
middlewares = ["myAuth"]
[http.routers.api.tls]
[http.middlewares.myAuth.basicAuth]
users = [
"test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/"
]
```
2019-10-14 17:26:05 +02:00
2019-09-20 16:44:04 +02:00
```yaml tab="File (YAML)"
## static configuration
# traefik.yaml
entryPoints:
2020-01-23 16:36:08 +01:00
websecure:
2019-09-20 16:44:04 +02:00
address: ':443'
api: {}
providers:
file:
2019-12-09 11:48:05 +01:00
directory: /path/to/dynamic/config
2019-09-20 16:44:04 +02:00
##---------------------##
## dynamic configuration
2019-12-09 11:48:05 +01:00
# /path/to/dynamic/config/dynamic-conf.yaml
2019-09-20 16:44:04 +02:00
http:
routers:
api:
rule: Host(`traefik.docker.localhost` )
entrypoints:
2020-01-23 16:36:08 +01:00
- websecure
2019-09-20 16:44:04 +02:00
service: api@internal
middlewares:
- myAuth
tls: {}
middlewares:
myAuth:
basicAuth:
users:
- 'test:$apr1$H6uskkkW$IgXLP6ewTrSuBkTrqE8wj/'
2019-10-14 17:26:05 +02:00
```
2019-08-31 09:28:04 +02:00
## Providers
2019-09-18 08:38:05 +02:00
Supported [providers ](../providers/overview.md ), for now:
2019-08-31 09:28:04 +02:00
2019-09-20 16:44:04 +02:00
* [ ] Azure Service Fabric
* [ ] BoltDB
* [ ] Consul
2019-10-16 10:36:04 +02:00
* [x] Consul Catalog
2019-09-20 16:44:04 +02:00
* [x] Docker
* [ ] DynamoDB
* [ ] ECS
* [ ] Etcd
* [ ] Eureka
* [x] File
* [x] Kubernetes Ingress (without annotations)
* [x] Kubernetes IngressRoute
* [x] Marathon
* [ ] Mesos
* [x] Rancher
* [x] Rest
* [ ] Zookeeper
2019-10-14 17:26:05 +02:00
## Some Tips You Should Know
2019-09-20 16:44:04 +02:00
* Different sources of static configuration (file, CLI flags, ...) cannot be [mixed ](../getting-started/configuration-overview.md#the-static-configuration ).
* Now, configuration elements can be referenced between different providers by using the provider namespace notation: `@<provider>` .
For instance, a router named `myrouter` in a File Provider can refer to a service named `myservice` defined in Docker Provider with the following notation: `myservice@docker` .
* Middlewares are applied in the same order as their declaration in router.
2019-09-23 04:30:04 -04:00
* If you have any questions feel free to join our [community forum ](https://community.containo.us ).