Merge v2.4 into master
This commit is contained in:
commit
1b21f0723f
59 changed files with 1615 additions and 1400 deletions
6
.github/workflows/documentation.yml
vendored
6
.github/workflows/documentation.yml
vendored
|
@ -32,6 +32,9 @@ jobs:
|
||||||
- name: Install Structor ${{ env.STRUCTOR_VERSION }}
|
- name: Install Structor ${{ env.STRUCTOR_VERSION }}
|
||||||
run: curl -sSfL https://raw.githubusercontent.com/traefik/structor/master/godownloader.sh | sh -s -- -b $HOME/bin ${STRUCTOR_VERSION}
|
run: curl -sSfL https://raw.githubusercontent.com/traefik/structor/master/godownloader.sh | sh -s -- -b $HOME/bin ${STRUCTOR_VERSION}
|
||||||
|
|
||||||
|
- name: Install Seo-doc
|
||||||
|
run: curl -sSfL https://raw.githubusercontent.com/traefik/seo-doc/master/godownloader.sh | sh -s -- -b "${HOME}/bin"
|
||||||
|
|
||||||
- name: Install Mixtus ${{ env.MIXTUS_VERSION }}
|
- name: Install Mixtus ${{ env.MIXTUS_VERSION }}
|
||||||
run: curl -sSfL https://raw.githubusercontent.com/traefik/mixtus/master/godownloader.sh | sh -s -- -b $HOME/bin ${MIXTUS_VERSION}
|
run: curl -sSfL https://raw.githubusercontent.com/traefik/mixtus/master/godownloader.sh | sh -s -- -b $HOME/bin ${MIXTUS_VERSION}
|
||||||
|
|
||||||
|
@ -40,6 +43,9 @@ jobs:
|
||||||
env:
|
env:
|
||||||
STRUCTOR_LATEST_TAG: ${{ secrets.STRUCTOR_LATEST_TAG }}
|
STRUCTOR_LATEST_TAG: ${{ secrets.STRUCTOR_LATEST_TAG }}
|
||||||
|
|
||||||
|
- name: Apply seo
|
||||||
|
run: $HOME/bin/seo -path=./site
|
||||||
|
|
||||||
- name: Publish documentation
|
- name: Publish documentation
|
||||||
run: $HOME/bin/mixtus --dst-doc-path="./traefik" --dst-owner=traefik --dst-repo-name=doc --git-user-email="30906710+traefiker@users.noreply.github.com" --git-user-name=traefiker --src-doc-path="./site" --src-owner=containous --src-repo-name=traefik
|
run: $HOME/bin/mixtus --dst-doc-path="./traefik" --dst-owner=traefik --dst-repo-name=doc --git-user-email="30906710+traefiker@users.noreply.github.com" --git-user-name=traefiker --src-doc-path="./site" --src-owner=containous --src-repo-name=traefik
|
||||||
env:
|
env:
|
||||||
|
|
15
CHANGELOG.md
15
CHANGELOG.md
|
@ -1,3 +1,18 @@
|
||||||
|
## [v2.4.3](https://github.com/traefik/traefik/tree/v2.4.3) (2021-02-15)
|
||||||
|
[All Commits](https://github.com/traefik/traefik/compare/v2.4.2...v2.4.3)
|
||||||
|
|
||||||
|
**Bug fixes:**
|
||||||
|
- **[acme]** Fix TLS challenge timeout and validation error ([#7879](https://github.com/traefik/traefik/pull/7879) by [ldez](https://github.com/ldez))
|
||||||
|
- **[consulcatalog]** Fixed typo in consul catalog tests ([#7865](https://github.com/traefik/traefik/pull/7865) by [apollo13](https://github.com/apollo13))
|
||||||
|
- **[middleware]** Apply content type exclusion on response ([#7888](https://github.com/traefik/traefik/pull/7888) by [jbdoumenjou](https://github.com/jbdoumenjou))
|
||||||
|
|
||||||
|
**Documentation:**
|
||||||
|
- **[middleware]** Add HEAD as available option for Method ([#7858](https://github.com/traefik/traefik/pull/7858) by [mlandauer](https://github.com/mlandauer))
|
||||||
|
- **[middleware]** Middleware documentation fixes ([#7808](https://github.com/traefik/traefik/pull/7808) by [Ullaakut](https://github.com/Ullaakut))
|
||||||
|
- **[provider]** Add missing doc about servers transport ([#7894](https://github.com/traefik/traefik/pull/7894) by [ldez](https://github.com/ldez))
|
||||||
|
- **[provider]** Provider documentation fixes ([#7823](https://github.com/traefik/traefik/pull/7823) by [Ullaakut](https://github.com/Ullaakut))
|
||||||
|
- Fix the static reference documentation for the internal redirection router ([#7860](https://github.com/traefik/traefik/pull/7860) by [jbdoumenjou](https://github.com/jbdoumenjou))
|
||||||
|
|
||||||
## [v2.4.2](https://github.com/traefik/traefik/tree/v2.4.2) (2021-02-02)
|
## [v2.4.2](https://github.com/traefik/traefik/tree/v2.4.2) (2021-02-02)
|
||||||
[All Commits](https://github.com/traefik/traefik/compare/v2.4.1...v2.4.2)
|
[All Commits](https://github.com/traefik/traefik/compare/v2.4.1...v2.4.2)
|
||||||
|
|
||||||
|
|
|
@ -188,7 +188,9 @@ func setupServer(staticConfiguration *static.Configuration) (*server.Server, err
|
||||||
|
|
||||||
tlsManager := traefiktls.NewManager()
|
tlsManager := traefiktls.NewManager()
|
||||||
httpChallengeProvider := acme.NewChallengeHTTP()
|
httpChallengeProvider := acme.NewChallengeHTTP()
|
||||||
tlsChallengeProvider := acme.NewChallengeTLSALPN(time.Duration(staticConfiguration.Providers.ProvidersThrottleDuration))
|
|
||||||
|
// we need to wait at least 2 times the ProvidersThrottleDuration to be sure to handle the challenge.
|
||||||
|
tlsChallengeProvider := acme.NewChallengeTLSALPN(time.Duration(staticConfiguration.Providers.ProvidersThrottleDuration) * 2)
|
||||||
err = providerAggregator.AddProvider(tlsChallengeProvider)
|
err = providerAggregator.AddProvider(tlsChallengeProvider)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
|
|
@ -5,7 +5,7 @@ Prefixing the Path
|
||||||
|
|
||||||
![AddPrefix](../assets/img/middleware/addprefix.png)
|
![AddPrefix](../assets/img/middleware/addprefix.png)
|
||||||
|
|
||||||
The AddPrefix middleware updates the URL Path of the request before forwarding it.
|
The AddPrefix middleware updates the path of a request before forwarding it.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
@ -64,4 +64,4 @@ http:
|
||||||
### `prefix`
|
### `prefix`
|
||||||
|
|
||||||
`prefix` is the string to add before the current path in the requested URL.
|
`prefix` is the string to add before the current path in the requested URL.
|
||||||
It should include the leading slash (`/`).
|
It should include a leading slash (`/`).
|
||||||
|
|
|
@ -5,7 +5,7 @@ Adding Basic Authentication
|
||||||
|
|
||||||
![BasicAuth](../assets/img/middleware/basicauth.png)
|
![BasicAuth](../assets/img/middleware/basicauth.png)
|
||||||
|
|
||||||
The BasicAuth middleware is a quick way to restrict access to your services to known users.
|
The BasicAuth middleware restricts access to your services to known users.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
@ -81,7 +81,7 @@ Passwords must be hashed using MD5, SHA1, or BCrypt.
|
||||||
|
|
||||||
### `users`
|
### `users`
|
||||||
|
|
||||||
The `users` option is an array of authorized users. Each user will be declared using the `name:hashed-password` format.
|
The `users` option is an array of authorized users. Each user must be declared using the `name:hashed-password` format.
|
||||||
|
|
||||||
!!! note ""
|
!!! note ""
|
||||||
|
|
||||||
|
|
|
@ -5,22 +5,22 @@ How to Read the Request before Forwarding It
|
||||||
|
|
||||||
![Buffering](../assets/img/middleware/buffering.png)
|
![Buffering](../assets/img/middleware/buffering.png)
|
||||||
|
|
||||||
The Buffering middleware gives you control on how you want to read the requests before sending them to services.
|
The Buffering middleware limits the size of requests that can be forwarded to services.
|
||||||
|
|
||||||
With Buffering, Traefik reads the entire request into memory (possibly buffering large requests into disk), and rejects requests that are over a specified limit.
|
With Buffering, Traefik reads the entire request into memory (possibly buffering large requests into disk), and rejects requests that are over a specified size limit.
|
||||||
|
|
||||||
This can help services deal with large data (multipart/form-data for example), and can minimize time spent sending data to a service.
|
This can help services avoid large amounts of data (`multipart/form-data` for example), and can minimize the time spent sending data to a service.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
# Sets the maximum request body to 2Mb
|
# Sets the maximum request body to 2MB
|
||||||
labels:
|
labels:
|
||||||
- "traefik.http.middlewares.limit.buffering.maxRequestBodyBytes=2000000"
|
- "traefik.http.middlewares.limit.buffering.maxRequestBodyBytes=2000000"
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="Kubernetes"
|
```yaml tab="Kubernetes"
|
||||||
# Sets the maximum request body to 2Mb
|
# Sets the maximum request body to 2MB
|
||||||
apiVersion: traefik.containo.us/v1alpha1
|
apiVersion: traefik.containo.us/v1alpha1
|
||||||
kind: Middleware
|
kind: Middleware
|
||||||
metadata:
|
metadata:
|
||||||
|
@ -31,7 +31,7 @@ spec:
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="Consul Catalog"
|
```yaml tab="Consul Catalog"
|
||||||
# Sets the maximum request body to 2Mb
|
# Sets the maximum request body to 2MB
|
||||||
- "traefik.http.middlewares.limit.buffering.maxRequestBodyBytes=2000000"
|
- "traefik.http.middlewares.limit.buffering.maxRequestBodyBytes=2000000"
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -42,20 +42,20 @@ spec:
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="Rancher"
|
```yaml tab="Rancher"
|
||||||
# Sets the maximum request body to 2Mb
|
# Sets the maximum request body to 2MB
|
||||||
labels:
|
labels:
|
||||||
- "traefik.http.middlewares.limit.buffering.maxRequestBodyBytes=2000000"
|
- "traefik.http.middlewares.limit.buffering.maxRequestBodyBytes=2000000"
|
||||||
```
|
```
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
# Sets the maximum request body to 2Mb
|
# Sets the maximum request body to 2MB
|
||||||
[http.middlewares]
|
[http.middlewares]
|
||||||
[http.middlewares.limit.buffering]
|
[http.middlewares.limit.buffering]
|
||||||
maxRequestBodyBytes = 2000000
|
maxRequestBodyBytes = 2000000
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
```yaml tab="File (YAML)"
|
||||||
# Sets the maximum request body to 2Mb
|
# Sets the maximum request body to 2MB
|
||||||
http:
|
http:
|
||||||
middlewares:
|
middlewares:
|
||||||
limit:
|
limit:
|
||||||
|
@ -67,9 +67,9 @@ http:
|
||||||
|
|
||||||
### `maxRequestBodyBytes`
|
### `maxRequestBodyBytes`
|
||||||
|
|
||||||
With the `maxRequestBodyBytes` option, you can configure the maximum allowed body size for the request (in Bytes).
|
The `maxRequestBodyBytes` option configures the maximum allowed body size for the request (in bytes).
|
||||||
|
|
||||||
If the request exceeds the allowed size, it is not forwarded to the service and the client gets a `413 (Request Entity Too Large)` response.
|
If the request exceeds the allowed size, it is not forwarded to the service, and the client gets a `413 (Request Entity Too Large)` response.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -117,7 +117,7 @@ http:
|
||||||
|
|
||||||
### `memRequestBodyBytes`
|
### `memRequestBodyBytes`
|
||||||
|
|
||||||
You can configure a threshold (in Bytes) from which the request will be buffered on disk instead of in memory with the `memRequestBodyBytes` option.
|
You can configure a threshold (in bytes) from which the request will be buffered on disk instead of in memory with the `memRequestBodyBytes` option.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -165,7 +165,7 @@ http:
|
||||||
|
|
||||||
### `maxResponseBodyBytes`
|
### `maxResponseBodyBytes`
|
||||||
|
|
||||||
With the `maxResponseBodyBytes` option, you can configure the maximum allowed response size from the service (in Bytes).
|
The `maxResponseBodyBytes` option configures the maximum allowed response size from the service (in bytes).
|
||||||
|
|
||||||
If the response exceeds the allowed size, it is not forwarded to the client. The client gets a `413 (Request Entity Too Large) response` instead.
|
If the response exceeds the allowed size, it is not forwarded to the client. The client gets a `413 (Request Entity Too Large) response` instead.
|
||||||
|
|
||||||
|
@ -215,7 +215,7 @@ http:
|
||||||
|
|
||||||
### `memResponseBodyBytes`
|
### `memResponseBodyBytes`
|
||||||
|
|
||||||
You can configure a threshold (in Bytes) from which the response will be buffered on disk instead of in memory with the `memResponseBodyBytes` option.
|
You can configure a threshold (in bytes) from which the response will be buffered on disk instead of in memory with the `memResponseBodyBytes` option.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -263,9 +263,9 @@ http:
|
||||||
|
|
||||||
### `retryExpression`
|
### `retryExpression`
|
||||||
|
|
||||||
You can have the Buffering middleware replay the request with the help of the `retryExpression` option.
|
You can have the Buffering middleware replay the request using `retryExpression`.
|
||||||
|
|
||||||
??? example "Retries once in case of a network error"
|
??? example "Retries once in the case of a network error"
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -315,4 +315,4 @@ The retry expression is defined as a logical combination of the functions below
|
||||||
|
|
||||||
- `Attempts()` number of attempts (the first one counts)
|
- `Attempts()` number of attempts (the first one counts)
|
||||||
- `ResponseCode()` response code of the service
|
- `ResponseCode()` response code of the service
|
||||||
- `IsNetworkError()` - if the response code is related to networking error
|
- `IsNetworkError()` whether the response code is related to networking error
|
||||||
|
|
|
@ -10,7 +10,7 @@ It makes reusing the same groups easier.
|
||||||
|
|
||||||
## Configuration Example
|
## Configuration Example
|
||||||
|
|
||||||
Example "A Chain for WhiteList, BasicAuth, and HTTPS"
|
Below is an example of a Chain containing `WhiteList`, `BasicAuth`, and `RedirectScheme`.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -30,11 +30,9 @@ kind: IngressRoute
|
||||||
metadata:
|
metadata:
|
||||||
name: test
|
name: test
|
||||||
namespace: default
|
namespace: default
|
||||||
|
|
||||||
spec:
|
spec:
|
||||||
entryPoints:
|
entryPoints:
|
||||||
- web
|
- web
|
||||||
|
|
||||||
routes:
|
routes:
|
||||||
- match: Host(`mydomain`)
|
- match: Host(`mydomain`)
|
||||||
kind: Rule
|
kind: Rule
|
||||||
|
|
|
@ -5,24 +5,21 @@ Don't Waste Time Calling Unhealthy Services
|
||||||
|
|
||||||
![CircuitBreaker](../assets/img/middleware/circuitbreaker.png)
|
![CircuitBreaker](../assets/img/middleware/circuitbreaker.png)
|
||||||
|
|
||||||
The circuit breaker protects your system from stacking requests to unhealthy services (resulting in cascading failures).
|
The circuit breaker protects your system from stacking requests to unhealthy services, resulting in cascading failures.
|
||||||
|
|
||||||
When your system is healthy, the circuit is closed (normal operations).
|
When your system is healthy, the circuit is closed (normal operations).
|
||||||
When your system becomes unhealthy, the circuit becomes open and the requests are no longer forwarded (but handled by a fallback mechanism).
|
When your system becomes unhealthy, the circuit opens, and the requests are no longer forwarded, but instead are handled by a fallback mechanism.
|
||||||
|
|
||||||
To assess if your system is healthy, the circuit breaker constantly monitors the services.
|
To assess if your system is healthy, the circuit breaker constantly monitors the services.
|
||||||
|
|
||||||
!!! note ""
|
!!! note ""
|
||||||
|
|
||||||
- The CircuitBreaker only analyses what happens _after_ it is positioned in the middleware chain. What happens _before_ has no impact on its state.
|
The CircuitBreaker only analyzes what happens _after_ its position within the middleware chain. What happens _before_ has no impact on its state.
|
||||||
- The CircuitBreaker only affects the routers that use it. Routers that don't use the CircuitBreaker won't be affected by its state.
|
|
||||||
|
|
||||||
!!! important
|
!!! important
|
||||||
|
|
||||||
Each router will eventually gets its own instance of a given circuit breaker.
|
Each router gets its own instance of a given circuit breaker.
|
||||||
|
One circuit breaker instance can be open while the other remains closed: their state is not shared.
|
||||||
If two different routers refer to the same circuit breaker definition, they will get one instance each.
|
|
||||||
It means that one circuit breaker can be open while the other stays closed: their state is not shared.
|
|
||||||
|
|
||||||
This is the expected behavior, we want you to be able to define what makes a service healthy without having to declare a circuit breaker for each route.
|
This is the expected behavior, we want you to be able to define what makes a service healthy without having to declare a circuit breaker for each route.
|
||||||
|
|
||||||
|
@ -90,44 +87,45 @@ There are three possible states for your circuit breaker:
|
||||||
|
|
||||||
While the circuit is closed, the circuit breaker only collects metrics to analyze the behavior of the requests.
|
While the circuit is closed, the circuit breaker only collects metrics to analyze the behavior of the requests.
|
||||||
|
|
||||||
At specified intervals (`checkPeriod`), it will evaluate `expression` to decide if its state must change.
|
At specified intervals (`checkPeriod`), the circuit breaker evaluates `expression` to decide if its state must change.
|
||||||
|
|
||||||
### Open
|
### Open
|
||||||
|
|
||||||
While open, the fallback mechanism takes over the normal service calls for a duration of `FallbackDuration`.
|
While open, the fallback mechanism takes over the normal service calls for a duration of `FallbackDuration`.
|
||||||
After this duration, it will enter the recovering state.
|
After this duration, it enters the recovering state.
|
||||||
|
|
||||||
### Recovering
|
### Recovering
|
||||||
|
|
||||||
While recovering, the circuit breaker will progressively send requests to your service again (in a linear way, for `RecoveryDuration`).
|
While recovering, the circuit breaker sends linearly increasing amounts of requests to your service (for `RecoveryDuration`).
|
||||||
If your service fails during recovery, the circuit breaker becomes open again.
|
If your service fails during recovery, the circuit breaker opens again.
|
||||||
If the service operates normally during the whole recovering duration, then the circuit breaker returns to close.
|
If the service operates normally during the entire recovery duration, then the circuit breaker closes.
|
||||||
|
|
||||||
## Configuration Options
|
## Configuration Options
|
||||||
|
|
||||||
### Configuring the Trigger
|
### Configuring the Trigger
|
||||||
|
|
||||||
You can specify an `expression` that, once matched, will trigger the circuit breaker (and apply the fallback mechanism instead of calling your services).
|
You can specify an `expression` that, once matched, opens the circuit breaker and applies the fallback mechanism instead of calling your services.
|
||||||
|
|
||||||
The `expression` can check three different metrics:
|
The `expression` option can check three different metrics:
|
||||||
|
|
||||||
- The network error ratio (`NetworkErrorRatio`)
|
- The network error ratio (`NetworkErrorRatio`)
|
||||||
- The status code ratio (`ResponseCodeRatio`)
|
- The status code ratio (`ResponseCodeRatio`)
|
||||||
- The latency at quantile, in milliseconds (`LatencyAtQuantileMS`)
|
- The latency at a quantile in milliseconds (`LatencyAtQuantileMS`)
|
||||||
|
|
||||||
#### `NetworkErrorRatio`
|
#### `NetworkErrorRatio`
|
||||||
|
|
||||||
If you want the circuit breaker to trigger at a 30% ratio of network errors, the expression will be `NetworkErrorRatio() > 0.30`
|
If you want the circuit breaker to open at a 30% ratio of network errors, the `expression` is `NetworkErrorRatio() > 0.30`
|
||||||
|
|
||||||
#### `ResponseCodeRatio`
|
#### `ResponseCodeRatio`
|
||||||
|
|
||||||
You can trigger the circuit breaker based on the ratio of a given range of status codes.
|
You can configure the circuit breaker to open based on the ratio of a given range of status codes.
|
||||||
|
|
||||||
The `ResponseCodeRatio` accepts four parameters, `from`, `to`, `dividedByFrom`, `dividedByTo`.
|
The `ResponseCodeRatio` accepts four parameters, `from`, `to`, `dividedByFrom`, `dividedByTo`.
|
||||||
|
|
||||||
The operation that will be computed is sum(`to` -> `from`) / sum (`dividedByFrom` -> `dividedByTo`).
|
The operation that will be computed is sum(`to` -> `from`) / sum (`dividedByFrom` -> `dividedByTo`).
|
||||||
|
|
||||||
!!! note ""
|
!!! note ""
|
||||||
|
|
||||||
If sum (`dividedByFrom` -> `dividedByTo`) equals 0, then `ResponseCodeRatio` returns 0.
|
If sum (`dividedByFrom` -> `dividedByTo`) equals 0, then `ResponseCodeRatio` returns 0.
|
||||||
|
|
||||||
`from`is inclusive, `to` is exclusive.
|
`from`is inclusive, `to` is exclusive.
|
||||||
|
@ -136,17 +134,17 @@ For example, the expression `ResponseCodeRatio(500, 600, 0, 600) > 0.25` will tr
|
||||||
|
|
||||||
#### `LatencyAtQuantileMS`
|
#### `LatencyAtQuantileMS`
|
||||||
|
|
||||||
You can trigger the circuit breaker when a given proportion of your requests become too slow.
|
You can configure the circuit breaker to open when a given proportion of your requests become too slow.
|
||||||
|
|
||||||
For example, the expression `LatencyAtQuantileMS(50.0) > 100` will trigger the circuit breaker when the median latency (quantile 50) reaches 100MS.
|
For example, the expression `LatencyAtQuantileMS(50.0) > 100` opens the circuit breaker when the median latency (quantile 50) reaches 100ms.
|
||||||
|
|
||||||
!!! note ""
|
!!! note ""
|
||||||
|
|
||||||
You must provide a float number (with the trailing .0) for the quantile value
|
You must provide a floating point number (with the trailing .0) for the quantile value
|
||||||
|
|
||||||
#### Using multiple metrics
|
#### Using Multiple Metrics
|
||||||
|
|
||||||
You can combine multiple metrics using operators in your expression.
|
You can combine multiple metrics using operators in your `expression`.
|
||||||
|
|
||||||
Supported operators are:
|
Supported operators are:
|
||||||
|
|
||||||
|
@ -168,7 +166,7 @@ Here is the list of supported operators:
|
||||||
|
|
||||||
### Fallback mechanism
|
### Fallback mechanism
|
||||||
|
|
||||||
The fallback mechanism returns a `HTTP 503 Service Unavailable` to the client (instead of calling the target service).
|
The fallback mechanism returns a `HTTP 503 Service Unavailable` to the client instead of calling the target service.
|
||||||
This behavior cannot be configured.
|
This behavior cannot be configured.
|
||||||
|
|
||||||
### `CheckPeriod`
|
### `CheckPeriod`
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
# Compress
|
# Compress
|
||||||
|
|
||||||
Compressing the Response before Sending it to the Client
|
Compress Responses before Sending them to the Client
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
![Compress](../assets/img/middleware/compress.png)
|
![Compress](../assets/img/middleware/compress.png)
|
||||||
|
|
||||||
The Compress middleware enables the gzip compression.
|
The Compress middleware uses gzip compression.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
@ -58,22 +58,22 @@ http:
|
||||||
|
|
||||||
!!! info
|
!!! info
|
||||||
|
|
||||||
Responses are compressed when:
|
Responses are compressed when the following criteria are all met:
|
||||||
|
|
||||||
* The response body is larger than `1400` bytes.
|
* The response body is larger than `1400` bytes.
|
||||||
* The `Accept-Encoding` request header contains `gzip`.
|
* The `Accept-Encoding` request header contains `gzip`.
|
||||||
* The response is not already compressed, i.e. the `Content-Encoding` response header is not already set.
|
* The response is not already compressed, i.e. the `Content-Encoding` response header is not already set.
|
||||||
|
|
||||||
If Content-Type header is not defined, or empty, the compress middleware will automatically [detect](https://mimesniff.spec.whatwg.org/) a content type.
|
If the `Content-Type` header is not defined, or empty, the compress middleware will automatically [detect](https://mimesniff.spec.whatwg.org/) a content type.
|
||||||
It will also set accordingly the `Content-Type` header with the detected MIME type.
|
It will also set the `Content-Type` header according to the detected MIME type.
|
||||||
|
|
||||||
## Configuration Options
|
## Configuration Options
|
||||||
|
|
||||||
### `excludedContentTypes`
|
### `excludedContentTypes`
|
||||||
|
|
||||||
`excludedContentTypes` specifies a list of content types to compare the `Content-Type` header of the incoming requests to before compressing.
|
`excludedContentTypes` specifies a list of content types to compare the `Content-Type` header of the incoming requests and responses before compressing.
|
||||||
|
|
||||||
The requests with content types defined in `excludedContentTypes` are not compressed.
|
The responses with content types defined in `excludedContentTypes` are not compressed.
|
||||||
|
|
||||||
Content types are compared in a case-insensitive, whitespace-ignored manner.
|
Content types are compared in a case-insensitive, whitespace-ignored manner.
|
||||||
|
|
||||||
|
|
|
@ -1,18 +1,19 @@
|
||||||
|
|
||||||
# ContentType
|
# ContentType
|
||||||
|
|
||||||
Handling ContentType auto-detection
|
Handling Content-Type auto-detection
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
The Content-Type middleware - or rather its unique `autoDetect` option -
|
The Content-Type middleware - or rather its `autoDetect` option -
|
||||||
specifies whether to let the `Content-Type` header,
|
specifies whether to let the `Content-Type` header,
|
||||||
if it has not been set by the backend,
|
if it has not been defined by the backend,
|
||||||
be automatically set to a value derived from the contents of the response.
|
be automatically set to a value derived from the contents of the response.
|
||||||
|
|
||||||
As a proxy, the default behavior should be to leave the header alone,
|
As a proxy, the default behavior should be to leave the header alone,
|
||||||
regardless of what the backend did with it.
|
regardless of what the backend did with it.
|
||||||
However, the historic default was to always auto-detect and set the header if it was nil,
|
However, the historic default was to always auto-detect and set the header if it was not already defined,
|
||||||
and it is going to be kept that way in order to support users currently relying on it.
|
and altering this behavior would be a breaking change which would impact many users.
|
||||||
|
|
||||||
This middleware exists to enable the correct behavior until at least the default one can be changed in a future version.
|
This middleware exists to enable the correct behavior until at least the default one can be changed in a future version.
|
||||||
|
|
||||||
!!! info
|
!!! info
|
||||||
|
|
|
@ -5,7 +5,7 @@ Adding Digest Authentication
|
||||||
|
|
||||||
![BasicAuth](../assets/img/middleware/digestauth.png)
|
![BasicAuth](../assets/img/middleware/digestauth.png)
|
||||||
|
|
||||||
The DigestAuth middleware is a quick way to restrict access to your services to known users.
|
The DigestAuth middleware restricts access to your services to known users.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
|
|
@ -86,27 +86,29 @@ http:
|
||||||
```
|
```
|
||||||
|
|
||||||
!!! note ""
|
!!! note ""
|
||||||
|
|
||||||
In this example, the error page URL is based on the status code (`query=/{status}.html`).
|
In this example, the error page URL is based on the status code (`query=/{status}.html`).
|
||||||
|
|
||||||
## Configuration Options
|
## Configuration Options
|
||||||
|
|
||||||
### `status`
|
### `status`
|
||||||
|
|
||||||
The `status` that will trigger the error page.
|
The `status` option defines which status or range of statuses should result in an error page.
|
||||||
|
|
||||||
The status code ranges are inclusive (`500-599` will trigger with every code between `500` and `599`, `500` and `599` included).
|
The status code ranges are inclusive (`500-599` will trigger with every code between `500` and `599`, `500` and `599` included).
|
||||||
|
|
||||||
!!! note ""
|
!!! note ""
|
||||||
|
|
||||||
You can define either a status code like `500` or ranges with a syntax like `500-599`.
|
You can define either a status code as a number (`500`) or ranges by separating two codes with a dash (`500-599`).
|
||||||
|
|
||||||
### `service`
|
### `service`
|
||||||
|
|
||||||
The service that will serve the new requested error page.
|
The service that will serve the new requested error page.
|
||||||
|
|
||||||
!!! note ""
|
!!! note ""
|
||||||
In kubernetes, you need to reference a kubernetes service instead of a traefik service.
|
|
||||||
|
In Kubernetes, you need to reference a Kubernetes Service instead of a Traefik service.
|
||||||
|
|
||||||
### `query`
|
### `query`
|
||||||
|
|
||||||
The URL for the error page (hosted by `service`). You can use `{status}` in the query, that will be replaced by the received status code.
|
The URL for the error page (hosted by `service`). You can use the `{status}` variable in the `query` option in order to insert the status code in the URL.
|
||||||
|
|
|
@ -1,12 +1,12 @@
|
||||||
# ForwardAuth
|
# ForwardAuth
|
||||||
|
|
||||||
Using an External Service to Check for Credentials
|
Using an External Service to Forward Authentication
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
![AuthForward](../assets/img/middleware/authforward.png)
|
![AuthForward](../assets/img/middleware/authforward.png)
|
||||||
|
|
||||||
The ForwardAuth middleware delegate the authentication to an external service.
|
The ForwardAuth middleware delegates authentication to an external service.
|
||||||
If the service response code is 2XX, access is granted and the original request is performed.
|
If the service answers with a 2XX code, access is granted, and the original request is performed.
|
||||||
Otherwise, the response from the authentication server is returned.
|
Otherwise, the response from the authentication server is returned.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
@ -125,7 +125,7 @@ http:
|
||||||
|
|
||||||
### `trustForwardHeader`
|
### `trustForwardHeader`
|
||||||
|
|
||||||
Set the `trustForwardHeader` option to `true` to trust all the existing `X-Forwarded-*` headers.
|
Set the `trustForwardHeader` option to `true` to trust all `X-Forwarded-*` headers.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -176,7 +176,8 @@ http:
|
||||||
|
|
||||||
### `authResponseHeaders`
|
### `authResponseHeaders`
|
||||||
|
|
||||||
The `authResponseHeaders` option is the list of the headers to copy from the authentication server to the request. All incoming request's headers in this list are deleted from the request before any copy happens.
|
The `authResponseHeaders` option is the list of headers to copy from the authentication server response and set on
|
||||||
|
forwarded request, replacing any existing conflicting headers.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -231,9 +232,10 @@ http:
|
||||||
|
|
||||||
### `authResponseHeadersRegex`
|
### `authResponseHeadersRegex`
|
||||||
|
|
||||||
The `authResponseHeadersRegex` option is the regex to match the headers that should be copied from the authentication server to the request. All incoming request's headers matching this regex are deleted from the request before any copy happens.
|
The `authResponseHeadersRegex` option is the regex to match headers to copy from the authentication server response and
|
||||||
It allows partial matching of the regular expression against the header's key.
|
set on forwarded request, after stripping all headers that match the regex.
|
||||||
You should use start of string (`^`) and end of string (`$`) anchors to ensure a full match against the header's key.
|
It allows partial matching of the regular expression against the header key.
|
||||||
|
The start of string (`^`) and end of string (`$`) anchors should be used to ensure a full match against the header key.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -286,7 +288,7 @@ http:
|
||||||
|
|
||||||
The `authRequestHeaders` option is the list of the headers to copy from the request to the authentication server.
|
The `authRequestHeaders` option is the list of the headers to copy from the request to the authentication server.
|
||||||
It allows filtering headers that should not be passed to the authentication server.
|
It allows filtering headers that should not be passed to the authentication server.
|
||||||
If not set or empty then all request headers will be passed.
|
If not set or empty then all request headers are passed.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -409,12 +411,15 @@ http:
|
||||||
|
|
||||||
#### `tls.caOptional`
|
#### `tls.caOptional`
|
||||||
|
|
||||||
Policy used for the secured connection with TLS Client Authentication to the authentication server.
|
The value of `tls.caOptional` defines which policy should be used for the secure connection with TLS Client Authentication to the authentication server.
|
||||||
Requires `tls.ca` to be defined.
|
|
||||||
|
|
||||||
- `true`: VerifyClientCertIfGiven
|
!!! warning ""
|
||||||
- `false`: RequireAndVerifyClientCert
|
|
||||||
- if `tls.ca` is undefined NoClientCert
|
If `tls.ca` is undefined, this option will be ignored, and no client certificate will be requested during the handshake. Any provided certificate will thus never be verified.
|
||||||
|
|
||||||
|
When this option is set to `true`, a client certificate is requested during the handshake but is not required. If a certificate is sent, it is required to be valid.
|
||||||
|
|
||||||
|
When this option is set to `false`, a client certificate is requested during the handshake, and at least one valid certificate should be sent by the client.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -468,7 +473,7 @@ http:
|
||||||
|
|
||||||
#### `tls.cert`
|
#### `tls.cert`
|
||||||
|
|
||||||
Public certificate used for the secured connection to the authentication server.
|
The public certificate used for the secure connection to the authentication server.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -538,11 +543,12 @@ http:
|
||||||
```
|
```
|
||||||
|
|
||||||
!!! info
|
!!! info
|
||||||
For security reasons, the field doesn't exist for Kubernetes IngressRoute, and one should use the `secret` field instead.
|
|
||||||
|
For security reasons, the field does not exist for Kubernetes IngressRoute, and one should use the `secret` field instead.
|
||||||
|
|
||||||
#### `tls.key`
|
#### `tls.key`
|
||||||
|
|
||||||
Private certificate used for the secure connection to the authentication server.
|
The private certificate used for the secure connection to the authentication server.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -612,11 +618,12 @@ http:
|
||||||
```
|
```
|
||||||
|
|
||||||
!!! info
|
!!! info
|
||||||
For security reasons, the field doesn't exist for Kubernetes IngressRoute, and one should use the `secret` field instead.
|
|
||||||
|
For security reasons, the field does not exist for Kubernetes IngressRoute, and one should use the `secret` field instead.
|
||||||
|
|
||||||
#### `tls.insecureSkipVerify`
|
#### `tls.insecureSkipVerify`
|
||||||
|
|
||||||
If `insecureSkipVerify` is `true`, TLS for the connection to authentication server accepts any certificate presented by the server and any host name in that certificate.
|
If `insecureSkipVerify` is `true`, the TLS connection to the authentication server accepts any certificate presented by the server regardless of the hostnames it covers.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
|
|
@ -1,17 +1,17 @@
|
||||||
# Headers
|
# Headers
|
||||||
|
|
||||||
Adding Headers to the Request / Response
|
Managing Request/Response headers
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
![Headers](../assets/img/middleware/headers.png)
|
![Headers](../assets/img/middleware/headers.png)
|
||||||
|
|
||||||
The Headers middleware can manage the requests/responses headers.
|
The Headers middleware manages the headers of requests and responses.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
### Adding Headers to the Request and the Response
|
### Adding Headers to the Request and the Response
|
||||||
|
|
||||||
Add the `X-Script-Name` header to the proxied request and the `X-Custom-Response-Header` to the response
|
The following example adds the `X-Script-Name` header to the proxied request and the `X-Custom-Response-Header` header to the response
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -72,8 +72,8 @@ http:
|
||||||
|
|
||||||
### Adding and Removing Headers
|
### Adding and Removing Headers
|
||||||
|
|
||||||
`X-Script-Name` header added to the proxied request, the `X-Custom-Request-Header` header removed from the request,
|
In the following example, requests are proxied with an extra `X-Script-Name` header while their `X-Custom-Request-Header` header gets stripped,
|
||||||
and the `X-Custom-Response-Header` header removed from the response.
|
and responses are stripped of their `X-Custom-Response-Header` header.
|
||||||
|
|
||||||
Please note that it is not possible to remove headers through the use of labels (Docker, Rancher, Marathon, ...) for now.
|
Please note that it is not possible to remove headers through the use of labels (Docker, Rancher, Marathon, ...) for now.
|
||||||
|
|
||||||
|
@ -135,8 +135,8 @@ http:
|
||||||
|
|
||||||
### Using Security Headers
|
### Using Security Headers
|
||||||
|
|
||||||
Security related headers (HSTS headers, SSL redirection, Browser XSS filter, etc) can be added and configured in a manner similar to the custom headers above.
|
Security-related headers (HSTS headers, SSL redirection, Browser XSS filter, etc) can be managed similarly to custom headers as shown above.
|
||||||
This functionality allows for some easy security features to quickly be set.
|
This functionality makes it possible to easily use security features by adding headers.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -274,18 +274,20 @@ http:
|
||||||
### General
|
### General
|
||||||
|
|
||||||
!!! warning
|
!!! warning
|
||||||
If the custom header name is the same as one header name of the request or response, it will be replaced.
|
|
||||||
|
Custom headers will overwrite existing headers if they have identical names.
|
||||||
|
|
||||||
!!! note ""
|
!!! note ""
|
||||||
The detailed documentation for the security headers can be found in [unrolled/secure](https://github.com/unrolled/secure#available-options).
|
|
||||||
|
The detailed documentation for security headers can be found in [unrolled/secure](https://github.com/unrolled/secure#available-options).
|
||||||
|
|
||||||
### `customRequestHeaders`
|
### `customRequestHeaders`
|
||||||
|
|
||||||
The `customRequestHeaders` option lists the Header names and values to apply to the request.
|
The `customRequestHeaders` option lists the header names and values to apply to the request.
|
||||||
|
|
||||||
### `customResponseHeaders`
|
### `customResponseHeaders`
|
||||||
|
|
||||||
The `customResponseHeaders` option lists the Header names and values to apply to the response.
|
The `customResponseHeaders` option lists the header names and values to apply to the response.
|
||||||
|
|
||||||
### `accessControlAllowCredentials`
|
### `accessControlAllowCredentials`
|
||||||
|
|
||||||
|
@ -303,25 +305,26 @@ The `accessControlAllowMethods` indicates which methods can be used during requ
|
||||||
|
|
||||||
The `accessControlAllowOriginList` indicates whether a resource can be shared by returning different values.
|
The `accessControlAllowOriginList` indicates whether a resource can be shared by returning different values.
|
||||||
|
|
||||||
A wildcard origin `*` can also be configured, and will match all requests.
|
A wildcard origin `*` can also be configured, and matches all requests.
|
||||||
If this value is set by a backend server, it will be overwritten by Traefik
|
If this value is set by a backend service, it will be overwritten by Traefik.
|
||||||
|
|
||||||
This value can contain a list of allowed origins.
|
This value can contain a list of allowed origins.
|
||||||
|
|
||||||
More information including how to use the settings can be found on:
|
More information including how to use the settings can be found at:
|
||||||
|
|
||||||
- [Mozilla.org](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Access-Control-Allow-Origin)
|
- [Mozilla.org](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Access-Control-Allow-Origin)
|
||||||
- [w3](https://fetch.spec.whatwg.org/#http-access-control-allow-origin)
|
- [w3](https://fetch.spec.whatwg.org/#http-access-control-allow-origin)
|
||||||
- [IETF](https://tools.ietf.org/html/rfc6454#section-7.1)
|
- [IETF](https://tools.ietf.org/html/rfc6454#section-7.1)
|
||||||
|
|
||||||
Traefik no longer supports the null value, as it is [no longer recommended as a return value](https://w3c.github.io/webappsec-cors-for-developers/#avoid-returning-access-control-allow-origin-null).
|
Traefik no longer supports the `null` value, as it is [no longer recommended as a return value](https://w3c.github.io/webappsec-cors-for-developers/#avoid-returning-access-control-allow-origin-null).
|
||||||
|
|
||||||
### `accessControlAllowOriginListRegex`
|
### `accessControlAllowOriginListRegex`
|
||||||
|
|
||||||
The `accessControlAllowOriginListRegex` option is the counterpart of the `accessControlAllowOriginList` option with regular expressions instead of origin values.
|
The `accessControlAllowOriginListRegex` option is the counterpart of the `accessControlAllowOriginList` option with regular expressions instead of origin values.
|
||||||
It will allow all origin that contains any match of a regular expression in the `accessControlAllowOriginList`.
|
It allows all origins that contain any match of a regular expression in the `accessControlAllowOriginList`.
|
||||||
|
|
||||||
!!! tip
|
!!! tip
|
||||||
|
|
||||||
Regular expressions can be tested using online tools such as [Go Playground](https://play.golang.org/p/mWU9p-wk2ru) or the [Regex101](https://regex101.com/r/58sIgx/2).
|
Regular expressions can be tested using online tools such as [Go Playground](https://play.golang.org/p/mWU9p-wk2ru) or the [Regex101](https://regex101.com/r/58sIgx/2).
|
||||||
|
|
||||||
### `accessControlExposeHeaders`
|
### `accessControlExposeHeaders`
|
||||||
|
@ -330,11 +333,11 @@ The `accessControlExposeHeaders` indicates which headers are safe to expose to t
|
||||||
|
|
||||||
### `accessControlMaxAge`
|
### `accessControlMaxAge`
|
||||||
|
|
||||||
The `accessControlMaxAge` indicates how long (in seconds) a preflight request can be cached.
|
The `accessControlMaxAge` indicates how many seconds a preflight request can be cached for.
|
||||||
|
|
||||||
### `addVaryHeader`
|
### `addVaryHeader`
|
||||||
|
|
||||||
The `addVaryHeader` is used in conjunction with `accessControlAllowOriginList` to determine whether the vary header should be added or modified to demonstrate that server responses can differ based on the value of the origin header.
|
The `addVaryHeader` is used in conjunction with `accessControlAllowOriginList` to determine whether the `Vary` header should be added or modified to demonstrate that server responses can differ based on the value of the origin header.
|
||||||
|
|
||||||
### `allowedHosts`
|
### `allowedHosts`
|
||||||
|
|
||||||
|
@ -346,50 +349,50 @@ The `hostsProxyHeaders` option is a set of header keys that may hold a proxied h
|
||||||
|
|
||||||
### `sslRedirect`
|
### `sslRedirect`
|
||||||
|
|
||||||
The `sslRedirect` is set to true, then only allow https requests.
|
The `sslRedirect` only allow HTTPS requests when set to `true`.
|
||||||
|
|
||||||
### `sslTemporaryRedirect`
|
### `sslTemporaryRedirect`
|
||||||
|
|
||||||
Set the `sslTemporaryRedirect` to `true` to force an SSL redirection using a 302 (instead of a 301).
|
Set `sslTemporaryRedirect` to `true` to force an SSL redirection using a 302 (instead of a 301).
|
||||||
|
|
||||||
### `sslHost`
|
### `sslHost`
|
||||||
|
|
||||||
The `sslHost` option is the host name that is used to redirect http requests to https.
|
The `sslHost` option is the host name that is used to redirect HTTP requests to HTTPS.
|
||||||
|
|
||||||
### `sslProxyHeaders`
|
### `sslProxyHeaders`
|
||||||
|
|
||||||
The `sslProxyHeaders` option is set of header keys with associated values that would indicate a valid https request.
|
The `sslProxyHeaders` option is set of header keys with associated values that would indicate a valid HTTPS request.
|
||||||
Useful when using other proxies with header like: `"X-Forwarded-Proto": "https"`.
|
It can be useful when using other proxies (example: `"X-Forwarded-Proto": "https"`).
|
||||||
|
|
||||||
### `sslForceHost`
|
### `sslForceHost`
|
||||||
|
|
||||||
Set `sslForceHost` to true and set SSLHost to forced requests to use `SSLHost` even the ones that are already using SSL.
|
Set `sslForceHost` to `true` and set `sslHost` to force requests to use `SSLHost` regardless of whether they already use SSL.
|
||||||
|
|
||||||
### `stsSeconds`
|
### `stsSeconds`
|
||||||
|
|
||||||
The `stsSeconds` is the max-age of the Strict-Transport-Security header.
|
The `stsSeconds` is the max-age of the `Strict-Transport-Security` header.
|
||||||
If set to 0, would NOT include the header.
|
If set to `0`, the header is not set.
|
||||||
|
|
||||||
### `stsIncludeSubdomains`
|
### `stsIncludeSubdomains`
|
||||||
|
|
||||||
The `stsIncludeSubdomains` is set to true, the `includeSubDomains` directive will be appended to the Strict-Transport-Security header.
|
If the `stsIncludeSubdomains` is set to `true`, the `includeSubDomains` directive is appended to the `Strict-Transport-Security` header.
|
||||||
|
|
||||||
### `stsPreload`
|
### `stsPreload`
|
||||||
|
|
||||||
Set `stsPreload` to true to have the `preload` flag appended to the Strict-Transport-Security header.
|
Set `stsPreload` to `true` to have the `preload` flag appended to the `Strict-Transport-Security` header.
|
||||||
|
|
||||||
### `forceSTSHeader`
|
### `forceSTSHeader`
|
||||||
|
|
||||||
Set `forceSTSHeader` to true, to add the STS header even when the connection is HTTP.
|
Set `forceSTSHeader` to `true` to add the STS header even when the connection is HTTP.
|
||||||
|
|
||||||
### `frameDeny`
|
### `frameDeny`
|
||||||
|
|
||||||
Set `frameDeny` to true to add the `X-Frame-Options` header with the value of `DENY`.
|
Set `frameDeny` to `true` to add the `X-Frame-Options` header with the value of `DENY`.
|
||||||
|
|
||||||
### `customFrameOptionsValue`
|
### `customFrameOptionsValue`
|
||||||
|
|
||||||
The `customFrameOptionsValue` allows the `X-Frame-Options` header value to be set with a custom value.
|
The `customFrameOptionsValue` allows the `X-Frame-Options` header value to be set with a custom value.
|
||||||
This overrides the FrameDeny option.
|
This overrides the `FrameDeny` option.
|
||||||
|
|
||||||
### `contentTypeNosniff`
|
### `contentTypeNosniff`
|
||||||
|
|
||||||
|
@ -402,7 +405,7 @@ Set `browserXssFilter` to true to add the `X-XSS-Protection` header with the val
|
||||||
### `customBrowserXSSValue`
|
### `customBrowserXSSValue`
|
||||||
|
|
||||||
The `customBrowserXssValue` option allows the `X-XSS-Protection` header value to be set with a custom value.
|
The `customBrowserXssValue` option allows the `X-XSS-Protection` header value to be set with a custom value.
|
||||||
This overrides the BrowserXssFilter option.
|
This overrides the `BrowserXssFilter` option.
|
||||||
|
|
||||||
### `contentSecurityPolicy`
|
### `contentSecurityPolicy`
|
||||||
|
|
||||||
|
@ -414,7 +417,7 @@ The `publicKey` implements HPKP to prevent MITM attacks with forged certificates
|
||||||
|
|
||||||
### `referrerPolicy`
|
### `referrerPolicy`
|
||||||
|
|
||||||
The `referrerPolicy` allows sites to control when browsers will pass the Referer header to other sites.
|
The `referrerPolicy` allows sites to control whether browsers forward the `Referer` header to other sites.
|
||||||
|
|
||||||
### `featurePolicy`
|
### `featurePolicy`
|
||||||
|
|
||||||
|
@ -422,7 +425,6 @@ The `featurePolicy` allows sites to control browser features.
|
||||||
|
|
||||||
### `isDevelopment`
|
### `isDevelopment`
|
||||||
|
|
||||||
Set `isDevelopment` to true when developing.
|
Set `isDevelopment` to `true` when developing to mitigate the unwanted effects of the `AllowedHosts`, SSL, and STS options.
|
||||||
The AllowedHosts, SSL, and STS options can cause some unwanted effects.
|
Usually testing takes place using HTTP, not HTTPS, and on `localhost`, not your production domain.
|
||||||
Usually testing happens on http, not https, and on localhost, not your production domain.
|
If you would like your development environment to mimic production with complete Host blocking, SSL redirects, and STS headers, leave this as `false`.
|
||||||
If you would like your development environment to mimic production with complete Host blocking, SSL redirects, and STS headers, leave this as false.
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ Limiting the Number of Simultaneous In-Flight Requests
|
||||||
|
|
||||||
![InFlightReq](../assets/img/middleware/inflightreq.png)
|
![InFlightReq](../assets/img/middleware/inflightreq.png)
|
||||||
|
|
||||||
To proactively prevent services from being overwhelmed with high load, a limit on the number of simultaneous in-flight requests can be applied.
|
To proactively prevent services from being overwhelmed with high load, the number of allowed simultaneous in-flight requests can be limited.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
@ -62,7 +62,7 @@ http:
|
||||||
### `amount`
|
### `amount`
|
||||||
|
|
||||||
The `amount` option defines the maximum amount of allowed simultaneous in-flight request.
|
The `amount` option defines the maximum amount of allowed simultaneous in-flight request.
|
||||||
The middleware will return an `HTTP 429 Too Many Requests` if there are already `amount` requests in progress (based on the same `sourceCriterion` strategy).
|
The middleware responds with `HTTP 429 Too Many Requests` if there are already `amount` requests in progress (based on the same `sourceCriterion` strategy).
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
@ -114,24 +114,24 @@ http:
|
||||||
|
|
||||||
### `sourceCriterion`
|
### `sourceCriterion`
|
||||||
|
|
||||||
SourceCriterion defines what criterion is used to group requests as originating from a common source.
|
The `sourceCriterion` option defines what criterion is used to group requests as originating from a common source.
|
||||||
The precedence order is `ipStrategy`, then `requestHeaderName`, then `requestHost`.
|
The precedence order is `ipStrategy`, then `requestHeaderName`, then `requestHost`.
|
||||||
If none are set, the default is to use the `requestHost`.
|
If none are set, the default is to use the `requestHost`.
|
||||||
|
|
||||||
#### `sourceCriterion.ipStrategy`
|
#### `sourceCriterion.ipStrategy`
|
||||||
|
|
||||||
The `ipStrategy` option defines two parameters that sets how Traefik will determine the client IP: `depth`, and `excludedIPs`.
|
The `ipStrategy` option defines two parameters that configures how Traefik determines the client IP: `depth`, and `excludedIPs`.
|
||||||
|
|
||||||
##### `ipStrategy.depth`
|
##### `ipStrategy.depth`
|
||||||
|
|
||||||
The `depth` option tells Traefik to use the `X-Forwarded-For` header and take the IP located at the `depth` position (starting from the right).
|
The `depth` option tells Traefik to use the `X-Forwarded-For` header and select the IP located at the `depth` position (starting from the right).
|
||||||
|
|
||||||
- If `depth` is greater than the total number of IPs in `X-Forwarded-For`, then the client IP will be empty.
|
- If `depth` is greater than the total number of IPs in `X-Forwarded-For`, then the client IP is empty.
|
||||||
- `depth` is ignored if its value is lesser than or equal to 0.
|
- `depth` is ignored if its value is less than or equal to 0.
|
||||||
|
|
||||||
!!! example "Example of Depth & X-Forwarded-For"
|
!!! example "Example of Depth & X-Forwarded-For"
|
||||||
|
|
||||||
If `depth` was equal to 2, and the request `X-Forwarded-For` header was `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` then the "real" client IP would be `"10.0.0.1"` (at depth 4) but the IP used as the criterion would be `"12.0.0.1"` (`depth=2`).
|
If `depth` is set to 2, and the request `X-Forwarded-For` header is `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` then the "real" client IP is `"10.0.0.1"` (at depth 4) but the IP used as the criterion is `"12.0.0.1"` (`depth=2`).
|
||||||
|
|
||||||
| `X-Forwarded-For` | `depth` | clientIP |
|
| `X-Forwarded-For` | `depth` | clientIP |
|
||||||
|-----------------------------------------|---------|--------------|
|
|-----------------------------------------|---------|--------------|
|
||||||
|
@ -190,7 +190,7 @@ http:
|
||||||
|
|
||||||
##### `ipStrategy.excludedIPs`
|
##### `ipStrategy.excludedIPs`
|
||||||
|
|
||||||
`excludedIPs` tells Traefik to scan the `X-Forwarded-For` header and pick the first IP not in the list.
|
`excludedIPs` configures Traefik to scan the `X-Forwarded-For` header and select the first IP not in the list.
|
||||||
|
|
||||||
!!! important "If `depth` is specified, `excludedIPs` is ignored."
|
!!! important "If `depth` is specified, `excludedIPs` is ignored."
|
||||||
|
|
||||||
|
@ -259,7 +259,7 @@ http:
|
||||||
|
|
||||||
#### `sourceCriterion.requestHeaderName`
|
#### `sourceCriterion.requestHeaderName`
|
||||||
|
|
||||||
Requests having the same value for the given header are grouped as coming from the same source.
|
Name of the header used to group incoming requests.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
|
|
@ -70,14 +70,25 @@ The `sourceRange` option sets the allowed IPs (or ranges of allowed IPs by using
|
||||||
|
|
||||||
### `ipStrategy`
|
### `ipStrategy`
|
||||||
|
|
||||||
The `ipStrategy` option defines two parameters that sets how Traefik will determine the client IP: `depth`, and `excludedIPs`.
|
The `ipStrategy` option defines two parameters that set how Traefik determines the client IP: `depth`, and `excludedIPs`.
|
||||||
|
|
||||||
#### `ipStrategy.depth`
|
#### `ipStrategy.depth`
|
||||||
|
|
||||||
The `depth` option tells Traefik to use the `X-Forwarded-For` header and take the IP located at the `depth` position (starting from the right).
|
The `depth` option tells Traefik to use the `X-Forwarded-For` header and take the IP located at the `depth` position (starting from the right).
|
||||||
|
|
||||||
|
- If `depth` is greater than the total number of IPs in `X-Forwarded-For`, then the client IP will be empty.
|
||||||
|
- `depth` is ignored if its value is less than or equal to 0.
|
||||||
|
|
||||||
!!! example "Examples of Depth & X-Forwarded-For"
|
!!! example "Examples of Depth & X-Forwarded-For"
|
||||||
|
|
||||||
|
If `depth` is set to 2, and the request `X-Forwarded-For` header is `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` then the "real" client IP is `"10.0.0.1"` (at depth 4) but the IP used for the whitelisting is `"12.0.0.1"` (`depth=2`).
|
||||||
|
|
||||||
|
| `X-Forwarded-For` | `depth` | clientIP |
|
||||||
|
|-----------------------------------------|---------|--------------|
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `1` | `"13.0.0.1"` |
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `3` | `"11.0.0.1"` |
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `5` | `""` |
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
# Whitelisting Based on `X-Forwarded-For` with `depth=2`
|
# Whitelisting Based on `X-Forwarded-For` with `depth=2`
|
||||||
labels:
|
labels:
|
||||||
|
@ -142,23 +153,22 @@ The `depth` option tells Traefik to use the `X-Forwarded-For` header and take th
|
||||||
depth: 2
|
depth: 2
|
||||||
```
|
```
|
||||||
|
|
||||||
If `depth` was equal to 2, and the request `X-Forwarded-For` header was `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` then the "real" client IP would be `"10.0.0.1"` (at depth 4) but the IP used for the whitelisting would be `"12.0.0.1"` (`depth=2`).
|
|
||||||
|
|
||||||
??? example "More examples"
|
|
||||||
|
|
||||||
| `X-Forwarded-For` | `depth` | clientIP |
|
|
||||||
|-----------------------------------------|---------|--------------|
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `1` | `"13.0.0.1"` |
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `3` | `"11.0.0.1"` |
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `5` | `""` |
|
|
||||||
|
|
||||||
!!! info
|
|
||||||
|
|
||||||
- If `depth` is greater than the total number of IPs in `X-Forwarded-For`, then the client IP will be empty.
|
|
||||||
- `depth` is ignored if its value is lesser than or equal to 0.
|
|
||||||
|
|
||||||
#### `ipStrategy.excludedIPs`
|
#### `ipStrategy.excludedIPs`
|
||||||
|
|
||||||
|
`excludedIPs` configures Traefik to scan the `X-Forwarded-For` header and select the first IP not in the list.
|
||||||
|
|
||||||
|
!!! important "If `depth` is specified, `excludedIPs` is ignored."
|
||||||
|
|
||||||
|
!!! example "Example of ExcludedIPs & X-Forwarded-For"
|
||||||
|
|
||||||
|
| `X-Forwarded-For` | `excludedIPs` | clientIP |
|
||||||
|
|-----------------------------------------|-----------------------|--------------|
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"12.0.0.1,13.0.0.1"` | `"11.0.0.1"` |
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"15.0.0.1,13.0.0.1"` | `"12.0.0.1"` |
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"10.0.0.1,13.0.0.1"` | `"12.0.0.1"` |
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"15.0.0.1,16.0.0.1"` | `"13.0.0.1"` |
|
||||||
|
| `"10.0.0.1,11.0.0.1"` | `"10.0.0.1,11.0.0.1"` | `""` |
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
# Exclude from `X-Forwarded-For`
|
# Exclude from `X-Forwarded-For`
|
||||||
labels:
|
labels:
|
||||||
|
@ -215,17 +225,3 @@ http:
|
||||||
- "127.0.0.1/32"
|
- "127.0.0.1/32"
|
||||||
- "192.168.1.7"
|
- "192.168.1.7"
|
||||||
```
|
```
|
||||||
|
|
||||||
`excludedIPs` tells Traefik to scan the `X-Forwarded-For` header and pick the first IP not in the list.
|
|
||||||
|
|
||||||
!!! important "If `depth` is specified, `excludedIPs` is ignored."
|
|
||||||
|
|
||||||
!!! example "Examples of ExcludedIPs & X-Forwarded-For"
|
|
||||||
|
|
||||||
| `X-Forwarded-For` | `excludedIPs` | clientIP |
|
|
||||||
|-----------------------------------------|-----------------------|--------------|
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"12.0.0.1,13.0.0.1"` | `"11.0.0.1"` |
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"15.0.0.1,13.0.0.1"` | `"12.0.0.1"` |
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"10.0.0.1,13.0.0.1"` | `"12.0.0.1"` |
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"15.0.0.1,16.0.0.1"` | `"13.0.0.1"` |
|
|
||||||
| `"10.0.0.1,11.0.0.1"` | `"10.0.0.1,11.0.0.1"` | `""` |
|
|
||||||
|
|
|
@ -7,7 +7,7 @@ Adding Client Certificates in a Header
|
||||||
TODO: add schema
|
TODO: add schema
|
||||||
-->
|
-->
|
||||||
|
|
||||||
PassTLSClientCert adds in header the selected data from the passed client tls certificate.
|
PassTLSClientCert adds the selected data from the passed client TLS certificate to a header.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
@ -251,9 +251,9 @@ PassTLSClientCert can add two headers to the request:
|
||||||
* These options only work accordingly to the [MutualTLS configuration](../https/tls.md#client-authentication-mtls).
|
* These options only work accordingly to the [MutualTLS configuration](../https/tls.md#client-authentication-mtls).
|
||||||
That is to say, only the certificates that match the `clientAuth.clientAuthType` policy are passed.
|
That is to say, only the certificates that match the `clientAuth.clientAuthType` policy are passed.
|
||||||
|
|
||||||
In the following example, you can see a complete certificate. We will use each part of it to explain the middleware options.
|
The following example shows a complete certificate and explains each of the middleware options.
|
||||||
|
|
||||||
??? example "A complete client tls certificate"
|
??? example "A complete client TLS certificate"
|
||||||
|
|
||||||
```
|
```
|
||||||
Certificate:
|
Certificate:
|
||||||
|
@ -359,7 +359,7 @@ In the following example, you can see a complete certificate. We will use each p
|
||||||
|
|
||||||
### `pem`
|
### `pem`
|
||||||
|
|
||||||
The `pem` option sets the `X-Forwarded-Tls-Client-Cert` header with the escape certificate.
|
The `pem` option sets the `X-Forwarded-Tls-Client-Cert` header with the escaped certificate.
|
||||||
|
|
||||||
In the example, it is the part between `-----BEGIN CERTIFICATE-----` and `-----END CERTIFICATE-----` delimiters:
|
In the example, it is the part between `-----BEGIN CERTIFICATE-----` and `-----END CERTIFICATE-----` delimiters:
|
||||||
|
|
||||||
|
@ -416,8 +416,9 @@ In the example, it is the part between `-----BEGIN CERTIFICATE-----` and `-----E
|
||||||
|
|
||||||
### `info`
|
### `info`
|
||||||
|
|
||||||
The `info` option select the specific client certificate details you want to add to the `X-Forwarded-Tls-Client-Cert-Info` header.
|
The `info` option selects the specific client certificate details you want to add to the `X-Forwarded-Tls-Client-Cert-Info` header.
|
||||||
The value of the header will be an escaped concatenation of all the selected certificate details.
|
|
||||||
|
The value of the header is an escaped concatenation of all the selected certificate details.
|
||||||
|
|
||||||
The following example shows an unescaped result that uses all the available fields:
|
The following example shows an unescaped result that uses all the available fields:
|
||||||
|
|
||||||
|
@ -433,14 +434,14 @@ Subject="DC=org,DC=cheese,C=FR,C=US,ST=Cheese org state,ST=Cheese com state,L=TO
|
||||||
|
|
||||||
Set the `info.notAfter` option to `true` to add the `Not After` information from the `Validity` part.
|
Set the `info.notAfter` option to `true` to add the `Not After` information from the `Validity` part.
|
||||||
|
|
||||||
The data are taken from the following certificate part:
|
The data is taken from the following certificate part:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
Validity
|
Validity
|
||||||
Not After : Dec 5 11:10:16 2020 GMT
|
Not After : Dec 5 11:10:16 2020 GMT
|
||||||
```
|
```
|
||||||
|
|
||||||
The escape `notAfter` info part will be like:
|
The escaped `notAfter` info part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
NA="1607166616"
|
NA="1607166616"
|
||||||
|
@ -450,14 +451,14 @@ NA="1607166616"
|
||||||
|
|
||||||
Set the `info.notBefore` option to `true` to add the `Not Before` information from the `Validity` part.
|
Set the `info.notBefore` option to `true` to add the `Not Before` information from the `Validity` part.
|
||||||
|
|
||||||
The data are taken from the following certificate part:
|
The data is taken from the following certificate part:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
Validity
|
Validity
|
||||||
Not Before: Dec 6 11:10:16 2018 GMT
|
Not Before: Dec 6 11:10:16 2018 GMT
|
||||||
```
|
```
|
||||||
|
|
||||||
The escape `notBefore` info part will be like:
|
The escaped `notBefore` info part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
NB="1544094616"
|
NB="1544094616"
|
||||||
|
@ -467,28 +468,28 @@ NB="1544094616"
|
||||||
|
|
||||||
Set the `info.sans` option to `true` to add the `Subject Alternative Name` information from the `Subject Alternative Name` part.
|
Set the `info.sans` option to `true` to add the `Subject Alternative Name` information from the `Subject Alternative Name` part.
|
||||||
|
|
||||||
The data are taken from the following certificate part:
|
The data is taken from the following certificate part:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
X509v3 Subject Alternative Name:
|
X509v3 Subject Alternative Name:
|
||||||
DNS:*.example.org, DNS:*.example.net, DNS:*.example.com, IP Address:10.0.1.0, IP Address:10.0.1.2, email:test@example.org, email:test@example.net
|
DNS:*.example.org, DNS:*.example.net, DNS:*.example.com, IP Address:10.0.1.0, IP Address:10.0.1.2, email:test@example.org, email:test@example.net
|
||||||
```
|
```
|
||||||
|
|
||||||
The escape SANs info part will be like:
|
The escape SANs info part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
SAN="*.example.org,*.example.net,*.example.com,test@example.org,test@example.net,10.0.1.0,10.0.1.2"
|
SAN="*.example.org,*.example.net,*.example.com,test@example.org,test@example.net,10.0.1.0,10.0.1.2"
|
||||||
```
|
```
|
||||||
|
|
||||||
!!! info "multiple values"
|
!!! info "Multiple values"
|
||||||
|
|
||||||
All the SANs data are separated by a `,`.
|
The SANs are separated by a `,`.
|
||||||
|
|
||||||
#### `info.subject`
|
#### `info.subject`
|
||||||
|
|
||||||
The `info.subject` select the specific client certificate subject details you want to add to the `X-Forwarded-Tls-Client-Cert-Info` header.
|
The `info.subject` selects the specific client certificate subject details you want to add to the `X-Forwarded-Tls-Client-Cert-Info` header.
|
||||||
|
|
||||||
The data are taken from the following certificate part :
|
The data is taken from the following certificate part:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
Subject: DC=org, DC=cheese, O=Cheese, O=Cheese 2, OU=Simple Signing Section, OU=Simple Signing Section 2, CN=*.example.org, CN=*.example.com, C=FR, C=US, L=TOULOUSE, L=LYON, ST=Cheese org state, ST=Cheese com state/emailAddress=cert@example.org/emailAddress=cert@sexample.com
|
Subject: DC=org, DC=cheese, O=Cheese, O=Cheese 2, OU=Simple Signing Section, OU=Simple Signing Section 2, CN=*.example.org, CN=*.example.com, C=FR, C=US, L=TOULOUSE, L=LYON, ST=Cheese org state, ST=Cheese com state/emailAddress=cert@example.org/emailAddress=cert@sexample.com
|
||||||
|
@ -496,9 +497,11 @@ Subject: DC=org, DC=cheese, O=Cheese, O=Cheese 2, OU=Simple Signing Section, OU=
|
||||||
|
|
||||||
##### `info.subject.country`
|
##### `info.subject.country`
|
||||||
|
|
||||||
Set the `info.subject.country` option to true to add the `country` information into the subject.
|
Set the `info.subject.country` option to `true` to add the `country` information into the subject.
|
||||||
The data are taken from the subject part with the `C` key.
|
|
||||||
The escape country info in the subject part will be like :
|
The data is taken from the subject part with the `C` key.
|
||||||
|
|
||||||
|
The escape country info in the subject part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
C=FR,C=US
|
C=FR,C=US
|
||||||
|
@ -506,11 +509,11 @@ C=FR,C=US
|
||||||
|
|
||||||
##### `info.subject.province`
|
##### `info.subject.province`
|
||||||
|
|
||||||
Set the `info.subject.province` option to true to add the `province` information into the subject.
|
Set the `info.subject.province` option to `true` to add the `province` information into the subject.
|
||||||
|
|
||||||
The data are taken from the subject part with the `ST` key.
|
The data is taken from the subject part with the `ST` key.
|
||||||
|
|
||||||
The escape province info in the subject part will be like :
|
The escape province info in the subject part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
ST=Cheese org state,ST=Cheese com state
|
ST=Cheese org state,ST=Cheese com state
|
||||||
|
@ -518,11 +521,11 @@ ST=Cheese org state,ST=Cheese com state
|
||||||
|
|
||||||
##### `info.subject.locality`
|
##### `info.subject.locality`
|
||||||
|
|
||||||
Set the `info.subject.locality` option to true to add the `locality` information into the subject.
|
Set the `info.subject.locality` option to `true` to add the `locality` information into the subject.
|
||||||
|
|
||||||
The data are taken from the subject part with the `L` key.
|
The data is taken from the subject part with the `L` key.
|
||||||
|
|
||||||
The escape locality info in the subject part will be like :
|
The escape locality info in the subject part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
L=TOULOUSE,L=LYON
|
L=TOULOUSE,L=LYON
|
||||||
|
@ -530,11 +533,11 @@ L=TOULOUSE,L=LYON
|
||||||
|
|
||||||
##### `info.subject.organization`
|
##### `info.subject.organization`
|
||||||
|
|
||||||
Set the `info.subject.organization` option to true to add the `organization` information into the subject.
|
Set the `info.subject.organization` option to `true` to add the `organization` information into the subject.
|
||||||
|
|
||||||
The data are taken from the subject part with the `O` key.
|
The data is taken from the subject part with the `O` key.
|
||||||
|
|
||||||
The escape organization info in the subject part will be like :
|
The escape organization info in the subject part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
O=Cheese,O=Cheese 2
|
O=Cheese,O=Cheese 2
|
||||||
|
@ -542,11 +545,11 @@ O=Cheese,O=Cheese 2
|
||||||
|
|
||||||
##### `info.subject.commonName`
|
##### `info.subject.commonName`
|
||||||
|
|
||||||
Set the `info.subject.commonName` option to true to add the `commonName` information into the subject.
|
Set the `info.subject.commonName` option to `true` to add the `commonName` information into the subject.
|
||||||
|
|
||||||
The data are taken from the subject part with the `CN` key.
|
The data is taken from the subject part with the `CN` key.
|
||||||
|
|
||||||
The escape common name info in the subject part will be like :
|
The escape common name info in the subject part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
CN=*.example.com
|
CN=*.example.com
|
||||||
|
@ -554,11 +557,11 @@ CN=*.example.com
|
||||||
|
|
||||||
##### `info.subject.serialNumber`
|
##### `info.subject.serialNumber`
|
||||||
|
|
||||||
Set the `info.subject.serialNumber` option to true to add the `serialNumber` information into the subject.
|
Set the `info.subject.serialNumber` option to `true` to add the `serialNumber` information into the subject.
|
||||||
|
|
||||||
The data are taken from the subject part with the `SN` key.
|
The data is taken from the subject part with the `SN` key.
|
||||||
|
|
||||||
The escape serial number info in the subject part will be like :
|
The escape serial number info in the subject part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
SN=1234567890
|
SN=1234567890
|
||||||
|
@ -566,11 +569,11 @@ SN=1234567890
|
||||||
|
|
||||||
##### `info.subject.domainComponent`
|
##### `info.subject.domainComponent`
|
||||||
|
|
||||||
Set the `info.subject.domainComponent` option to true to add the `domainComponent` information into the subject.
|
Set the `info.subject.domainComponent` option to `true` to add the `domainComponent` information into the subject.
|
||||||
|
|
||||||
The data are taken from the subject part with the `DC` key.
|
The data is taken from the subject part with the `DC` key.
|
||||||
|
|
||||||
The escape domaincomponent info in the subject part will be like :
|
The escape domain component info in the subject part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
DC=org,DC=cheese
|
DC=org,DC=cheese
|
||||||
|
@ -578,9 +581,9 @@ DC=org,DC=cheese
|
||||||
|
|
||||||
#### `info.issuer`
|
#### `info.issuer`
|
||||||
|
|
||||||
The `info.issuer` select the specific client certificate issuer details you want to add to the `X-Forwarded-Tls-Client-Cert-Info` header.
|
The `info.issuer` selects the specific client certificate issuer details you want to add to the `X-Forwarded-Tls-Client-Cert-Info` header.
|
||||||
|
|
||||||
The data are taken from the following certificate part :
|
The data is taken from the following certificate part:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
Issuer: DC=org, DC=cheese, O=Cheese, O=Cheese 2, OU=Simple Signing Section, OU=Simple Signing Section 2, CN=Simple Signing CA, CN=Simple Signing CA 2, C=FR, C=US, L=TOULOUSE, L=LYON, ST=Signing State, ST=Signing State 2/emailAddress=simple@signing.com/emailAddress=simple2@signing.com
|
Issuer: DC=org, DC=cheese, O=Cheese, O=Cheese 2, OU=Simple Signing Section, OU=Simple Signing Section 2, CN=Simple Signing CA, CN=Simple Signing CA 2, C=FR, C=US, L=TOULOUSE, L=LYON, ST=Signing State, ST=Signing State 2/emailAddress=simple@signing.com/emailAddress=simple2@signing.com
|
||||||
|
@ -588,9 +591,11 @@ Issuer: DC=org, DC=cheese, O=Cheese, O=Cheese 2, OU=Simple Signing Section, OU=S
|
||||||
|
|
||||||
##### `info.issuer.country`
|
##### `info.issuer.country`
|
||||||
|
|
||||||
Set the `info.issuer.country` option to true to add the `country` information into the issuer.
|
Set the `info.issuer.country` option to `true` to add the `country` information into the issuer.
|
||||||
The data are taken from the issuer part with the `C` key.
|
|
||||||
The escape country info in the issuer part will be like :
|
The data is taken from the issuer part with the `C` key.
|
||||||
|
|
||||||
|
The escape country info in the issuer part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
C=FR,C=US
|
C=FR,C=US
|
||||||
|
@ -598,11 +603,11 @@ C=FR,C=US
|
||||||
|
|
||||||
##### `info.issuer.province`
|
##### `info.issuer.province`
|
||||||
|
|
||||||
Set the `info.issuer.province` option to true to add the `province` information into the issuer.
|
Set the `info.issuer.province` option to `true` to add the `province` information into the issuer.
|
||||||
|
|
||||||
The data are taken from the issuer part with the `ST` key.
|
The data is taken from the issuer part with the `ST` key.
|
||||||
|
|
||||||
The escape province info in the issuer part will be like :
|
The escape province info in the issuer part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
ST=Signing State,ST=Signing State 2
|
ST=Signing State,ST=Signing State 2
|
||||||
|
@ -610,11 +615,11 @@ ST=Signing State,ST=Signing State 2
|
||||||
|
|
||||||
##### `info.issuer.locality`
|
##### `info.issuer.locality`
|
||||||
|
|
||||||
Set the `info.issuer.locality` option to true to add the `locality` information into the issuer.
|
Set the `info.issuer.locality` option to `true` to add the `locality` information into the issuer.
|
||||||
|
|
||||||
The data are taken from the issuer part with the `L` key.
|
The data is taken from the issuer part with the `L` key.
|
||||||
|
|
||||||
The escape locality info in the issuer part will be like :
|
The escape locality info in the issuer part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
L=TOULOUSE,L=LYON
|
L=TOULOUSE,L=LYON
|
||||||
|
@ -622,11 +627,11 @@ L=TOULOUSE,L=LYON
|
||||||
|
|
||||||
##### `info.issuer.organization`
|
##### `info.issuer.organization`
|
||||||
|
|
||||||
Set the `info.issuer.organization` option to true to add the `organization` information into the issuer.
|
Set the `info.issuer.organization` option to `true` to add the `organization` information into the issuer.
|
||||||
|
|
||||||
The data are taken from the issuer part with the `O` key.
|
The data is taken from the issuer part with the `O` key.
|
||||||
|
|
||||||
The escape organization info in the issuer part will be like :
|
The escape organization info in the issuer part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
O=Cheese,O=Cheese 2
|
O=Cheese,O=Cheese 2
|
||||||
|
@ -634,11 +639,11 @@ O=Cheese,O=Cheese 2
|
||||||
|
|
||||||
##### `info.issuer.commonName`
|
##### `info.issuer.commonName`
|
||||||
|
|
||||||
Set the `info.issuer.commonName` option to true to add the `commonName` information into the issuer.
|
Set the `info.issuer.commonName` option to `true` to add the `commonName` information into the issuer.
|
||||||
|
|
||||||
The data are taken from the issuer part with the `CN` key.
|
The data is taken from the issuer part with the `CN` key.
|
||||||
|
|
||||||
The escape common name info in the issuer part will be like :
|
The escape common name info in the issuer part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
CN=Simple Signing CA 2
|
CN=Simple Signing CA 2
|
||||||
|
@ -646,11 +651,11 @@ CN=Simple Signing CA 2
|
||||||
|
|
||||||
##### `info.issuer.serialNumber`
|
##### `info.issuer.serialNumber`
|
||||||
|
|
||||||
Set the `info.issuer.serialNumber` option to true to add the `serialNumber` information into the issuer.
|
Set the `info.issuer.serialNumber` option to `true` to add the `serialNumber` information into the issuer.
|
||||||
|
|
||||||
The data are taken from the issuer part with the `SN` key.
|
The data is taken from the issuer part with the `SN` key.
|
||||||
|
|
||||||
The escape serial number info in the issuer part will be like :
|
The escape serial number info in the issuer part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
SN=1234567890
|
SN=1234567890
|
||||||
|
@ -658,11 +663,11 @@ SN=1234567890
|
||||||
|
|
||||||
##### `info.issuer.domainComponent`
|
##### `info.issuer.domainComponent`
|
||||||
|
|
||||||
Set the `info.issuer.domainComponent` option to true to add the `domainComponent` information into the issuer.
|
Set the `info.issuer.domainComponent` option to `true` to add the `domainComponent` information into the issuer.
|
||||||
|
|
||||||
The data are taken from the issuer part with the `DC` key.
|
The data is taken from the issuer part with the `DC` key.
|
||||||
|
|
||||||
The escape domain component info in the issuer part will be like :
|
The escape domain component info in the issuer part is formatted as below:
|
||||||
|
|
||||||
```text
|
```text
|
||||||
DC=org,DC=cheese
|
DC=org,DC=cheese
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
To Control the Number of Requests Going to a Service
|
To Control the Number of Requests Going to a Service
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
The RateLimit middleware ensures that services will receive a _fair_ number of requests, and allows one to define what fair is.
|
The RateLimit middleware ensures that services will receive a _fair_ amount of requests, and allows one to define what fair is.
|
||||||
|
|
||||||
## Configuration Example
|
## Configuration Example
|
||||||
|
|
||||||
|
@ -74,7 +74,7 @@ http:
|
||||||
|
|
||||||
### `average`
|
### `average`
|
||||||
|
|
||||||
`average` is the maximum rate, by default in requests by second, allowed for the given source.
|
`average` is the maximum rate, by default in requests per second, allowed from a given source.
|
||||||
|
|
||||||
It defaults to `0`, which means no rate limiting.
|
It defaults to `0`, which means no rate limiting.
|
||||||
|
|
||||||
|
@ -249,24 +249,24 @@ http:
|
||||||
|
|
||||||
### `sourceCriterion`
|
### `sourceCriterion`
|
||||||
|
|
||||||
SourceCriterion defines what criterion is used to group requests as originating from a common source.
|
The `sourceCriterion` option defines what criterion is used to group requests as originating from a common source.
|
||||||
The precedence order is `ipStrategy`, then `requestHeaderName`, then `requestHost`.
|
The precedence order is `ipStrategy`, then `requestHeaderName`, then `requestHost`.
|
||||||
If none are set, the default is to use the request's remote address field (as an `ipStrategy`).
|
If none are set, the default is to use the request's remote address field (as an `ipStrategy`).
|
||||||
|
|
||||||
#### `sourceCriterion.ipStrategy`
|
#### `sourceCriterion.ipStrategy`
|
||||||
|
|
||||||
The `ipStrategy` option defines two parameters that sets how Traefik will determine the client IP: `depth`, and `excludedIPs`.
|
The `ipStrategy` option defines two parameters that configures how Traefik determines the client IP: `depth`, and `excludedIPs`.
|
||||||
|
|
||||||
##### `ipStrategy.depth`
|
##### `ipStrategy.depth`
|
||||||
|
|
||||||
The `depth` option tells Traefik to use the `X-Forwarded-For` header and take the IP located at the `depth` position (starting from the right).
|
The `depth` option tells Traefik to use the `X-Forwarded-For` header and select the IP located at the `depth` position (starting from the right).
|
||||||
|
|
||||||
- If `depth` is greater than the total number of IPs in `X-Forwarded-For`, then the client IP will be empty.
|
- If `depth` is greater than the total number of IPs in `X-Forwarded-For`, then the client IP is empty.
|
||||||
- `depth` is ignored if its value is lesser than or equal to 0.
|
- `depth` is ignored if its value is less than or equal to 0.
|
||||||
|
|
||||||
!!! example "Example of Depth & X-Forwarded-For"
|
!!! example "Example of Depth & X-Forwarded-For"
|
||||||
|
|
||||||
If `depth` was equal to 2, and the request `X-Forwarded-For` header was `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` then the "real" client IP would be `"10.0.0.1"` (at depth 4) but the IP used as the criterion would be `"12.0.0.1"` (`depth=2`).
|
If `depth` is set to 2, and the request `X-Forwarded-For` header is `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` then the "real" client IP is `"10.0.0.1"` (at depth 4) but the IP used as the criterion is `"12.0.0.1"` (`depth=2`).
|
||||||
|
|
||||||
| `X-Forwarded-For` | `depth` | clientIP |
|
| `X-Forwarded-For` | `depth` | clientIP |
|
||||||
|-----------------------------------------|---------|--------------|
|
|-----------------------------------------|---------|--------------|
|
||||||
|
@ -274,8 +274,71 @@ The `depth` option tells Traefik to use the `X-Forwarded-For` header and take th
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `3` | `"11.0.0.1"` |
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `3` | `"11.0.0.1"` |
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `5` | `""` |
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `5` | `""` |
|
||||||
|
|
||||||
|
```yaml tab="Docker"
|
||||||
|
labels:
|
||||||
|
- "traefik.http.middlewares.test-ratelimit.ratelimit.sourcecriterion.ipstrategy.depth=2"
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml tab="Kubernetes"
|
||||||
|
apiVersion: traefik.containo.us/v1alpha1
|
||||||
|
kind: Middleware
|
||||||
|
metadata:
|
||||||
|
name: test-ratelimit
|
||||||
|
spec:
|
||||||
|
rateLimit:
|
||||||
|
sourceCriterion:
|
||||||
|
ipStrategy:
|
||||||
|
depth: 2
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml tab="Consul Catalog"
|
||||||
|
- "traefik.http.middlewares.test-ratelimit.ratelimit.sourcecriterion.ipstrategy.depth=2"
|
||||||
|
```
|
||||||
|
|
||||||
|
```json tab="Marathon"
|
||||||
|
"labels": {
|
||||||
|
"traefik.http.middlewares.test-ratelimit.ratelimit.sourcecriterion.ipstrategy.depth": "2"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml tab="Rancher"
|
||||||
|
labels:
|
||||||
|
- "traefik.http.middlewares.test-ratelimit.ratelimit.sourcecriterion.ipstrategy.depth=2"
|
||||||
|
```
|
||||||
|
|
||||||
|
```toml tab="File (TOML)"
|
||||||
|
[http.middlewares]
|
||||||
|
[http.middlewares.test-ratelimit.rateLimit]
|
||||||
|
[http.middlewares.test-ratelimit.rateLimit.sourceCriterion.ipStrategy]
|
||||||
|
depth = 2
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml tab="File (YAML)"
|
||||||
|
http:
|
||||||
|
middlewares:
|
||||||
|
test-ratelimit:
|
||||||
|
rateLimit:
|
||||||
|
sourceCriterion:
|
||||||
|
ipStrategy:
|
||||||
|
depth: 2
|
||||||
|
```
|
||||||
|
|
||||||
##### `ipStrategy.excludedIPs`
|
##### `ipStrategy.excludedIPs`
|
||||||
|
|
||||||
|
`excludedIPs` configures Traefik to scan the `X-Forwarded-For` header and select the first IP not in the list.
|
||||||
|
|
||||||
|
!!! important "If `depth` is specified, `excludedIPs` is ignored."
|
||||||
|
|
||||||
|
!!! example "Example of ExcludedIPs & X-Forwarded-For"
|
||||||
|
|
||||||
|
| `X-Forwarded-For` | `excludedIPs` | clientIP |
|
||||||
|
|-----------------------------------------|-----------------------|--------------|
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"12.0.0.1,13.0.0.1"` | `"11.0.0.1"` |
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"15.0.0.1,13.0.0.1"` | `"12.0.0.1"` |
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"10.0.0.1,13.0.0.1"` | `"12.0.0.1"` |
|
||||||
|
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"15.0.0.1,16.0.0.1"` | `"13.0.0.1"` |
|
||||||
|
| `"10.0.0.1,11.0.0.1"` | `"10.0.0.1,11.0.0.1"` | `""` |
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
- "traefik.http.middlewares.test-ratelimit.ratelimit.sourcecriterion.ipstrategy.excludedips=127.0.0.1/32, 192.168.1.7"
|
- "traefik.http.middlewares.test-ratelimit.ratelimit.sourcecriterion.ipstrategy.excludedips=127.0.0.1/32, 192.168.1.7"
|
||||||
|
@ -329,23 +392,9 @@ http:
|
||||||
- "192.168.1.7"
|
- "192.168.1.7"
|
||||||
```
|
```
|
||||||
|
|
||||||
`excludedIPs` tells Traefik to scan the `X-Forwarded-For` header and pick the first IP not in the list.
|
|
||||||
|
|
||||||
!!! important "If `depth` is specified, `excludedIPs` is ignored."
|
|
||||||
|
|
||||||
!!! example "Example of ExcludedIPs & X-Forwarded-For"
|
|
||||||
|
|
||||||
| `X-Forwarded-For` | `excludedIPs` | clientIP |
|
|
||||||
|-----------------------------------------|-----------------------|--------------|
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"12.0.0.1,13.0.0.1"` | `"11.0.0.1"` |
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"15.0.0.1,13.0.0.1"` | `"12.0.0.1"` |
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"10.0.0.1,13.0.0.1"` | `"12.0.0.1"` |
|
|
||||||
| `"10.0.0.1,11.0.0.1,12.0.0.1,13.0.0.1"` | `"15.0.0.1,16.0.0.1"` | `"13.0.0.1"` |
|
|
||||||
| `"10.0.0.1,11.0.0.1"` | `"10.0.0.1,11.0.0.1"` | `""` |
|
|
||||||
|
|
||||||
#### `sourceCriterion.requestHeaderName`
|
#### `sourceCriterion.requestHeaderName`
|
||||||
|
|
||||||
Requests having the same value for the given header are grouped as coming from the same source.
|
Name of the header used to group incoming requests.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
|
|
|
@ -7,7 +7,7 @@ Redirecting the Client to a Different Location
|
||||||
TODO: add schema
|
TODO: add schema
|
||||||
-->
|
-->
|
||||||
|
|
||||||
RegexRedirect redirect a request from an url to another with regex matching and replacement.
|
The RedirectRegex redirects a request using regex matching and replacement.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
@ -73,6 +73,10 @@ http:
|
||||||
|
|
||||||
## Configuration Options
|
## Configuration Options
|
||||||
|
|
||||||
|
!!! tip
|
||||||
|
|
||||||
|
Regular expressions and replacements can be tested using online tools such as [Go Playground](https://play.golang.org/p/mWU9p-wk2ru) or the [Regex101](https://regex101.com/r/58sIgx/2).
|
||||||
|
|
||||||
### `permanent`
|
### `permanent`
|
||||||
|
|
||||||
Set the `permanent` option to `true` to apply a permanent redirection.
|
Set the `permanent` option to `true` to apply a permanent redirection.
|
||||||
|
@ -81,14 +85,10 @@ Set the `permanent` option to `true` to apply a permanent redirection.
|
||||||
|
|
||||||
The `regex` option is the regular expression to match and capture elements from the request URL.
|
The `regex` option is the regular expression to match and capture elements from the request URL.
|
||||||
|
|
||||||
!!! warning
|
|
||||||
|
|
||||||
Care should be taken when defining replacement expand variables: `$1x` is equivalent to `${1x}`, not `${1}x` (see [Regexp.Expand](https://golang.org/pkg/regexp/#Regexp.Expand)), so use `${1}` syntax.
|
|
||||||
|
|
||||||
!!! tip
|
|
||||||
|
|
||||||
Regular expressions and replacements can be tested using online tools such as [Go Playground](https://play.golang.org/p/mWU9p-wk2ru) or the [Regex101](https://regex101.com/r/58sIgx/2).
|
|
||||||
|
|
||||||
### `replacement`
|
### `replacement`
|
||||||
|
|
||||||
The `replacement` option defines how to modify the URL to have the new target URL.
|
The `replacement` option defines how to modify the URL to have the new target URL.
|
||||||
|
|
||||||
|
!!! warning
|
||||||
|
|
||||||
|
Care should be taken when defining replacement expand variables: `$1x` is equivalent to `${1x}`, not `${1}x` (see [Regexp.Expand](https://golang.org/pkg/regexp/#Regexp.Expand)), so use `${1}` syntax.
|
||||||
|
|
|
@ -7,7 +7,7 @@ Redirecting the Client to a Different Scheme/Port
|
||||||
TODO: add schema
|
TODO: add schema
|
||||||
-->
|
-->
|
||||||
|
|
||||||
RedirectScheme redirect request from a scheme to another.
|
RedirectScheme redirects requests from a scheme/port to another.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
@ -135,7 +135,7 @@ http:
|
||||||
|
|
||||||
### `scheme`
|
### `scheme`
|
||||||
|
|
||||||
The `scheme` option defines the scheme of the new url.
|
The `scheme` option defines the scheme of the new URL.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
# Redirect to https
|
# Redirect to https
|
||||||
|
@ -190,7 +190,7 @@ http:
|
||||||
|
|
||||||
### `port`
|
### `port`
|
||||||
|
|
||||||
The `port` option defines the port of the new url.
|
The `port` option defines the port of the new URL.
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
# Redirect to https
|
# Redirect to https
|
||||||
|
|
|
@ -7,18 +7,18 @@ Updating the Path Before Forwarding the Request
|
||||||
TODO: add schema
|
TODO: add schema
|
||||||
-->
|
-->
|
||||||
|
|
||||||
Replace the path of the request url.
|
Replace the path of the request URL.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
# Replace the path by /foo
|
# Replace the path with /foo
|
||||||
labels:
|
labels:
|
||||||
- "traefik.http.middlewares.test-replacepath.replacepath.path=/foo"
|
- "traefik.http.middlewares.test-replacepath.replacepath.path=/foo"
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="Kubernetes"
|
```yaml tab="Kubernetes"
|
||||||
# Replace the path by /foo
|
# Replace the path with /foo
|
||||||
apiVersion: traefik.containo.us/v1alpha1
|
apiVersion: traefik.containo.us/v1alpha1
|
||||||
kind: Middleware
|
kind: Middleware
|
||||||
metadata:
|
metadata:
|
||||||
|
@ -29,7 +29,7 @@ spec:
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="Consul Catalog"
|
```yaml tab="Consul Catalog"
|
||||||
# Replace the path by /foo
|
# Replace the path with /foo
|
||||||
- "traefik.http.middlewares.test-replacepath.replacepath.path=/foo"
|
- "traefik.http.middlewares.test-replacepath.replacepath.path=/foo"
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -40,20 +40,20 @@ spec:
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="Rancher"
|
```yaml tab="Rancher"
|
||||||
# Replace the path by /foo
|
# Replace the path with /foo
|
||||||
labels:
|
labels:
|
||||||
- "traefik.http.middlewares.test-replacepath.replacepath.path=/foo"
|
- "traefik.http.middlewares.test-replacepath.replacepath.path=/foo"
|
||||||
```
|
```
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
# Replace the path by /foo
|
# Replace the path with /foo
|
||||||
[http.middlewares]
|
[http.middlewares]
|
||||||
[http.middlewares.test-replacepath.replacePath]
|
[http.middlewares.test-replacepath.replacePath]
|
||||||
path = "/foo"
|
path = "/foo"
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
```yaml tab="File (YAML)"
|
||||||
# Replace the path by /foo
|
# Replace the path with /foo
|
||||||
http:
|
http:
|
||||||
middlewares:
|
middlewares:
|
||||||
test-replacepath:
|
test-replacepath:
|
||||||
|
@ -67,9 +67,9 @@ http:
|
||||||
|
|
||||||
The ReplacePath middleware will:
|
The ReplacePath middleware will:
|
||||||
|
|
||||||
- replace the actual path by the specified one.
|
- replace the actual path with the specified one.
|
||||||
- store the original path in a `X-Replaced-Path` header.
|
- store the original path in a `X-Replaced-Path` header.
|
||||||
|
|
||||||
### `path`
|
### `path`
|
||||||
|
|
||||||
The `path` option defines the path to use as replacement in the request url.
|
The `path` option defines the path to use as replacement in the request URL.
|
||||||
|
|
|
@ -7,7 +7,7 @@ Updating the Path Before Forwarding the Request (Using a Regex)
|
||||||
TODO: add schema
|
TODO: add schema
|
||||||
-->
|
-->
|
||||||
|
|
||||||
The ReplaceRegex replace a path from an url to another with regex matching and replacement.
|
The ReplaceRegex replaces the path of a URL using regex matching and replacement.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
@ -51,7 +51,7 @@ labels:
|
||||||
```
|
```
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
# Redirect with domain replacement
|
# Replace path with regex
|
||||||
[http.middlewares]
|
[http.middlewares]
|
||||||
[http.middlewares.test-replacepathregex.replacePathRegex]
|
[http.middlewares.test-replacepathregex.replacePathRegex]
|
||||||
regex = "^/foo/(.*)"
|
regex = "^/foo/(.*)"
|
||||||
|
@ -59,7 +59,7 @@ labels:
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
```yaml tab="File (YAML)"
|
||||||
# Redirect with domain replacement
|
# Replace path with regex
|
||||||
http:
|
http:
|
||||||
middlewares:
|
middlewares:
|
||||||
test-replacepathregex:
|
test-replacepathregex:
|
||||||
|
@ -74,21 +74,21 @@ http:
|
||||||
|
|
||||||
The ReplacePathRegex middleware will:
|
The ReplacePathRegex middleware will:
|
||||||
|
|
||||||
- replace the matching path by the specified one.
|
- replace the matching path with the specified one.
|
||||||
- store the original path in a `X-Replaced-Path` header.
|
- store the original path in a `X-Replaced-Path` header.
|
||||||
|
|
||||||
|
!!! tip
|
||||||
|
|
||||||
|
Regular expressions and replacements can be tested using online tools such as [Go Playground](https://play.golang.org/p/mWU9p-wk2ru) or [Regex101](https://regex101.com/r/58sIgx/2).
|
||||||
|
|
||||||
### `regex`
|
### `regex`
|
||||||
|
|
||||||
The `regex` option is the regular expression to match and capture the path from the request URL.
|
The `regex` option is the regular expression to match and capture the path from the request URL.
|
||||||
|
|
||||||
|
### `replacement`
|
||||||
|
|
||||||
|
The `replacement` option defines the replacement path format, which can include captured variables.
|
||||||
|
|
||||||
!!! warning
|
!!! warning
|
||||||
|
|
||||||
Care should be taken when defining replacement expand variables: `$1x` is equivalent to `${1x}`, not `${1}x` (see [Regexp.Expand](https://golang.org/pkg/regexp/#Regexp.Expand)), so use `${1}` syntax.
|
Care should be taken when defining replacement expand variables: `$1x` is equivalent to `${1x}`, not `${1}x` (see [Regexp.Expand](https://golang.org/pkg/regexp/#Regexp.Expand)), so use `${1}` syntax.
|
||||||
|
|
||||||
!!! tip
|
|
||||||
|
|
||||||
Regular expressions and replacements can be tested using online tools such as [Go Playground](https://play.golang.org/p/mWU9p-wk2ru) or the [Regex101](https://regex101.com/r/58sIgx/2).
|
|
||||||
|
|
||||||
### `replacement`
|
|
||||||
|
|
||||||
The `replacement` option defines how to modify the path to have the new target path.
|
|
||||||
|
|
|
@ -7,21 +7,21 @@ Retrying until it Succeeds
|
||||||
TODO: add schema
|
TODO: add schema
|
||||||
-->
|
-->
|
||||||
|
|
||||||
The Retry middleware is in charge of reissuing a request a given number of times to a backend server if that server does not reply.
|
The Retry middleware reissues requests a given number of times to a backend server if that server does not reply.
|
||||||
To be clear, as soon as the server answers, the middleware stops retrying, regardless of the response status.
|
As soon as the server answers, the middleware stops retrying, regardless of the response status.
|
||||||
The Retry middleware has an optional configuration for exponential backoff.
|
The Retry middleware has an optional configuration to enable an exponential backoff.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
# Retry to send request 4 times with exponential backoff
|
# Retry 4 times with exponential backoff
|
||||||
labels:
|
labels:
|
||||||
- "traefik.http.middlewares.test-retry.retry.attempts=4"
|
- "traefik.http.middlewares.test-retry.retry.attempts=4"
|
||||||
- "traefik.http.middlewares.test-retry.retry.initialinterval=100ms"
|
- "traefik.http.middlewares.test-retry.retry.initialinterval=100ms"
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="Kubernetes"
|
```yaml tab="Kubernetes"
|
||||||
# Retry to send request 4 times with exponential backoff
|
# Retry 4 times with exponential backoff
|
||||||
apiVersion: traefik.containo.us/v1alpha1
|
apiVersion: traefik.containo.us/v1alpha1
|
||||||
kind: Middleware
|
kind: Middleware
|
||||||
metadata:
|
metadata:
|
||||||
|
@ -33,7 +33,7 @@ spec:
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="Consul Catalog"
|
```yaml tab="Consul Catalog"
|
||||||
# Retry to send request 4 times with exponential backoff
|
# Retry 4 times with exponential backoff
|
||||||
- "traefik.http.middlewares.test-retry.retry.attempts=4"
|
- "traefik.http.middlewares.test-retry.retry.attempts=4"
|
||||||
- "traefik.http.middlewares.test-retry.retry.initialinterval=100ms"
|
- "traefik.http.middlewares.test-retry.retry.initialinterval=100ms"
|
||||||
```
|
```
|
||||||
|
@ -46,14 +46,14 @@ spec:
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="Rancher"
|
```yaml tab="Rancher"
|
||||||
# Retry to send request 4 times with exponential backoff
|
# Retry 4 times with exponential backoff
|
||||||
labels:
|
labels:
|
||||||
- "traefik.http.middlewares.test-retry.retry.attempts=4"
|
- "traefik.http.middlewares.test-retry.retry.attempts=4"
|
||||||
- "traefik.http.middlewares.test-retry.retry.initialinterval=100ms"
|
- "traefik.http.middlewares.test-retry.retry.initialinterval=100ms"
|
||||||
```
|
```
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
# Retry to send request 4 times
|
# Retry 4 times with exponential backoff
|
||||||
[http.middlewares]
|
[http.middlewares]
|
||||||
[http.middlewares.test-retry.retry]
|
[http.middlewares.test-retry.retry]
|
||||||
attempts = 4
|
attempts = 4
|
||||||
|
@ -61,7 +61,7 @@ labels:
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
```yaml tab="File (YAML)"
|
||||||
# Retry to send request 4 times with exponential backoff
|
# Retry 4 times with exponential backoff
|
||||||
http:
|
http:
|
||||||
middlewares:
|
middlewares:
|
||||||
test-retry:
|
test-retry:
|
||||||
|
@ -80,4 +80,7 @@ The `attempts` option defines how many times the request should be retried.
|
||||||
|
|
||||||
### `initialInterval`
|
### `initialInterval`
|
||||||
|
|
||||||
The `initialInterval` option defines the first wait time in the exponential backoff series (provided in seconds or as a valid duration format, see [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration)). The maximum interval is calculated as twice the `initialInterval`. If unspecified, requests will be retried immediately.
|
The `initialInterval` option defines the first wait time in the exponential backoff series. The maximum interval is
|
||||||
|
calculated as twice the `initialInterval`. If unspecified, requests will be retried immediately.
|
||||||
|
|
||||||
|
The value of initialInterval should be provided in seconds or as a valid duration format, see [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration).
|
||||||
|
|
|
@ -69,32 +69,55 @@ http:
|
||||||
|
|
||||||
### General
|
### General
|
||||||
|
|
||||||
The StripPrefix middleware will:
|
The StripPrefix middleware strips the matching path prefix and stores it in a `X-Forwarded-Prefix` header.
|
||||||
|
|
||||||
- strip the matching path prefix.
|
|
||||||
- store the matching path prefix in a `X-Forwarded-Prefix` header.
|
|
||||||
|
|
||||||
!!! tip
|
!!! tip
|
||||||
|
|
||||||
Use a `StripPrefix` middleware if your backend listens on the root path (`/`) but should be routeable on a specific prefix.
|
Use a `StripPrefix` middleware if your backend listens on the root path (`/`) but should be exposed on a specific prefix.
|
||||||
|
|
||||||
### `prefixes`
|
### `prefixes`
|
||||||
|
|
||||||
The `prefixes` option defines the prefixes to strip from the request URL.
|
The `prefixes` option defines the prefixes to strip from the request URL.
|
||||||
|
|
||||||
For instance, `/products` would match `/products` but also `/products/shoes` and `/products/shirts`.
|
For instance, `/products` also matches `/products/shoes` and `/products/shirts`.
|
||||||
|
|
||||||
Since the path is stripped prior to forwarding, your backend is expected to listen on `/`.
|
If your backend is serving assets (e.g., images or JavaScript files), it can use the `X-Forwarded-Prefix` header to properly construct relative URLs.
|
||||||
|
Using the previous example, the backend should return `/products/shoes/image.png` (and not `/images.png`, which Traefik would likely not be able to associate with the same backend).
|
||||||
If your backend is serving assets (e.g., images or Javascript files), chances are it must return properly constructed relative URLs.
|
|
||||||
Continuing on the example, the backend should return `/products/shoes/image.png` (and not `/images.png` which Traefik would likely not be able to associate with the same backend).
|
|
||||||
|
|
||||||
The `X-Forwarded-Prefix` header can be queried to build such URLs dynamically.
|
|
||||||
|
|
||||||
### `forceSlash`
|
### `forceSlash`
|
||||||
|
|
||||||
_Optional, Default=true_
|
_Optional, Default=true_
|
||||||
|
|
||||||
|
The `forceSlash` option ensures the resulting stripped path is not the empty string, by replacing it with `/` when necessary.
|
||||||
|
|
||||||
|
This option was added to keep the initial (non-intuitive) behavior of this middleware, in order to avoid introducing a breaking change.
|
||||||
|
|
||||||
|
It is recommended to explicitly set `forceSlash` to `false`.
|
||||||
|
|
||||||
|
??? info "Behavior examples"
|
||||||
|
|
||||||
|
- `forceSlash=true`
|
||||||
|
|
||||||
|
| Path | Prefix to strip | Result |
|
||||||
|
|------------|-----------------|--------|
|
||||||
|
| `/` | `/` | `/` |
|
||||||
|
| `/foo` | `/foo` | `/` |
|
||||||
|
| `/foo/` | `/foo` | `/` |
|
||||||
|
| `/foo/` | `/foo/` | `/` |
|
||||||
|
| `/bar` | `/foo` | `/bar` |
|
||||||
|
| `/foo/bar` | `/foo` | `/bar` |
|
||||||
|
|
||||||
|
- `forceSlash=false`
|
||||||
|
|
||||||
|
| Path | Prefix to strip | Result |
|
||||||
|
|------------|-----------------|--------|
|
||||||
|
| `/` | `/` | empty |
|
||||||
|
| `/foo` | `/foo` | empty |
|
||||||
|
| `/foo/` | `/foo` | `/` |
|
||||||
|
| `/foo/` | `/foo/` | empty |
|
||||||
|
| `/bar` | `/foo` | `/bar` |
|
||||||
|
| `/foo/bar` | `/foo` | `/bar` |
|
||||||
|
|
||||||
```yaml tab="Docker"
|
```yaml tab="Docker"
|
||||||
labels:
|
labels:
|
||||||
- "traefik.http.middlewares.example.stripprefix.prefixes=/foobar"
|
- "traefik.http.middlewares.example.stripprefix.prefixes=/foobar"
|
||||||
|
@ -142,33 +165,3 @@ http:
|
||||||
- "/foobar"
|
- "/foobar"
|
||||||
forceSlash: false
|
forceSlash: false
|
||||||
```
|
```
|
||||||
|
|
||||||
The `forceSlash` option makes sure that the resulting stripped path is not the empty string, by replacing it with `/` when necessary.
|
|
||||||
|
|
||||||
This option was added to keep the initial (non-intuitive) behavior of this middleware, in order to avoid introducing a breaking change.
|
|
||||||
|
|
||||||
It's recommended to explicitly set `forceSlash` to `false`.
|
|
||||||
|
|
||||||
??? info "Behavior examples"
|
|
||||||
|
|
||||||
- `forceSlash=true`
|
|
||||||
|
|
||||||
| Path | Prefix to strip | Result |
|
|
||||||
|------------|-----------------|--------|
|
|
||||||
| `/` | `/` | `/` |
|
|
||||||
| `/foo` | `/foo` | `/` |
|
|
||||||
| `/foo/` | `/foo` | `/` |
|
|
||||||
| `/foo/` | `/foo/` | `/` |
|
|
||||||
| `/bar` | `/foo` | `/bar` |
|
|
||||||
| `/foo/bar` | `/foo` | `/bar` |
|
|
||||||
|
|
||||||
- `forceSlash=false`
|
|
||||||
|
|
||||||
| Path | Prefix to strip | Result |
|
|
||||||
|------------|-----------------|--------|
|
|
||||||
| `/` | `/` | empty |
|
|
||||||
| `/foo` | `/foo` | empty |
|
|
||||||
| `/foo/` | `/foo` | `/` |
|
|
||||||
| `/foo/` | `/foo/` | empty |
|
|
||||||
| `/bar` | `/foo` | `/bar` |
|
|
||||||
| `/foo/bar` | `/foo` | `/bar` |
|
|
||||||
|
|
|
@ -57,14 +57,11 @@ http:
|
||||||
|
|
||||||
### General
|
### General
|
||||||
|
|
||||||
The StripPrefixRegex middleware will:
|
The StripPrefixRegex middleware strips the matching path prefix and stores it in a `X-Forwarded-Prefix` header.
|
||||||
|
|
||||||
- strip the matching path prefix.
|
|
||||||
- store the matching path prefix in a `X-Forwarded-Prefix` header.
|
|
||||||
|
|
||||||
!!! tip
|
!!! tip
|
||||||
|
|
||||||
Use a `stripPrefixRegex` middleware if your backend listens on the root path (`/`) but should be routeable on a specific prefix.
|
Use a `stripPrefixRegex` middleware if your backend listens on the root path (`/`) but should be exposed on a specific prefix.
|
||||||
|
|
||||||
### `regex`
|
### `regex`
|
||||||
|
|
||||||
|
@ -74,12 +71,7 @@ The `regex` option is the regular expression to match the path prefix from the r
|
||||||
|
|
||||||
Regular expressions can be tested using online tools such as [Go Playground](https://play.golang.org/p/mWU9p-wk2ru) or the [Regex101](https://regex101.com/r/58sIgx/2).
|
Regular expressions can be tested using online tools such as [Go Playground](https://play.golang.org/p/mWU9p-wk2ru) or the [Regex101](https://regex101.com/r/58sIgx/2).
|
||||||
|
|
||||||
For instance, `/products` would match `/products` but also `/products/shoes` and `/products/shirts`.
|
For instance, `/products` also matches `/products/shoes` and `/products/shirts`.
|
||||||
|
|
||||||
Since the path is stripped prior to forwarding, your backend is expected to listen on `/`.
|
If your backend is serving assets (e.g., images or JavaScript files), it can use the `X-Forwarded-Prefix` header to properly construct relative URLs.
|
||||||
|
Using the previous example, the backend should return `/products/shoes/image.png` (and not `/images.png`, which Traefik would likely not be able to associate with the same backend).
|
||||||
If your backend is serving assets (e.g., images or Javascript files), chances are it must return properly constructed relative URLs.
|
|
||||||
|
|
||||||
Continuing on the example, the backend should return `/products/shoes/image.png` (and not `/images.png` which Traefik would likely not be able to associate with the same backend).
|
|
||||||
|
|
||||||
The `X-Forwarded-Prefix` header can be queried to build such URLs dynamically.
|
|
||||||
|
|
|
@ -42,6 +42,8 @@ See the dedicated section in [routing](../routing/providers/consul-catalog.md).
|
||||||
|
|
||||||
_Optional, Default=15s_
|
_Optional, Default=15s_
|
||||||
|
|
||||||
|
Defines the polling interval.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
refreshInterval = "30s"
|
refreshInterval = "30s"
|
||||||
|
@ -60,12 +62,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Defines the polling interval.
|
|
||||||
|
|
||||||
### `prefix`
|
### `prefix`
|
||||||
|
|
||||||
_required, Default="traefik"_
|
_required, Default="traefik"_
|
||||||
|
|
||||||
|
The prefix for Consul Catalog tags defining Traefik labels.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
prefix = "test"
|
prefix = "test"
|
||||||
|
@ -84,12 +86,18 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
The prefix for Consul Catalog tags defining traefik labels.
|
|
||||||
|
|
||||||
### `requireConsistent`
|
### `requireConsistent`
|
||||||
|
|
||||||
_Optional, Default=false_
|
_Optional, Default=false_
|
||||||
|
|
||||||
|
Forces the read to be fully consistent.
|
||||||
|
|
||||||
|
!!! note ""
|
||||||
|
|
||||||
|
It is more expensive due to an extra round-trip but prevents ever performing a stale read.
|
||||||
|
|
||||||
|
For more information, see the consul [documentation on consistency](https://www.consul.io/api-docs/features/consistency).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
requireConsistent = true
|
requireConsistent = true
|
||||||
|
@ -108,12 +116,18 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Forces the read to be fully consistent.
|
|
||||||
|
|
||||||
### `stale`
|
### `stale`
|
||||||
|
|
||||||
_Optional, Default=false_
|
_Optional, Default=false_
|
||||||
|
|
||||||
|
Use stale consistency for catalog reads.
|
||||||
|
|
||||||
|
!!! note ""
|
||||||
|
|
||||||
|
This makes reads very fast and scalable at the cost of a higher likelihood of stale values.
|
||||||
|
|
||||||
|
For more information, see the consul [documentation on consistency](https://www.consul.io/api-docs/features/consistency).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
stale = true
|
stale = true
|
||||||
|
@ -132,12 +146,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Use stale consistency for catalog reads.
|
|
||||||
|
|
||||||
### `cache`
|
### `cache`
|
||||||
|
|
||||||
_Optional, Default=false_
|
_Optional, Default=false_
|
||||||
|
|
||||||
|
Use local agent caching for catalog reads.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
cache = true
|
cache = true
|
||||||
|
@ -156,14 +170,14 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Use local agent caching for catalog reads.
|
|
||||||
|
|
||||||
### `endpoint`
|
### `endpoint`
|
||||||
|
|
||||||
Defines the Consul server endpoint.
|
Defines the Consul server endpoint.
|
||||||
|
|
||||||
#### `address`
|
#### `address`
|
||||||
|
|
||||||
|
Defines the address of the Consul server.
|
||||||
|
|
||||||
_Optional, Default="127.0.0.1:8500"_
|
_Optional, Default="127.0.0.1:8500"_
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
|
@ -186,12 +200,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Defines the address of the Consul server.
|
|
||||||
|
|
||||||
#### `scheme`
|
#### `scheme`
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Defines the URI scheme for the Consul server.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
[providers.consulCatalog.endpoint]
|
[providers.consulCatalog.endpoint]
|
||||||
|
@ -212,12 +226,13 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Defines the URI scheme for the Consul server.
|
|
||||||
|
|
||||||
#### `datacenter`
|
#### `datacenter`
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Defines the datacenter to use.
|
||||||
|
If not provided in Traefik, Consul uses the default agent datacenter.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
[providers.consulCatalog.endpoint]
|
[providers.consulCatalog.endpoint]
|
||||||
|
@ -238,13 +253,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Defines the Data center to use.
|
|
||||||
If not provided, the default agent data center is used.
|
|
||||||
|
|
||||||
#### `token`
|
#### `token`
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Token is used to provide a per-request ACL token which overwrites the agent's default token.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
[providers.consulCatalog.endpoint]
|
[providers.consulCatalog.endpoint]
|
||||||
|
@ -265,12 +279,13 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Token is used to provide a per-request ACL token which overrides the agent's default token.
|
|
||||||
|
|
||||||
#### `endpointWaitTime`
|
#### `endpointWaitTime`
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Limits the duration for which a Watch can block.
|
||||||
|
If not provided, the agent default values will be used.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
[providers.consulCatalog.endpoint]
|
[providers.consulCatalog.endpoint]
|
||||||
|
@ -291,18 +306,17 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
WaitTime limits how long a Watch will block.
|
|
||||||
If not provided, the agent default values will be used
|
|
||||||
|
|
||||||
#### `httpAuth`
|
#### `httpAuth`
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
|
||||||
Used to authenticate http client with HTTP Basic Authentication.
|
Used to authenticate the HTTP client using HTTP Basic Authentication.
|
||||||
|
|
||||||
##### `username`
|
##### `username`
|
||||||
|
|
||||||
_Optional_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Username to use for HTTP Basic Authentication.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog.endpoint.httpAuth]
|
[providers.consulCatalog.endpoint.httpAuth]
|
||||||
|
@ -321,11 +335,11 @@ providers:
|
||||||
--providers.consulcatalog.endpoint.httpauth.username=test
|
--providers.consulcatalog.endpoint.httpauth.username=test
|
||||||
```
|
```
|
||||||
|
|
||||||
Username to use for HTTP Basic Authentication
|
|
||||||
|
|
||||||
##### `password`
|
##### `password`
|
||||||
|
|
||||||
_Optional_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Password to use for HTTP Basic Authentication.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog.endpoint.httpAuth]
|
[providers.consulCatalog.endpoint.httpAuth]
|
||||||
|
@ -344,8 +358,6 @@ providers:
|
||||||
--providers.consulcatalog.endpoint.httpauth.password=test
|
--providers.consulcatalog.endpoint.httpauth.password=test
|
||||||
```
|
```
|
||||||
|
|
||||||
Password to use for HTTP Basic Authentication
|
|
||||||
|
|
||||||
#### `tls`
|
#### `tls`
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
@ -356,6 +368,8 @@ Defines TLS options for Consul server endpoint.
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
|
||||||
|
`ca` is the path to the CA certificate used for Consul communication, defaults to the system bundle if not specified.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog.endpoint.tls]
|
[providers.consulCatalog.endpoint.tls]
|
||||||
ca = "path/to/ca.crt"
|
ca = "path/to/ca.crt"
|
||||||
|
@ -373,12 +387,20 @@ providers:
|
||||||
--providers.consulcatalog.endpoint.tls.ca=path/to/ca.crt
|
--providers.consulcatalog.endpoint.tls.ca=path/to/ca.crt
|
||||||
```
|
```
|
||||||
|
|
||||||
`ca` is the path to the CA certificate used for Consul communication, defaults to the system bundle if not specified.
|
|
||||||
|
|
||||||
##### `caOptional`
|
##### `caOptional`
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
|
||||||
|
The value of `tls.caOptional` defines which policy should be used for the secure connection with TLS Client Authentication to Consul.
|
||||||
|
|
||||||
|
!!! warning ""
|
||||||
|
|
||||||
|
If `tls.ca` is undefined, this option will be ignored, and no client certificate will be requested during the handshake. Any provided certificate will thus never be verified.
|
||||||
|
|
||||||
|
When this option is set to `true`, a client certificate is requested during the handshake but is not required. If a certificate is sent, it is required to be valid.
|
||||||
|
|
||||||
|
When this option is set to `false`, a client certificate is requested during the handshake, and at least one valid certificate should be sent by the client.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog.endpoint.tls]
|
[providers.consulCatalog.endpoint.tls]
|
||||||
caOptional = true
|
caOptional = true
|
||||||
|
@ -396,17 +418,14 @@ providers:
|
||||||
--providers.consulcatalog.endpoint.tls.caoptional=true
|
--providers.consulcatalog.endpoint.tls.caoptional=true
|
||||||
```
|
```
|
||||||
|
|
||||||
Policy followed for the secured connection with TLS Client Authentication to Consul.
|
|
||||||
Requires `tls.ca` to be defined.
|
|
||||||
|
|
||||||
- `true`: VerifyClientCertIfGiven
|
|
||||||
- `false`: RequireAndVerifyClientCert
|
|
||||||
- if `tls.ca` is undefined NoClientCert
|
|
||||||
|
|
||||||
##### `cert`
|
##### `cert`
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
|
||||||
|
`cert` is the path to the public certificate to use for Consul communication.
|
||||||
|
|
||||||
|
When using this option, setting the `key` option is required.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog.endpoint.tls]
|
[providers.consulCatalog.endpoint.tls]
|
||||||
cert = "path/to/foo.cert"
|
cert = "path/to/foo.cert"
|
||||||
|
@ -427,13 +446,14 @@ providers:
|
||||||
--providers.consulcatalog.endpoint.tls.key=path/to/foo.key
|
--providers.consulcatalog.endpoint.tls.key=path/to/foo.key
|
||||||
```
|
```
|
||||||
|
|
||||||
`cert` is the path to the public certificate for Consul communication.
|
|
||||||
If this is set then you need to also set `key.
|
|
||||||
|
|
||||||
##### `key`
|
##### `key`
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
|
||||||
|
`key` is the path to the private key for Consul communication.
|
||||||
|
|
||||||
|
When using this option, setting the `cert` option is required.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog.endpoint.tls]
|
[providers.consulCatalog.endpoint.tls]
|
||||||
cert = "path/to/foo.cert"
|
cert = "path/to/foo.cert"
|
||||||
|
@ -454,13 +474,12 @@ providers:
|
||||||
--providers.consulcatalog.endpoint.tls.key=path/to/foo.key
|
--providers.consulcatalog.endpoint.tls.key=path/to/foo.key
|
||||||
```
|
```
|
||||||
|
|
||||||
`key` is the path to the private key for Consul communication.
|
|
||||||
If this is set then you need to also set `cert`.
|
|
||||||
|
|
||||||
##### `insecureSkipVerify`
|
##### `insecureSkipVerify`
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
|
||||||
|
If `insecureSkipVerify` is `true`, the TLS connection to Consul accepts any certificate presented by the server regardless of the hostnames it covers.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog.endpoint.tls]
|
[providers.consulCatalog.endpoint.tls]
|
||||||
insecureSkipVerify = true
|
insecureSkipVerify = true
|
||||||
|
@ -478,12 +497,15 @@ providers:
|
||||||
--providers.consulcatalog.endpoint.tls.insecureskipverify=true
|
--providers.consulcatalog.endpoint.tls.insecureskipverify=true
|
||||||
```
|
```
|
||||||
|
|
||||||
If `insecureSkipVerify` is `true`, TLS for the connection to Consul server accepts any certificate presented by the server and any host name in that certificate.
|
|
||||||
|
|
||||||
### `exposedByDefault`
|
### `exposedByDefault`
|
||||||
|
|
||||||
_Optional, Default=true_
|
_Optional, Default=true_
|
||||||
|
|
||||||
|
Expose Consul Catalog services by default in Traefik.
|
||||||
|
If set to `false`, services that don't have a `traefik.enable=true` tag will be ignored from the resulting routing configuration.
|
||||||
|
|
||||||
|
For additional information, refer to [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
exposedByDefault = false
|
exposedByDefault = false
|
||||||
|
@ -502,15 +524,20 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Expose Consul Catalog services by default in Traefik.
|
|
||||||
If set to false, services that don't have a `traefik.enable=true` tag will be ignored from the resulting routing configuration.
|
|
||||||
|
|
||||||
See also [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
|
||||||
|
|
||||||
### `defaultRule`
|
### `defaultRule`
|
||||||
|
|
||||||
_Optional, Default=```Host(`{{ normalize .Name }}`)```_
|
_Optional, Default=```Host(`{{ normalize .Name }}`)```_
|
||||||
|
|
||||||
|
The default host rule for all services.
|
||||||
|
|
||||||
|
For a given service, if no routing rule was defined by a tag, it is defined by this `defaultRule` instead.
|
||||||
|
The `defaultRule` must be set to a valid [Go template](https://golang.org/pkg/text/template/),
|
||||||
|
and can include [sprig template functions](http://masterminds.github.io/sprig/).
|
||||||
|
The service name can be accessed with the `Name` identifier,
|
||||||
|
and the template has access to all the labels (i.e. tags beginning with the `prefix`) defined on this service.
|
||||||
|
|
||||||
|
The option can be overridden on an instance basis with the `traefik.http.routers.{name-of-your-choice}.rule` tag.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consulCatalog]
|
[providers.consulCatalog]
|
||||||
defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
|
defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
|
||||||
|
@ -529,41 +556,13 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
The default host rule for all services.
|
|
||||||
|
|
||||||
For a given service if no routing rule was defined by a tag, it is defined by this defaultRule instead.
|
|
||||||
It must be a valid [Go template](https://golang.org/pkg/text/template/),
|
|
||||||
augmented with the [sprig template functions](http://masterminds.github.io/sprig/).
|
|
||||||
The service name can be accessed as the `Name` identifier,
|
|
||||||
and the template has access to all the labels (i.e. tags beginning with the `prefix`) defined on this service.
|
|
||||||
|
|
||||||
The option can be overridden on an instance basis with the `traefik.http.routers.{name-of-your-choice}.rule` tag.
|
|
||||||
|
|
||||||
### `constraints`
|
### `constraints`
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
The `constraints` option can be set to an expression that Traefik matches against the service tags to determine whether
|
||||||
[providers.consulCatalog]
|
to create any route for that service. If none of the service tags match the expression, no route for that service is
|
||||||
constraints = "Tag(`a.tag.name`)"
|
created. If the expression is empty, all detected services are included.
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
|
||||||
providers:
|
|
||||||
consulCatalog:
|
|
||||||
constraints: "Tag(`a.tag.name`)"
|
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
```bash tab="CLI"
|
|
||||||
--providers.consulcatalog.constraints="Tag(`a.tag.name`)"
|
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
Constraints is an expression that Traefik matches against the service's tags to determine whether to create any route for that service.
|
|
||||||
That is to say, if none of the service's tags match the expression, no route for that service is created.
|
|
||||||
If the expression is empty, all detected services are included.
|
|
||||||
|
|
||||||
The expression syntax is based on the ```Tag(`tag`)```, and ```TagRegex(`tag`)``` functions,
|
The expression syntax is based on the ```Tag(`tag`)```, and ```TagRegex(`tag`)``` functions,
|
||||||
as well as the usual boolean logic, as shown in examples below.
|
as well as the usual boolean logic, as shown in examples below.
|
||||||
|
@ -600,4 +599,22 @@ as well as the usual boolean logic, as shown in examples below.
|
||||||
constraints = "TagRegex(`a\.tag\.t.+`)"
|
constraints = "TagRegex(`a\.tag\.t.+`)"
|
||||||
```
|
```
|
||||||
|
|
||||||
See also [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
```toml tab="File (TOML)"
|
||||||
|
[providers.consulCatalog]
|
||||||
|
constraints = "Tag(`a.tag.name`)"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml tab="File (YAML)"
|
||||||
|
providers:
|
||||||
|
consulCatalog:
|
||||||
|
constraints: "Tag(`a.tag.name`)"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash tab="CLI"
|
||||||
|
--providers.consulcatalog.constraints="Tag(`a.tag.name`)"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
For additional information, refer to [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
||||||
|
|
|
@ -35,10 +35,10 @@ providers:
|
||||||
|
|
||||||
### `rootKey`
|
### `rootKey`
|
||||||
|
|
||||||
Defines the root key of the configuration.
|
|
||||||
|
|
||||||
_Required, Default="traefik"_
|
_Required, Default="traefik"_
|
||||||
|
|
||||||
|
Defines the root key of the configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consul]
|
[providers.consul]
|
||||||
rootKey = "traefik"
|
rootKey = "traefik"
|
||||||
|
@ -56,10 +56,10 @@ providers:
|
||||||
|
|
||||||
### `username`
|
### `username`
|
||||||
|
|
||||||
Defines a username to connect with Consul.
|
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Defines a username to connect to Consul with.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consul]
|
[providers.consul]
|
||||||
# ...
|
# ...
|
||||||
|
@ -81,7 +81,7 @@ providers:
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
Defines a password to connect with Consul.
|
Defines a password with which to connect to Consul.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consul]
|
[providers.consul]
|
||||||
|
@ -106,7 +106,7 @@ _Optional_
|
||||||
|
|
||||||
#### `tls.ca`
|
#### `tls.ca`
|
||||||
|
|
||||||
Certificate Authority used for the secured connection to Consul.
|
Certificate Authority used for the secure connection to Consul.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consul.tls]
|
[providers.consul.tls]
|
||||||
|
@ -126,12 +126,15 @@ providers:
|
||||||
|
|
||||||
#### `tls.caOptional`
|
#### `tls.caOptional`
|
||||||
|
|
||||||
Policy followed for the secured connection with TLS Client Authentication to Consul.
|
The value of `tls.caOptional` defines which policy should be used for the secure connection with TLS Client Authentication to Consul.
|
||||||
Requires `tls.ca` to be defined.
|
|
||||||
|
|
||||||
- `true`: VerifyClientCertIfGiven
|
!!! warning ""
|
||||||
- `false`: RequireAndVerifyClientCert
|
|
||||||
- if `tls.ca` is undefined NoClientCert
|
If `tls.ca` is undefined, this option will be ignored, and no client certificate will be requested during the handshake. Any provided certificate will thus never be verified.
|
||||||
|
|
||||||
|
When this option is set to `true`, a client certificate is requested during the handshake but is not required. If a certificate is sent, it is required to be valid.
|
||||||
|
|
||||||
|
When this option is set to `false`, a client certificate is requested during the handshake, and at least one valid certificate should be sent by the client.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consul.tls]
|
[providers.consul.tls]
|
||||||
|
@ -151,7 +154,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.cert`
|
#### `tls.cert`
|
||||||
|
|
||||||
Public certificate used for the secured connection to Consul.
|
Public certificate used for the secure connection to Consul.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consul.tls]
|
[providers.consul.tls]
|
||||||
|
@ -174,7 +177,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.key`
|
#### `tls.key`
|
||||||
|
|
||||||
Private certificate used for the secured connection to Consul.
|
Private certificate used for the secure connection to Consul.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consul.tls]
|
[providers.consul.tls]
|
||||||
|
@ -197,7 +200,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.insecureSkipVerify`
|
#### `tls.insecureSkipVerify`
|
||||||
|
|
||||||
If `insecureSkipVerify` is `true`, TLS for the connection to Consul accepts any certificate presented by the server and any host name in that certificate.
|
If `insecureSkipVerify` is `true`, the TLS connection to Consul accepts any certificate presented by the server regardless of the hostnames it covers.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.consul.tls]
|
[providers.consul.tls]
|
||||||
|
|
|
@ -11,7 +11,8 @@ Traefik works with both [Docker (standalone) Engine](https://docs.docker.com/eng
|
||||||
and [Docker Swarm Mode](https://docs.docker.com/engine/swarm/).
|
and [Docker Swarm Mode](https://docs.docker.com/engine/swarm/).
|
||||||
|
|
||||||
!!! tip "The Quick Start Uses Docker"
|
!!! tip "The Quick Start Uses Docker"
|
||||||
If you haven't already, maybe you'd like to go through the [quick start](../getting-started/quick-start.md) that uses the docker provider!
|
|
||||||
|
If you have not already read it, maybe you would like to go through the [quick start guide](../getting-started/quick-start.md) that uses the Docker provider.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
@ -102,17 +103,18 @@ When using Docker Compose, labels are specified by the directive
|
||||||
["services" objects](https://docs.docker.com/compose/compose-file/compose-file-v3/#service-configuration-reference).
|
["services" objects](https://docs.docker.com/compose/compose-file/compose-file-v3/#service-configuration-reference).
|
||||||
|
|
||||||
!!! tip "Not Only Docker"
|
!!! tip "Not Only Docker"
|
||||||
|
|
||||||
Please note that any tool like Nomad, Terraform, Ansible, etc.
|
Please note that any tool like Nomad, Terraform, Ansible, etc.
|
||||||
that is able to define a Docker container with labels can work
|
that is able to define a Docker container with labels can work
|
||||||
with Traefik & the Docker provider.
|
with Traefik and the Docker provider.
|
||||||
|
|
||||||
### Port Detection
|
### Port Detection
|
||||||
|
|
||||||
Traefik retrieves the private IP and port of containers from the Docker API.
|
Traefik retrieves the private IP and port of containers from the Docker API.
|
||||||
|
|
||||||
Ports detection works as follows:
|
Port detection works as follows:
|
||||||
|
|
||||||
- If a container [exposes](https://docs.docker.com/engine/reference/builder/#expose) only one port,
|
- If a container [exposes](https://docs.docker.com/engine/reference/builder/#expose) a single port,
|
||||||
then Traefik uses this port for private communication.
|
then Traefik uses this port for private communication.
|
||||||
- If a container [exposes](https://docs.docker.com/engine/reference/builder/#expose) multiple ports,
|
- If a container [exposes](https://docs.docker.com/engine/reference/builder/#expose) multiple ports,
|
||||||
or does not expose any port, then you must manually specify which port Traefik should use for communication
|
or does not expose any port, then you must manually specify which port Traefik should use for communication
|
||||||
|
@ -126,12 +128,11 @@ the IP address of the host is resolved as follows:
|
||||||
|
|
||||||
<!-- TODO: verify and document the swarm mode case with container.Node.IPAddress coming from the API -->
|
<!-- TODO: verify and document the swarm mode case with container.Node.IPAddress coming from the API -->
|
||||||
- try a lookup of `host.docker.internal`
|
- try a lookup of `host.docker.internal`
|
||||||
- otherwise fall back to `127.0.0.1`
|
- if the lookup was unsuccessful, fall back to `127.0.0.1`
|
||||||
|
|
||||||
On Linux, (and until [github.com/moby/moby/pull/40007](https://github.com/moby/moby/pull/40007) is included in a release),
|
On Linux, for versions of Docker older than 20.10.0, for `host.docker.internal` to be defined, it should be provided
|
||||||
for `host.docker.internal` to be defined, it should be provided as an `extra_host` to the Traefik container,
|
as an `extra_host` to the Traefik container, using the `--add-host` flag. For example, to set it to the IP address of
|
||||||
using the `--add-host` flag. For example, to set it to the IP address of the bridge interface (docker0 by default):
|
the bridge interface (`docker0` by default): `--add-host=host.docker.internal:172.17.0.1`
|
||||||
`--add-host=host.docker.internal:172.17.0.1`
|
|
||||||
|
|
||||||
### Docker API Access
|
### Docker API Access
|
||||||
|
|
||||||
|
@ -145,9 +146,10 @@ You can specify which Docker API Endpoint to use with the directive [`endpoint`]
|
||||||
If Traefik is attacked, then the attacker might get access to the underlying host.
|
If Traefik is attacked, then the attacker might get access to the underlying host.
|
||||||
{: #security-note }
|
{: #security-note }
|
||||||
|
|
||||||
As explained in the Docker documentation: ([Docker Daemon Attack Surface page](https://docs.docker.com/engine/security/#docker-daemon-attack-surface)):
|
As explained in the [Docker Daemon Attack Surface documentation](https://docs.docker.com/engine/security/#docker-daemon-attack-surface):
|
||||||
|
|
||||||
!!! quote
|
!!! quote
|
||||||
|
|
||||||
[...] only **trusted** users should be allowed to control your Docker daemon [...]
|
[...] only **trusted** users should be allowed to control your Docker daemon [...]
|
||||||
|
|
||||||
??? success "Solutions"
|
??? success "Solutions"
|
||||||
|
@ -155,7 +157,7 @@ You can specify which Docker API Endpoint to use with the directive [`endpoint`]
|
||||||
Expose the Docker socket over TCP or SSH, instead of the default Unix socket file.
|
Expose the Docker socket over TCP or SSH, instead of the default Unix socket file.
|
||||||
It allows different implementation levels of the [AAA (Authentication, Authorization, Accounting) concepts](https://en.wikipedia.org/wiki/AAA_(computer_security)), depending on your security assessment:
|
It allows different implementation levels of the [AAA (Authentication, Authorization, Accounting) concepts](https://en.wikipedia.org/wiki/AAA_(computer_security)), depending on your security assessment:
|
||||||
|
|
||||||
- Authentication with Client Certificates as described in ["Protect the Docker daemon socket."](https://docs.docker.com/engine/security/https/)
|
- Authentication with Client Certificates as described in ["Protect the Docker daemon socket."](https://docs.docker.com/engine/security/protect-access/)
|
||||||
- Authorize and filter requests to restrict possible actions with [the TecnativaDocker Socket Proxy](https://github.com/Tecnativa/docker-socket-proxy).
|
- Authorize and filter requests to restrict possible actions with [the TecnativaDocker Socket Proxy](https://github.com/Tecnativa/docker-socket-proxy).
|
||||||
- Authorization with the [Docker Authorization Plugin Mechanism](https://web.archive.org/web/20190920092526/https://docs.docker.com/engine/extend/plugins_authorization/)
|
- Authorization with the [Docker Authorization Plugin Mechanism](https://web.archive.org/web/20190920092526/https://docs.docker.com/engine/extend/plugins_authorization/)
|
||||||
- Accounting at networking level, by exposing the socket only inside a Docker private network, only available for Traefik.
|
- Accounting at networking level, by exposing the socket only inside a Docker private network, only available for Traefik.
|
||||||
|
@ -165,6 +167,7 @@ You can specify which Docker API Endpoint to use with the directive [`endpoint`]
|
||||||
- SSH public key authentication (SSH is supported with Docker > 18.09)
|
- SSH public key authentication (SSH is supported with Docker > 18.09)
|
||||||
|
|
||||||
??? info "More Resources and Examples"
|
??? info "More Resources and Examples"
|
||||||
|
|
||||||
- ["Paranoid about mounting /var/run/docker.sock?"](https://medium.com/@containeroo/traefik-2-0-paranoid-about-mounting-var-run-docker-sock-22da9cb3e78c)
|
- ["Paranoid about mounting /var/run/docker.sock?"](https://medium.com/@containeroo/traefik-2-0-paranoid-about-mounting-var-run-docker-sock-22da9cb3e78c)
|
||||||
- [Traefik and Docker: A Discussion with Docker Captain, Bret Fisher](https://blog.traefik.io/traefik-and-docker-a-discussion-with-docker-captain-bret-fisher-7f0b9a54ff88)
|
- [Traefik and Docker: A Discussion with Docker Captain, Bret Fisher](https://blog.traefik.io/traefik-and-docker-a-discussion-with-docker-captain-bret-fisher-7f0b9a54ff88)
|
||||||
- [KubeCon EU 2018 Keynote, Running with Scissors, from Liz Rice](https://www.youtube.com/watch?v=ltrV-Qmh3oY)
|
- [KubeCon EU 2018 Keynote, Running with Scissors, from Liz Rice](https://www.youtube.com/watch?v=ltrV-Qmh3oY)
|
||||||
|
@ -194,15 +197,15 @@ This behavior is only enabled for docker-compose version 3+ ([Compose file refer
|
||||||
|
|
||||||
Docker Swarm does not provide any [port detection](#port-detection) information to Traefik.
|
Docker Swarm does not provide any [port detection](#port-detection) information to Traefik.
|
||||||
|
|
||||||
Therefore you **must** specify the port to use for communication by using the label `traefik.http.services.<service_name>.loadbalancer.server.port`
|
Therefore, you **must** specify the port to use for communication by using the label `traefik.http.services.<service_name>.loadbalancer.server.port`
|
||||||
(Check the reference for this label in the [routing section for Docker](../routing/providers/docker.md#port)).
|
(Check the reference for this label in the [routing section for Docker](../routing/providers/docker.md#port)).
|
||||||
|
|
||||||
### Docker API Access
|
### Docker API Access
|
||||||
|
|
||||||
Docker Swarm Mode follows the same rules as Docker [API Access](#docker-api-access).
|
Docker Swarm Mode follows the same rules as Docker [API Access](#docker-api-access).
|
||||||
|
|
||||||
As the Swarm API is only exposed on the [manager nodes](https://docs.docker.com/engine/swarm/how-swarm-mode-works/nodes/#manager-nodes), you should schedule Traefik on the Swarm manager nodes by default,
|
Since the Swarm API is only exposed on the [manager nodes](https://docs.docker.com/engine/swarm/how-swarm-mode-works/nodes/#manager-nodes),
|
||||||
by deploying Traefik with a constraint on the node's "role":
|
these are the nodes that Traefik should be scheduled on by deploying Traefik with a constraint on the node "role":
|
||||||
|
|
||||||
```shell tab="With Docker CLI"
|
```shell tab="With Docker CLI"
|
||||||
docker service create \
|
docker service create \
|
||||||
|
@ -238,21 +241,6 @@ services:
|
||||||
|
|
||||||
_Required, Default="unix:///var/run/docker.sock"_
|
_Required, Default="unix:///var/run/docker.sock"_
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
|
||||||
[providers.docker]
|
|
||||||
endpoint = "unix:///var/run/docker.sock"
|
|
||||||
```
|
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
|
||||||
providers:
|
|
||||||
docker:
|
|
||||||
endpoint: "unix:///var/run/docker.sock"
|
|
||||||
```
|
|
||||||
|
|
||||||
```bash tab="CLI"
|
|
||||||
--providers.docker.endpoint=unix:///var/run/docker.sock
|
|
||||||
```
|
|
||||||
|
|
||||||
See the sections [Docker API Access](#docker-api-access) and [Docker Swarm API Access](#docker-api-access_1) for more information.
|
See the sections [Docker API Access](#docker-api-access) and [Docker Swarm API Access](#docker-api-access_1) for more information.
|
||||||
|
|
||||||
??? example "Using the docker.sock"
|
??? example "Using the docker.sock"
|
||||||
|
@ -315,34 +303,31 @@ See the sections [Docker API Access](#docker-api-access) and [Docker Swarm API A
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
### `useBindPortIP`
|
|
||||||
|
|
||||||
_Optional, Default=false_
|
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker]
|
[providers.docker]
|
||||||
useBindPortIP = true
|
endpoint = "unix:///var/run/docker.sock"
|
||||||
# ...
|
|
||||||
```
|
```
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
```yaml tab="File (YAML)"
|
||||||
providers:
|
providers:
|
||||||
docker:
|
docker:
|
||||||
useBindPortIP: true
|
endpoint: "unix:///var/run/docker.sock"
|
||||||
# ...
|
|
||||||
```
|
```
|
||||||
|
|
||||||
```bash tab="CLI"
|
```bash tab="CLI"
|
||||||
--providers.docker.useBindPortIP=true
|
--providers.docker.endpoint=unix:///var/run/docker.sock
|
||||||
# ...
|
|
||||||
```
|
```
|
||||||
|
|
||||||
Traefik routes requests to the IP/Port of the matching container.
|
### `useBindPortIP`
|
||||||
|
|
||||||
|
_Optional, Default=false_
|
||||||
|
|
||||||
|
Traefik routes requests to the IP/port of the matching container.
|
||||||
When setting `useBindPortIP=true`, you tell Traefik to use the IP/Port attached to the container's _binding_ instead of its inner network IP/Port.
|
When setting `useBindPortIP=true`, you tell Traefik to use the IP/Port attached to the container's _binding_ instead of its inner network IP/Port.
|
||||||
|
|
||||||
When used in conjunction with the `traefik.http.services.<name>.loadbalancer.server.port` label (that tells Traefik to route requests to a specific port),
|
When used in conjunction with the `traefik.http.services.<name>.loadbalancer.server.port` label (that tells Traefik to route requests to a specific port),
|
||||||
Traefik tries to find a binding on port `traefik.http.services.<name>.loadbalancer.server.port`.
|
Traefik tries to find a binding on port `traefik.http.services.<name>.loadbalancer.server.port`.
|
||||||
If it can't find such a binding, Traefik falls back on the internal network IP of the container,
|
If it cannot find such a binding, Traefik falls back on the internal network IP of the container,
|
||||||
but still uses the `traefik.http.services.<name>.loadbalancer.server.port` that is set in the label.
|
but still uses the `traefik.http.services.<name>.loadbalancer.server.port` that is set in the label.
|
||||||
|
|
||||||
??? example "Examples of `usebindportip` in different situations."
|
??? example "Examples of `usebindportip` in different situations."
|
||||||
|
@ -365,10 +350,33 @@ but still uses the `traefik.http.services.<name>.loadbalancer.server.port` that
|
||||||
- `ExtPort` stands for "external Port found in the binding"
|
- `ExtPort` stands for "external Port found in the binding"
|
||||||
- `IntPort` stands for "internal network container's port."
|
- `IntPort` stands for "internal network container's port."
|
||||||
|
|
||||||
|
```toml tab="File (TOML)"
|
||||||
|
[providers.docker]
|
||||||
|
useBindPortIP = true
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml tab="File (YAML)"
|
||||||
|
providers:
|
||||||
|
docker:
|
||||||
|
useBindPortIP: true
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash tab="CLI"
|
||||||
|
--providers.docker.useBindPortIP=true
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
### `exposedByDefault`
|
### `exposedByDefault`
|
||||||
|
|
||||||
_Optional, Default=true_
|
_Optional, Default=true_
|
||||||
|
|
||||||
|
Expose containers by default through Traefik.
|
||||||
|
If set to `false`, containers that do not have a `traefik.enable=true` label are ignored from the resulting routing configuration.
|
||||||
|
|
||||||
|
For additional information, refer to [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker]
|
[providers.docker]
|
||||||
exposedByDefault = false
|
exposedByDefault = false
|
||||||
|
@ -387,14 +395,13 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Expose containers by default through Traefik.
|
|
||||||
If set to false, containers that don't have a `traefik.enable=true` label will be ignored from the resulting routing configuration.
|
|
||||||
|
|
||||||
See also [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
|
||||||
|
|
||||||
### `network`
|
### `network`
|
||||||
|
|
||||||
_Optional, Default=empty_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Defines a default docker network to use for connections to all containers.
|
||||||
|
|
||||||
|
This option can be overridden on a per-container basis with the `traefik.docker.network` label.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker]
|
[providers.docker]
|
||||||
|
@ -414,14 +421,17 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Defines a default docker network to use for connections to all containers.
|
|
||||||
|
|
||||||
This option can be overridden on a container basis with the `traefik.docker.network` label.
|
|
||||||
|
|
||||||
### `defaultRule`
|
### `defaultRule`
|
||||||
|
|
||||||
_Optional, Default=```Host(`{{ normalize .Name }}`)```_
|
_Optional, Default=```Host(`{{ normalize .Name }}`)```_
|
||||||
|
|
||||||
|
The `defaultRule` option defines what routing rule to apply to a container if no rule is defined by a label.
|
||||||
|
|
||||||
|
It must be a valid [Go template](https://golang.org/pkg/text/template/), and can use
|
||||||
|
[sprig template functions](http://masterminds.github.io/sprig/).
|
||||||
|
The container service name can be accessed with the `Name` identifier,
|
||||||
|
and the template has access to all the labels defined on this container.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker]
|
[providers.docker]
|
||||||
defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
|
defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
|
||||||
|
@ -440,16 +450,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
For a given container if no routing rule was defined by a label, it is defined by this defaultRule instead.
|
|
||||||
It must be a valid [Go template](https://golang.org/pkg/text/template/),
|
|
||||||
augmented with the [sprig template functions](http://masterminds.github.io/sprig/).
|
|
||||||
The container service name can be accessed as the `Name` identifier,
|
|
||||||
and the template has access to all the labels defined on this container.
|
|
||||||
|
|
||||||
### `swarmMode`
|
### `swarmMode`
|
||||||
|
|
||||||
_Optional, Default=false_
|
_Optional, Default=false_
|
||||||
|
|
||||||
|
Enables the Swarm Mode (instead of standalone Docker).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker]
|
[providers.docker]
|
||||||
swarmMode = true
|
swarmMode = true
|
||||||
|
@ -468,12 +474,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Activates the Swarm Mode (instead of standalone Docker).
|
|
||||||
|
|
||||||
### `swarmModeRefreshSeconds`
|
### `swarmModeRefreshSeconds`
|
||||||
|
|
||||||
_Optional, Default=15_
|
_Optional, Default=15_
|
||||||
|
|
||||||
|
Defines the polling interval (in seconds) for Swarm Mode.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker]
|
[providers.docker]
|
||||||
swarmModeRefreshSeconds = 30
|
swarmModeRefreshSeconds = 30
|
||||||
|
@ -492,12 +498,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Defines the polling interval (in seconds) in Swarm Mode.
|
|
||||||
|
|
||||||
### `httpClientTimeout`
|
### `httpClientTimeout`
|
||||||
|
|
||||||
_Optional, Default=0_
|
_Optional, Default=0_
|
||||||
|
|
||||||
|
Defines the client timeout (in seconds) for HTTP connections. If its value is `0`, no timeout is set.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker]
|
[providers.docker]
|
||||||
httpClientTimeout = 300
|
httpClientTimeout = 300
|
||||||
|
@ -516,12 +522,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Defines the client timeout (in seconds) for HTTP connections. If zero, no timeout is set.
|
|
||||||
|
|
||||||
### `watch`
|
### `watch`
|
||||||
|
|
||||||
_Optional, Default=true_
|
_Optional, Default=true_
|
||||||
|
|
||||||
|
Watch Docker Swarm events.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker]
|
[providers.docker]
|
||||||
watch = false
|
watch = false
|
||||||
|
@ -540,35 +546,16 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Watch Docker Swarm events.
|
|
||||||
|
|
||||||
### `constraints`
|
### `constraints`
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
The `constraints` option can be set to an expression that Traefik matches against the container tags to determine whether
|
||||||
[providers.docker]
|
to create any route for that container. If none of the container tags match the expression, no route for that container is
|
||||||
constraints = "Label(`a.label.name`,`foo`)"
|
created. If the expression is empty, all detected containers are included.
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
The expression syntax is based on the ```Tag(`tag`)```, and ```TagRegex(`tag`)``` functions,
|
||||||
providers:
|
as well as the usual boolean logic, as shown in examples below.
|
||||||
docker:
|
|
||||||
constraints: "Label(`a.label.name`,`foo`)"
|
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
```bash tab="CLI"
|
|
||||||
--providers.docker.constraints=Label(`a.label.name`,`foo`)
|
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
Constraints is an expression that Traefik matches against the container's labels to determine whether to create any route for that container.
|
|
||||||
That is to say, if none of the container's labels match the expression, no route for the container is created.
|
|
||||||
If the expression is empty, all detected containers are included.
|
|
||||||
|
|
||||||
The expression syntax is based on the `Label("key", "value")`, and `LabelRegex("key", "value")` functions, as well as the usual boolean logic, as shown in examples below.
|
|
||||||
|
|
||||||
??? example "Constraints Expression Examples"
|
??? example "Constraints Expression Examples"
|
||||||
|
|
||||||
|
@ -602,7 +589,25 @@ The expression syntax is based on the `Label("key", "value")`, and `LabelRegex("
|
||||||
constraints = "LabelRegex(`a.label.name`, `a.+`)"
|
constraints = "LabelRegex(`a.label.name`, `a.+`)"
|
||||||
```
|
```
|
||||||
|
|
||||||
See also [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
For additional information, refer to [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
||||||
|
|
||||||
|
```toml tab="File (TOML)"
|
||||||
|
[providers.docker]
|
||||||
|
constraints = "Label(`a.label.name`,`foo`)"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml tab="File (YAML)"
|
||||||
|
providers:
|
||||||
|
docker:
|
||||||
|
constraints: "Label(`a.label.name`,`foo`)"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash tab="CLI"
|
||||||
|
--providers.docker.constraints=Label(`a.label.name`,`foo`)
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
### `tls`
|
### `tls`
|
||||||
|
|
||||||
|
@ -610,7 +615,7 @@ _Optional_
|
||||||
|
|
||||||
#### `tls.ca`
|
#### `tls.ca`
|
||||||
|
|
||||||
Certificate Authority used for the secured connection to Docker.
|
Certificate Authority used for the secure connection to Docker.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker.tls]
|
[providers.docker.tls]
|
||||||
|
@ -630,12 +635,15 @@ providers:
|
||||||
|
|
||||||
#### `tls.caOptional`
|
#### `tls.caOptional`
|
||||||
|
|
||||||
Policy followed for the secured connection with TLS Client Authentication to Docker.
|
The value of `tls.caOptional` defines which policy should be used for the secure connection with TLS Client Authentication to Docker.
|
||||||
Requires `tls.ca` to be defined.
|
|
||||||
|
|
||||||
- `true`: VerifyClientCertIfGiven
|
!!! warning ""
|
||||||
- `false`: RequireAndVerifyClientCert
|
|
||||||
- if `tls.ca` is undefined NoClientCert
|
If `tls.ca` is undefined, this option will be ignored, and no client certificate will be requested during the handshake. Any provided certificate will thus never be verified.
|
||||||
|
|
||||||
|
When this option is set to `true`, a client certificate is requested during the handshake but is not required. If a certificate is sent, it is required to be valid.
|
||||||
|
|
||||||
|
When this option is set to `false`, a client certificate is requested during the handshake, and at least one valid certificate should be sent by the client.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker.tls]
|
[providers.docker.tls]
|
||||||
|
@ -655,7 +663,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.cert`
|
#### `tls.cert`
|
||||||
|
|
||||||
Public certificate used for the secured connection to Docker.
|
Public certificate used for the secure connection to Docker.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker.tls]
|
[providers.docker.tls]
|
||||||
|
@ -678,7 +686,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.key`
|
#### `tls.key`
|
||||||
|
|
||||||
Private certificate used for the secured connection to Docker.
|
Private certificate used for the secure connection to Docker.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker.tls]
|
[providers.docker.tls]
|
||||||
|
@ -701,7 +709,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.insecureSkipVerify`
|
#### `tls.insecureSkipVerify`
|
||||||
|
|
||||||
If `insecureSkipVerify` is `true`, TLS for the connection to Docker accepts any certificate presented by the server and any host name in that certificate.
|
If `insecureSkipVerify` is `true`, the TLS connection to Docker accepts any certificate presented by the server regardless of the hostnames it covers.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.docker.tls]
|
[providers.docker.tls]
|
||||||
|
|
|
@ -52,12 +52,17 @@ Traefik needs the following policy to read ECS information:
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
## Provider configuration
|
## Provider Configuration
|
||||||
|
|
||||||
### `autoDiscoverClusters`
|
### `autoDiscoverClusters`
|
||||||
|
|
||||||
_Optional, Default=false_
|
_Optional, Default=false_
|
||||||
|
|
||||||
|
Search for services in cluster list.
|
||||||
|
|
||||||
|
- If set to `true` service discovery is disabled on configured clusters, but enabled for all other clusters.
|
||||||
|
- If set to `false` service discovery is enabled on configured clusters only.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.ecs]
|
[providers.ecs]
|
||||||
autoDiscoverClusters = true
|
autoDiscoverClusters = true
|
||||||
|
@ -76,15 +81,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Search for services in clusters list.
|
|
||||||
|
|
||||||
- If set to `true` the configured clusters will be ignored and the clusters will be discovered.
|
|
||||||
- If set to `false` the services will be discovered only in configured clusters.
|
|
||||||
|
|
||||||
### `clusters`
|
### `clusters`
|
||||||
|
|
||||||
_Optional, Default=["default"]_
|
_Optional, Default=["default"]_
|
||||||
|
|
||||||
|
Search for services in cluster list.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.ecs]
|
[providers.ecs]
|
||||||
clusters = ["default"]
|
clusters = ["default"]
|
||||||
|
@ -104,12 +106,14 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Search for services in clusters list.
|
|
||||||
|
|
||||||
### `exposedByDefault`
|
### `exposedByDefault`
|
||||||
|
|
||||||
_Optional, Default=true_
|
_Optional, Default=true_
|
||||||
|
|
||||||
|
Expose ECS services by default in Traefik.
|
||||||
|
|
||||||
|
If set to `false`, services that do not have a `traefik.enable=true` label are ignored from the resulting routing configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.ecs]
|
[providers.ecs]
|
||||||
exposedByDefault = false
|
exposedByDefault = false
|
||||||
|
@ -128,13 +132,17 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Expose ECS services by default in Traefik.
|
|
||||||
If set to false, services that don't have a `traefik.enable=true` label will be ignored from the resulting routing configuration.
|
|
||||||
|
|
||||||
### `defaultRule`
|
### `defaultRule`
|
||||||
|
|
||||||
_Optional, Default=```Host(`{{ normalize .Name }}`)```_
|
_Optional, Default=```Host(`{{ normalize .Name }}`)```_
|
||||||
|
|
||||||
|
The `defaultRule` option defines what routing rule to apply to a container if no rule is defined by a label.
|
||||||
|
|
||||||
|
It must be a valid [Go template](https://golang.org/pkg/text/template/), and can use
|
||||||
|
[sprig template functions](http://masterminds.github.io/sprig/).
|
||||||
|
The container service name can be accessed with the `Name` identifier,
|
||||||
|
and the template has access to all the labels defined on this container.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.ecs]
|
[providers.ecs]
|
||||||
defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
|
defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
|
||||||
|
@ -153,16 +161,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
For a given container if no routing rule was defined by a label, it is defined by this defaultRule instead.
|
|
||||||
It must be a valid [Go template](https://golang.org/pkg/text/template/),
|
|
||||||
augmented with the [sprig template functions](http://masterminds.github.io/sprig/).
|
|
||||||
The service name can be accessed as the `Name` identifier,
|
|
||||||
and the template has access to all the labels defined on this container.
|
|
||||||
|
|
||||||
### `refreshSeconds`
|
### `refreshSeconds`
|
||||||
|
|
||||||
_Optional, Default=15_
|
_Optional, Default=15_
|
||||||
|
|
||||||
|
Polling interval (in seconds).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.ecs]
|
[providers.ecs]
|
||||||
refreshSeconds = 15
|
refreshSeconds = 15
|
||||||
|
@ -181,12 +185,19 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Polling interval (in seconds).
|
|
||||||
|
|
||||||
### Credentials
|
### Credentials
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
|
||||||
|
If `region` is not provided, it is resolved from the EC2 metadata endpoint for EC2 tasks.
|
||||||
|
In a FARGATE context it is resolved from the `AWS_REGION` environment variable.
|
||||||
|
|
||||||
|
If `accessKeyID` and `secretAccessKey` are not provided, credentials are resolved in the following order:
|
||||||
|
|
||||||
|
- Using the environment variables `AWS_ACCESS_KEY_ID`, `AWS_SECRET_ACCESS_KEY`, and `AWS_SESSION_TOKEN`.
|
||||||
|
- Using shared credentials, determined by `AWS_PROFILE` and `AWS_SHARED_CREDENTIALS_FILE`, defaults to `default` and `~/.aws/credentials`.
|
||||||
|
- Using EC2 instance role or ECS task role
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.ecs]
|
[providers.ecs]
|
||||||
region = "us-east-1"
|
region = "us-east-1"
|
||||||
|
@ -209,12 +220,3 @@ providers:
|
||||||
--providers.ecs.secretAccessKey="123"
|
--providers.ecs.secretAccessKey="123"
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
If `region` is not provided, it will be resolved from the EC2 metadata endpoint for EC2 tasks.
|
|
||||||
In a FARGATE context it will be resolved from the `AWS_REGION` env variable.
|
|
||||||
|
|
||||||
If `accessKeyID` / `secretAccessKey` are not provided, credentials will be resolved in the following order:
|
|
||||||
|
|
||||||
- From environment variables `AWS_ACCESS_KEY_ID`, `AWS_SECRET_ACCESS_KEY`, and `AWS_SESSION_TOKEN`.
|
|
||||||
- Shared credentials, determined by `AWS_PROFILE` and `AWS_SHARED_CREDENTIALS_FILE`, defaults to default and `~/.aws/credentials`.
|
|
||||||
- EC2 instance role or ECS task role
|
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
A Story of KV store & Containers
|
A Story of KV store & Containers
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
Store your configuration in Etcd and let Traefik do the rest!
|
Store your configuration in etcd and let Traefik do the rest!
|
||||||
|
|
||||||
## Routing Configuration
|
## Routing Configuration
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@ See the dedicated section in [routing](../routing/providers/kv.md).
|
||||||
|
|
||||||
_Required, Default="127.0.0.1:2379"_
|
_Required, Default="127.0.0.1:2379"_
|
||||||
|
|
||||||
Defines how to access to Etcd.
|
Defines how to access etcd.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.etcd]
|
[providers.etcd]
|
||||||
|
@ -35,10 +35,10 @@ providers:
|
||||||
|
|
||||||
### `rootKey`
|
### `rootKey`
|
||||||
|
|
||||||
Defines the root key of the configuration.
|
|
||||||
|
|
||||||
_Required, Default="traefik"_
|
_Required, Default="traefik"_
|
||||||
|
|
||||||
|
Defines the root key of the configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.etcd]
|
[providers.etcd]
|
||||||
rootKey = "traefik"
|
rootKey = "traefik"
|
||||||
|
@ -56,10 +56,10 @@ providers:
|
||||||
|
|
||||||
### `username`
|
### `username`
|
||||||
|
|
||||||
Defines a username to connect with Etcd.
|
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Defines a username with which to connect to etcd.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.etcd]
|
[providers.etcd]
|
||||||
# ...
|
# ...
|
||||||
|
@ -81,7 +81,7 @@ providers:
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
Defines a password to connect with Etcd.
|
Defines a password with which to connect to etcd.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.etcd]
|
[providers.etcd]
|
||||||
|
@ -106,7 +106,7 @@ _Optional_
|
||||||
|
|
||||||
#### `tls.ca`
|
#### `tls.ca`
|
||||||
|
|
||||||
Certificate Authority used for the secured connection to Etcd.
|
Certificate Authority used for the secure connection to etcd.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.etcd.tls]
|
[providers.etcd.tls]
|
||||||
|
@ -126,12 +126,15 @@ providers:
|
||||||
|
|
||||||
#### `tls.caOptional`
|
#### `tls.caOptional`
|
||||||
|
|
||||||
Policy followed for the secured connection with TLS Client Authentication to Etcd.
|
The value of `tls.caOptional` defines which policy should be used for the secure connection with TLS Client Authentication to etcd.
|
||||||
Requires `tls.ca` to be defined.
|
|
||||||
|
|
||||||
- `true`: VerifyClientCertIfGiven
|
!!! warning ""
|
||||||
- `false`: RequireAndVerifyClientCert
|
|
||||||
- if `tls.ca` is undefined NoClientCert
|
If `tls.ca` is undefined, this option will be ignored, and no client certificate will be requested during the handshake. Any provided certificate will thus never be verified.
|
||||||
|
|
||||||
|
When this option is set to `true`, a client certificate is requested during the handshake but is not required. If a certificate is sent, it is required to be valid.
|
||||||
|
|
||||||
|
When this option is set to `false`, a client certificate is requested during the handshake, and at least one valid certificate should be sent by the client.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.etcd.tls]
|
[providers.etcd.tls]
|
||||||
|
@ -151,7 +154,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.cert`
|
#### `tls.cert`
|
||||||
|
|
||||||
Public certificate used for the secured connection to Etcd.
|
Public certificate used for the secure connection to etcd.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.etcd.tls]
|
[providers.etcd.tls]
|
||||||
|
@ -174,7 +177,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.key`
|
#### `tls.key`
|
||||||
|
|
||||||
Private certificate used for the secured connection to Etcd.
|
Private certificate used for the secure connection to etcd.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.etcd.tls]
|
[providers.etcd.tls]
|
||||||
|
@ -197,7 +200,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.insecureSkipVerify`
|
#### `tls.insecureSkipVerify`
|
||||||
|
|
||||||
If `insecureSkipVerify` is `true`, TLS for the connection to Etcd accepts any certificate presented by the server and any host name in that certificate.
|
If `insecureSkipVerify` is `true`, the TLS connection to etcd accepts any certificate presented by the server regardless of the hostnames it covers.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.etcd.tls]
|
[providers.etcd.tls]
|
||||||
|
|
|
@ -4,16 +4,16 @@ Good Old Configuration File
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
The file provider lets you define the [dynamic configuration](./overview.md) in a TOML or YAML file.
|
The file provider lets you define the [dynamic configuration](./overview.md) in a TOML or YAML file.
|
||||||
You can write one of these mutually exclusive configuration elements:
|
|
||||||
|
|
||||||
* In [a dedicated file](#filename)
|
It supports providing configuration through a [single configuration file]](#filename) or [multiple separate files](#directory).
|
||||||
* In [several dedicated files](#directory)
|
|
||||||
|
|
||||||
!!! info
|
!!! info
|
||||||
|
|
||||||
The file provider is the default format used throughout the documentation to show samples of the configuration for many features.
|
The file provider is the default format used throughout the documentation to show samples of the configuration for many features.
|
||||||
|
|
||||||
!!! tip
|
!!! tip
|
||||||
The file provider can be a good location for common elements you'd like to re-use from other providers; e.g. declaring whitelist middlewares, basic authentication, ...
|
|
||||||
|
The file provider can be a good solution for reusing common elements from other providers (e.g. declaring whitelist middlewares, basic authentication, ...)
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
|
@ -98,7 +98,7 @@ You can write one of these mutually exclusive configuration elements:
|
||||||
|
|
||||||
## Provider Configuration
|
## Provider Configuration
|
||||||
|
|
||||||
If you're in a hurry, maybe you'd rather go through the [dynamic configuration](../reference/dynamic-configuration/file.md) references and the [static configuration](../reference/static-configuration/overview.md).
|
For an overview of all the options that can be set with the file provider, see the [dynamic configuration](../reference/dynamic-configuration/file.md) and [static configuration](../reference/static-configuration/overview.md) references.
|
||||||
|
|
||||||
!!! warning "Limitations"
|
!!! warning "Limitations"
|
||||||
|
|
||||||
|
@ -107,13 +107,13 @@ If you're in a hurry, maybe you'd rather go through the [dynamic configuration](
|
||||||
If you use a mounted/bound file system in your orchestrator (like docker or kubernetes), the way the files are linked may be a source of errors.
|
If you use a mounted/bound file system in your orchestrator (like docker or kubernetes), the way the files are linked may be a source of errors.
|
||||||
If the link between the file systems is broken, when a source file/directory is changed/renamed, nothing will be reported to the linked file/directory, so the file system notifications will be neither triggered nor caught.
|
If the link between the file systems is broken, when a source file/directory is changed/renamed, nothing will be reported to the linked file/directory, so the file system notifications will be neither triggered nor caught.
|
||||||
|
|
||||||
For example, in docker, if the host file is renamed, the link to the mounted file will be broken and the container's file will not be updated.
|
For example, in Docker, if the host file is renamed, the link to the mounted file is broken and the container's file is no longer updated.
|
||||||
To avoid this kind of issue, a good practice is to:
|
To avoid this kind of issue, it is recommended to:
|
||||||
|
|
||||||
* set the Traefik [**directory**](#directory) configuration with the parent directory
|
* set the Traefik [**directory**](#directory) configuration with the parent directory
|
||||||
* mount/bind the parent directory
|
* mount/bind the parent directory
|
||||||
|
|
||||||
As it is very difficult to listen to all file system notifications, Traefik use [fsnotify](https://github.com/fsnotify/fsnotify).
|
As it is very difficult to listen to all file system notifications, Traefik uses [fsnotify](https://github.com/fsnotify/fsnotify).
|
||||||
If using a directory with a mounted directory does not fix your issue, please check your file system compatibility with fsnotify.
|
If using a directory with a mounted directory does not fix your issue, please check your file system compatibility with fsnotify.
|
||||||
|
|
||||||
### `filename`
|
### `filename`
|
||||||
|
@ -121,8 +121,9 @@ If you're in a hurry, maybe you'd rather go through the [dynamic configuration](
|
||||||
Defines the path to the configuration file.
|
Defines the path to the configuration file.
|
||||||
|
|
||||||
!!! warning ""
|
!!! warning ""
|
||||||
`filename` and `directory` are mutually exclusive.
|
|
||||||
The recommendation is to use `directory`.
|
The `filename` and `directory` options are mutually exclusive.
|
||||||
|
It is recommended to use `directory`.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers]
|
[providers]
|
||||||
|
@ -145,8 +146,9 @@ providers:
|
||||||
Defines the path to the directory that contains the configuration files.
|
Defines the path to the directory that contains the configuration files.
|
||||||
|
|
||||||
!!! warning ""
|
!!! warning ""
|
||||||
`filename` and `directory` are mutually exclusive.
|
|
||||||
The recommendation is to use `directory`.
|
The `filename` and `directory` options are mutually exclusive.
|
||||||
|
It is recommended to use `directory`.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers]
|
[providers]
|
||||||
|
@ -191,14 +193,15 @@ providers:
|
||||||
### Go Templating
|
### Go Templating
|
||||||
|
|
||||||
!!! warning
|
!!! warning
|
||||||
|
|
||||||
Go Templating only works with dedicated dynamic configuration files.
|
Go Templating only works with dedicated dynamic configuration files.
|
||||||
Templating does not work in the Traefik main static configuration file.
|
Templating does not work in the Traefik main static configuration file.
|
||||||
|
|
||||||
Traefik supports using Go templating to automatically generate repetitive portions of configuration files.
|
Traefik supports using Go templating to automatically generate repetitive sections of configuration files.
|
||||||
These sections must be valid [Go templates](https://golang.org/pkg/text/template/),
|
These sections must be a valid [Go template](https://golang.org/pkg/text/template/), and can use
|
||||||
augmented with the [Sprig template functions](http://masterminds.github.io/sprig/).
|
[sprig template functions](http://masterminds.github.io/sprig/).
|
||||||
|
|
||||||
To illustrate, it's possible to easily define multiple routers, services, and TLS certificates as described in the following examples:
|
To illustrate, it is possible to easily define multiple routers, services, and TLS certificates as described in the following examples:
|
||||||
|
|
||||||
??? example "Configuring Using Templating"
|
??? example "Configuring Using Templating"
|
||||||
|
|
||||||
|
@ -212,7 +215,6 @@ To illustrate, it's possible to easily define multiple routers, services, and TL
|
||||||
# ...
|
# ...
|
||||||
{{ end }}
|
{{ end }}
|
||||||
|
|
||||||
|
|
||||||
[http.services]
|
[http.services]
|
||||||
{{ range $i, $e := until 100 }}
|
{{ range $i, $e := until 100 }}
|
||||||
[http.services.service{{ $e }}]
|
[http.services.service{{ $e }}]
|
||||||
|
@ -227,7 +229,6 @@ To illustrate, it's possible to easily define multiple routers, services, and TL
|
||||||
# ...
|
# ...
|
||||||
{{ end }}
|
{{ end }}
|
||||||
|
|
||||||
|
|
||||||
[tcp.services]
|
[tcp.services]
|
||||||
{{ range $i, $e := until 100 }}
|
{{ range $i, $e := until 100 }}
|
||||||
[http.services.service{{ $e }}]
|
[http.services.service{{ $e }}]
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Traefik & HTTP
|
# Traefik & HTTP
|
||||||
|
|
||||||
Provide your [dynamic configuration](./overview.md) via an HTTP(s) endpoint and let Traefik do the rest!
|
Provide your [dynamic configuration](./overview.md) via an HTTP(S) endpoint and let Traefik do the rest!
|
||||||
|
|
||||||
## Routing Configuration
|
## Routing Configuration
|
||||||
|
|
||||||
|
@ -12,7 +12,7 @@ The HTTP provider uses the same configuration as the [File Provider](./file.md)
|
||||||
|
|
||||||
_Required_
|
_Required_
|
||||||
|
|
||||||
Defines the HTTP(s) endpoint to poll.
|
Defines the HTTP(S) endpoint to poll.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.http]
|
[providers.http]
|
||||||
|
@ -78,7 +78,7 @@ _Optional_
|
||||||
|
|
||||||
#### `tls.ca`
|
#### `tls.ca`
|
||||||
|
|
||||||
Certificate Authority used for the secured connection to the configured Endpoint.
|
Certificate Authority used for the secure connection to the configured endpoint.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.http.tls]
|
[providers.http.tls]
|
||||||
|
@ -98,12 +98,15 @@ providers:
|
||||||
|
|
||||||
#### `tls.caOptional`
|
#### `tls.caOptional`
|
||||||
|
|
||||||
Policy followed for the secured connection with TLS Client Authentication to the configured Endpoint.
|
The value of `tls.caOptional` defines which policy should be used for the secure connection with TLS Client Authentication to the configured endpoint.
|
||||||
Requires `tls.ca` to be defined.
|
|
||||||
|
|
||||||
- `true`: VerifyClientCertIfGiven
|
!!! warning ""
|
||||||
- `false`: RequireAndVerifyClientCert
|
|
||||||
- if `tls.ca` is undefined NoClientCert
|
If `tls.ca` is undefined, this option will be ignored, and no client certificate will be requested during the handshake. Any provided certificate will thus never be verified.
|
||||||
|
|
||||||
|
When this option is set to `true`, a client certificate is requested during the handshake but is not required. If a certificate is sent, it is required to be valid.
|
||||||
|
|
||||||
|
When this option is set to `false`, a client certificate is requested during the handshake, and at least one valid certificate should be sent by the client.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.http.tls]
|
[providers.http.tls]
|
||||||
|
@ -123,7 +126,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.cert`
|
#### `tls.cert`
|
||||||
|
|
||||||
Public certificate used for the secured connection to the configured Endpoint.
|
Public certificate used for the secure connection to the configured endpoint.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.http.tls]
|
[providers.http.tls]
|
||||||
|
@ -146,7 +149,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.key`
|
#### `tls.key`
|
||||||
|
|
||||||
Private certificate used for the secured connection to the configured Endpoint.
|
Private certificate used for the secure connection to the configured endpoint.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.http.tls]
|
[providers.http.tls]
|
||||||
|
@ -169,8 +172,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.insecureSkipVerify`
|
#### `tls.insecureSkipVerify`
|
||||||
|
|
||||||
If `insecureSkipVerify` is `true`, TLS connection to the configured Endpoint accepts any certificate presented by the
|
If `insecureSkipVerify` is `true`, the TLS connection to the endpoint accepts any certificate presented by the server regardless of the hostnames it covers.
|
||||||
server and any host name in that certificate.
|
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.http.tls]
|
[providers.http.tls]
|
||||||
|
|
|
@ -3,10 +3,11 @@
|
||||||
The Kubernetes Ingress Controller, The Custom Resource Way.
|
The Kubernetes Ingress Controller, The Custom Resource Way.
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
Traefik used to support Kubernetes only through the [Kubernetes Ingress provider](./kubernetes-ingress.md), which is a Kubernetes Ingress controller in the strict sense of the term.
|
In early versions, Traefik supported Kubernetes only through the [Kubernetes Ingress provider](./kubernetes-ingress.md), which is a Kubernetes Ingress controller in the strict sense of the term.
|
||||||
|
|
||||||
However, as the community expressed the need to benefit from Traefik features without resorting to (lots of) annotations,
|
However, as the community expressed the need to benefit from Traefik features without resorting to (lots of) annotations,
|
||||||
we ended up writing a [Custom Resource Definition](https://kubernetes.io/docs/concepts/extend-kubernetes/api-extension/custom-resources/) (alias CRD in the following) for an IngressRoute type, defined below, in order to provide a better way to configure access to a Kubernetes cluster.
|
the Traefik engineering team developed a [Custom Resource Definition](https://kubernetes.io/docs/concepts/extend-kubernetes/api-extension/custom-resources/)
|
||||||
|
(CRD) for an IngressRoute type, defined below, in order to provide a better way to configure access to a Kubernetes cluster.
|
||||||
|
|
||||||
## Configuration Requirements
|
## Configuration Requirements
|
||||||
|
|
||||||
|
@ -17,7 +18,7 @@ we ended up writing a [Custom Resource Definition](https://kubernetes.io/docs/co
|
||||||
* Use [Helm Chart](../getting-started/install-traefik.md#use-the-helm-chart) or use a custom Traefik Deployment
|
* Use [Helm Chart](../getting-started/install-traefik.md#use-the-helm-chart) or use a custom Traefik Deployment
|
||||||
* Enable the kubernetesCRD provider
|
* Enable the kubernetesCRD provider
|
||||||
* Apply the needed kubernetesCRD provider [configuration](#provider-configuration)
|
* Apply the needed kubernetesCRD provider [configuration](#provider-configuration)
|
||||||
* Add all needed traefik custom [resources](../reference/dynamic-configuration/kubernetes-crd.md#resources)
|
* Add all necessary Traefik custom [resources](../reference/dynamic-configuration/kubernetes-crd.md#resources)
|
||||||
|
|
||||||
??? example "Initializing Resource Definition and RBAC"
|
??? example "Initializing Resource Definition and RBAC"
|
||||||
|
|
||||||
|
@ -37,13 +38,11 @@ Traefik uses [Custom Resource Definition](https://kubernetes.io/docs/concepts/ex
|
||||||
Traefik Custom Resource Definitions are a Kubernetes implementation of the Traefik concepts. The main particularities are:
|
Traefik Custom Resource Definitions are a Kubernetes implementation of the Traefik concepts. The main particularities are:
|
||||||
|
|
||||||
* The usage of `name` **and** `namespace` to refer to another Kubernetes resource.
|
* The usage of `name` **and** `namespace` to refer to another Kubernetes resource.
|
||||||
* The usage of [secret](https://kubernetes.io/docs/concepts/configuration/secret/) for sensible data like:
|
* The usage of [secret](https://kubernetes.io/docs/concepts/configuration/secret/) for sensitive data (TLS certificates and credentials).
|
||||||
* TLS certificate.
|
|
||||||
* Authentication data.
|
|
||||||
* The structure of the configuration.
|
* The structure of the configuration.
|
||||||
* The obligation to declare all the [definitions](../reference/dynamic-configuration/kubernetes-crd.md#definitions).
|
* The requirement to declare all the [definitions](../reference/dynamic-configuration/kubernetes-crd.md#definitions).
|
||||||
|
|
||||||
The Traefik CRD are building blocks which you can assemble according to your needs.
|
The Traefik CRDs are building blocks that you can assemble according to your needs.
|
||||||
See the list of CRDs in the dedicated [routing section](../routing/providers/kubernetes-crd.md).
|
See the list of CRDs in the dedicated [routing section](../routing/providers/kubernetes-crd.md).
|
||||||
|
|
||||||
## LetsEncrypt Support with the Custom Resource Definition Provider
|
## LetsEncrypt Support with the Custom Resource Definition Provider
|
||||||
|
@ -51,23 +50,36 @@ See the list of CRDs in the dedicated [routing section](../routing/providers/kub
|
||||||
By design, Traefik is a stateless application, meaning that it only derives its configuration from the environment it runs in, without additional configuration.
|
By design, Traefik is a stateless application, meaning that it only derives its configuration from the environment it runs in, without additional configuration.
|
||||||
For this reason, users can run multiple instances of Traefik at the same time to achieve HA, as is a common pattern in the kubernetes ecosystem.
|
For this reason, users can run multiple instances of Traefik at the same time to achieve HA, as is a common pattern in the kubernetes ecosystem.
|
||||||
|
|
||||||
When using a single instance of Traefik with LetsEncrypt, no issues should be encountered, however this could be a single point of failure.
|
When using a single instance of Traefik with Let's Encrypt, you should encounter no issues. However, this could be a single point of failure.
|
||||||
Unfortunately, it is not possible to run multiple instances of Traefik 2.0 with LetsEncrypt enabled, because there is no way to ensure that the correct instance of Traefik will receive the challenge request, and subsequent responses.
|
Unfortunately, it is not possible to run multiple instances of Traefik Proxy 2.0 with Let's Encrypt enabled, because there is no way to ensure that the correct instance of Traefik will receive the challenge request and subsequent responses.
|
||||||
Previous versions of Traefik used a [KV store](https://doc.traefik.io/traefik/v1.7/configuration/acme/#storage) to attempt to achieve this, but due to sub-optimal performance was dropped as a feature in 2.0.
|
Previous versions of Traefik used a [KV store](https://doc.traefik.io/traefik/v1.7/configuration/acme/#storage) to attempt to achieve this, but due to sub-optimal performance that feature was dropped in 2.0.
|
||||||
|
|
||||||
If you require LetsEncrypt with HA in a kubernetes environment, we recommend using [Traefik Enterprise](https://traefik.io/traefik-enterprise/) where distributed LetsEncrypt is a supported feature.
|
If you need Let's Encrypt with HA in a Kubernetes environment, we recommend using [Traefik Enterprise](https://traefik.io/traefik-enterprise/), which includes distributed Let's Encrypt as a supported feature.
|
||||||
|
|
||||||
If you want to continue to run Traefik Community Edition, LetsEncrypt HA can be achieved by using a Certificate Controller such as [Cert-Manager](https://docs.cert-manager.io/en/latest/index.html).
|
If you want to keep using Traefik Proxy, high availability for Let's Encrypt can be achieved by using a Certificate Controller such as [Cert-Manager](https://docs.cert-manager.io/en/latest/index.html).
|
||||||
When using Cert-Manager to manage certificates, it will create secrets in your namespaces that can be referenced as TLS secrets in your [ingress objects](https://kubernetes.io/docs/concepts/services-networking/ingress/#tls).
|
When using Cert-Manager to manage certificates, it creates secrets in your namespaces that can be referenced as TLS secrets in your [ingress objects](https://kubernetes.io/docs/concepts/services-networking/ingress/#tls).
|
||||||
When using the Traefik Kubernetes CRD Provider, unfortunately Cert-Manager cannot interface directly with the CRDs _yet_, but this is being worked on by our team.
|
When using the Traefik Kubernetes CRD Provider, unfortunately Cert-Manager cannot yet interface directly with the CRDs.
|
||||||
A workaround is to enable the [Kubernetes Ingress provider](./kubernetes-ingress.md) to allow Cert-Manager to create ingress objects to complete the challenges.
|
A workaround is to enable the [Kubernetes Ingress provider](./kubernetes-ingress.md) to allow Cert-Manager to create ingress objects to complete the challenges.
|
||||||
Please note that this still requires manual intervention to create the certificates through Cert-Manager, but once created, Cert-Manager will keep the certificate renewed.
|
Please note that this still requires manual intervention to create the certificates through Cert-Manager, but once the certificates are created, Cert-Manager keeps them renewed.
|
||||||
|
|
||||||
## Provider Configuration
|
## Provider Configuration
|
||||||
|
|
||||||
### `endpoint`
|
### `endpoint`
|
||||||
|
|
||||||
_Optional, Default=empty_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
The Kubernetes server endpoint URL.
|
||||||
|
|
||||||
|
When deployed into Kubernetes, Traefik reads the environment variables `KUBERNETES_SERVICE_HOST` and `KUBERNETES_SERVICE_PORT` or `KUBECONFIG` to construct the endpoint.
|
||||||
|
|
||||||
|
The access token is looked up in `/var/run/secrets/kubernetes.io/serviceaccount/token` and the SSL CA certificate in `/var/run/secrets/kubernetes.io/serviceaccount/ca.crt`.
|
||||||
|
Both are mounted automatically when deployed inside Kubernetes.
|
||||||
|
|
||||||
|
The endpoint may be specified to override the environment variable values inside a cluster.
|
||||||
|
|
||||||
|
When the environment variables are not found, Traefik tries to connect to the Kubernetes API server with an external-cluster client.
|
||||||
|
In this case, the endpoint is required.
|
||||||
|
Specifically, it may be set to the URL used by `kubectl proxy` to connect to a Kubernetes cluster using the granted authentication and authorization of the associated kubeconfig.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesCRD]
|
[providers.kubernetesCRD]
|
||||||
|
@ -86,22 +98,11 @@ providers:
|
||||||
--providers.kubernetescrd.endpoint=http://localhost:8080
|
--providers.kubernetescrd.endpoint=http://localhost:8080
|
||||||
```
|
```
|
||||||
|
|
||||||
The Kubernetes server endpoint as URL.
|
|
||||||
|
|
||||||
When deployed into Kubernetes, Traefik will read the environment variables `KUBERNETES_SERVICE_HOST` and `KUBERNETES_SERVICE_PORT` or `KUBECONFIG` to construct the endpoint.
|
|
||||||
|
|
||||||
The access token will be looked up in `/var/run/secrets/kubernetes.io/serviceaccount/token` and the SSL CA certificate in `/var/run/secrets/kubernetes.io/serviceaccount/ca.crt`.
|
|
||||||
Both are provided mounted automatically when deployed inside Kubernetes.
|
|
||||||
|
|
||||||
The endpoint may be specified to override the environment variable values inside a cluster.
|
|
||||||
|
|
||||||
When the environment variables are not found, Traefik will try to connect to the Kubernetes API server with an external-cluster client.
|
|
||||||
In this case, the endpoint is required.
|
|
||||||
Specifically, it may be set to the URL used by `kubectl proxy` to connect to a Kubernetes cluster using the granted authentication and authorization of the associated kubeconfig.
|
|
||||||
|
|
||||||
### `token`
|
### `token`
|
||||||
|
|
||||||
_Optional, Default=empty_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Bearer token used for the Kubernetes client configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesCRD]
|
[providers.kubernetesCRD]
|
||||||
|
@ -120,11 +121,12 @@ providers:
|
||||||
--providers.kubernetescrd.token=mytoken
|
--providers.kubernetescrd.token=mytoken
|
||||||
```
|
```
|
||||||
|
|
||||||
Bearer token used for the Kubernetes client configuration.
|
|
||||||
|
|
||||||
### `certAuthFilePath`
|
### `certAuthFilePath`
|
||||||
|
|
||||||
_Optional, Default=empty_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Path to the certificate authority file.
|
||||||
|
Used for the Kubernetes client configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesCRD]
|
[providers.kubernetesCRD]
|
||||||
|
@ -143,12 +145,12 @@ providers:
|
||||||
--providers.kubernetescrd.certauthfilepath=/my/ca.crt
|
--providers.kubernetescrd.certauthfilepath=/my/ca.crt
|
||||||
```
|
```
|
||||||
|
|
||||||
Path to the certificate authority file.
|
|
||||||
Used for the Kubernetes client configuration.
|
|
||||||
|
|
||||||
### `namespaces`
|
### `namespaces`
|
||||||
|
|
||||||
_Optional, Default: all namespaces (empty array)_
|
_Optional, Default: []_
|
||||||
|
|
||||||
|
Array of namespaces to watch.
|
||||||
|
If left empty, watches all namespaces if the value of `namespaces`.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesCRD]
|
[providers.kubernetesCRD]
|
||||||
|
@ -169,11 +171,20 @@ providers:
|
||||||
--providers.kubernetescrd.namespaces=default,production
|
--providers.kubernetescrd.namespaces=default,production
|
||||||
```
|
```
|
||||||
|
|
||||||
Array of namespaces to watch.
|
|
||||||
|
|
||||||
### `labelselector`
|
### `labelselector`
|
||||||
|
|
||||||
_Optional,Default: empty (process all resources)_
|
_Optional, Default: ""_
|
||||||
|
|
||||||
|
A label selector can be defined to filter on specific resource objects only,
|
||||||
|
this applies only to Traefik [Custom Resources](../routing/providers/kubernetes-crd.md#custom-resource-definition-crd)
|
||||||
|
and has no effect on Kubernetes `Secrets`, `Endpoints` and `Services`.
|
||||||
|
If left empty, Traefik processes all resource objects in the configured namespaces.
|
||||||
|
|
||||||
|
See [label-selectors](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors) for details.
|
||||||
|
|
||||||
|
!!! warning
|
||||||
|
|
||||||
|
Because the label selector is applied to all Traefik Custom Resources, they all must match the filter.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesCRD]
|
[providers.kubernetesCRD]
|
||||||
|
@ -192,20 +203,14 @@ providers:
|
||||||
--providers.kubernetescrd.labelselector="app=traefik"
|
--providers.kubernetescrd.labelselector="app=traefik"
|
||||||
```
|
```
|
||||||
|
|
||||||
By default, Traefik processes all resource objects in the configured namespaces.
|
|
||||||
A label selector can be defined to filter on specific resource objects only,
|
|
||||||
this will apply only on Traefik [Custom Resources](../routing/providers/kubernetes-crd.md#custom-resource-definition-crd)
|
|
||||||
and has no effect on Kubernetes `Secrets`, `Endpoints` and `Services`.
|
|
||||||
|
|
||||||
See [label-selectors](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors) for details.
|
|
||||||
|
|
||||||
!!! warning
|
|
||||||
|
|
||||||
As the LabelSelector is applied to all Traefik Custom Resources, they all must match the filter.
|
|
||||||
|
|
||||||
### `ingressClass`
|
### `ingressClass`
|
||||||
|
|
||||||
_Optional, Default: empty_
|
_Optional, Default: ""_
|
||||||
|
|
||||||
|
Value of `kubernetes.io/ingress.class` annotation that identifies resource objects to be processed.
|
||||||
|
|
||||||
|
If the parameter is set, only resources containing an annotation with the same value are processed.
|
||||||
|
Otherwise, resources missing the annotation, having an empty value, or the value `traefik` are processed.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesCRD]
|
[providers.kubernetesCRD]
|
||||||
|
@ -224,14 +229,17 @@ providers:
|
||||||
--providers.kubernetescrd.ingressclass=traefik-internal
|
--providers.kubernetescrd.ingressclass=traefik-internal
|
||||||
```
|
```
|
||||||
|
|
||||||
Value of `kubernetes.io/ingress.class` annotation that identifies resource objects to be processed.
|
|
||||||
|
|
||||||
If the parameter is non-empty, only resources containing an annotation with the same value are processed.
|
|
||||||
Otherwise, resources missing the annotation, having an empty value, or the value `traefik` are processed.
|
|
||||||
|
|
||||||
### `throttleDuration`
|
### `throttleDuration`
|
||||||
|
|
||||||
_Optional, Default: 0 (no throttling)_
|
_Optional, Default: 0_
|
||||||
|
|
||||||
|
The `throttleDuration` option defines how often the provider is allowed to handle events from Kubernetes. This prevents
|
||||||
|
a Kubernetes cluster that updates many times per second from continuously changing your Traefik configuration.
|
||||||
|
|
||||||
|
If left empty, the provider does not apply any throttling and does not drop any Kubernetes events.
|
||||||
|
|
||||||
|
The value of `throttleDuration` should be provided in seconds or as a valid duration format,
|
||||||
|
see [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesCRD]
|
[providers.kubernetesCRD]
|
||||||
|
@ -254,6 +262,12 @@ providers:
|
||||||
|
|
||||||
_Optional, Default: true_
|
_Optional, Default: true_
|
||||||
|
|
||||||
|
If the parameter is set to `false`, IngressRoutes are not able to reference any resources in other namespaces than theirs.
|
||||||
|
|
||||||
|
!!! warning "Deprecation"
|
||||||
|
|
||||||
|
Please note that the default value for this option will be set to `false` in a future version.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesCRD]
|
[providers.kubernetesCRD]
|
||||||
allowCrossNamespace = false
|
allowCrossNamespace = false
|
||||||
|
@ -271,13 +285,6 @@ providers:
|
||||||
--providers.kubernetescrd.allowCrossNamespace=false
|
--providers.kubernetescrd.allowCrossNamespace=false
|
||||||
```
|
```
|
||||||
|
|
||||||
If the parameter is set to `false`, an IngressRoute will not be able to reference any resources
|
## Full Example
|
||||||
in another namespace than the IngressRoute namespace.
|
|
||||||
|
|
||||||
!!! warning "Deprecation"
|
For additional information, refer to the [full example](../user-guides/crd-acme/index.md) with Let's Encrypt.
|
||||||
|
|
||||||
Please notice that the default value for this option will be set to `false` in a future version.
|
|
||||||
|
|
||||||
## Further
|
|
||||||
|
|
||||||
Also see the [full example](../user-guides/crd-acme/index.md) with Let's Encrypt.
|
|
||||||
|
|
|
@ -3,17 +3,17 @@
|
||||||
The Kubernetes Gateway API, The Experimental Way.
|
The Kubernetes Gateway API, The Experimental Way.
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
Gateway API is the evolution of Kubernetes APIs that relate to `Services`, e.g. `Ingress`.
|
Gateway API is the evolution of Kubernetes APIs that relate to `Services`, such as `Ingress`.
|
||||||
The Gateway API project is part of Kubernetes, working under SIG-NETWORK.
|
The Gateway API project is part of Kubernetes, working under SIG-NETWORK.
|
||||||
|
|
||||||
The Kubernetes Gateway provider is a Traefik implementation of the [service apis](https://github.com/kubernetes-sigs/service-apis)
|
The Kubernetes Gateway provider is a Traefik implementation of the [Service APIs](https://kubernetes-sigs.github.io/gateway-api/)
|
||||||
specifications from the Kubernetes SIGs.
|
specifications from the Kubernetes Special Interest Groups (SIGs).
|
||||||
|
|
||||||
This provider is proposed as an experimental feature and partially supports the service apis [v0.1.0](https://github.com/kubernetes-sigs/service-apis/releases/tag/v0.1.0) specification.
|
This provider is proposed as an experimental feature and partially supports the Service APIs [v0.1.0](https://github.com/kubernetes-sigs/service-apis/releases/tag/v0.1.0) specification.
|
||||||
|
|
||||||
!!! warning "Enabling The Experimental Kubernetes Gateway Provider"
|
!!! warning "Enabling The Experimental Kubernetes Gateway Provider"
|
||||||
|
|
||||||
As this provider is in experimental stage, it needs to be activated in the experimental section of the static configuration.
|
Since this provider is still experimental, it needs to be activated in the experimental section of the static configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[experimental]
|
[experimental]
|
||||||
|
@ -71,25 +71,27 @@ This provider is proposed as an experimental feature and partially supports the
|
||||||
--8<-- "content/reference/dynamic-configuration/kubernetes-gateway-rbac.yml"
|
--8<-- "content/reference/dynamic-configuration/kubernetes-gateway-rbac.yml"
|
||||||
```
|
```
|
||||||
|
|
||||||
The Kubernetes Service APIs provides several [guides](https://kubernetes-sigs.github.io/service-apis/guides/) of how to use their API.
|
The Kubernetes Service APIs project provides several [guides](https://kubernetes-sigs.github.io/gateway-api/guides/) on how to use the APIs.
|
||||||
Those guides will help you to go further than the example above.
|
These guides can help you to go further than the example above.
|
||||||
The [getting started](https://kubernetes-sigs.github.io/service-apis/getting-started/) show you how to install the CRDs from their repository.
|
The [getting started guide](https://kubernetes-sigs.github.io/gateway-api/getting-started/) details how to install the CRDs from their repository.
|
||||||
Thus, keep in mind that the Traefik Gateway provider only supports the `v0.1.0`.
|
|
||||||
|
|
||||||
For now, the Traefik Gateway Provider could be used to achieve the following set-up guides:
|
!!! note ""
|
||||||
|
|
||||||
* [Simple Gateway](https://kubernetes-sigs.github.io/service-apis/simple-gateway/)
|
Keep in mind that the Traefik Gateway provider only supports the `v0.1.0`.
|
||||||
* [HTTP routing](https://kubernetes-sigs.github.io/service-apis/http-routing/)
|
|
||||||
* [TLS](https://kubernetes-sigs.github.io/service-apis/tls/) (Partial support: only on listeners with terminate mode)
|
For now, the Traefik Gateway Provider can be used while following the below guides:
|
||||||
|
|
||||||
|
* [Simple Gateway](https://kubernetes-sigs.github.io/gateway-api/simple-gateway/)
|
||||||
|
* [HTTP routing](https://kubernetes-sigs.github.io/gateway-api/http-routing/)
|
||||||
|
* [TLS](https://kubernetes-sigs.github.io/gateway-api/tls/) (Partial support: only on listeners with terminate mode)
|
||||||
|
|
||||||
## Resource Configuration
|
## Resource Configuration
|
||||||
|
|
||||||
When using Kubernetes Gateway API as a provider,
|
When using Kubernetes Gateway API as a provider, Traefik uses Kubernetes
|
||||||
Traefik uses Kubernetes
|
[Custom Resource Definitions](https://kubernetes.io/docs/concepts/extend-kubernetes/api-extension/custom-resources/)
|
||||||
[Custom Resource Definition](https://kubernetes.io/docs/concepts/extend-kubernetes/api-extension/custom-resources/)
|
|
||||||
to retrieve its routing configuration.
|
to retrieve its routing configuration.
|
||||||
|
|
||||||
All concepts can be found in the official API concepts [documentation](https://kubernetes-sigs.github.io/service-apis/api-overview/).
|
All concepts can be found in the official API concepts [documentation](https://kubernetes-sigs.github.io/gateway-api/api-overview/).
|
||||||
Traefik implements the following resources:
|
Traefik implements the following resources:
|
||||||
|
|
||||||
* `GatewayClass` defines a set of Gateways that share a common configuration and behaviour.
|
* `GatewayClass` defines a set of Gateways that share a common configuration and behaviour.
|
||||||
|
@ -100,7 +102,20 @@ Traefik implements the following resources:
|
||||||
|
|
||||||
### `endpoint`
|
### `endpoint`
|
||||||
|
|
||||||
_Optional, Default=empty_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
The Kubernetes server endpoint URL.
|
||||||
|
|
||||||
|
When deployed into Kubernetes, Traefik reads the environment variables `KUBERNETES_SERVICE_HOST` and `KUBERNETES_SERVICE_PORT` or `KUBECONFIG` to construct the endpoint.
|
||||||
|
|
||||||
|
The access token is looked up in `/var/run/secrets/kubernetes.io/serviceaccount/token` and the SSL CA certificate in `/var/run/secrets/kubernetes.io/serviceaccount/ca.crt`.
|
||||||
|
Both are mounted automatically when deployed inside Kubernetes.
|
||||||
|
|
||||||
|
The endpoint may be specified to override the environment variable values inside a cluster.
|
||||||
|
|
||||||
|
When the environment variables are not found, Traefik tries to connect to the Kubernetes API server with an external-cluster client.
|
||||||
|
In this case, the endpoint is required.
|
||||||
|
Specifically, it may be set to the URL used by `kubectl proxy` to connect to a Kubernetes cluster using the granted authentication and authorization of the associated kubeconfig.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesGateway]
|
[providers.kubernetesGateway]
|
||||||
|
@ -119,22 +134,11 @@ providers:
|
||||||
--providers.kubernetesgateway.endpoint=http://localhost:8080
|
--providers.kubernetesgateway.endpoint=http://localhost:8080
|
||||||
```
|
```
|
||||||
|
|
||||||
The Kubernetes server endpoint as URL.
|
|
||||||
|
|
||||||
When deployed into Kubernetes, Traefik will read the environment variables `KUBERNETES_SERVICE_HOST` and `KUBERNETES_SERVICE_PORT` or `KUBECONFIG` to construct the endpoint.
|
|
||||||
|
|
||||||
The access token will be looked up in `/var/run/secrets/kubernetes.io/serviceaccount/token` and the SSL CA certificate in `/var/run/secrets/kubernetes.io/serviceaccount/ca.crt`.
|
|
||||||
Both are mounted automatically when deployed inside Kubernetes.
|
|
||||||
|
|
||||||
The endpoint may be specified to override the environment variable values inside a cluster.
|
|
||||||
|
|
||||||
When the environment variables are not found, Traefik will try to connect to the Kubernetes API server with an external-cluster client.
|
|
||||||
In this case, the endpoint is required.
|
|
||||||
Specifically, it may be set to the URL used by `kubectl proxy` to connect to a Kubernetes cluster using the granted authentication and authorization of the associated kubeconfig.
|
|
||||||
|
|
||||||
### `token`
|
### `token`
|
||||||
|
|
||||||
_Optional, Default=empty_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Bearer token used for the Kubernetes client configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesGateway]
|
[providers.kubernetesGateway]
|
||||||
|
@ -153,11 +157,12 @@ providers:
|
||||||
--providers.kubernetesgateway.token=mytoken
|
--providers.kubernetesgateway.token=mytoken
|
||||||
```
|
```
|
||||||
|
|
||||||
Bearer token used for the Kubernetes client configuration.
|
|
||||||
|
|
||||||
### `certAuthFilePath`
|
### `certAuthFilePath`
|
||||||
|
|
||||||
_Optional, Default=empty_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Path to the certificate authority file.
|
||||||
|
Used for the Kubernetes client configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesGateway]
|
[providers.kubernetesGateway]
|
||||||
|
@ -176,12 +181,12 @@ providers:
|
||||||
--providers.kubernetesgateway.certauthfilepath=/my/ca.crt
|
--providers.kubernetesgateway.certauthfilepath=/my/ca.crt
|
||||||
```
|
```
|
||||||
|
|
||||||
Path to the certificate authority file.
|
|
||||||
Used for the Kubernetes client configuration.
|
|
||||||
|
|
||||||
### `namespaces`
|
### `namespaces`
|
||||||
|
|
||||||
_Optional, Default: all namespaces (empty array)_
|
_Optional, Default: []_
|
||||||
|
|
||||||
|
Array of namespaces to watch.
|
||||||
|
If left empty, watches all namespaces if the value of `namespaces`.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesGateway]
|
[providers.kubernetesGateway]
|
||||||
|
@ -202,11 +207,14 @@ providers:
|
||||||
--providers.kubernetesgateway.namespaces=default,production
|
--providers.kubernetesgateway.namespaces=default,production
|
||||||
```
|
```
|
||||||
|
|
||||||
Array of namespaces to watch.
|
|
||||||
|
|
||||||
### `labelselector`
|
### `labelselector`
|
||||||
|
|
||||||
_Optional, Default: empty (process all resources)_
|
_Optional, Default: ""_
|
||||||
|
|
||||||
|
A label selector can be defined to filter on specific GatewayClass objects only.
|
||||||
|
If left empty, Traefik processes all GatewayClass objects in the configured namespaces.
|
||||||
|
|
||||||
|
See [label-selectors](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors) for details.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesGateway]
|
[providers.kubernetesGateway]
|
||||||
|
@ -225,14 +233,17 @@ providers:
|
||||||
--providers.kubernetesgateway.labelselector="app=traefik"
|
--providers.kubernetesgateway.labelselector="app=traefik"
|
||||||
```
|
```
|
||||||
|
|
||||||
By default, Traefik processes all resource objects in the configured namespaces.
|
|
||||||
A label selector can be defined to filter on specific GatewayClass objects only.
|
|
||||||
|
|
||||||
See [label-selectors](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors) for details.
|
|
||||||
|
|
||||||
### `throttleDuration`
|
### `throttleDuration`
|
||||||
|
|
||||||
_Optional, Default: 0 (no throttling)_
|
_Optional, Default: 0_
|
||||||
|
|
||||||
|
The `throttleDuration` option defines how often the provider is allowed to handle events from Kubernetes. This prevents
|
||||||
|
a Kubernetes cluster that updates many times per second from continuously changing your Traefik configuration.
|
||||||
|
|
||||||
|
If left empty, the provider does not apply any throttling and does not drop any Kubernetes events.
|
||||||
|
|
||||||
|
The value of `throttleDuration` should be provided in seconds or as a valid duration format,
|
||||||
|
see [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesGateway]
|
[providers.kubernetesGateway]
|
||||||
|
|
|
@ -12,7 +12,7 @@ See the dedicated section in [routing](../routing/providers/kubernetes-ingress.m
|
||||||
|
|
||||||
## Enabling and Using the Provider
|
## Enabling and Using the Provider
|
||||||
|
|
||||||
As usual, the provider is enabled through the static configuration:
|
You can enable the provider in the static configuration:
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesIngress]
|
[providers.kubernetesIngress]
|
||||||
|
@ -29,7 +29,7 @@ providers:
|
||||||
|
|
||||||
The provider then watches for incoming ingresses events, such as the example below,
|
The provider then watches for incoming ingresses events, such as the example below,
|
||||||
and derives the corresponding dynamic configuration from it,
|
and derives the corresponding dynamic configuration from it,
|
||||||
which in turn will create the resulting routers, services, handlers, etc.
|
which in turn creates the resulting routers, services, handlers, etc.
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
```yaml tab="File (YAML)"
|
||||||
kind: Ingress
|
kind: Ingress
|
||||||
|
@ -61,26 +61,39 @@ without additional configuration.
|
||||||
For this reason, users can run multiple instances of Traefik at the same time to achieve HA,
|
For this reason, users can run multiple instances of Traefik at the same time to achieve HA,
|
||||||
as is a common pattern in the kubernetes ecosystem.
|
as is a common pattern in the kubernetes ecosystem.
|
||||||
|
|
||||||
When using a single instance of Traefik with LetsEncrypt, no issues should be encountered,
|
When using a single instance of Traefik Proxy with Let's Encrypt, you should encounter no issues.
|
||||||
however this could be a single point of failure.
|
However, this could be a single point of failure.
|
||||||
Unfortunately, it is not possible to run multiple instances of Traefik 2.0 with LetsEncrypt enabled,
|
Unfortunately, it is not possible to run multiple instances of Traefik 2.0 with Let's Encrypt enabled,
|
||||||
because there is no way to ensure that the correct instance of Traefik will receive the challenge request, and subsequent responses.
|
because there is no way to ensure that the correct instance of Traefik receives the challenge request, and subsequent responses.
|
||||||
Previous versions of Traefik used a [KV store](https://doc.traefik.io/traefik/v1.7/configuration/acme/#storage) to attempt to achieve this,
|
Previous versions of Traefik used a [KV store](https://doc.traefik.io/traefik/v1.7/configuration/acme/#storage) to attempt to achieve this,
|
||||||
but due to sub-optimal performance was dropped as a feature in 2.0.
|
but due to sub-optimal performance that feature was dropped in 2.0.
|
||||||
|
|
||||||
If you require LetsEncrypt with HA in a kubernetes environment,
|
If you need Let's Encrypt with high availability in a Kubernetes environment,
|
||||||
we recommend using [Traefik Enterprise](https://traefik.io/traefik-enterprise/) where distributed LetsEncrypt is a supported feature.
|
we recommend using [Traefik Enterprise](https://traefik.io/traefik-enterprise/) which includes distributed Let's Encrypt as a supported feature.
|
||||||
|
|
||||||
If you are wanting to continue to run Traefik Community Edition,
|
If you want to keep using Traefik Proxy,
|
||||||
LetsEncrypt HA can be achieved by using a Certificate Controller such as [Cert-Manager](https://docs.cert-manager.io/en/latest/index.html).
|
LetsEncrypt HA can be achieved by using a Certificate Controller such as [Cert-Manager](https://docs.cert-manager.io/en/latest/index.html).
|
||||||
When using Cert-Manager to manage certificates,
|
When using Cert-Manager to manage certificates,
|
||||||
it will create secrets in your namespaces that can be referenced as TLS secrets in your [ingress objects](https://kubernetes.io/docs/concepts/services-networking/ingress/#tls).
|
it creates secrets in your namespaces that can be referenced as TLS secrets in your [ingress objects](https://kubernetes.io/docs/concepts/services-networking/ingress/#tls).
|
||||||
|
|
||||||
## Provider Configuration
|
## Provider Configuration
|
||||||
|
|
||||||
### `endpoint`
|
### `endpoint`
|
||||||
|
|
||||||
_Optional, Default=empty_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
The Kubernetes server endpoint URL.
|
||||||
|
|
||||||
|
When deployed into Kubernetes, Traefik reads the environment variables `KUBERNETES_SERVICE_HOST` and `KUBERNETES_SERVICE_PORT` or `KUBECONFIG` to construct the endpoint.
|
||||||
|
|
||||||
|
The access token is looked up in `/var/run/secrets/kubernetes.io/serviceaccount/token` and the SSL CA certificate in `/var/run/secrets/kubernetes.io/serviceaccount/ca.crt`.
|
||||||
|
Both are mounted automatically when deployed inside Kubernetes.
|
||||||
|
|
||||||
|
The endpoint may be specified to override the environment variable values inside a cluster.
|
||||||
|
|
||||||
|
When the environment variables are not found, Traefik tries to connect to the Kubernetes API server with an external-cluster client.
|
||||||
|
In this case, the endpoint is required.
|
||||||
|
Specifically, it may be set to the URL used by `kubectl proxy` to connect to a Kubernetes cluster using the granted authentication and authorization of the associated kubeconfig.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesIngress]
|
[providers.kubernetesIngress]
|
||||||
|
@ -99,21 +112,11 @@ providers:
|
||||||
--providers.kubernetesingress.endpoint=http://localhost:8080
|
--providers.kubernetesingress.endpoint=http://localhost:8080
|
||||||
```
|
```
|
||||||
|
|
||||||
The Kubernetes server endpoint as URL, which is only used when the behavior based on environment variables described below does not apply.
|
|
||||||
|
|
||||||
When deployed into Kubernetes, Traefik reads the environment variables `KUBERNETES_SERVICE_HOST` and `KUBERNETES_SERVICE_PORT` or `KUBECONFIG` to construct the endpoint.
|
|
||||||
|
|
||||||
The access token is looked up in `/var/run/secrets/kubernetes.io/serviceaccount/token` and the SSL CA certificate in `/var/run/secrets/kubernetes.io/serviceaccount/ca.crt`.
|
|
||||||
They are both provided automatically as mounts in the pod where Traefik is deployed.
|
|
||||||
|
|
||||||
When the environment variables are not found, Traefik tries to connect to the Kubernetes API server with an external-cluster client.
|
|
||||||
In which case, the endpoint is required.
|
|
||||||
Specifically, it may be set to the URL used by `kubectl proxy` to connect to a Kubernetes cluster using the granted authentication
|
|
||||||
and authorization of the associated kubeconfig.
|
|
||||||
|
|
||||||
### `token`
|
### `token`
|
||||||
|
|
||||||
_Optional, Default=empty_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Bearer token used for the Kubernetes client configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesIngress]
|
[providers.kubernetesIngress]
|
||||||
|
@ -132,11 +135,12 @@ providers:
|
||||||
--providers.kubernetesingress.token=mytoken
|
--providers.kubernetesingress.token=mytoken
|
||||||
```
|
```
|
||||||
|
|
||||||
Bearer token used for the Kubernetes client configuration.
|
|
||||||
|
|
||||||
### `certAuthFilePath`
|
### `certAuthFilePath`
|
||||||
|
|
||||||
_Optional, Default=empty_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Path to the certificate authority file.
|
||||||
|
Used for the Kubernetes client configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesIngress]
|
[providers.kubernetesIngress]
|
||||||
|
@ -155,12 +159,12 @@ providers:
|
||||||
--providers.kubernetesingress.certauthfilepath=/my/ca.crt
|
--providers.kubernetesingress.certauthfilepath=/my/ca.crt
|
||||||
```
|
```
|
||||||
|
|
||||||
Path to the certificate authority file.
|
|
||||||
Used for the Kubernetes client configuration.
|
|
||||||
|
|
||||||
### `namespaces`
|
### `namespaces`
|
||||||
|
|
||||||
_Optional, Default: all namespaces (empty array)_
|
_Optional, Default: []_
|
||||||
|
|
||||||
|
Array of namespaces to watch.
|
||||||
|
If left empty, watches all namespaces if the value of `namespaces`.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesIngress]
|
[providers.kubernetesIngress]
|
||||||
|
@ -181,11 +185,14 @@ providers:
|
||||||
--providers.kubernetesingress.namespaces=default,production
|
--providers.kubernetesingress.namespaces=default,production
|
||||||
```
|
```
|
||||||
|
|
||||||
Array of namespaces to watch.
|
|
||||||
|
|
||||||
### `labelSelector`
|
### `labelSelector`
|
||||||
|
|
||||||
_Optional,Default: empty (process all Ingresses)_
|
_Optional, Default: ""_
|
||||||
|
|
||||||
|
A label selector can be defined to filter on specific Ingress objects only.
|
||||||
|
If left empty, Traefik processes all Ingress objects in the configured namespaces.
|
||||||
|
|
||||||
|
See [label-selectors](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors) for details.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesIngress]
|
[providers.kubernetesIngress]
|
||||||
|
@ -204,36 +211,14 @@ providers:
|
||||||
--providers.kubernetesingress.labelselector="app=traefik"
|
--providers.kubernetesingress.labelselector="app=traefik"
|
||||||
```
|
```
|
||||||
|
|
||||||
By default, Traefik processes all `Ingress` objects in the configured namespaces.
|
|
||||||
A label selector can be defined to filter on specific `Ingress` objects only.
|
|
||||||
|
|
||||||
See [label-selectors](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors) for details.
|
|
||||||
|
|
||||||
### `ingressClass`
|
### `ingressClass`
|
||||||
|
|
||||||
_Optional, Default: empty_
|
_Optional, Default: ""_
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
|
||||||
[providers.kubernetesIngress]
|
|
||||||
ingressClass = "traefik-internal"
|
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
|
||||||
providers:
|
|
||||||
kubernetesIngress:
|
|
||||||
ingressClass: "traefik-internal"
|
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
```bash tab="CLI"
|
|
||||||
--providers.kubernetesingress.ingressclass=traefik-internal
|
|
||||||
```
|
|
||||||
|
|
||||||
Value of `kubernetes.io/ingress.class` annotation that identifies Ingress objects to be processed.
|
Value of `kubernetes.io/ingress.class` annotation that identifies Ingress objects to be processed.
|
||||||
|
|
||||||
If the parameter is non-empty, only Ingresses containing an annotation with the same value are processed.
|
If the parameter is set, only Ingresses containing an annotation with the same value are processed.
|
||||||
Otherwise, Ingresses missing the annotation, having an empty value, or with the value `traefik` are processed.
|
Otherwise, Ingresses missing the annotation, having an empty value, or the value `traefik` are processed.
|
||||||
|
|
||||||
!!! info "Kubernetes 1.18+"
|
!!! info "Kubernetes 1.18+"
|
||||||
|
|
||||||
|
@ -269,11 +254,30 @@ Otherwise, Ingresses missing the annotation, having an empty value, or with the
|
||||||
servicePort: 80
|
servicePort: 80
|
||||||
```
|
```
|
||||||
|
|
||||||
|
```toml tab="File (TOML)"
|
||||||
|
[providers.kubernetesIngress]
|
||||||
|
ingressClass = "traefik-internal"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml tab="File (YAML)"
|
||||||
|
providers:
|
||||||
|
kubernetesIngress:
|
||||||
|
ingressClass: "traefik-internal"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash tab="CLI"
|
||||||
|
--providers.kubernetesingress.ingressclass=traefik-internal
|
||||||
|
```
|
||||||
|
|
||||||
### `ingressEndpoint`
|
### `ingressEndpoint`
|
||||||
|
|
||||||
#### `hostname`
|
#### `hostname`
|
||||||
|
|
||||||
_Optional, Default: empty_
|
_Optional, Default: ""_
|
||||||
|
|
||||||
|
Hostname used for Kubernetes Ingress endpoints.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesIngress.ingressEndpoint]
|
[providers.kubernetesIngress.ingressEndpoint]
|
||||||
|
@ -293,11 +297,11 @@ providers:
|
||||||
--providers.kubernetesingress.ingressendpoint.hostname=example.net
|
--providers.kubernetesingress.ingressendpoint.hostname=example.net
|
||||||
```
|
```
|
||||||
|
|
||||||
Hostname used for Kubernetes Ingress endpoints.
|
|
||||||
|
|
||||||
#### `ip`
|
#### `ip`
|
||||||
|
|
||||||
_Optional, Default: empty_
|
_Optional, Default: ""_
|
||||||
|
|
||||||
|
IP used for Kubernetes Ingress endpoints.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesIngress.ingressEndpoint]
|
[providers.kubernetesIngress.ingressEndpoint]
|
||||||
|
@ -317,11 +321,12 @@ providers:
|
||||||
--providers.kubernetesingress.ingressendpoint.ip=1.2.3.4
|
--providers.kubernetesingress.ingressendpoint.ip=1.2.3.4
|
||||||
```
|
```
|
||||||
|
|
||||||
IP used for Kubernetes Ingress endpoints.
|
|
||||||
|
|
||||||
#### `publishedService`
|
#### `publishedService`
|
||||||
|
|
||||||
_Optional, Default: empty_
|
_Optional, Default: ""_
|
||||||
|
|
||||||
|
Published Kubernetes Service to copy status from.
|
||||||
|
Format: `namespace/servicename`.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesIngress.ingressEndpoint]
|
[providers.kubernetesIngress.ingressEndpoint]
|
||||||
|
@ -341,12 +346,17 @@ providers:
|
||||||
--providers.kubernetesingress.ingressendpoint.publishedservice=namespace/foo-service
|
--providers.kubernetesingress.ingressendpoint.publishedservice=namespace/foo-service
|
||||||
```
|
```
|
||||||
|
|
||||||
Published Kubernetes Service to copy status from.
|
|
||||||
Format: `namespace/servicename`.
|
|
||||||
|
|
||||||
### `throttleDuration`
|
### `throttleDuration`
|
||||||
|
|
||||||
_Optional, Default: 0 (no throttling)_
|
_Optional, Default: 0_
|
||||||
|
|
||||||
|
The `throttleDuration` option defines how often the provider is allowed to handle events from Kubernetes. This prevents
|
||||||
|
a Kubernetes cluster that updates many times per second from continuously changing your Traefik configuration.
|
||||||
|
|
||||||
|
If left empty, the provider does not apply any throttling and does not drop any Kubernetes events.
|
||||||
|
|
||||||
|
The value of `throttleDuration` should be provided in seconds or as a valid duration format,
|
||||||
|
see [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.kubernetesIngress]
|
[providers.kubernetesIngress]
|
||||||
|
@ -367,5 +377,5 @@ providers:
|
||||||
|
|
||||||
### Further
|
### Further
|
||||||
|
|
||||||
If one wants to know more about the various aspects of the Ingress spec that Traefik supports,
|
To learn more about the various aspects of the Ingress specification that Traefik supports,
|
||||||
many examples of Ingresses definitions are located in the tests [data](https://github.com/traefik/traefik/tree/v2.4/pkg/provider/kubernetes/ingress/fixtures) of the Traefik repository.
|
many examples of Ingresses definitions are located in the test [examples](https://github.com/traefik/traefik/tree/v2.4/pkg/provider/kubernetes/ingress/fixtures) of the Traefik repository.
|
||||||
|
|
|
@ -3,13 +3,13 @@
|
||||||
Traefik can be configured to use Marathon as a provider.
|
Traefik can be configured to use Marathon as a provider.
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
See also [Marathon user guide](../user-guides/marathon.md).
|
For additional information, refer to [Marathon user guide](../user-guides/marathon.md).
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
??? example "Configuring Marathon & Deploying / Exposing Applications"
|
??? example "Configuring Marathon & Deploying / Exposing Applications"
|
||||||
|
|
||||||
Enabling the marathon provider
|
Enabling the Marathon provider
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
|
@ -24,7 +24,7 @@ See also [Marathon user guide](../user-guides/marathon.md).
|
||||||
--providers.marathon=true
|
--providers.marathon=true
|
||||||
```
|
```
|
||||||
|
|
||||||
Attaching labels to marathon applications
|
Attaching labels to Marathon applications
|
||||||
|
|
||||||
```json
|
```json
|
||||||
{
|
{
|
||||||
|
@ -59,6 +59,8 @@ See the dedicated section in [routing](../routing/providers/marathon.md).
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
|
||||||
|
Enables Marathon basic authentication.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon.basic]
|
[providers.marathon.basic]
|
||||||
httpBasicAuthUser = "foo"
|
httpBasicAuthUser = "foo"
|
||||||
|
@ -78,12 +80,14 @@ providers:
|
||||||
--providers.marathon.basic.httpbasicpassword=bar
|
--providers.marathon.basic.httpbasicpassword=bar
|
||||||
```
|
```
|
||||||
|
|
||||||
Enables Marathon basic authentication.
|
|
||||||
|
|
||||||
### `dcosToken`
|
### `dcosToken`
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
|
||||||
|
Datacenter Operating System (DCOS) Token for DCOS environment.
|
||||||
|
|
||||||
|
If set, it overrides the Authorization header.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
dcosToken = "xxxxxx"
|
dcosToken = "xxxxxx"
|
||||||
|
@ -101,14 +105,20 @@ providers:
|
||||||
--providers.marathon.dcosToken=xxxxxx
|
--providers.marathon.dcosToken=xxxxxx
|
||||||
```
|
```
|
||||||
|
|
||||||
DCOSToken for DCOS environment.
|
|
||||||
|
|
||||||
If set, it overrides the Authorization header.
|
|
||||||
|
|
||||||
### `defaultRule`
|
### `defaultRule`
|
||||||
|
|
||||||
_Optional, Default=```Host(`{{ normalize .Name }}`)```_
|
_Optional, Default=```Host(`{{ normalize .Name }}`)```_
|
||||||
|
|
||||||
|
The default host rule for all services.
|
||||||
|
|
||||||
|
For a given application, if no routing rule was defined by a label, it is defined by this `defaultRule` instead.
|
||||||
|
|
||||||
|
It must be a valid [Go template](https://golang.org/pkg/text/template/),
|
||||||
|
and can include [sprig template functions](http://masterminds.github.io/sprig/).
|
||||||
|
|
||||||
|
The app ID can be accessed with the `Name` identifier,
|
||||||
|
and the template has access to all the labels defined on this Marathon application.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
|
defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
|
||||||
|
@ -127,18 +137,16 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
For a given application if no routing rule was defined by a label, it is defined by this defaultRule instead.
|
|
||||||
|
|
||||||
It must be a valid [Go template](https://golang.org/pkg/text/template/),
|
|
||||||
augmented with the [sprig template functions](http://masterminds.github.io/sprig/).
|
|
||||||
|
|
||||||
The app ID can be accessed as the Name identifier,
|
|
||||||
and the template has access to all the labels defined on this Marathon application.
|
|
||||||
|
|
||||||
### `dialerTimeout`
|
### `dialerTimeout`
|
||||||
|
|
||||||
_Optional, Default=5s_
|
_Optional, Default=5s_
|
||||||
|
|
||||||
|
Amount of time the Marathon provider should wait before timing out,
|
||||||
|
when trying to open a TCP connection to a Marathon master.
|
||||||
|
|
||||||
|
The value of `dialerTimeout` should be provided in seconds or as a valid duration format,
|
||||||
|
see [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
dialerTimeout = "10s"
|
dialerTimeout = "10s"
|
||||||
|
@ -156,18 +164,14 @@ providers:
|
||||||
--providers.marathon.dialerTimeout=10s
|
--providers.marathon.dialerTimeout=10s
|
||||||
```
|
```
|
||||||
|
|
||||||
Overrides DialerTimeout.
|
|
||||||
|
|
||||||
Amount of time the Marathon provider should wait before timing out,
|
|
||||||
when trying to open a TCP connection to a Marathon master.
|
|
||||||
|
|
||||||
Can be provided in a format supported by [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration),
|
|
||||||
or directly as a number of seconds.
|
|
||||||
|
|
||||||
### `endpoint`
|
### `endpoint`
|
||||||
|
|
||||||
_Optional, Default=http://127.0.0.1:8080_
|
_Optional, Default=http://127.0.0.1:8080_
|
||||||
|
|
||||||
|
Marathon server endpoint.
|
||||||
|
|
||||||
|
You can optionally specify multiple endpoints.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
endpoint = "http://10.241.1.71:8080,10.241.1.72:8080,10.241.1.73:8080"
|
endpoint = "http://10.241.1.71:8080,10.241.1.72:8080,10.241.1.73:8080"
|
||||||
|
@ -185,14 +189,16 @@ providers:
|
||||||
--providers.marathon.endpoint=http://10.241.1.71:8080,10.241.1.72:8080,10.241.1.73:8080
|
--providers.marathon.endpoint=http://10.241.1.71:8080,10.241.1.72:8080,10.241.1.73:8080
|
||||||
```
|
```
|
||||||
|
|
||||||
Marathon server endpoint.
|
|
||||||
|
|
||||||
You can optionally specify multiple endpoints:
|
|
||||||
|
|
||||||
### `exposedByDefault`
|
### `exposedByDefault`
|
||||||
|
|
||||||
_Optional, Default=true_
|
_Optional, Default=true_
|
||||||
|
|
||||||
|
Exposes Marathon applications by default through Traefik.
|
||||||
|
|
||||||
|
If set to `false`, applications that do not have a `traefik.enable=true` label are ignored from the resulting routing configuration.
|
||||||
|
|
||||||
|
For additional information, refer to [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
exposedByDefault = false
|
exposedByDefault = false
|
||||||
|
@ -211,41 +217,18 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Exposes Marathon applications by default through Traefik.
|
|
||||||
|
|
||||||
If set to false, applications that don't have a `traefik.enable=true` label will be ignored from the resulting routing configuration.
|
|
||||||
|
|
||||||
See also [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
|
||||||
|
|
||||||
### `constraints`
|
### `constraints`
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
The `constraints` option can be set to an expression that Traefik matches against the application labels to determine whether
|
||||||
[providers.marathon]
|
to create any route for that application. If none of the application labels match the expression, no route for that application is
|
||||||
constraints = "Label(`a.label.name`,`foo`)"
|
created. In addition, the expression is also matched against the application constraints, such as described
|
||||||
# ...
|
in [Marathon constraints](https://mesosphere.github.io/marathon/docs/constraints.html).
|
||||||
```
|
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
|
||||||
providers:
|
|
||||||
marathon:
|
|
||||||
constraints: "Label(`a.label.name`,`foo`)"
|
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
```bash tab="CLI"
|
|
||||||
--providers.marathon.constraints=Label(`a.label.name`,`foo`)
|
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
Constraints is an expression that Traefik matches against the application's labels to determine whether to create any route for that application.
|
|
||||||
That is to say, if none of the application's labels match the expression, no route for the application is created.
|
|
||||||
In addition, the expression also matched against the application's constraints, such as described in [Marathon constraints](https://mesosphere.github.io/marathon/docs/constraints.html).
|
|
||||||
If the expression is empty, all detected applications are included.
|
If the expression is empty, all detected applications are included.
|
||||||
|
|
||||||
The expression syntax is based on the `Label("key", "value")`, and `LabelRegex("key", "value")`, as well as the usual boolean logic.
|
The expression syntax is based on the `Label("key", "value")`, and `LabelRegex("key", "value")` functions, as well as the usual boolean logic.
|
||||||
In addition, to match against marathon constraints, the function `MarathonConstraint("field:operator:value")` can be used, where the field, operator, and value parts are joined together in a single string with the `:` separator.
|
In addition, to match against Marathon constraints, the function `MarathonConstraint("field:operator:value")` can be used, where the field, operator, and value parts are concatenated in a single string using the `:` separator.
|
||||||
|
|
||||||
??? example "Constraints Expression Examples"
|
??? example "Constraints Expression Examples"
|
||||||
|
|
||||||
|
@ -289,12 +272,34 @@ In addition, to match against marathon constraints, the function `MarathonConstr
|
||||||
constraints = "MarathonConstraint(`A:B:C`) && Label(`a.label.name`, `value`)"
|
constraints = "MarathonConstraint(`A:B:C`) && Label(`a.label.name`, `value`)"
|
||||||
```
|
```
|
||||||
|
|
||||||
See also [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
For additional information, refer to [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
||||||
|
|
||||||
|
```toml tab="File (TOML)"
|
||||||
|
[providers.marathon]
|
||||||
|
constraints = "Label(`a.label.name`,`foo`)"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml tab="File (YAML)"
|
||||||
|
providers:
|
||||||
|
marathon:
|
||||||
|
constraints: "Label(`a.label.name`,`foo`)"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash tab="CLI"
|
||||||
|
--providers.marathon.constraints=Label(`a.label.name`,`foo`)
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
### `forceTaskHostname`
|
### `forceTaskHostname`
|
||||||
|
|
||||||
_Optional, Default=false_
|
_Optional, Default=false_
|
||||||
|
|
||||||
|
By default, the task IP address (as returned by the Marathon API) is used as backend server if an IP-per-task configuration can be found;
|
||||||
|
otherwise, the name of the host running the task is used.
|
||||||
|
The latter behavior can be enforced by setting this option to `true`.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
forceTaskHostname = true
|
forceTaskHostname = true
|
||||||
|
@ -313,14 +318,14 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
By default, a task's IP address (as returned by the Marathon API) is used as backend server if an IP-per-task configuration can be found;
|
|
||||||
otherwise, the name of the host running the task is used.
|
|
||||||
The latter behavior can be enforced by enabling this switch.
|
|
||||||
|
|
||||||
### `keepAlive`
|
### `keepAlive`
|
||||||
|
|
||||||
_Optional, Default=10s_
|
_Optional, Default=10s_
|
||||||
|
|
||||||
|
Set the TCP Keep Alive duration for the Marathon HTTP Client.
|
||||||
|
The value of `keepAlive` should be provided in seconds or as a valid duration format,
|
||||||
|
see [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
keepAlive = "30s"
|
keepAlive = "30s"
|
||||||
|
@ -339,14 +344,16 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Set the TCP Keep Alive interval for the Marathon HTTP Client.
|
|
||||||
Can be provided in a format supported by [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration),
|
|
||||||
or directly as a number of seconds.
|
|
||||||
|
|
||||||
### `respectReadinessChecks`
|
### `respectReadinessChecks`
|
||||||
|
|
||||||
_Optional, Default=false_
|
_Optional, Default=false_
|
||||||
|
|
||||||
|
Applications may define readiness checks which are probed by Marathon during deployments periodically, and these check results are exposed via the API.
|
||||||
|
Enabling `respectReadinessChecks` causes Traefik to filter out tasks whose readiness checks have not succeeded.
|
||||||
|
Note that the checks are only valid during deployments.
|
||||||
|
|
||||||
|
See the Marathon guide for details.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
respectReadinessChecks = true
|
respectReadinessChecks = true
|
||||||
|
@ -365,16 +372,16 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Applications may define readiness checks which are probed by Marathon during deployments periodically, and these check results are exposed via the API.
|
|
||||||
Enabling respectReadinessChecks causes Traefik to filter out tasks whose readiness checks have not succeeded.
|
|
||||||
Note that the checks are only valid at deployment times.
|
|
||||||
|
|
||||||
See the Marathon guide for details.
|
|
||||||
|
|
||||||
### `responseHeaderTimeout`
|
### `responseHeaderTimeout`
|
||||||
|
|
||||||
_Optional, Default=60s_
|
_Optional, Default=60s_
|
||||||
|
|
||||||
|
Amount of time the Marathon provider should wait before timing out when waiting for the first response header
|
||||||
|
from a Marathon master.
|
||||||
|
|
||||||
|
The value of `responseHeaderTimeout` should be provided in seconds or as a valid duration format,
|
||||||
|
see [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
responseHeaderTimeout = "66s"
|
responseHeaderTimeout = "66s"
|
||||||
|
@ -393,19 +400,13 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Overrides ResponseHeaderTimeout.
|
|
||||||
Amount of time the Marathon provider should wait before timing out,
|
|
||||||
when waiting for the first response header from a Marathon master.
|
|
||||||
|
|
||||||
Can be provided in a format supported by [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration), or directly as a number of seconds.
|
|
||||||
|
|
||||||
### `tls`
|
### `tls`
|
||||||
|
|
||||||
_Optional_
|
_Optional_
|
||||||
|
|
||||||
#### `tls.ca`
|
#### `tls.ca`
|
||||||
|
|
||||||
Certificate Authority used for the secured connection to Marathon.
|
Certificate Authority used for the secure connection to Marathon.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon.tls]
|
[providers.marathon.tls]
|
||||||
|
@ -425,12 +426,15 @@ providers:
|
||||||
|
|
||||||
#### `tls.caOptional`
|
#### `tls.caOptional`
|
||||||
|
|
||||||
Policy followed for the secured connection to Marathon with TLS Client Authentication.
|
The value of `tls.caOptional` defines which policy should be used for the secure connection with TLS Client Authentication to Marathon.
|
||||||
Requires `tls.ca` to be defined.
|
|
||||||
|
|
||||||
- `true`: VerifyClientCertIfGiven
|
!!! warning ""
|
||||||
- `false`: RequireAndVerifyClientCert
|
|
||||||
- if `tls.ca` is undefined NoClientCert
|
If `tls.ca` is undefined, this option will be ignored, and no client certificate will be requested during the handshake. Any provided certificate will thus never be verified.
|
||||||
|
|
||||||
|
When this option is set to `true`, a client certificate is requested during the handshake but is not required. If a certificate is sent, it is required to be valid.
|
||||||
|
|
||||||
|
When this option is set to `false`, a client certificate is requested during the handshake, and at least one valid certificate should be sent by the client.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon.tls]
|
[providers.marathon.tls]
|
||||||
|
@ -450,7 +454,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.cert`
|
#### `tls.cert`
|
||||||
|
|
||||||
Public certificate used for the secured connection to Marathon.
|
Public certificate used for the secure connection to Marathon.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon.tls]
|
[providers.marathon.tls]
|
||||||
|
@ -473,7 +477,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.key`
|
#### `tls.key`
|
||||||
|
|
||||||
Private certificate used for the secured connection to Marathon.
|
Private certificate used for the secure connection to Marathon.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon.tls]
|
[providers.marathon.tls]
|
||||||
|
@ -496,7 +500,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.insecureSkipVerify`
|
#### `tls.insecureSkipVerify`
|
||||||
|
|
||||||
If `insecureSkipVerify` is `true`, TLS for the connection to Marathon accepts any certificate presented by the server and any host name in that certificate.
|
If `insecureSkipVerify` is `true`, the TLS connection to Marathon accepts any certificate presented by the server regardless of the hostnames it covers.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon.tls]
|
[providers.marathon.tls]
|
||||||
|
@ -518,6 +522,12 @@ providers:
|
||||||
|
|
||||||
_Optional, Default=5s_
|
_Optional, Default=5s_
|
||||||
|
|
||||||
|
Amount of time the Marathon provider should wait before timing out,
|
||||||
|
when waiting for the TLS handshake to complete.
|
||||||
|
|
||||||
|
The value of `tlsHandshakeTimeout` should be provided in seconds or as a valid duration format,
|
||||||
|
see [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
responseHeaderTimeout = "10s"
|
responseHeaderTimeout = "10s"
|
||||||
|
@ -536,17 +546,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Overrides TLSHandshakeTimeout.
|
|
||||||
|
|
||||||
Amount of time the Marathon provider should wait before timing out,
|
|
||||||
when waiting for the TLS handshake to complete.
|
|
||||||
Can be provided in a format supported by [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration),
|
|
||||||
or directly as a number of seconds.
|
|
||||||
|
|
||||||
### `trace`
|
### `trace`
|
||||||
|
|
||||||
_Optional, Default=false_
|
_Optional, Default=false_
|
||||||
|
|
||||||
|
Displays additional provider logs when available.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
trace = true
|
trace = true
|
||||||
|
@ -565,12 +570,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Displays additional provider logs (if available).
|
|
||||||
|
|
||||||
### `watch`
|
### `watch`
|
||||||
|
|
||||||
_Optional, Default=true_
|
_Optional, Default=true_
|
||||||
|
|
||||||
|
When set to `true`, watches for Marathon changes.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.marathon]
|
[providers.marathon]
|
||||||
watch = false
|
watch = false
|
||||||
|
@ -588,5 +593,3 @@ providers:
|
||||||
--providers.marathon.watch=false
|
--providers.marathon.watch=false
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Enables watching for Marathon changes.
|
|
||||||
|
|
|
@ -7,30 +7,28 @@ Traefik's Many Friends
|
||||||
|
|
||||||
Configuration discovery in Traefik is achieved through _Providers_.
|
Configuration discovery in Traefik is achieved through _Providers_.
|
||||||
|
|
||||||
The _providers_ are existing infrastructure components, whether orchestrators, container engines, cloud providers, or key-value stores.
|
The _providers_ are infrastructure components, whether orchestrators, container engines, cloud providers, or key-value stores.
|
||||||
The idea is that Traefik will query the providers' API in order to find relevant information about routing,
|
The idea is that Traefik queries the provider APIs in order to find relevant information about routing,
|
||||||
and each time Traefik detects a change, it dynamically updates the routes.
|
and when Traefik detects a change, it dynamically updates the routes.
|
||||||
|
|
||||||
Deploy and forget is Traefik's credo.
|
|
||||||
|
|
||||||
## Orchestrators
|
## Orchestrators
|
||||||
|
|
||||||
Even if each provider is different, we can categorize them in four groups:
|
While each provider is different, you can think of each as belonging to one of four categories:
|
||||||
|
|
||||||
- Label based (each deployed container has a set of labels attached to it)
|
- Label-based: each deployed container has a set of labels attached to it
|
||||||
- Key-Value based (each deployed container updates a key-value store with relevant information)
|
- Key-Value-based: each deployed container updates a key-value store with relevant information
|
||||||
- Annotation based (a separate object, with annotations, defines the characteristics of the container)
|
- Annotation-based: a separate object, with annotations, defines the characteristics of the container
|
||||||
- File based (the good old configuration file)
|
- File-based: uses files to define configuration
|
||||||
|
|
||||||
## Provider Namespace
|
## Provider Namespace
|
||||||
|
|
||||||
When you declare certain objects, in Traefik dynamic configuration,
|
When you declare certain objects in the Traefik dynamic configuration,
|
||||||
such as middleware, service, TLS options or servers transport, they live in its provider's namespace.
|
such as middleware, services, TLS options or server transports, they reside in their provider's namespace.
|
||||||
For example, if you declare a middleware using a Docker label, under the hoods, it will reside in the docker provider namespace.
|
For example, if you declare a middleware using a Docker label, it resides in the Docker provider namespace.
|
||||||
|
|
||||||
If you use multiple providers and wish to reference such an object declared in another provider
|
If you use multiple providers and wish to reference such an object declared in another provider
|
||||||
(aka referencing a cross-provider object, e.g. middleware), then you'll have to append the `@` separator,
|
(e.g. referencing a cross-provider object like middleware), then the object name should be suffixed by the `@`
|
||||||
followed by the provider name to the object name.
|
separator, and the provider name.
|
||||||
|
|
||||||
```text
|
```text
|
||||||
<resource-name>@<provider-name>
|
<resource-name>@<provider-name>
|
||||||
|
@ -39,14 +37,15 @@ followed by the provider name to the object name.
|
||||||
!!! important "Kubernetes Namespace"
|
!!! important "Kubernetes Namespace"
|
||||||
|
|
||||||
As Kubernetes also has its own notion of namespace,
|
As Kubernetes also has its own notion of namespace,
|
||||||
one should not confuse the "provider namespace" with the "kubernetes namespace" of a resource when in the context of a cross-provider usage.
|
one should not confuse the _provider namespace_ with the _Kubernetes Namespace_ of a resource when in the context of cross-provider usage.
|
||||||
In this case, since the definition of a traefik dynamic configuration object is not in kubernetes,
|
|
||||||
specifying a "kubernetes namespace" when referring to the resource does not make any sense,
|
|
||||||
and therefore this specification would be ignored even if present.
|
|
||||||
On the other hand, if you, say, declare a middleware as a Custom Resource in Kubernetes and use the non-crd Ingress objects,
|
|
||||||
you'll have to add the Kubernetes namespace of the middleware to the annotation like this `<middleware-namespace>-<middleware-name>@kubernetescrd`.
|
|
||||||
|
|
||||||
!!! abstract "Referencing a Traefik dynamic configuration object from Another Provider"
|
In this case, since the definition of a Traefik dynamic configuration object is not in Kubernetes,
|
||||||
|
specifying a Kubernetes Namespace when referring to the resource does not make any sense.
|
||||||
|
|
||||||
|
On the other hand, if you were to declare a middleware as a Custom Resource in Kubernetes and use the non-CRD Ingress objects,
|
||||||
|
you would have to add the Kubernetes Namespace of the middleware to the annotation like this `<middleware-namespace>-<middleware-name>@kubernetescrd`.
|
||||||
|
|
||||||
|
!!! abstract "Referencing a Traefik Dynamic Configuration Object from Another Provider"
|
||||||
|
|
||||||
Declaring the add-foo-prefix in the file provider.
|
Declaring the add-foo-prefix in the file provider.
|
||||||
|
|
||||||
|
@ -143,26 +142,32 @@ Below is the list of the currently supported providers in Traefik.
|
||||||
|
|
||||||
!!! info "More Providers"
|
!!! info "More Providers"
|
||||||
|
|
||||||
The current version of Traefik doesn't support (yet) every provider.
|
The current version of Traefik does not yet support every provider that Traefik v1.7 did.
|
||||||
See the [previous version (v1.7)](https://doc.traefik.io/traefik/v1.7/) for more providers.
|
See the [previous version (v1.7)](https://doc.traefik.io/traefik/v1.7/) for more providers.
|
||||||
|
|
||||||
### Configuration reload frequency
|
### Configuration Reload Frequency
|
||||||
|
|
||||||
|
#### `providers.providersThrottleDuration`
|
||||||
|
|
||||||
|
_Optional, Default: 2s_
|
||||||
|
|
||||||
In some cases, some providers might undergo a sudden burst of changes,
|
In some cases, some providers might undergo a sudden burst of changes,
|
||||||
which would generate a lot of configuration change events.
|
which would generate a lot of configuration change events.
|
||||||
If Traefik took them all into account,
|
If Traefik took them all into account,
|
||||||
that would trigger a lot more configuration reloads than what is necessary,
|
that would trigger a lot more configuration reloads than is necessary,
|
||||||
or even useful.
|
or even useful.
|
||||||
|
|
||||||
In order to mitigate that, the `providers.providersThrottleDuration` option can be set.
|
In order to mitigate that, the `providers.providersThrottleDuration` option can be set.
|
||||||
It is the duration that Traefik waits for, after a configuration reload,
|
It is the duration that Traefik waits for, after a configuration reload,
|
||||||
before taking into account any new configuration refresh event.
|
before taking into account any new configuration refresh event.
|
||||||
If any event arrives during that duration, only the most recent one is taken into account,
|
If multiple events occur within this time, only the most recent one is taken into account,
|
||||||
and all the previous others are dropped.
|
and all others are discarded.
|
||||||
|
|
||||||
This option cannot be set per provider,
|
This option cannot be set per provider,
|
||||||
but the throttling algorithm applies independently to each of them.
|
but the throttling algorithm applies to each of them independently.
|
||||||
It defaults to 2 seconds.
|
|
||||||
|
The value of `providers.providersThrottleDuration` should be provided in seconds or as a valid duration format,
|
||||||
|
see [time.ParseDuration](https://golang.org/pkg/time/#ParseDuration).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers]
|
[providers]
|
||||||
|
@ -184,17 +189,18 @@ TODO (document TCP VS HTTP dynamic configuration)
|
||||||
|
|
||||||
## Restrict the Scope of Service Discovery
|
## Restrict the Scope of Service Discovery
|
||||||
|
|
||||||
By default Traefik will create routes for all detected containers.
|
By default, Traefik creates routes for all detected containers.
|
||||||
|
|
||||||
If you want to limit the scope of Traefik's service discovery,
|
If you want to limit the scope of the Traefik service discovery,
|
||||||
i.e. disallow route creation for some containers,
|
i.e. disallow route creation for some containers,
|
||||||
you can do so in two different ways:
|
you can do so in two different ways:
|
||||||
either with the generic configuration option `exposedByDefault`,
|
|
||||||
or with a finer granularity mechanism based on constraints.
|
- the generic configuration option `exposedByDefault`,
|
||||||
|
- a finer granularity mechanism based on constraints.
|
||||||
|
|
||||||
### `exposedByDefault` and `traefik.enable`
|
### `exposedByDefault` and `traefik.enable`
|
||||||
|
|
||||||
List of providers that support that feature:
|
List of providers that support these features:
|
||||||
|
|
||||||
- [Docker](./docker.md#exposedbydefault)
|
- [Docker](./docker.md#exposedbydefault)
|
||||||
- [Consul Catalog](./consul-catalog.md#exposedbydefault)
|
- [Consul Catalog](./consul-catalog.md#exposedbydefault)
|
||||||
|
@ -211,3 +217,4 @@ List of providers that support constraints:
|
||||||
- [Marathon](./marathon.md#constraints)
|
- [Marathon](./marathon.md#constraints)
|
||||||
- [Kubernetes CRD](./kubernetes-crd.md#labelselector)
|
- [Kubernetes CRD](./kubernetes-crd.md#labelselector)
|
||||||
- [Kubernetes Ingress](./kubernetes-ingress.md#labelselector)
|
- [Kubernetes Ingress](./kubernetes-ingress.md#labelselector)
|
||||||
|
- [Kubernetes Gateway](./kubernetes-gateway.md#labelselector)
|
||||||
|
|
|
@ -10,13 +10,13 @@ Attach labels to your services and let Traefik do the rest!
|
||||||
!!! important "This provider is specific to Rancher 1.x."
|
!!! important "This provider is specific to Rancher 1.x."
|
||||||
|
|
||||||
Rancher 2.x requires Kubernetes and does not have a metadata endpoint of its own for Traefik to query.
|
Rancher 2.x requires Kubernetes and does not have a metadata endpoint of its own for Traefik to query.
|
||||||
As such, Rancher 2.x users should utilize the [Kubernetes provider](./kubernetes-crd.md) directly.
|
As such, Rancher 2.x users should utilize the [Kubernetes CRD provider](./kubernetes-crd.md) directly.
|
||||||
|
|
||||||
## Configuration Examples
|
## Configuration Examples
|
||||||
|
|
||||||
??? example "Configuring Rancher & Deploying / Exposing Services"
|
??? example "Configuring Rancher & Deploying / Exposing Services"
|
||||||
|
|
||||||
Enabling the rancher provider
|
Enabling the Rancher provider
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.rancher]
|
[providers.rancher]
|
||||||
|
@ -45,7 +45,8 @@ See the dedicated section in [routing](../routing/providers/rancher.md).
|
||||||
## Provider Configuration
|
## Provider Configuration
|
||||||
|
|
||||||
??? tip "Browse the Reference"
|
??? tip "Browse the Reference"
|
||||||
If you're in a hurry, maybe you'd rather go through the configuration reference:
|
|
||||||
|
For an overview of all the options that can be set with the Rancher provider, see the following snippets:
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
--8<-- "content/providers/rancher.toml"
|
--8<-- "content/providers/rancher.toml"
|
||||||
|
@ -63,6 +64,11 @@ See the dedicated section in [routing](../routing/providers/rancher.md).
|
||||||
|
|
||||||
_Optional, Default=true_
|
_Optional, Default=true_
|
||||||
|
|
||||||
|
Expose Rancher services by default in Traefik.
|
||||||
|
If set to `false`, services that do not have a `traefik.enable=true` label are ignored from the resulting routing configuration.
|
||||||
|
|
||||||
|
For additional information, refer to [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.rancher]
|
[providers.rancher]
|
||||||
exposedByDefault = false
|
exposedByDefault = false
|
||||||
|
@ -81,15 +87,21 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Expose Rancher services by default in Traefik.
|
|
||||||
If set to false, services that don't have a `traefik.enable=true` label will be ignored from the resulting routing configuration.
|
|
||||||
|
|
||||||
See also [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
|
||||||
|
|
||||||
### `defaultRule`
|
### `defaultRule`
|
||||||
|
|
||||||
_Optional, Default=```Host(`{{ normalize .Name }}`)```_
|
_Optional, Default=```Host(`{{ normalize .Name }}`)```_
|
||||||
|
|
||||||
|
The default host rule for all services.
|
||||||
|
|
||||||
|
The `defaultRule` option defines what routing rule to apply to a container if no rule is defined by a label.
|
||||||
|
|
||||||
|
It must be a valid [Go template](https://golang.org/pkg/text/template/), and can use
|
||||||
|
[sprig template functions](http://masterminds.github.io/sprig/).
|
||||||
|
The service name can be accessed with the `Name` identifier,
|
||||||
|
and the template has access to all the labels defined on this container.
|
||||||
|
|
||||||
|
This option can be overridden on a container basis with the `traefik.http.routers.Router1.rule` label.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.rancher]
|
[providers.rancher]
|
||||||
defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
|
defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
|
||||||
|
@ -108,20 +120,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
The default host rule for all services.
|
|
||||||
|
|
||||||
For a given container if no routing rule was defined by a label, it is defined by this defaultRule instead.
|
|
||||||
It must be a valid [Go template](https://golang.org/pkg/text/template/),
|
|
||||||
augmented with the [sprig template functions](http://masterminds.github.io/sprig/).
|
|
||||||
The service name can be accessed as the `Name` identifier,
|
|
||||||
and the template has access to all the labels defined on this container.
|
|
||||||
|
|
||||||
This option can be overridden on a container basis with the `traefik.http.routers.Router1.rule` label.
|
|
||||||
|
|
||||||
### `enableServiceHealthFilter`
|
### `enableServiceHealthFilter`
|
||||||
|
|
||||||
_Optional, Default=true_
|
_Optional, Default=true_
|
||||||
|
|
||||||
|
Filter out services with unhealthy states and inactive states.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.rancher]
|
[providers.rancher]
|
||||||
enableServiceHealthFilter = false
|
enableServiceHealthFilter = false
|
||||||
|
@ -140,12 +144,12 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Filter services with unhealthy states and inactive states.
|
|
||||||
|
|
||||||
### `refreshSeconds`
|
### `refreshSeconds`
|
||||||
|
|
||||||
_Optional, Default=15_
|
_Optional, Default=15_
|
||||||
|
|
||||||
|
Defines the polling interval (in seconds).
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.rancher]
|
[providers.rancher]
|
||||||
refreshSeconds = 30
|
refreshSeconds = 30
|
||||||
|
@ -164,12 +168,13 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Defines the polling interval (in seconds).
|
|
||||||
|
|
||||||
### `intervalPoll`
|
### `intervalPoll`
|
||||||
|
|
||||||
_Optional, Default=false_
|
_Optional, Default=false_
|
||||||
|
|
||||||
|
Poll the Rancher metadata service for changes every `rancher.refreshSeconds`,
|
||||||
|
which is less accurate than the default long polling technique which provides near instantaneous updates to Traefik.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.rancher]
|
[providers.rancher]
|
||||||
intervalPoll = true
|
intervalPoll = true
|
||||||
|
@ -188,12 +193,11 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Poll the Rancher metadata service for changes every `rancher.refreshSeconds`,
|
|
||||||
which is less accurate than the default long polling technique which will provide near instantaneous updates to Traefik.
|
|
||||||
|
|
||||||
### `prefix`
|
### `prefix`
|
||||||
|
|
||||||
_Optional, Default=/latest_
|
_Optional, Default="/latest"_
|
||||||
|
|
||||||
|
Prefix used for accessing the Rancher metadata service.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.rancher]
|
[providers.rancher]
|
||||||
|
@ -213,35 +217,16 @@ providers:
|
||||||
# ...
|
# ...
|
||||||
```
|
```
|
||||||
|
|
||||||
Prefix used for accessing the Rancher metadata service
|
|
||||||
|
|
||||||
### `constraints`
|
### `constraints`
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
The `constraints` option can be set to an expression that Traefik matches against the container labels to determine whether
|
||||||
[providers.rancher]
|
to create any route for that container. If none of the container tags match the expression, no route for that container is
|
||||||
constraints = "Label(`a.label.name`,`foo`)"
|
created. If the expression is empty, all detected containers are included.
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
```yaml tab="File (YAML)"
|
The expression syntax is based on the `Label("key", "value")`, and `LabelRegex("key", "value")` functions, as well as
|
||||||
providers:
|
the usual boolean logic, as shown in examples below.
|
||||||
rancher:
|
|
||||||
constraints: "Label(`a.label.name`,`foo`)"
|
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
```bash tab="CLI"
|
|
||||||
--providers.rancher.constraints=Label(`a.label.name`,`foo`)
|
|
||||||
# ...
|
|
||||||
```
|
|
||||||
|
|
||||||
Constraints is an expression that Traefik matches against the container's labels to determine whether to create any route for that container.
|
|
||||||
That is to say, if none of the container's labels match the expression, no route for the container is created.
|
|
||||||
If the expression is empty, all detected containers are included.
|
|
||||||
|
|
||||||
The expression syntax is based on the `Label("key", "value")`, and `LabelRegex("key", "value")` functions, as well as the usual boolean logic, as shown in examples below.
|
|
||||||
|
|
||||||
??? example "Constraints Expression Examples"
|
??? example "Constraints Expression Examples"
|
||||||
|
|
||||||
|
@ -275,4 +260,22 @@ The expression syntax is based on the `Label("key", "value")`, and `LabelRegex("
|
||||||
constraints = "LabelRegex(`a.label.name`, `a.+`)"
|
constraints = "LabelRegex(`a.label.name`, `a.+`)"
|
||||||
```
|
```
|
||||||
|
|
||||||
See also [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
For additional information, refer to [Restrict the Scope of Service Discovery](./overview.md#restrict-the-scope-of-service-discovery).
|
||||||
|
|
||||||
|
```toml tab="File (TOML)"
|
||||||
|
[providers.rancher]
|
||||||
|
constraints = "Label(`a.label.name`,`foo`)"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml tab="File (YAML)"
|
||||||
|
providers:
|
||||||
|
rancher:
|
||||||
|
constraints: "Label(`a.label.name`,`foo`)"
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash tab="CLI"
|
||||||
|
--providers.rancher.constraints=Label(`a.label.name`,`foo`)
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
|
@ -35,10 +35,10 @@ providers:
|
||||||
|
|
||||||
### `rootKey`
|
### `rootKey`
|
||||||
|
|
||||||
Defines the root key of the configuration.
|
|
||||||
|
|
||||||
_Required, Default="traefik"_
|
_Required, Default="traefik"_
|
||||||
|
|
||||||
|
Defines the root key of the configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.redis]
|
[providers.redis]
|
||||||
rootKey = "traefik"
|
rootKey = "traefik"
|
||||||
|
@ -56,10 +56,10 @@ providers:
|
||||||
|
|
||||||
### `username`
|
### `username`
|
||||||
|
|
||||||
Defines a username to connect with Redis.
|
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Defines a username to connect with Redis.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.redis]
|
[providers.redis]
|
||||||
# ...
|
# ...
|
||||||
|
@ -106,7 +106,7 @@ _Optional_
|
||||||
|
|
||||||
#### `tls.ca`
|
#### `tls.ca`
|
||||||
|
|
||||||
Certificate Authority used for the secured connection to Redis.
|
Certificate Authority used for the secure connection to Redis.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.redis.tls]
|
[providers.redis.tls]
|
||||||
|
@ -126,12 +126,15 @@ providers:
|
||||||
|
|
||||||
#### `tls.caOptional`
|
#### `tls.caOptional`
|
||||||
|
|
||||||
Policy followed for the secured connection with TLS Client Authentication to Redis.
|
The value of `tls.caOptional` defines which policy should be used for the secure connection with TLS Client Authentication to Redis.
|
||||||
Requires `tls.ca` to be defined.
|
|
||||||
|
|
||||||
- `true`: VerifyClientCertIfGiven
|
!!! warning ""
|
||||||
- `false`: RequireAndVerifyClientCert
|
|
||||||
- if `tls.ca` is undefined NoClientCert
|
If `tls.ca` is undefined, this option will be ignored, and no client certificate will be requested during the handshake. Any provided certificate will thus never be verified.
|
||||||
|
|
||||||
|
When this option is set to `true`, a client certificate is requested during the handshake but is not required. If a certificate is sent, it is required to be valid.
|
||||||
|
|
||||||
|
When this option is set to `false`, a client certificate is requested during the handshake, and at least one valid certificate should be sent by the client.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.redis.tls]
|
[providers.redis.tls]
|
||||||
|
@ -151,7 +154,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.cert`
|
#### `tls.cert`
|
||||||
|
|
||||||
Public certificate used for the secured connection to Redis.
|
Public certificate used for the secure connection to Redis.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.redis.tls]
|
[providers.redis.tls]
|
||||||
|
@ -174,7 +177,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.key`
|
#### `tls.key`
|
||||||
|
|
||||||
Private certificate used for the secured connection to Redis.
|
Private certificate used for the secure connection to Redis.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.redis.tls]
|
[providers.redis.tls]
|
||||||
|
@ -197,7 +200,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.insecureSkipVerify`
|
#### `tls.insecureSkipVerify`
|
||||||
|
|
||||||
If `insecureSkipVerify` is `true`, TLS for the connection to Redis accepts any certificate presented by the server and any host name in that certificate.
|
If `insecureSkipVerify` is `true`, the TLS connection to Redis accepts any certificate presented by the server regardless of the hostnames it covers.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.redis.tls]
|
[providers.redis.tls]
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Traefik & ZooKeeper
|
# Traefik & ZooKeeper
|
||||||
|
|
||||||
A Story of KV store & Containers
|
A Story of KV Store & Containers
|
||||||
{: .subtitle }
|
{: .subtitle }
|
||||||
|
|
||||||
Store your configuration in ZooKeeper and let Traefik do the rest!
|
Store your configuration in ZooKeeper and let Traefik do the rest!
|
||||||
|
@ -35,10 +35,10 @@ providers:
|
||||||
|
|
||||||
### `rootKey`
|
### `rootKey`
|
||||||
|
|
||||||
Defines the root key of the configuration.
|
|
||||||
|
|
||||||
_Required, Default="traefik"_
|
_Required, Default="traefik"_
|
||||||
|
|
||||||
|
Defines the root key of the configuration.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.zooKeeper]
|
[providers.zooKeeper]
|
||||||
rootKey = "traefik"
|
rootKey = "traefik"
|
||||||
|
@ -56,10 +56,10 @@ providers:
|
||||||
|
|
||||||
### `username`
|
### `username`
|
||||||
|
|
||||||
Defines a username to connect with ZooKeeper.
|
|
||||||
|
|
||||||
_Optional, Default=""_
|
_Optional, Default=""_
|
||||||
|
|
||||||
|
Defines a username to connect with ZooKeeper.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.zooKeeper]
|
[providers.zooKeeper]
|
||||||
# ...
|
# ...
|
||||||
|
@ -106,7 +106,7 @@ _Optional_
|
||||||
|
|
||||||
#### `tls.ca`
|
#### `tls.ca`
|
||||||
|
|
||||||
Certificate Authority used for the secured connection to ZooKeeper.
|
Certificate Authority used for the secure connection to ZooKeeper.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.zooKeeper.tls]
|
[providers.zooKeeper.tls]
|
||||||
|
@ -126,12 +126,15 @@ providers:
|
||||||
|
|
||||||
#### `tls.caOptional`
|
#### `tls.caOptional`
|
||||||
|
|
||||||
Policy followed for the secured connection with TLS Client Authentication to ZooKeeper.
|
The value of `tls.caOptional` defines which policy should be used for the secure connection with TLS Client Authentication to Zookeeper.
|
||||||
Requires `tls.ca` to be defined.
|
|
||||||
|
|
||||||
- `true`: VerifyClientCertIfGiven
|
!!! warning ""
|
||||||
- `false`: RequireAndVerifyClientCert
|
|
||||||
- if `tls.ca` is undefined NoClientCert
|
If `tls.ca` is undefined, this option will be ignored, and no client certificate will be requested during the handshake. Any provided certificate will thus never be verified.
|
||||||
|
|
||||||
|
When this option is set to `true`, a client certificate is requested during the handshake but is not required. If a certificate is sent, it is required to be valid.
|
||||||
|
|
||||||
|
When this option is set to `false`, a client certificate is requested during the handshake, and at least one valid certificate should be sent by the client.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.zooKeeper.tls]
|
[providers.zooKeeper.tls]
|
||||||
|
@ -151,7 +154,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.cert`
|
#### `tls.cert`
|
||||||
|
|
||||||
Public certificate used for the secured connection to ZooKeeper.
|
Public certificate used for the secure connection to ZooKeeper.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.zooKeeper.tls]
|
[providers.zooKeeper.tls]
|
||||||
|
@ -174,7 +177,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.key`
|
#### `tls.key`
|
||||||
|
|
||||||
Private certificate used for the secured connection to ZooKeeper.
|
Private certificate used for the secure connection to ZooKeeper.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.zooKeeper.tls]
|
[providers.zooKeeper.tls]
|
||||||
|
@ -197,7 +200,7 @@ providers:
|
||||||
|
|
||||||
#### `tls.insecureSkipVerify`
|
#### `tls.insecureSkipVerify`
|
||||||
|
|
||||||
If `insecureSkipVerify` is `true`, TLS for the connection to ZooKeeper accepts any certificate presented by the server and any host name in that certificate.
|
If `insecureSkipVerify` is `true`, the TLS connection to Zookeeper accepts any certificate presented by the server regardless of the hostnames it covers.
|
||||||
|
|
||||||
```toml tab="File (TOML)"
|
```toml tab="File (TOML)"
|
||||||
[providers.zooKeeper.tls]
|
[providers.zooKeeper.tls]
|
||||||
|
|
|
@ -121,7 +121,7 @@ Default middlewares for the routers linked to the entry point.
|
||||||
Applies a permanent redirection. (Default: ```true```)
|
Applies a permanent redirection. (Default: ```true```)
|
||||||
|
|
||||||
`--entrypoints.<name>.http.redirections.entrypoint.priority`:
|
`--entrypoints.<name>.http.redirections.entrypoint.priority`:
|
||||||
Priority of the generated router. (Default: ```2147483647```)
|
Priority of the generated router. (Default: ```2147483646```)
|
||||||
|
|
||||||
`--entrypoints.<name>.http.redirections.entrypoint.scheme`:
|
`--entrypoints.<name>.http.redirections.entrypoint.scheme`:
|
||||||
Scheme used for the redirection. (Default: ```https```)
|
Scheme used for the redirection. (Default: ```https```)
|
||||||
|
|
|
@ -121,7 +121,7 @@ Default middlewares for the routers linked to the entry point.
|
||||||
Applies a permanent redirection. (Default: ```true```)
|
Applies a permanent redirection. (Default: ```true```)
|
||||||
|
|
||||||
`TRAEFIK_ENTRYPOINTS_<NAME>_HTTP_REDIRECTIONS_ENTRYPOINT_PRIORITY`:
|
`TRAEFIK_ENTRYPOINTS_<NAME>_HTTP_REDIRECTIONS_ENTRYPOINT_PRIORITY`:
|
||||||
Priority of the generated router. (Default: ```2147483647```)
|
Priority of the generated router. (Default: ```2147483646```)
|
||||||
|
|
||||||
`TRAEFIK_ENTRYPOINTS_<NAME>_HTTP_REDIRECTIONS_ENTRYPOINT_SCHEME`:
|
`TRAEFIK_ENTRYPOINTS_<NAME>_HTTP_REDIRECTIONS_ENTRYPOINT_SCHEME`:
|
||||||
Scheme used for the redirection. (Default: ```https```)
|
Scheme used for the redirection. (Default: ```https```)
|
||||||
|
|
|
@ -130,8 +130,17 @@ you'd add the tag `traefik.http.services.{name-of-your-choice}.loadbalancer.pass
|
||||||
traefik.http.services.myservice.loadbalancer.server.scheme=http
|
traefik.http.services.myservice.loadbalancer.server.scheme=http
|
||||||
```
|
```
|
||||||
|
|
||||||
|
??? info "`traefik.http.services.<service_name>.loadbalancer.serverstransport`"
|
||||||
|
|
||||||
|
See [serverstransport](../services/index.md#serverstransport) for more information.
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
traefik.http.services.<service_name>.loadbalancer.serverstransport=foobar
|
||||||
|
```
|
||||||
|
|
||||||
??? info "`traefik.http.services.<service_name>.loadbalancer.passhostheader`"
|
??? info "`traefik.http.services.<service_name>.loadbalancer.passhostheader`"
|
||||||
<!-- TODO doc passHostHeader in services page -->
|
|
||||||
|
See [pass Host header](../services/index.md#pass-host-header) for more information.
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
traefik.http.services.myservice.loadbalancer.passhostheader=true
|
traefik.http.services.myservice.loadbalancer.passhostheader=true
|
||||||
|
@ -201,12 +210,12 @@ you'd add the tag `traefik.http.services.{name-of-your-choice}.loadbalancer.pass
|
||||||
traefik.http.services.myservice.loadbalancer.healthcheck.followredirects=true
|
traefik.http.services.myservice.loadbalancer.healthcheck.followredirects=true
|
||||||
```
|
```
|
||||||
|
|
||||||
??? info "`traefik.http.services.<service_name>.loadbalancer.sticky`"
|
??? info "`traefik.http.services.<service_name>.loadbalancer.sticky.cookie`"
|
||||||
|
|
||||||
See [sticky sessions](../services/index.md#sticky-sessions) for more information.
|
See [sticky sessions](../services/index.md#sticky-sessions) for more information.
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
traefik.http.services.myservice.loadbalancer.sticky=true
|
traefik.http.services.myservice.loadbalancer.sticky.cookie=true
|
||||||
```
|
```
|
||||||
|
|
||||||
??? info "`traefik.http.services.<service_name>.loadbalancer.sticky.cookie.httponly`"
|
??? info "`traefik.http.services.<service_name>.loadbalancer.sticky.cookie.httponly`"
|
||||||
|
@ -242,9 +251,8 @@ you'd add the tag `traefik.http.services.{name-of-your-choice}.loadbalancer.pass
|
||||||
```
|
```
|
||||||
|
|
||||||
??? info "`traefik.http.services.<service_name>.loadbalancer.responseforwarding.flushinterval`"
|
??? info "`traefik.http.services.<service_name>.loadbalancer.responseforwarding.flushinterval`"
|
||||||
<!-- TODO doc responseforwarding in services page -->
|
|
||||||
|
|
||||||
FlushInterval specifies the flush interval to flush to the client while copying the response body.
|
See [response forwarding](../services/index.md#response-forwarding) for more information.
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
traefik.http.services.myservice.loadbalancer.responseforwarding.flushinterval=10
|
traefik.http.services.myservice.loadbalancer.responseforwarding.flushinterval=10
|
||||||
|
|
|
@ -133,6 +133,14 @@ you'd add the label `traefik.http.services.{name-of-your-choice}.loadbalancer.pa
|
||||||
traefik.http.services.myservice.loadbalancer.server.scheme=http
|
traefik.http.services.myservice.loadbalancer.server.scheme=http
|
||||||
```
|
```
|
||||||
|
|
||||||
|
??? info "`traefik.http.services.<service_name>.loadbalancer.serverstransport`"
|
||||||
|
|
||||||
|
See [serverstransport](../services/index.md#serverstransport) for more information.
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
traefik.http.services.<service_name>.loadbalancer.serverstransport=foobar
|
||||||
|
```
|
||||||
|
|
||||||
??? info "`traefik.http.services.<service_name>.loadbalancer.passhostheader`"
|
??? info "`traefik.http.services.<service_name>.loadbalancer.passhostheader`"
|
||||||
|
|
||||||
See [pass Host header](../services/index.md#pass-host-header) for more information.
|
See [pass Host header](../services/index.md#pass-host-header) for more information.
|
||||||
|
@ -205,12 +213,12 @@ you'd add the label `traefik.http.services.{name-of-your-choice}.loadbalancer.pa
|
||||||
traefik.http.services.myservice.loadbalancer.healthcheck.followredirects=true
|
traefik.http.services.myservice.loadbalancer.healthcheck.followredirects=true
|
||||||
```
|
```
|
||||||
|
|
||||||
??? info "`traefik.http.services.<service_name>.loadbalancer.sticky`"
|
??? info "`traefik.http.services.<service_name>.loadbalancer.sticky.cookie`"
|
||||||
|
|
||||||
See [sticky sessions](../services/index.md#sticky-sessions) for more information.
|
See [sticky sessions](../services/index.md#sticky-sessions) for more information.
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
traefik.http.services.myservice.loadbalancer.sticky=true
|
traefik.http.services.myservice.loadbalancer.sticky.cookie=true
|
||||||
```
|
```
|
||||||
|
|
||||||
??? info "`traefik.http.services.<service_name>.loadbalancer.sticky.cookie.httponly`"
|
??? info "`traefik.http.services.<service_name>.loadbalancer.sticky.cookie.httponly`"
|
||||||
|
|
|
@ -35,14 +35,14 @@ You can find an excerpt of the supported Kubernetes Gateway API resources in the
|
||||||
|
|
||||||
| Kind | Purpose | Concept Behind |
|
| Kind | Purpose | Concept Behind |
|
||||||
|------------------------------------|---------------------------------------------------------------------------|-------------------------------------------------------------------------------------------|
|
|------------------------------------|---------------------------------------------------------------------------|-------------------------------------------------------------------------------------------|
|
||||||
| [GatewayClass](#kind-gatewayclass) | Defines a set of Gateways that share a common configuration and behaviour | [GatewayClass](https://kubernetes-sigs.github.io/service-apis/api-overview/#gatewayclass) |
|
| [GatewayClass](#kind-gatewayclass) | Defines a set of Gateways that share a common configuration and behaviour | [GatewayClass](https://kubernetes-sigs.github.io/gateway-api/api-overview/#gatewayclass) |
|
||||||
| [Gateway](#kind-gateway) | Describes how traffic can be translated to Services within the cluster | [Gateway](https://kubernetes-sigs.github.io/service-apis/api-overview/#gateway) |
|
| [Gateway](#kind-gateway) | Describes how traffic can be translated to Services within the cluster | [Gateway](https://kubernetes-sigs.github.io/gateway-api/api-overview/#gateway) |
|
||||||
| [HTTPRoute](#kind-httproute) | HTTP rules for mapping requests from a Gateway to Kubernetes Services | [Route](https://kubernetes-sigs.github.io/service-apis/api-overview/#httptcpfooroute) |
|
| [HTTPRoute](#kind-httproute) | HTTP rules for mapping requests from a Gateway to Kubernetes Services | [Route](https://kubernetes-sigs.github.io/gateway-api/api-overview/#httptcpfooroute) |
|
||||||
|
|
||||||
### Kind: `GatewayClass`
|
### Kind: `GatewayClass`
|
||||||
|
|
||||||
`GatewayClass` is cluster-scoped resource defined by the infrastructure provider. This resource represents a class of Gateways that can be instantiated.
|
`GatewayClass` is cluster-scoped resource defined by the infrastructure provider. This resource represents a class of Gateways that can be instantiated.
|
||||||
More details on the GatewayClass [official documentation](https://kubernetes-sigs.github.io/service-apis/gatewayclass/).
|
More details on the GatewayClass [official documentation](https://kubernetes-sigs.github.io/gateway-api/gatewayclass/).
|
||||||
|
|
||||||
The `GatewayClass` should be declared by the infrastructure provider, otherwise please register the `GatewayClass`
|
The `GatewayClass` should be declared by the infrastructure provider, otherwise please register the `GatewayClass`
|
||||||
[definition](../../reference/dynamic-configuration/kubernetes-gateway.md#definitions) in the Kubernetes cluster before
|
[definition](../../reference/dynamic-configuration/kubernetes-gateway.md#definitions) in the Kubernetes cluster before
|
||||||
|
@ -65,7 +65,7 @@ creating `GatewayClass` objects.
|
||||||
|
|
||||||
A `Gateway` is 1:1 with the life cycle of the configuration of infrastructure. When a user creates a Gateway,
|
A `Gateway` is 1:1 with the life cycle of the configuration of infrastructure. When a user creates a Gateway,
|
||||||
some load balancing infrastructure is provisioned or configured by the GatewayClass controller.
|
some load balancing infrastructure is provisioned or configured by the GatewayClass controller.
|
||||||
More details on the Gateway [official documentation](https://kubernetes-sigs.github.io/service-apis/gateway/).
|
More details on the Gateway [official documentation](https://kubernetes-sigs.github.io/gateway-api/gateway/).
|
||||||
|
|
||||||
Register the `Gateway` [definition](../../reference/dynamic-configuration/kubernetes-gateway.md#definitions) in the
|
Register the `Gateway` [definition](../../reference/dynamic-configuration/kubernetes-gateway.md#definitions) in the
|
||||||
Kubernetes cluster before creating `Gateway` objects.
|
Kubernetes cluster before creating `Gateway` objects.
|
||||||
|
|
|
@ -110,6 +110,14 @@ A Story of key & values
|
||||||
|-----------------------------------------------------------------|-----------------------------------------|
|
|-----------------------------------------------------------------|-----------------------------------------|
|
||||||
| `traefik/http/services/myservice/loadbalancer/servers/0/url` | `http://<ip-server-1>:<port-server-1>/` |
|
| `traefik/http/services/myservice/loadbalancer/servers/0/url` | `http://<ip-server-1>:<port-server-1>/` |
|
||||||
|
|
||||||
|
??? info "`traefik/http/services/<service_name>/loadbalancer/serverstransport`"
|
||||||
|
|
||||||
|
See [serverstransport](../services/index.md#serverstransport) for more information.
|
||||||
|
|
||||||
|
| Key (Path) | Value |
|
||||||
|
|-----------------------------------------------------------------|----------|
|
||||||
|
| `traefik/http/services/myservice/loadbalancer/serverstransport` | `foobar` |
|
||||||
|
|
||||||
??? info "`traefik/http/services/<service_name>/loadbalancer/passhostheader`"
|
??? info "`traefik/http/services/<service_name>/loadbalancer/passhostheader`"
|
||||||
|
|
||||||
See [pass Host header](../services/index.md#pass-host-header) for more information.
|
See [pass Host header](../services/index.md#pass-host-header) for more information.
|
||||||
|
|
|
@ -160,6 +160,14 @@ For example, to change the passHostHeader behavior, you'd add the label `"traefi
|
||||||
"traefik.http.services.myservice.loadbalancer.server.scheme": "http"
|
"traefik.http.services.myservice.loadbalancer.server.scheme": "http"
|
||||||
```
|
```
|
||||||
|
|
||||||
|
??? info "`traefik.http.services.<service_name>.loadbalancer.serverstransport`"
|
||||||
|
|
||||||
|
See [serverstransport](../services/index.md#serverstransport) for more information.
|
||||||
|
|
||||||
|
```json
|
||||||
|
"traefik.http.services.<service_name>.loadbalancer.serverstransport": "foobar"
|
||||||
|
```
|
||||||
|
|
||||||
??? info "`traefik.http.services.<service_name>.loadbalancer.passhostheader`"
|
??? info "`traefik.http.services.<service_name>.loadbalancer.passhostheader`"
|
||||||
|
|
||||||
See [pass Host header](../services/index.md#pass-host-header) for more information.
|
See [pass Host header](../services/index.md#pass-host-header) for more information.
|
||||||
|
|
|
@ -166,6 +166,14 @@ you'd add the label `traefik.http.services.{name-of-your-choice}.loadbalancer.pa
|
||||||
- "traefik.http.services.myservice.loadbalancer.server.scheme=http"
|
- "traefik.http.services.myservice.loadbalancer.server.scheme=http"
|
||||||
```
|
```
|
||||||
|
|
||||||
|
??? info "`traefik.http.services.<service_name>.loadbalancer.serverstransport`"
|
||||||
|
|
||||||
|
See [serverstransport](../services/index.md#serverstransport) for more information.
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
- "traefik.http.services.<service_name>.loadbalancer.serverstransport=foobar"
|
||||||
|
```
|
||||||
|
|
||||||
??? info "`traefik.http.services.<service_name>.loadbalancer.passhostheader`"
|
??? info "`traefik.http.services.<service_name>.loadbalancer.passhostheader`"
|
||||||
|
|
||||||
See [pass Host header](../services/index.md#pass-host-header) for more information.
|
See [pass Host header](../services/index.md#pass-host-header) for more information.
|
||||||
|
@ -195,7 +203,7 @@ you'd add the label `traefik.http.services.{name-of-your-choice}.loadbalancer.pa
|
||||||
See [health check](../services/index.md#health-check) for more information.
|
See [health check](../services/index.md#health-check) for more information.
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
- "traefik.http.services.myservice.loadbalancer.healthcheck.interval=10"
|
- "traefik.http.services.myservice.loadbalancer.healthcheck.interval=10s"
|
||||||
```
|
```
|
||||||
|
|
||||||
??? info "`traefik.http.services.<service_name>.loadbalancer.healthcheck.path`"
|
??? info "`traefik.http.services.<service_name>.loadbalancer.healthcheck.path`"
|
||||||
|
|
|
@ -235,7 +235,7 @@ The table below lists all the available matchers:
|
||||||
| ```Host(`example.com`, ...)``` | Check if the request domain (host header value) targets one of the given `domains`. |
|
| ```Host(`example.com`, ...)``` | Check if the request domain (host header value) targets one of the given `domains`. |
|
||||||
| ```HostHeader(`example.com`, ...)``` | Check if the request domain (host header value) targets one of the given `domains`. |
|
| ```HostHeader(`example.com`, ...)``` | Check if the request domain (host header value) targets one of the given `domains`. |
|
||||||
| ```HostRegexp(`example.com`, `{subdomain:[a-z]+}.example.com`, ...)``` | Check if the request domain matches the given `regexp`. |
|
| ```HostRegexp(`example.com`, `{subdomain:[a-z]+}.example.com`, ...)``` | Check if the request domain matches the given `regexp`. |
|
||||||
| ```Method(`GET`, ...)``` | Check if the request method is one of the given `methods` (`GET`, `POST`, `PUT`, `DELETE`, `PATCH`) |
|
| ```Method(`GET`, ...)``` | Check if the request method is one of the given `methods` (`GET`, `POST`, `PUT`, `DELETE`, `PATCH`, `HEAD`) |
|
||||||
| ```Path(`/path`, `/articles/{cat:[a-z]+}/{id:[0-9]+}`, ...)``` | Match exact request path. It accepts a sequence of literal and regular expression paths. |
|
| ```Path(`/path`, `/articles/{cat:[a-z]+}/{id:[0-9]+}`, ...)``` | Match exact request path. It accepts a sequence of literal and regular expression paths. |
|
||||||
| ```PathPrefix(`/products/`, `/articles/{cat:[a-z]+}/{id:[0-9]+}`)``` | Match request prefix path. It accepts a sequence of literal and regular expression prefix paths. |
|
| ```PathPrefix(`/products/`, `/articles/{cat:[a-z]+}/{id:[0-9]+}`)``` | Match request prefix path. It accepts a sequence of literal and regular expression prefix paths. |
|
||||||
| ```Query(`foo=bar`, `bar=baz`)``` | Match Query String parameters. It accepts a sequence of key=value pairs. |
|
| ```Query(`foo=bar`, `bar=baz`)``` | Match Query String parameters. It accepts a sequence of key=value pairs. |
|
||||||
|
|
|
@ -480,7 +480,7 @@ By default, `passHostHeader` is true.
|
||||||
services:
|
services:
|
||||||
Service01:
|
Service01:
|
||||||
loadBalancer:
|
loadBalancer:
|
||||||
serversTransport = "mytransport"
|
serversTransport: mytransport
|
||||||
```
|
```
|
||||||
|
|
||||||
!!! info default serversTransport
|
!!! info default serversTransport
|
||||||
|
|
4
go.mod
4
go.mod
|
@ -7,7 +7,6 @@ require (
|
||||||
github.com/ExpediaDotCom/haystack-client-go v0.0.0-20190315171017-e7edbdf53a61
|
github.com/ExpediaDotCom/haystack-client-go v0.0.0-20190315171017-e7edbdf53a61
|
||||||
github.com/Masterminds/sprig/v3 v3.2.0
|
github.com/Masterminds/sprig/v3 v3.2.0
|
||||||
github.com/Microsoft/hcsshim v0.8.7 // indirect
|
github.com/Microsoft/hcsshim v0.8.7 // indirect
|
||||||
github.com/NYTimes/gziphandler v1.1.1
|
|
||||||
github.com/Shopify/sarama v1.23.1 // indirect
|
github.com/Shopify/sarama v1.23.1 // indirect
|
||||||
github.com/abbot/go-http-auth v0.0.0-00010101000000-000000000000
|
github.com/abbot/go-http-auth v0.0.0-00010101000000-000000000000
|
||||||
github.com/abronan/valkeyrie v0.0.0-20200127174252-ef4277a138cd
|
github.com/abronan/valkeyrie v0.0.0-20200127174252-ef4277a138cd
|
||||||
|
@ -67,9 +66,10 @@ require (
|
||||||
github.com/prometheus/client_model v0.2.0
|
github.com/prometheus/client_model v0.2.0
|
||||||
github.com/rancher/go-rancher-metadata v0.0.0-20200311180630-7f4c936a06ac
|
github.com/rancher/go-rancher-metadata v0.0.0-20200311180630-7f4c936a06ac
|
||||||
github.com/sirupsen/logrus v1.7.0
|
github.com/sirupsen/logrus v1.7.0
|
||||||
github.com/stretchr/testify v1.6.1
|
github.com/stretchr/testify v1.7.0
|
||||||
github.com/stvp/go-udp-testing v0.0.0-20191102171040-06b61409b154
|
github.com/stvp/go-udp-testing v0.0.0-20191102171040-06b61409b154
|
||||||
github.com/tinylib/msgp v1.0.2 // indirect
|
github.com/tinylib/msgp v1.0.2 // indirect
|
||||||
|
github.com/traefik/gziphandler v1.1.2-0.20210212101304-175e0fad6888
|
||||||
github.com/traefik/paerser v0.1.1
|
github.com/traefik/paerser v0.1.1
|
||||||
github.com/traefik/yaegi v0.9.8
|
github.com/traefik/yaegi v0.9.8
|
||||||
github.com/uber/jaeger-client-go v2.25.0+incompatible
|
github.com/uber/jaeger-client-go v2.25.0+incompatible
|
||||||
|
|
6
go.sum
6
go.sum
|
@ -92,8 +92,6 @@ github.com/Microsoft/go-winio v0.4.15-0.20190919025122-fc70bd9a86b5/go.mod h1:tT
|
||||||
github.com/Microsoft/hcsshim v0.8.7 h1:ptnOoufxGSzauVTsdE+wMYnCWA301PdoN4xg5oRdZpg=
|
github.com/Microsoft/hcsshim v0.8.7 h1:ptnOoufxGSzauVTsdE+wMYnCWA301PdoN4xg5oRdZpg=
|
||||||
github.com/Microsoft/hcsshim v0.8.7/go.mod h1:OHd7sQqRFrYd3RmSgbgji+ctCwkbq2wbEYNSzOYtcBQ=
|
github.com/Microsoft/hcsshim v0.8.7/go.mod h1:OHd7sQqRFrYd3RmSgbgji+ctCwkbq2wbEYNSzOYtcBQ=
|
||||||
github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ=
|
github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ=
|
||||||
github.com/NYTimes/gziphandler v1.1.1 h1:ZUDjpQae29j0ryrS0u/B8HZfJBtBQHjqw2rQ2cqUQ3I=
|
|
||||||
github.com/NYTimes/gziphandler v1.1.1/go.mod h1:n/CVRwUEOgIxrgPvAQhUUr9oeUtvrhMomdKFjzJNB0c=
|
|
||||||
github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU=
|
github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU=
|
||||||
github.com/OpenDNS/vegadns2client v0.0.0-20180418235048-a3fa4a771d87 h1:xPMsUicZ3iosVPSIP7bW5EcGUzjiiMl1OYTe14y/R24=
|
github.com/OpenDNS/vegadns2client v0.0.0-20180418235048-a3fa4a771d87 h1:xPMsUicZ3iosVPSIP7bW5EcGUzjiiMl1OYTe14y/R24=
|
||||||
github.com/OpenDNS/vegadns2client v0.0.0-20180418235048-a3fa4a771d87/go.mod h1:iGLljf5n9GjT6kc0HBvyI1nOKnGQbNB66VzSNbK5iks=
|
github.com/OpenDNS/vegadns2client v0.0.0-20180418235048-a3fa4a771d87/go.mod h1:iGLljf5n9GjT6kc0HBvyI1nOKnGQbNB66VzSNbK5iks=
|
||||||
|
@ -932,6 +930,8 @@ github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81P
|
||||||
github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA=
|
github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA=
|
||||||
github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0=
|
github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0=
|
||||||
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
|
github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY=
|
||||||
|
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
github.com/stvp/go-udp-testing v0.0.0-20191102171040-06b61409b154 h1:XGopsea1Dw7ecQ8JscCNQXDGYAKDiWjDeXnpN/+BY9g=
|
github.com/stvp/go-udp-testing v0.0.0-20191102171040-06b61409b154 h1:XGopsea1Dw7ecQ8JscCNQXDGYAKDiWjDeXnpN/+BY9g=
|
||||||
github.com/stvp/go-udp-testing v0.0.0-20191102171040-06b61409b154/go.mod h1:7jxmlfBCDBXRzr0eAQJ48XC1hBu1np4CS5+cHEYfwpc=
|
github.com/stvp/go-udp-testing v0.0.0-20191102171040-06b61409b154/go.mod h1:7jxmlfBCDBXRzr0eAQJ48XC1hBu1np4CS5+cHEYfwpc=
|
||||||
github.com/syndtr/gocapability v0.0.0-20170704070218-db04d3cc01c8/go.mod h1:hkRG7XYTFWNJGYcbNJQlaLq0fg1yr4J4t/NcTQtrfww=
|
github.com/syndtr/gocapability v0.0.0-20170704070218-db04d3cc01c8/go.mod h1:hkRG7XYTFWNJGYcbNJQlaLq0fg1yr4J4t/NcTQtrfww=
|
||||||
|
@ -942,6 +942,8 @@ github.com/tinylib/msgp v1.0.2/go.mod h1:+d+yLhGm8mzTaHzB+wgMYrodPfmZrzkirds8fDW
|
||||||
github.com/tmc/grpc-websocket-proxy v0.0.0-20170815181823-89b8d40f7ca8/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
github.com/tmc/grpc-websocket-proxy v0.0.0-20170815181823-89b8d40f7ca8/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
||||||
github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5 h1:LnC5Kc/wtumK+WB441p7ynQJzVuNRJiqddSIE3IlSEQ=
|
github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5 h1:LnC5Kc/wtumK+WB441p7ynQJzVuNRJiqddSIE3IlSEQ=
|
||||||
github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
||||||
|
github.com/traefik/gziphandler v1.1.2-0.20210212101304-175e0fad6888 h1:GMY0C+M/w8xO+/NP3Kq6sroMd+z2KbbdVr1K8o2NLHk=
|
||||||
|
github.com/traefik/gziphandler v1.1.2-0.20210212101304-175e0fad6888/go.mod h1:sLqwoN03tkluITKL+lPEZbfsJQU2suYoKbrR/HeV9aM=
|
||||||
github.com/traefik/paerser v0.1.1 h1:Suj0iA4hTAV6E4Dh5/++TXAj5u6iTwydBlFssIUz+9w=
|
github.com/traefik/paerser v0.1.1 h1:Suj0iA4hTAV6E4Dh5/++TXAj5u6iTwydBlFssIUz+9w=
|
||||||
github.com/traefik/paerser v0.1.1/go.mod h1:yYnAgdEC2wJH5CgG75qGWC8SsFDEapg09o9RrA6FfrE=
|
github.com/traefik/paerser v0.1.1/go.mod h1:yYnAgdEC2wJH5CgG75qGWC8SsFDEapg09o9RrA6FfrE=
|
||||||
github.com/traefik/yaegi v0.9.8 h1:sYGKV2d911B9olVsSO/nRmjIh+CQTffAUSzoyv4a4/4=
|
github.com/traefik/yaegi v0.9.8 h1:sYGKV2d911B9olVsSO/nRmjIh+CQTffAUSzoyv4a4/4=
|
||||||
|
|
|
@ -482,10 +482,10 @@ func (s *ConsulCatalogSuite) TestSameServiceIDOnDifferentConsulAgent(c *check.C)
|
||||||
s.composeProject.Container(c, "whoami2").NetworkSettings.IPAddress))
|
s.composeProject.Container(c, "whoami2").NetworkSettings.IPAddress))
|
||||||
c.Assert(err, checker.IsNil)
|
c.Assert(err, checker.IsNil)
|
||||||
|
|
||||||
err = s.deregisterService("whoami1", false)
|
err = s.deregisterService("whoami", false)
|
||||||
c.Assert(err, checker.IsNil)
|
c.Assert(err, checker.IsNil)
|
||||||
|
|
||||||
err = s.deregisterService("whoami2", true)
|
err = s.deregisterService("whoami", true)
|
||||||
c.Assert(err, checker.IsNil)
|
c.Assert(err, checker.IsNil)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -6,8 +6,8 @@ import (
|
||||||
"mime"
|
"mime"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/NYTimes/gziphandler"
|
|
||||||
"github.com/opentracing/opentracing-go/ext"
|
"github.com/opentracing/opentracing-go/ext"
|
||||||
|
"github.com/traefik/gziphandler"
|
||||||
"github.com/traefik/traefik/v2/pkg/config/dynamic"
|
"github.com/traefik/traefik/v2/pkg/config/dynamic"
|
||||||
"github.com/traefik/traefik/v2/pkg/log"
|
"github.com/traefik/traefik/v2/pkg/log"
|
||||||
"github.com/traefik/traefik/v2/pkg/middlewares"
|
"github.com/traefik/traefik/v2/pkg/middlewares"
|
||||||
|
@ -52,7 +52,7 @@ func (c *compress) ServeHTTP(rw http.ResponseWriter, req *http.Request) {
|
||||||
c.next.ServeHTTP(rw, req)
|
c.next.ServeHTTP(rw, req)
|
||||||
} else {
|
} else {
|
||||||
ctx := middlewares.GetLoggerCtx(req.Context(), c.name, typeName)
|
ctx := middlewares.GetLoggerCtx(req.Context(), c.name, typeName)
|
||||||
gzipHandler(ctx, c.next).ServeHTTP(rw, req)
|
c.gzipHandler(ctx).ServeHTTP(rw, req)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,15 +60,16 @@ func (c *compress) GetTracingInformation() (string, ext.SpanKindEnum) {
|
||||||
return c.name, tracing.SpanKindNoneEnum
|
return c.name, tracing.SpanKindNoneEnum
|
||||||
}
|
}
|
||||||
|
|
||||||
func gzipHandler(ctx context.Context, h http.Handler) http.Handler {
|
func (c *compress) gzipHandler(ctx context.Context) http.Handler {
|
||||||
wrapper, err := gziphandler.GzipHandlerWithOpts(
|
wrapper, err := gziphandler.GzipHandlerWithOpts(
|
||||||
|
gziphandler.ContentTypeExceptions(c.excludes),
|
||||||
gziphandler.CompressionLevel(gzip.DefaultCompression),
|
gziphandler.CompressionLevel(gzip.DefaultCompression),
|
||||||
gziphandler.MinSize(gziphandler.DefaultMinSize))
|
gziphandler.MinSize(gziphandler.DefaultMinSize))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.FromContext(ctx).Error(err)
|
log.FromContext(ctx).Error(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return wrapper(h)
|
return wrapper(c.next)
|
||||||
}
|
}
|
||||||
|
|
||||||
func contains(values []string, val string) bool {
|
func contains(values []string, val string) bool {
|
||||||
|
|
|
@ -7,9 +7,9 @@ import (
|
||||||
"net/http/httptest"
|
"net/http/httptest"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/NYTimes/gziphandler"
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
"github.com/traefik/gziphandler"
|
||||||
"github.com/traefik/traefik/v2/pkg/config/dynamic"
|
"github.com/traefik/traefik/v2/pkg/config/dynamic"
|
||||||
"github.com/traefik/traefik/v2/pkg/testhelpers"
|
"github.com/traefik/traefik/v2/pkg/testhelpers"
|
||||||
)
|
)
|
||||||
|
@ -91,25 +91,26 @@ func TestShouldNotCompressWhenNoAcceptEncodingHeader(t *testing.T) {
|
||||||
func TestShouldNotCompressWhenSpecificContentType(t *testing.T) {
|
func TestShouldNotCompressWhenSpecificContentType(t *testing.T) {
|
||||||
baseBody := generateBytes(gziphandler.DefaultMinSize)
|
baseBody := generateBytes(gziphandler.DefaultMinSize)
|
||||||
|
|
||||||
next := http.HandlerFunc(func(rw http.ResponseWriter, r *http.Request) {
|
|
||||||
_, err := rw.Write(baseBody)
|
|
||||||
if err != nil {
|
|
||||||
http.Error(rw, err.Error(), http.StatusInternalServerError)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
desc string
|
desc string
|
||||||
conf dynamic.Compress
|
conf dynamic.Compress
|
||||||
reqContentType string
|
reqContentType string
|
||||||
|
respContentType string
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
desc: "text/event-stream",
|
desc: "Exclude Request Content-Type",
|
||||||
conf: dynamic.Compress{
|
conf: dynamic.Compress{
|
||||||
ExcludedContentTypes: []string{"text/event-stream"},
|
ExcludedContentTypes: []string{"text/event-stream"},
|
||||||
},
|
},
|
||||||
reqContentType: "text/event-stream",
|
reqContentType: "text/event-stream",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
desc: "Exclude Response Content-Type",
|
||||||
|
conf: dynamic.Compress{
|
||||||
|
ExcludedContentTypes: []string{"text/event-stream"},
|
||||||
|
},
|
||||||
|
respContentType: "text/event-stream",
|
||||||
|
},
|
||||||
{
|
{
|
||||||
desc: "application/grpc",
|
desc: "application/grpc",
|
||||||
conf: dynamic.Compress{},
|
conf: dynamic.Compress{},
|
||||||
|
@ -128,6 +129,17 @@ func TestShouldNotCompressWhenSpecificContentType(t *testing.T) {
|
||||||
req.Header.Add(contentTypeHeader, test.reqContentType)
|
req.Header.Add(contentTypeHeader, test.reqContentType)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
next := http.HandlerFunc(func(rw http.ResponseWriter, r *http.Request) {
|
||||||
|
if len(test.respContentType) > 0 {
|
||||||
|
rw.Header().Set(contentTypeHeader, test.respContentType)
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := rw.Write(baseBody)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(rw, err.Error(), http.StatusInternalServerError)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
handler, err := New(context.Background(), next, test.conf, "test")
|
handler, err := New(context.Background(), next, test.conf, "test")
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
|
|
@ -39,8 +39,8 @@ func NewChallengeTLSALPN(timeout time.Duration) *ChallengeTLSALPN {
|
||||||
|
|
||||||
// Present presents a challenge to obtain new ACME certificate.
|
// Present presents a challenge to obtain new ACME certificate.
|
||||||
func (c *ChallengeTLSALPN) Present(domain, _, keyAuth string) error {
|
func (c *ChallengeTLSALPN) Present(domain, _, keyAuth string) error {
|
||||||
log.WithoutContext().WithField(log.ProviderName, providerNameALPN).
|
logger := log.WithoutContext().WithField(log.ProviderName, providerNameALPN)
|
||||||
Debugf("TLS Challenge Present temp certificate for %s", domain)
|
logger.Debugf("TLS Challenge Present temp certificate for %s", domain)
|
||||||
|
|
||||||
certPEMBlock, keyPEMBlock, err := tlsalpn01.ChallengeBlocks(domain, keyAuth)
|
certPEMBlock, keyPEMBlock, err := tlsalpn01.ChallengeBlocks(domain, keyAuth)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -68,6 +68,12 @@ func (c *ChallengeTLSALPN) Present(domain, _, keyAuth string) error {
|
||||||
case t := <-timer.C:
|
case t := <-timer.C:
|
||||||
timer.Stop()
|
timer.Stop()
|
||||||
close(c.chans[string(certPEMBlock)])
|
close(c.chans[string(certPEMBlock)])
|
||||||
|
|
||||||
|
err = c.CleanUp(domain, "", keyAuth)
|
||||||
|
if err != nil {
|
||||||
|
logger.Errorf("Failed to clean up TLS challenge: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
errC = fmt.Errorf("timeout %s", t)
|
errC = fmt.Errorf("timeout %s", t)
|
||||||
case <-ch:
|
case <-ch:
|
||||||
// noop
|
// noop
|
||||||
|
|
|
@ -421,6 +421,7 @@ func (p *Provider) watchNewDomains(ctx context.Context) {
|
||||||
if route.TLS == nil || route.TLS.CertResolver != p.ResolverName {
|
if route.TLS == nil || route.TLS.CertResolver != p.ResolverName {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
ctxRouter := log.With(ctx, log.Str(log.RouterName, routerName), log.Str(log.Rule, route.Rule))
|
ctxRouter := log.With(ctx, log.Str(log.RouterName, routerName), log.Str(log.Rule, route.Rule))
|
||||||
|
|
||||||
tlsStore := "default"
|
tlsStore := "default"
|
||||||
|
@ -462,6 +463,7 @@ func (p *Provider) resolveCertificate(ctx context.Context, domain types.Domain,
|
||||||
if len(uncheckedDomains) == 0 {
|
if len(uncheckedDomains) == 0 {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
defer p.removeResolvingDomains(uncheckedDomains)
|
defer p.removeResolvingDomains(uncheckedDomains)
|
||||||
|
|
||||||
logger := log.FromContext(ctx)
|
logger := log.FromContext(ctx)
|
||||||
|
|
|
@ -4,11 +4,11 @@ RepositoryName = "traefik"
|
||||||
OutputType = "file"
|
OutputType = "file"
|
||||||
FileName = "traefik_changelog.md"
|
FileName = "traefik_changelog.md"
|
||||||
|
|
||||||
# example new bugfix v2.4.2
|
# example new bugfix v2.4.3
|
||||||
CurrentRef = "v2.4"
|
CurrentRef = "v2.4"
|
||||||
PreviousRef = "v2.4.1"
|
PreviousRef = "v2.4.2"
|
||||||
BaseBranch = "v2.4"
|
BaseBranch = "v2.4"
|
||||||
FutureCurrentRefName = "v2.4.2"
|
FutureCurrentRefName = "v2.4.3"
|
||||||
|
|
||||||
ThresholdPreviousRef = 10
|
ThresholdPreviousRef = 10
|
||||||
ThresholdCurrentRef = 10
|
ThresholdCurrentRef = 10
|
||||||
|
|
Loading…
Reference in a new issue