2018-11-14 09:18:03 +00:00
package static
import (
2019-07-19 09:52:04 +00:00
"fmt"
stdlog "log"
2018-11-27 16:42:04 +00:00
"strings"
"time"
2019-08-03 01:58:23 +00:00
"github.com/containous/traefik/v2/pkg/log"
"github.com/containous/traefik/v2/pkg/ping"
acmeprovider "github.com/containous/traefik/v2/pkg/provider/acme"
2019-10-15 15:34:08 +00:00
"github.com/containous/traefik/v2/pkg/provider/consulcatalog"
2019-08-03 01:58:23 +00:00
"github.com/containous/traefik/v2/pkg/provider/docker"
"github.com/containous/traefik/v2/pkg/provider/file"
"github.com/containous/traefik/v2/pkg/provider/kubernetes/crd"
"github.com/containous/traefik/v2/pkg/provider/kubernetes/ingress"
"github.com/containous/traefik/v2/pkg/provider/marathon"
"github.com/containous/traefik/v2/pkg/provider/rancher"
"github.com/containous/traefik/v2/pkg/provider/rest"
"github.com/containous/traefik/v2/pkg/tls"
"github.com/containous/traefik/v2/pkg/tracing/datadog"
2019-11-27 15:00:07 +00:00
"github.com/containous/traefik/v2/pkg/tracing/elastic"
2019-08-03 01:58:23 +00:00
"github.com/containous/traefik/v2/pkg/tracing/haystack"
"github.com/containous/traefik/v2/pkg/tracing/instana"
"github.com/containous/traefik/v2/pkg/tracing/jaeger"
"github.com/containous/traefik/v2/pkg/tracing/zipkin"
"github.com/containous/traefik/v2/pkg/types"
2019-02-18 06:52:03 +00:00
assetfs "github.com/elazarl/go-bindata-assetfs"
2019-08-03 01:58:23 +00:00
legolog "github.com/go-acme/lego/v3/log"
2019-07-19 09:52:04 +00:00
"github.com/sirupsen/logrus"
2018-11-14 09:18:03 +00:00
)
2018-11-27 16:42:04 +00:00
const (
// DefaultInternalEntryPointName the name of the default internal entry point
DefaultInternalEntryPointName = "traefik"
// DefaultGraceTimeout controls how long Traefik serves pending requests
// prior to shutting down.
DefaultGraceTimeout = 10 * time . Second
// DefaultIdleTimeout before closing an idle connection.
DefaultIdleTimeout = 180 * time . Second
// DefaultAcmeCAServer is the default ACME API endpoint
DefaultAcmeCAServer = "https://acme-v02.api.letsencrypt.org/directory"
)
// Configuration is the static configuration
2018-11-14 09:18:03 +00:00
type Configuration struct {
2019-07-01 09:30:05 +00:00
Global * Global ` description:"Global configuration options" json:"global,omitempty" toml:"global,omitempty" yaml:"global,omitempty" export:"true" `
2018-11-27 16:42:04 +00:00
2019-07-01 09:30:05 +00:00
ServersTransport * ServersTransport ` description:"Servers default transport." json:"serversTransport,omitempty" toml:"serversTransport,omitempty" yaml:"serversTransport,omitempty" export:"true" `
EntryPoints EntryPoints ` description:"Entry points definition." json:"entryPoints,omitempty" toml:"entryPoints,omitempty" yaml:"entryPoints,omitempty" export:"true" `
Providers * Providers ` description:"Providers configuration." json:"providers,omitempty" toml:"providers,omitempty" yaml:"providers,omitempty" export:"true" `
2018-11-14 09:18:03 +00:00
2019-07-01 09:30:05 +00:00
API * API ` description:"Enable api/dashboard." json:"api,omitempty" toml:"api,omitempty" yaml:"api,omitempty" label:"allowEmpty" export:"true" `
Metrics * types . Metrics ` description:"Enable a metrics exporter." json:"metrics,omitempty" toml:"metrics,omitempty" yaml:"metrics,omitempty" export:"true" `
Ping * ping . Handler ` description:"Enable ping." json:"ping,omitempty" toml:"ping,omitempty" yaml:"ping,omitempty" label:"allowEmpty" export:"true" `
2018-11-14 09:18:03 +00:00
2019-07-01 09:30:05 +00:00
Log * types . TraefikLog ` description:"Traefik log settings." json:"log,omitempty" toml:"log,omitempty" yaml:"log,omitempty" label:"allowEmpty" export:"true" `
AccessLog * types . AccessLog ` description:"Access log settings." json:"accessLog,omitempty" toml:"accessLog,omitempty" yaml:"accessLog,omitempty" label:"allowEmpty" export:"true" `
Tracing * Tracing ` description:"OpenTracing configuration." json:"tracing,omitempty" toml:"tracing,omitempty" yaml:"tracing,omitempty" label:"allowEmpty" export:"true" `
2018-11-14 09:18:03 +00:00
2019-07-01 09:30:05 +00:00
HostResolver * types . HostResolverConfig ` description:"Enable CNAME Flattening." json:"hostResolver,omitempty" toml:"hostResolver,omitempty" yaml:"hostResolver,omitempty" label:"allowEmpty" export:"true" `
2018-11-14 09:18:03 +00:00
2019-07-19 09:52:04 +00:00
CertificatesResolvers map [ string ] CertificateResolver ` description:"Certificates resolvers configuration." json:"certificatesResolvers,omitempty" toml:"certificatesResolvers,omitempty" yaml:"certificatesResolvers,omitempty" export:"true" `
}
// CertificateResolver contains the configuration for the different types of certificates resolver.
type CertificateResolver struct {
2019-07-01 09:30:05 +00:00
ACME * acmeprovider . Configuration ` description:"Enable ACME (Let's Encrypt): automatic SSL." json:"acme,omitempty" toml:"acme,omitempty" yaml:"acme,omitempty" export:"true" `
2018-11-14 09:18:03 +00:00
}
// Global holds the global configuration.
type Global struct {
2019-09-16 15:26:06 +00:00
CheckNewVersion bool ` description:"Periodically check if a new version has been released." json:"checkNewVersion,omitempty" toml:"checkNewVersion,omitempty" yaml:"checkNewVersion,omitempty" label:"allowEmpty" export:"true" `
SendAnonymousUsage bool ` description:"Periodically send anonymous usage statistics. If the option is not specified, it will be enabled by default." json:"sendAnonymousUsage,omitempty" toml:"sendAnonymousUsage,omitempty" yaml:"sendAnonymousUsage,omitempty" label:"allowEmpty" export:"true" `
2018-11-27 16:42:04 +00:00
}
// ServersTransport options to configure communication between Traefik and the servers
type ServersTransport struct {
2019-07-01 09:30:05 +00:00
InsecureSkipVerify bool ` description:"Disable SSL certificate verification." json:"insecureSkipVerify,omitempty" toml:"insecureSkipVerify,omitempty" yaml:"insecureSkipVerify,omitempty" export:"true" `
RootCAs [ ] tls . FileOrContent ` description:"Add cert file for self-signed certificate." json:"rootCAs,omitempty" toml:"rootCAs,omitempty" yaml:"rootCAs,omitempty" `
MaxIdleConnsPerHost int ` description:"If non-zero, controls the maximum idle (keep-alive) to keep per-host. If zero, DefaultMaxIdleConnsPerHost is used" json:"maxIdleConnsPerHost,omitempty" toml:"maxIdleConnsPerHost,omitempty" yaml:"maxIdleConnsPerHost,omitempty" export:"true" `
ForwardingTimeouts * ForwardingTimeouts ` description:"Timeouts for requests forwarded to the backend servers." json:"forwardingTimeouts,omitempty" toml:"forwardingTimeouts,omitempty" yaml:"forwardingTimeouts,omitempty" export:"true" `
2018-11-14 09:18:03 +00:00
}
// API holds the API configuration
type API struct {
2019-09-06 13:08:04 +00:00
Insecure bool ` description:"Activate API directly on the entryPoint named traefik." json:"insecure,omitempty" toml:"insecure,omitempty" yaml:"insecure,omitempty" export:"true" `
2019-07-19 10:28:07 +00:00
Dashboard bool ` description:"Activate dashboard." json:"dashboard,omitempty" toml:"dashboard,omitempty" yaml:"dashboard,omitempty" export:"true" `
Debug bool ` description:"Enable additional endpoints for debugging and profiling." json:"debug,omitempty" toml:"debug,omitempty" yaml:"debug,omitempty" export:"true" `
// TODO: Re-enable statistics
// Statistics *types.Statistics `description:"Enable more detailed statistics." json:"statistics,omitempty" toml:"statistics,omitempty" yaml:"statistics,omitempty" export:"true" label:"allowEmpty"`
DashboardAssets * assetfs . AssetFS ` json:"-" toml:"-" yaml:"-" label:"-" `
2019-06-17 09:48:05 +00:00
}
// SetDefaults sets the default values.
func ( a * API ) SetDefaults ( ) {
a . Dashboard = true
2018-11-14 09:18:03 +00:00
}
// RespondingTimeouts contains timeout configurations for incoming requests to the Traefik instance.
type RespondingTimeouts struct {
2019-07-01 09:30:05 +00:00
ReadTimeout types . Duration ` description:"ReadTimeout is the maximum duration for reading the entire request, including the body. If zero, no timeout is set." json:"readTimeout,omitempty" toml:"readTimeout,omitempty" yaml:"readTimeout,omitempty" export:"true" `
WriteTimeout types . Duration ` description:"WriteTimeout is the maximum duration before timing out writes of the response. If zero, no timeout is set." json:"writeTimeout,omitempty" toml:"writeTimeout,omitempty" yaml:"writeTimeout,omitempty" export:"true" `
IdleTimeout types . Duration ` description:"IdleTimeout is the maximum amount duration an idle (keep-alive) connection will remain idle before closing itself. If zero, no timeout is set." json:"idleTimeout,omitempty" toml:"idleTimeout,omitempty" yaml:"idleTimeout,omitempty" export:"true" `
2019-06-17 09:48:05 +00:00
}
// SetDefaults sets the default values.
func ( a * RespondingTimeouts ) SetDefaults ( ) {
a . IdleTimeout = types . Duration ( DefaultIdleTimeout )
2018-11-14 09:18:03 +00:00
}
// ForwardingTimeouts contains timeout configurations for forwarding requests to the backend servers.
type ForwardingTimeouts struct {
2019-07-01 09:30:05 +00:00
DialTimeout types . Duration ` description:"The amount of time to wait until a connection to a backend server can be established. If zero, no timeout exists." json:"dialTimeout,omitempty" toml:"dialTimeout,omitempty" yaml:"dialTimeout,omitempty" export:"true" `
ResponseHeaderTimeout types . Duration ` description:"The amount of time to wait for a server's response headers after fully writing the request (including its body, if any). If zero, no timeout exists." json:"responseHeaderTimeout,omitempty" toml:"responseHeaderTimeout,omitempty" yaml:"responseHeaderTimeout,omitempty" export:"true" `
IdleConnTimeout types . Duration ` description:"The maximum period for which an idle HTTP keep-alive connection will remain open before closing itself" json:"idleConnTimeout,omitempty" toml:"idleConnTimeout,omitempty" yaml:"idleConnTimeout,omitempty" export:"true" `
2019-06-17 09:48:05 +00:00
}
// SetDefaults sets the default values.
func ( f * ForwardingTimeouts ) SetDefaults ( ) {
f . DialTimeout = types . Duration ( 30 * time . Second )
2019-06-27 22:36:04 +00:00
f . IdleConnTimeout = types . Duration ( 90 * time . Second )
2018-11-14 09:18:03 +00:00
}
// LifeCycle contains configurations relevant to the lifecycle (such as the shutdown phase) of Traefik.
type LifeCycle struct {
2019-07-01 09:30:05 +00:00
RequestAcceptGraceTimeout types . Duration ` description:"Duration to keep accepting requests before Traefik initiates the graceful shutdown procedure." json:"requestAcceptGraceTimeout,omitempty" toml:"requestAcceptGraceTimeout,omitempty" yaml:"requestAcceptGraceTimeout,omitempty" export:"true" `
GraceTimeOut types . Duration ` description:"Duration to give active requests a chance to finish before Traefik stops." json:"graceTimeOut,omitempty" toml:"graceTimeOut,omitempty" yaml:"graceTimeOut,omitempty" export:"true" `
2019-06-17 09:48:05 +00:00
}
// SetDefaults sets the default values.
func ( a * LifeCycle ) SetDefaults ( ) {
a . GraceTimeOut = types . Duration ( DefaultGraceTimeout )
2018-11-14 09:18:03 +00:00
}
// Tracing holds the tracing configuration.
type Tracing struct {
2019-07-01 09:30:05 +00:00
ServiceName string ` description:"Set the name for this service." json:"serviceName,omitempty" toml:"serviceName,omitempty" yaml:"serviceName,omitempty" export:"true" `
SpanNameLimit int ` description:"Set the maximum character limit for Span names (default 0 = no limit)." json:"spanNameLimit,omitempty" toml:"spanNameLimit,omitempty" yaml:"spanNameLimit,omitempty" export:"true" `
Jaeger * jaeger . Config ` description:"Settings for Jaeger." json:"jaeger,omitempty" toml:"jaeger,omitempty" yaml:"jaeger,omitempty" export:"true" label:"allowEmpty" `
Zipkin * zipkin . Config ` description:"Settings for Zipkin." json:"zipkin,omitempty" toml:"zipkin,omitempty" yaml:"zipkin,omitempty" export:"true" label:"allowEmpty" `
2019-09-02 10:18:04 +00:00
Datadog * datadog . Config ` description:"Settings for Datadog." json:"datadog,omitempty" toml:"datadog,omitempty" yaml:"datadog,omitempty" export:"true" label:"allowEmpty" `
2019-07-01 09:30:05 +00:00
Instana * instana . Config ` description:"Settings for Instana." json:"instana,omitempty" toml:"instana,omitempty" yaml:"instana,omitempty" export:"true" label:"allowEmpty" `
Haystack * haystack . Config ` description:"Settings for Haystack." json:"haystack,omitempty" toml:"haystack,omitempty" yaml:"haystack,omitempty" export:"true" label:"allowEmpty" `
2019-11-27 15:00:07 +00:00
Elastic * elastic . Config ` description:"Settings for Elastic." json:"elastic,omitempty" toml:"elastic,omitempty" yaml:"elastic,omitempty" export:"true" label:"allowEmpty" `
2019-06-17 09:48:05 +00:00
}
// SetDefaults sets the default values.
func ( t * Tracing ) SetDefaults ( ) {
t . ServiceName = "traefik"
t . SpanNameLimit = 0
2018-11-14 09:18:03 +00:00
}
2018-11-27 16:42:04 +00:00
// Providers contains providers configuration
type Providers struct {
2019-10-15 15:34:08 +00:00
ProvidersThrottleDuration types . Duration ` description:"Backends throttle duration: minimum duration between 2 events from providers before applying a new configuration. It avoids unnecessary reloads if multiples events are sent in a short amount of time." json:"providersThrottleDuration,omitempty" toml:"providersThrottleDuration,omitempty" yaml:"providersThrottleDuration,omitempty" export:"true" `
Docker * docker . Provider ` description:"Enable Docker backend with default settings." json:"docker,omitempty" toml:"docker,omitempty" yaml:"docker,omitempty" export:"true" label:"allowEmpty" `
File * file . Provider ` description:"Enable File backend with default settings." json:"file,omitempty" toml:"file,omitempty" yaml:"file,omitempty" export:"true" `
Marathon * marathon . Provider ` description:"Enable Marathon backend with default settings." json:"marathon,omitempty" toml:"marathon,omitempty" yaml:"marathon,omitempty" export:"true" label:"allowEmpty" `
KubernetesIngress * ingress . Provider ` description:"Enable Kubernetes backend with default settings." json:"kubernetesIngress,omitempty" toml:"kubernetesIngress,omitempty" yaml:"kubernetesIngress,omitempty" export:"true" label:"allowEmpty" `
KubernetesCRD * crd . Provider ` description:"Enable Kubernetes backend with default settings." json:"kubernetesCRD,omitempty" toml:"kubernetesCRD,omitempty" yaml:"kubernetesCRD,omitempty" export:"true" label:"allowEmpty" `
Rest * rest . Provider ` description:"Enable Rest backend with default settings." json:"rest,omitempty" toml:"rest,omitempty" yaml:"rest,omitempty" export:"true" label:"allowEmpty" `
Rancher * rancher . Provider ` description:"Enable Rancher backend with default settings." json:"rancher,omitempty" toml:"rancher,omitempty" yaml:"rancher,omitempty" export:"true" label:"allowEmpty" `
2019-10-29 11:32:05 +00:00
ConsulCatalog * consulcatalog . Provider ` description:"Enable ConsulCatalog backend with default settings." json:"consulCatalog,omitempty" toml:"consulCatalog,omitempty" yaml:"consulCatalog,omitempty" `
2018-11-27 16:42:04 +00:00
}
// SetEffectiveConfiguration adds missing configuration parameters derived from existing ones.
// It also takes care of maintaining backwards compatibility.
2019-07-15 08:22:03 +00:00
func ( c * Configuration ) SetEffectiveConfiguration ( ) {
2018-11-27 16:42:04 +00:00
if len ( c . EntryPoints ) == 0 {
2019-06-17 09:48:05 +00:00
ep := & EntryPoint { Address : ":80" }
ep . SetDefaults ( )
2018-11-27 16:42:04 +00:00
c . EntryPoints = EntryPoints {
2019-06-17 09:48:05 +00:00
"http" : ep ,
2018-11-27 16:42:04 +00:00
}
}
2019-09-06 13:08:04 +00:00
if ( c . API != nil && c . API . Insecure ) ||
2019-11-14 15:40:05 +00:00
( c . Ping != nil && ! c . Ping . ManualRouting && c . Ping . EntryPoint == DefaultInternalEntryPointName ) ||
( c . Metrics != nil && c . Metrics . Prometheus != nil && ! c . Metrics . Prometheus . ManualRouting && c . Metrics . Prometheus . EntryPoint == DefaultInternalEntryPointName ) ||
( c . Providers != nil && c . Providers . Rest != nil && c . Providers . Rest . Insecure ) {
2018-11-27 16:42:04 +00:00
if _ , ok := c . EntryPoints [ DefaultInternalEntryPointName ] ; ! ok {
2019-06-17 09:48:05 +00:00
ep := & EntryPoint { Address : ":8080" }
ep . SetDefaults ( )
c . EntryPoints [ DefaultInternalEntryPointName ] = ep
2018-11-27 16:42:04 +00:00
}
}
2019-01-29 16:08:08 +00:00
if c . Providers . Docker != nil {
if c . Providers . Docker . SwarmModeRefreshSeconds <= 0 {
2019-06-17 09:48:05 +00:00
c . Providers . Docker . SwarmModeRefreshSeconds = types . Duration ( 15 * time . Second )
2019-01-29 16:08:08 +00:00
}
}
2019-04-05 10:22:04 +00:00
if c . Providers . Rancher != nil {
if c . Providers . Rancher . RefreshSeconds <= 0 {
c . Providers . Rancher . RefreshSeconds = 15
}
}
2018-11-27 16:42:04 +00:00
c . initACMEProvider ( )
}
func ( c * Configuration ) initACMEProvider ( ) {
2019-07-19 09:52:04 +00:00
for _ , resolver := range c . CertificatesResolvers {
if resolver . ACME != nil {
resolver . ACME . CAServer = getSafeACMECAServer ( resolver . ACME . CAServer )
2018-11-27 16:42:04 +00:00
}
2019-07-19 09:52:04 +00:00
}
2018-11-27 16:42:04 +00:00
2019-07-19 09:52:04 +00:00
legolog . Logger = stdlog . New ( log . WithoutContext ( ) . WriterLevel ( logrus . DebugLevel ) , "legolog: " , 0 )
}
2018-11-27 16:42:04 +00:00
2019-07-19 09:52:04 +00:00
// ValidateConfiguration validate that configuration is coherent
func ( c * Configuration ) ValidateConfiguration ( ) error {
var acmeEmail string
for name , resolver := range c . CertificatesResolvers {
if resolver . ACME == nil {
continue
2018-11-27 16:42:04 +00:00
}
2019-07-19 09:52:04 +00:00
if len ( resolver . ACME . Storage ) == 0 {
return fmt . Errorf ( "unable to initialize certificates resolver %q with no storage location for the certificates" , name )
2018-11-27 16:42:04 +00:00
}
2019-07-19 09:52:04 +00:00
if acmeEmail != "" && resolver . ACME . Email != acmeEmail {
return fmt . Errorf ( "unable to initialize certificates resolver %q, all the acme resolvers must use the same email" , name )
2018-11-27 16:42:04 +00:00
}
2019-07-19 09:52:04 +00:00
acmeEmail = resolver . ACME . Email
2018-11-27 16:42:04 +00:00
}
2019-07-19 09:52:04 +00:00
return nil
2018-11-27 16:42:04 +00:00
}
func getSafeACMECAServer ( caServerSrc string ) string {
if len ( caServerSrc ) == 0 {
return DefaultAcmeCAServer
}
if strings . HasPrefix ( caServerSrc , "https://acme-v01.api.letsencrypt.org" ) {
caServer := strings . Replace ( caServerSrc , "v01" , "v02" , 1 )
2019-09-13 17:28:04 +00:00
log . WithoutContext ( ) .
Warnf ( "The CA server %[1]q refers to a v01 endpoint of the ACME API, please change to %[2]q. Fallback to %[2]q." , caServerSrc , caServer )
2018-11-27 16:42:04 +00:00
return caServer
}
if strings . HasPrefix ( caServerSrc , "https://acme-staging.api.letsencrypt.org" ) {
caServer := strings . Replace ( caServerSrc , "https://acme-staging.api.letsencrypt.org" , "https://acme-staging-v02.api.letsencrypt.org" , 1 )
2019-09-13 17:28:04 +00:00
log . WithoutContext ( ) .
Warnf ( "The CA server %[1]q refers to a v01 endpoint of the ACME API, please change to %[2]q. Fallback to %[2]q." , caServerSrc , caServer )
2018-11-27 16:42:04 +00:00
return caServer
}
return caServerSrc
}