Traefik & Nomad Service Discovery

Configuration Example

You can enable the Nomad provider with as detailed below:

File (YAML)

  1. providers:
  2. nomad: {}

File (TOML)

  1. [providers.nomad]

CLI

  1. --providers.nomad=true

Attaching tags to services:

  1. ...
  2. service {
  3. name = "myService"
  4. tags = [
  5. "traefik.http.routers.my-router.rule=Host(`example.com`)",
  6. ]
  7. }
  8. ...

Configuration Options

FieldDescriptionDefaultRequired
providers.providersThrottleDurationMinimum amount of time to wait for, after a configuration reload, before taking into account any new configuration refresh event.
If multiple events occur within this time, only the most recent one is taken into account, and all others are discarded.
This option cannot be set per provider, but the throttling algorithm applies to each of them independently.
2sNo
providers.nomad.namespacesDefines the namespaces in which the nomad services will be discovered.“”No
providers.nomad.refreshIntervalDefines the polling interval. This option is ignored when the watch option is enabled15sNo
providers.nomad.watchEnables the watch mode to refresh the configuration on a per-event basis.falseNo
providers.nomad.throttleDurationDefines how often the provider is allowed to handle service events from Nomad. This option is only compatible when the watch option is enabled0sNo
providers.nomad.defaultRuleThe Default Host rule for all services. See here for more information“Host({{ normalize .Name }})”No
providers.nomad.constraintsDefines an expression that Traefik matches against the container labels to determine whether to create any route for that container. See here for more information.“”No
providers.nomad.exposedByDefaultExpose Nomad services by default in Traefik. If set to false, services that do not have a traefik.enable=true tag will be ignored from the resulting routing configuration. See here for additional informationtrueNo
providers.nomad.allowEmptyServicesInstructs the provider to create any servers load balancer defined for Docker containers regardless of the healthiness of the corresponding containers.falseNo
providers.nomad.prefixDefines the prefix for Nomad service tags defining Traefik labels.traefikyes
providers.nomad.staleInstructs Traefik to use stale consistency for Nomad service API reads. See here for more informationfalseNo
providers.nomad.endpoint.addressDefines the Address of the Nomad server.http://127.0.0.1:4646No
providers.nomad.endpoint.tokenDefines a per-request ACL token if Nomad ACLs are enabled. See here for more information“”No
providers.nomad.endpoint.endpointWaitTimeDefines a duration for which a watch can block. If not provided, the agent default values will be used.“”No
providers.nomad.endpoint.tlsDefines the TLS configuration used for the secure connection to the Nomad APi.-No
providers.nomad.endpoint.tls.caDefines the path to the certificate authority used for the secure connection to the Nomad API, it defaults to the system bundle.“”No
providers.nomad.endpoint.tls.certDefines the path to the public certificate used for the secure connection to the Nomad API. When using this option, setting the key option is required.‘“Yes
providers.nomad.endpoint.tls.keyDefines the path to the private key used for the secure connection to the Nomad API. When using this option, setting the cert option is required.“”Yes
providers.nomad.endpoint.tls.insecureSkipVerifyInstructs the provider to accept any certificate presented by Nomad when establishing a TLS connection, regardless of the hostnames the certificate covers.falseNo

namespaces

The namespaces option defines the namespaces in which the nomad services will be discovered. When using the namespaces option, the discovered object names will be suffixed as shown below:

  1. <resource-name>@nomad-<namespace>

Warning

One should only define either the namespaces option or the namespace option.

File (YAML)

  1. providers:
  2. nomad:
  3. namespaces:
  4. - "ns1"
  5. - "ns2"
  6. # ...

File (TOML)

  1. [providers.nomad]
  2. namespaces = ["ns1", "ns2"]
  3. # ...

CLI

  1. --providers.nomad.namespaces=ns1,ns2
  2. # ...

stale

Use stale consistency for Nomad service API reads.

This makes reads very fast and scalable at the cost of a higher likelihood of stale values.

For more information, see the Nomad documentation on consistency.

File (YAML)

  1. providers:
  2. nomad:
  3. stale: true
  4. # ...

File (TOML)

  1. [providers.nomad]
  2. stale = true
  3. # ...

CLI

  1. --providers.nomad.stale=true
  2. # ...

token

Token is used to provide a per-request ACL token, if Nomad ACLs are enabled. The appropriate ACL privilege for this token is ‘read-job’, as outlined in the Nomad documentation on ACL.

File (YAML)

  1. providers:
  2. nomad:
  3. endpoint:
  4. token: test
  5. # ...

File (TOML)

  1. [providers.nomad]
  2. [providers.nomad.endpoint]
  3. token = "test"
  4. # ...

CLI

  1. --providers.nomad.endpoint.token=test
  2. # ...

defaultRule

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, and can include sprig template functions. 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.

File (YAML)

  1. providers:
  2. nomad:
  3. defaultRule: "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
  4. # ...

File (TOML)

  1. [providers.nomad]
  2. defaultRule = "Host(`{{ .Name }}.{{ index .Labels \"customLabel\"}}`)"
  3. # ...

CLI

  1. --providers.nomad.defaultRule='Host(`{{ .Name }}.{{ index .Labels "customLabel"}}`)'
  2. # ...

Default rule and Traefik service

The exposure of the Traefik container, combined with the default rule mechanism, can lead to create a router targeting itself in a loop. In this case, to prevent an infinite loop, Traefik adds an internal middleware to refuse the request if it comes from the same router.

constraints

The constraints option can be set to an expression that Traefik matches against the service tags to determine whether to create any route for that service. If none of the service 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, as well as the usual boolean logic, as shown in examples below.

Constraints Expression Examples

  1. # Includes only services having the tag `a.tag.name=foo`
  2. constraints = "Tag(`a.tag.name=foo`)"
  1. # Excludes services having any tag `a.tag.name=foo`
  2. constraints = "!Tag(`a.tag.name=foo`)"
  1. # With logical AND.
  2. constraints = "Tag(`a.tag.name`) && Tag(`another.tag.name`)"
  1. # With logical OR.
  2. constraints = "Tag(`a.tag.name`) || Tag(`another.tag.name`)"
  1. # With logical AND and OR, with precedence set by parentheses.
  2. constraints = "Tag(`a.tag.name`) && (Tag(`another.tag.name`) || Tag(`yet.another.tag.name`))"
  1. # Includes only services having a tag matching the `a\.tag\.t.+` regular expression.
  2. constraints = "TagRegex(`a\.tag\.t.+`)"

File (YAML)

  1. providers:
  2. nomad:
  3. constraints: "Tag(`a.tag.name`)"
  4. # ...

File (TOML)

  1. [providers.nomad]
  2. constraints = "Tag(`a.tag.name`)"
  3. # ...

CLI

  1. --providers.nomad.constraints="Tag(`a.tag.name`)"
  2. # ...

For additional information, refer to Restrict the Scope of Service Discovery.

Routing Configuration

See the dedicated section in routing.