2018-06-11 11:36:03 +02:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2018-07-19 17:30:06 +02:00
|
|
|
"net"
|
2018-06-11 11:36:03 +02:00
|
|
|
"net/http"
|
|
|
|
"reflect"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/containous/mux"
|
|
|
|
"github.com/containous/traefik/configuration"
|
|
|
|
"github.com/containous/traefik/healthcheck"
|
2018-07-03 21:44:05 +07:00
|
|
|
"github.com/containous/traefik/hostresolver"
|
2018-06-11 11:36:03 +02:00
|
|
|
"github.com/containous/traefik/log"
|
|
|
|
"github.com/containous/traefik/metrics"
|
|
|
|
"github.com/containous/traefik/middlewares"
|
2018-06-30 00:28:25 +02:00
|
|
|
"github.com/containous/traefik/middlewares/pipelining"
|
2018-06-11 11:36:03 +02:00
|
|
|
"github.com/containous/traefik/rules"
|
|
|
|
traefiktls "github.com/containous/traefik/tls"
|
2018-07-06 02:30:03 -06:00
|
|
|
"github.com/containous/traefik/tls/generate"
|
2018-06-11 11:36:03 +02:00
|
|
|
"github.com/containous/traefik/types"
|
|
|
|
"github.com/eapache/channels"
|
2018-07-03 12:44:04 +02:00
|
|
|
"github.com/sirupsen/logrus"
|
2018-06-11 11:36:03 +02:00
|
|
|
"github.com/urfave/negroni"
|
|
|
|
"github.com/vulcand/oxy/forward"
|
|
|
|
)
|
|
|
|
|
|
|
|
// loadConfiguration manages dynamically frontends, backends and TLS configurations
|
|
|
|
func (s *Server) loadConfiguration(configMsg types.ConfigMessage) {
|
|
|
|
currentConfigurations := s.currentConfigurations.Get().(types.Configurations)
|
|
|
|
|
|
|
|
// Copy configurations to new map so we don't change current if LoadConfig fails
|
|
|
|
newConfigurations := make(types.Configurations)
|
|
|
|
for k, v := range currentConfigurations {
|
|
|
|
newConfigurations[k] = v
|
|
|
|
}
|
|
|
|
newConfigurations[configMsg.ProviderName] = configMsg.Configuration
|
|
|
|
|
|
|
|
s.metricsRegistry.ConfigReloadsCounter().Add(1)
|
|
|
|
|
|
|
|
newServerEntryPoints, err := s.loadConfig(newConfigurations, s.globalConfiguration)
|
|
|
|
if err != nil {
|
|
|
|
s.metricsRegistry.ConfigReloadsFailureCounter().Add(1)
|
|
|
|
s.metricsRegistry.LastConfigReloadFailureGauge().Set(float64(time.Now().Unix()))
|
|
|
|
log.Error("Error loading new configuration, aborted ", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
s.metricsRegistry.LastConfigReloadSuccessGauge().Set(float64(time.Now().Unix()))
|
|
|
|
|
|
|
|
for newServerEntryPointName, newServerEntryPoint := range newServerEntryPoints {
|
|
|
|
s.serverEntryPoints[newServerEntryPointName].httpRouter.UpdateHandler(newServerEntryPoint.httpRouter.GetHandler())
|
|
|
|
|
|
|
|
if s.entryPoints[newServerEntryPointName].Configuration.TLS == nil {
|
2018-07-06 02:30:03 -06:00
|
|
|
if newServerEntryPoint.certs.ContainsCertificates() {
|
2018-06-11 11:36:03 +02:00
|
|
|
log.Debugf("Certificates not added to non-TLS entryPoint %s.", newServerEntryPointName)
|
|
|
|
}
|
|
|
|
} else {
|
2018-07-06 02:30:03 -06:00
|
|
|
s.serverEntryPoints[newServerEntryPointName].certs.DynamicCerts.Set(newServerEntryPoint.certs.DynamicCerts.Get())
|
|
|
|
s.serverEntryPoints[newServerEntryPointName].certs.ResetCache()
|
2018-06-11 11:36:03 +02:00
|
|
|
}
|
|
|
|
log.Infof("Server configuration reloaded on %s", s.serverEntryPoints[newServerEntryPointName].httpServer.Addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
s.currentConfigurations.Set(newConfigurations)
|
|
|
|
|
|
|
|
for _, listener := range s.configurationListeners {
|
|
|
|
listener(*configMsg.Configuration)
|
|
|
|
}
|
|
|
|
|
|
|
|
s.postLoadConfiguration()
|
|
|
|
}
|
|
|
|
|
|
|
|
// loadConfig returns a new gorilla.mux Route from the specified global configuration and the dynamic
|
|
|
|
// provider configurations.
|
|
|
|
func (s *Server) loadConfig(configurations types.Configurations, globalConfiguration configuration.GlobalConfiguration) (map[string]*serverEntryPoint, error) {
|
|
|
|
redirectHandlers, err := s.buildEntryPointRedirect()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
serverEntryPoints := s.buildServerEntryPoints()
|
|
|
|
|
|
|
|
backendsHandlers := map[string]http.Handler{}
|
|
|
|
backendsHealthCheck := map[string]*healthcheck.BackendConfig{}
|
|
|
|
|
|
|
|
var postConfigs []handlerPostConfig
|
|
|
|
|
|
|
|
for providerName, config := range configurations {
|
|
|
|
frontendNames := sortedFrontendNamesForConfig(config)
|
|
|
|
|
|
|
|
for _, frontendName := range frontendNames {
|
|
|
|
frontendPostConfigs, err := s.loadFrontendConfig(providerName, frontendName, config,
|
2018-06-19 13:56:04 +02:00
|
|
|
redirectHandlers, serverEntryPoints,
|
2018-06-11 11:36:03 +02:00
|
|
|
backendsHandlers, backendsHealthCheck)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("%v. Skipping frontend %s...", err, frontendName)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(frontendPostConfigs) > 0 {
|
|
|
|
postConfigs = append(postConfigs, frontendPostConfigs...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, postConfig := range postConfigs {
|
|
|
|
err := postConfig(backendsHandlers)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("middleware post configuration error: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
healthcheck.GetHealthCheck(s.metricsRegistry).SetBackendsConfiguration(s.routinesPool.Ctx(), backendsHealthCheck)
|
|
|
|
|
|
|
|
// Get new certificates list sorted per entrypoints
|
|
|
|
// Update certificates
|
|
|
|
entryPointsCertificates, err := s.loadHTTPSConfiguration(configurations, globalConfiguration.DefaultEntryPoints)
|
|
|
|
// FIXME error management
|
|
|
|
|
|
|
|
// Sort routes and update certificates
|
|
|
|
for serverEntryPointName, serverEntryPoint := range serverEntryPoints {
|
|
|
|
serverEntryPoint.httpRouter.GetHandler().SortRoutes()
|
|
|
|
if _, exists := entryPointsCertificates[serverEntryPointName]; exists {
|
2018-07-06 02:30:03 -06:00
|
|
|
serverEntryPoint.certs.DynamicCerts.Set(entryPointsCertificates[serverEntryPointName])
|
2018-06-11 11:36:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return serverEntryPoints, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) loadFrontendConfig(
|
|
|
|
providerName string, frontendName string, config *types.Configuration,
|
2018-06-19 13:56:04 +02:00
|
|
|
redirectHandlers map[string]negroni.Handler, serverEntryPoints map[string]*serverEntryPoint,
|
2018-06-11 11:36:03 +02:00
|
|
|
backendsHandlers map[string]http.Handler, backendsHealthCheck map[string]*healthcheck.BackendConfig,
|
|
|
|
) ([]handlerPostConfig, error) {
|
|
|
|
|
|
|
|
frontend := config.Frontends[frontendName]
|
2018-07-03 21:44:05 +07:00
|
|
|
hostResolver := buildHostResolver(s.globalConfiguration)
|
2018-06-11 11:36:03 +02:00
|
|
|
|
|
|
|
if len(frontend.EntryPoints) == 0 {
|
|
|
|
return nil, fmt.Errorf("no entrypoint defined for frontend %s", frontendName)
|
|
|
|
}
|
|
|
|
|
|
|
|
backend := config.Backends[frontend.Backend]
|
|
|
|
if backend == nil {
|
|
|
|
return nil, fmt.Errorf("undefined backend '%s' for frontend %s", frontend.Backend, frontendName)
|
|
|
|
}
|
|
|
|
|
|
|
|
frontendHash, err := frontend.Hash()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error calculating hash value for frontend %s: %v", frontendName, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var postConfigs []handlerPostConfig
|
|
|
|
|
|
|
|
for _, entryPointName := range frontend.EntryPoints {
|
|
|
|
log.Debugf("Wiring frontend %s to entryPoint %s", frontendName, entryPointName)
|
|
|
|
|
|
|
|
entryPoint := s.entryPoints[entryPointName].Configuration
|
|
|
|
|
|
|
|
if backendsHandlers[entryPointName+providerName+frontendHash] == nil {
|
|
|
|
log.Debugf("Creating backend %s", frontend.Backend)
|
|
|
|
|
|
|
|
handlers, responseModifier, postConfig, err := s.buildMiddlewares(frontendName, frontend, config.Backends, entryPointName, entryPoint, providerName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if postConfig != nil {
|
|
|
|
postConfigs = append(postConfigs, postConfig)
|
|
|
|
}
|
|
|
|
|
2018-06-19 13:56:04 +02:00
|
|
|
fwd, err := s.buildForwarder(entryPointName, entryPoint, frontendName, frontend, responseModifier)
|
2018-06-11 11:36:03 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to create the forwarder for frontend %s: %v", frontendName, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
lb, healthCheckConfig, err := s.buildBalancerMiddlewares(frontendName, frontend, backend, fwd)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-09-17 20:40:04 +02:00
|
|
|
// Handler used by error pages
|
|
|
|
if backendsHandlers[entryPointName+providerName+frontend.Backend] == nil {
|
|
|
|
backendsHandlers[entryPointName+providerName+frontend.Backend] = lb
|
|
|
|
}
|
|
|
|
|
2018-06-11 11:36:03 +02:00
|
|
|
if healthCheckConfig != nil {
|
|
|
|
backendsHealthCheck[entryPointName+providerName+frontendHash] = healthCheckConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
n := negroni.New()
|
|
|
|
|
|
|
|
if _, exist := redirectHandlers[entryPointName]; exist {
|
|
|
|
n.Use(redirectHandlers[entryPointName])
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, handler := range handlers {
|
|
|
|
n.Use(handler)
|
|
|
|
}
|
|
|
|
|
|
|
|
n.UseHandler(lb)
|
|
|
|
|
|
|
|
backendsHandlers[entryPointName+providerName+frontendHash] = n
|
|
|
|
} else {
|
|
|
|
log.Debugf("Reusing backend %s [%s - %s - %s - %s]",
|
|
|
|
frontend.Backend, entryPointName, providerName, frontendName, frontendHash)
|
|
|
|
}
|
|
|
|
|
2018-07-03 21:44:05 +07:00
|
|
|
serverRoute, err := buildServerRoute(serverEntryPoints[entryPointName], frontendName, frontend, hostResolver)
|
2018-06-11 11:36:03 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
handler := buildMatcherMiddlewares(serverRoute, backendsHandlers[entryPointName+providerName+frontendHash])
|
|
|
|
serverRoute.Route.Handler(handler)
|
|
|
|
|
|
|
|
err = serverRoute.Route.GetError()
|
|
|
|
if err != nil {
|
|
|
|
// FIXME error management
|
|
|
|
log.Errorf("Error building route: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return postConfigs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) buildForwarder(entryPointName string, entryPoint *configuration.EntryPoint,
|
|
|
|
frontendName string, frontend *types.Frontend,
|
2018-06-19 13:56:04 +02:00
|
|
|
responseModifier modifyResponse) (http.Handler, error) {
|
2018-06-11 11:36:03 +02:00
|
|
|
|
|
|
|
roundTripper, err := s.getRoundTripper(entryPointName, frontend.PassTLSCert, entryPoint.TLS)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to create RoundTripper for frontend %s: %v", frontendName, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
rewriter, err := NewHeaderRewriter(entryPoint.ForwardedHeaders.TrustedIPs, entryPoint.ForwardedHeaders.Insecure)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error creating rewriter for frontend %s: %v", frontendName, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var fwd http.Handler
|
|
|
|
fwd, err = forward.New(
|
|
|
|
forward.Stream(true),
|
|
|
|
forward.PassHostHeader(frontend.PassHostHeader),
|
|
|
|
forward.RoundTripper(roundTripper),
|
|
|
|
forward.Rewriter(rewriter),
|
|
|
|
forward.ResponseModifier(responseModifier),
|
|
|
|
forward.BufferPool(s.bufferPool),
|
2018-07-19 17:30:06 +02:00
|
|
|
forward.WebsocketConnectionClosedHook(func(req *http.Request, conn net.Conn) {
|
|
|
|
server := req.Context().Value(http.ServerContextKey).(*http.Server)
|
|
|
|
if server != nil {
|
|
|
|
connState := server.ConnState
|
|
|
|
if connState != nil {
|
|
|
|
connState(conn, http.StateClosed)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}),
|
2018-06-11 11:36:03 +02:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error creating forwarder for frontend %s: %v", frontendName, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if s.tracingMiddleware.IsEnabled() {
|
|
|
|
tm := s.tracingMiddleware.NewForwarderMiddleware(frontendName, frontend.Backend)
|
|
|
|
|
|
|
|
next := fwd
|
|
|
|
fwd = http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
tm.ServeHTTP(w, r, next.ServeHTTP)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-06-30 00:28:25 +02:00
|
|
|
fwd = pipelining.NewPipelining(fwd)
|
|
|
|
|
2018-06-11 11:36:03 +02:00
|
|
|
return fwd, nil
|
|
|
|
}
|
|
|
|
|
2018-07-03 21:44:05 +07:00
|
|
|
func buildServerRoute(serverEntryPoint *serverEntryPoint, frontendName string, frontend *types.Frontend, hostResolver *hostresolver.Resolver) (*types.ServerRoute, error) {
|
2018-06-11 11:36:03 +02:00
|
|
|
serverRoute := &types.ServerRoute{Route: serverEntryPoint.httpRouter.GetHandler().NewRoute().Name(frontendName)}
|
|
|
|
|
|
|
|
priority := 0
|
|
|
|
for routeName, route := range frontend.Routes {
|
2018-07-03 21:44:05 +07:00
|
|
|
rls := rules.Rules{Route: serverRoute, HostResolver: hostResolver}
|
2018-06-11 11:36:03 +02:00
|
|
|
newRoute, err := rls.Parse(route.Rule)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error creating route for frontend %s: %v", frontendName, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
serverRoute.Route = newRoute
|
|
|
|
|
|
|
|
priority += len(route.Rule)
|
|
|
|
log.Debugf("Creating route %s %s", routeName, route.Rule)
|
|
|
|
}
|
|
|
|
|
|
|
|
if frontend.Priority > 0 {
|
|
|
|
serverRoute.Route.Priority(frontend.Priority)
|
|
|
|
} else {
|
|
|
|
serverRoute.Route.Priority(priority)
|
|
|
|
}
|
|
|
|
|
|
|
|
return serverRoute, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) preLoadConfiguration(configMsg types.ConfigMessage) {
|
|
|
|
providersThrottleDuration := time.Duration(s.globalConfiguration.ProvidersThrottleDuration)
|
|
|
|
s.defaultConfigurationValues(configMsg.Configuration)
|
|
|
|
currentConfigurations := s.currentConfigurations.Get().(types.Configurations)
|
|
|
|
|
2018-07-03 12:44:04 +02:00
|
|
|
if log.GetLevel() == logrus.DebugLevel {
|
|
|
|
jsonConf, _ := json.Marshal(configMsg.Configuration)
|
|
|
|
log.Debugf("Configuration received from provider %s: %s", configMsg.ProviderName, string(jsonConf))
|
|
|
|
}
|
2018-06-11 11:36:03 +02:00
|
|
|
|
|
|
|
if configMsg.Configuration == nil || configMsg.Configuration.Backends == nil && configMsg.Configuration.Frontends == nil && configMsg.Configuration.TLS == nil {
|
|
|
|
log.Infof("Skipping empty Configuration for provider %s", configMsg.ProviderName)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if reflect.DeepEqual(currentConfigurations[configMsg.ProviderName], configMsg.Configuration) {
|
|
|
|
log.Infof("Skipping same configuration for provider %s", configMsg.ProviderName)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
providerConfigUpdateCh, ok := s.providerConfigUpdateMap[configMsg.ProviderName]
|
|
|
|
if !ok {
|
|
|
|
providerConfigUpdateCh = make(chan types.ConfigMessage)
|
|
|
|
s.providerConfigUpdateMap[configMsg.ProviderName] = providerConfigUpdateCh
|
|
|
|
s.routinesPool.Go(func(stop chan bool) {
|
|
|
|
s.throttleProviderConfigReload(providersThrottleDuration, s.configurationValidatedChan, providerConfigUpdateCh, stop)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
providerConfigUpdateCh <- configMsg
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) defaultConfigurationValues(configuration *types.Configuration) {
|
|
|
|
if configuration == nil || configuration.Frontends == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
s.configureFrontends(configuration.Frontends)
|
|
|
|
configureBackends(configuration.Backends)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) configureFrontends(frontends map[string]*types.Frontend) {
|
|
|
|
defaultEntrypoints := s.globalConfiguration.DefaultEntryPoints
|
|
|
|
|
|
|
|
for frontendName, frontend := range frontends {
|
|
|
|
// default endpoints if not defined in frontends
|
|
|
|
if len(frontend.EntryPoints) == 0 {
|
|
|
|
frontend.EntryPoints = defaultEntrypoints
|
|
|
|
}
|
|
|
|
|
|
|
|
frontendEntryPoints, undefinedEntryPoints := s.filterEntryPoints(frontend.EntryPoints)
|
|
|
|
if len(undefinedEntryPoints) > 0 {
|
|
|
|
log.Errorf("Undefined entry point(s) '%s' for frontend %s", strings.Join(undefinedEntryPoints, ","), frontendName)
|
|
|
|
}
|
|
|
|
|
|
|
|
frontend.EntryPoints = frontendEntryPoints
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) filterEntryPoints(entryPoints []string) ([]string, []string) {
|
|
|
|
var frontendEntryPoints []string
|
|
|
|
var undefinedEntryPoints []string
|
|
|
|
|
|
|
|
for _, fepName := range entryPoints {
|
|
|
|
var exist bool
|
|
|
|
|
|
|
|
for epName := range s.entryPoints {
|
|
|
|
if epName == fepName {
|
|
|
|
exist = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if exist {
|
|
|
|
frontendEntryPoints = append(frontendEntryPoints, fepName)
|
|
|
|
} else {
|
|
|
|
undefinedEntryPoints = append(undefinedEntryPoints, fepName)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return frontendEntryPoints, undefinedEntryPoints
|
|
|
|
}
|
|
|
|
|
|
|
|
func configureBackends(backends map[string]*types.Backend) {
|
|
|
|
for backendName := range backends {
|
|
|
|
backend := backends[backendName]
|
|
|
|
if backend.LoadBalancer != nil && backend.LoadBalancer.Sticky {
|
|
|
|
log.Warnf("Deprecated configuration found: %s. Please use %s.", "backend.LoadBalancer.Sticky", "backend.LoadBalancer.Stickiness")
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := types.NewLoadBalancerMethod(backend.LoadBalancer)
|
|
|
|
if err == nil {
|
|
|
|
if backend.LoadBalancer != nil && backend.LoadBalancer.Stickiness == nil && backend.LoadBalancer.Sticky {
|
|
|
|
backend.LoadBalancer.Stickiness = &types.Stickiness{
|
|
|
|
CookieName: "_TRAEFIK_BACKEND",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log.Debugf("Backend %s: %v", backendName, err)
|
|
|
|
|
|
|
|
var stickiness *types.Stickiness
|
|
|
|
if backend.LoadBalancer != nil {
|
|
|
|
if backend.LoadBalancer.Stickiness == nil {
|
|
|
|
if backend.LoadBalancer.Sticky {
|
|
|
|
stickiness = &types.Stickiness{
|
|
|
|
CookieName: "_TRAEFIK_BACKEND",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
stickiness = backend.LoadBalancer.Stickiness
|
|
|
|
}
|
|
|
|
}
|
|
|
|
backend.LoadBalancer = &types.LoadBalancer{
|
|
|
|
Method: "wrr",
|
|
|
|
Stickiness: stickiness,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) listenConfigurations(stop chan bool) {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-stop:
|
|
|
|
return
|
|
|
|
case configMsg, ok := <-s.configurationValidatedChan:
|
|
|
|
if !ok || configMsg.Configuration == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
s.loadConfiguration(configMsg)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// throttleProviderConfigReload throttles the configuration reload speed for a single provider.
|
|
|
|
// It will immediately publish a new configuration and then only publish the next configuration after the throttle duration.
|
|
|
|
// Note that in the case it receives N new configs in the timeframe of the throttle duration after publishing,
|
|
|
|
// it will publish the last of the newly received configurations.
|
|
|
|
func (s *Server) throttleProviderConfigReload(throttle time.Duration, publish chan<- types.ConfigMessage, in <-chan types.ConfigMessage, stop chan bool) {
|
|
|
|
ring := channels.NewRingChannel(1)
|
|
|
|
defer ring.Close()
|
|
|
|
|
|
|
|
s.routinesPool.Go(func(stop chan bool) {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-stop:
|
|
|
|
return
|
|
|
|
case nextConfig := <-ring.Out():
|
2018-09-06 14:24:03 +02:00
|
|
|
if config, ok := nextConfig.(types.ConfigMessage); ok {
|
|
|
|
publish <- config
|
|
|
|
time.Sleep(throttle)
|
|
|
|
}
|
2018-06-11 11:36:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-stop:
|
|
|
|
return
|
|
|
|
case nextConfig := <-in:
|
|
|
|
ring.In() <- nextConfig
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildMatcherMiddlewares(serverRoute *types.ServerRoute, handler http.Handler) http.Handler {
|
|
|
|
// path replace - This needs to always be the very last on the handler chain (first in the order in this function)
|
|
|
|
// -- Replacing Path should happen at the very end of the Modifier chain, after all the Matcher+Modifiers ran
|
|
|
|
if len(serverRoute.ReplacePath) > 0 {
|
|
|
|
handler = &middlewares.ReplacePath{
|
|
|
|
Path: serverRoute.ReplacePath,
|
|
|
|
Handler: handler,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(serverRoute.ReplacePathRegex) > 0 {
|
|
|
|
sp := strings.Split(serverRoute.ReplacePathRegex, " ")
|
|
|
|
if len(sp) == 2 {
|
|
|
|
handler = middlewares.NewReplacePathRegexHandler(sp[0], sp[1], handler)
|
|
|
|
} else {
|
|
|
|
log.Warnf("Invalid syntax for ReplacePathRegex: %s. Separate the regular expression and the replacement by a space.", serverRoute.ReplacePathRegex)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// add prefix - This needs to always be right before ReplacePath on the chain (second in order in this function)
|
|
|
|
// -- Adding Path Prefix should happen after all *Strip Matcher+Modifiers ran, but before Replace (in case it's configured)
|
|
|
|
if len(serverRoute.AddPrefix) > 0 {
|
|
|
|
handler = &middlewares.AddPrefix{
|
|
|
|
Prefix: serverRoute.AddPrefix,
|
|
|
|
Handler: handler,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// strip prefix
|
|
|
|
if len(serverRoute.StripPrefixes) > 0 {
|
|
|
|
handler = &middlewares.StripPrefix{
|
|
|
|
Prefixes: serverRoute.StripPrefixes,
|
|
|
|
Handler: handler,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// strip prefix with regex
|
|
|
|
if len(serverRoute.StripPrefixesRegex) > 0 {
|
|
|
|
handler = middlewares.NewStripPrefixRegex(handler, serverRoute.StripPrefixesRegex)
|
|
|
|
}
|
|
|
|
|
|
|
|
return handler
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) postLoadConfiguration() {
|
|
|
|
if s.metricsRegistry.IsEnabled() {
|
|
|
|
activeConfig := s.currentConfigurations.Get().(types.Configurations)
|
|
|
|
metrics.OnConfigurationUpdate(activeConfig)
|
|
|
|
}
|
|
|
|
|
|
|
|
if s.globalConfiguration.ACME == nil || s.leadership == nil || !s.leadership.IsLeader() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if s.globalConfiguration.ACME.OnHostRule {
|
|
|
|
currentConfigurations := s.currentConfigurations.Get().(types.Configurations)
|
|
|
|
for _, config := range currentConfigurations {
|
|
|
|
for _, frontend := range config.Frontends {
|
|
|
|
|
|
|
|
// check if one of the frontend entrypoints is configured with TLS
|
|
|
|
// and is configured with ACME
|
|
|
|
acmeEnabled := false
|
|
|
|
for _, entryPoint := range frontend.EntryPoints {
|
|
|
|
if s.globalConfiguration.ACME.EntryPoint == entryPoint && s.entryPoints[entryPoint].Configuration.TLS != nil {
|
|
|
|
acmeEnabled = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if acmeEnabled {
|
|
|
|
for _, route := range frontend.Routes {
|
|
|
|
rls := rules.Rules{}
|
|
|
|
domains, err := rls.ParseDomains(route.Rule)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Error parsing domains: %v", err)
|
2018-09-07 10:11:30 +02:00
|
|
|
} else if len(domains) == 0 {
|
|
|
|
log.Debugf("No domain parsed in rule %q", route.Rule)
|
2018-06-11 11:36:03 +02:00
|
|
|
} else {
|
|
|
|
s.globalConfiguration.ACME.LoadCertificateForDomains(domains)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// loadHTTPSConfiguration add/delete HTTPS certificate managed dynamically
|
|
|
|
func (s *Server) loadHTTPSConfiguration(configurations types.Configurations, defaultEntryPoints configuration.DefaultEntryPoints) (map[string]map[string]*tls.Certificate, error) {
|
|
|
|
newEPCertificates := make(map[string]map[string]*tls.Certificate)
|
|
|
|
// Get all certificates
|
|
|
|
for _, config := range configurations {
|
|
|
|
if config.TLS != nil && len(config.TLS) > 0 {
|
|
|
|
if err := traefiktls.SortTLSPerEntryPoints(config.TLS, newEPCertificates, defaultEntryPoints); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return newEPCertificates, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) buildServerEntryPoints() map[string]*serverEntryPoint {
|
|
|
|
serverEntryPoints := make(map[string]*serverEntryPoint)
|
|
|
|
for entryPointName, entryPoint := range s.entryPoints {
|
|
|
|
serverEntryPoints[entryPointName] = &serverEntryPoint{
|
|
|
|
httpRouter: middlewares.NewHandlerSwitcher(s.buildDefaultHTTPRouter()),
|
|
|
|
onDemandListener: entryPoint.OnDemandListener,
|
2018-07-03 12:44:04 +02:00
|
|
|
tlsALPNGetter: entryPoint.TLSALPNGetter,
|
2018-06-11 11:36:03 +02:00
|
|
|
}
|
2018-07-06 02:30:03 -06:00
|
|
|
|
2018-06-11 11:36:03 +02:00
|
|
|
if entryPoint.CertificateStore != nil {
|
2018-07-06 02:30:03 -06:00
|
|
|
serverEntryPoints[entryPointName].certs = entryPoint.CertificateStore
|
2018-06-11 11:36:03 +02:00
|
|
|
} else {
|
2018-07-06 02:30:03 -06:00
|
|
|
serverEntryPoints[entryPointName].certs = traefiktls.NewCertificateStore()
|
|
|
|
}
|
|
|
|
|
|
|
|
if entryPoint.Configuration.TLS != nil {
|
|
|
|
serverEntryPoints[entryPointName].certs.SniStrict = entryPoint.Configuration.TLS.SniStrict
|
|
|
|
|
|
|
|
if entryPoint.Configuration.TLS.DefaultCertificate != nil {
|
2018-09-29 00:04:02 +02:00
|
|
|
cert, err := buildDefaultCertificate(entryPoint.Configuration.TLS.DefaultCertificate)
|
2018-07-06 02:30:03 -06:00
|
|
|
if err != nil {
|
2018-09-29 00:04:02 +02:00
|
|
|
log.Error(err)
|
|
|
|
continue
|
2018-07-06 02:30:03 -06:00
|
|
|
}
|
2018-09-29 00:04:02 +02:00
|
|
|
serverEntryPoints[entryPointName].certs.DefaultCertificate = cert
|
2018-07-06 02:30:03 -06:00
|
|
|
} else {
|
|
|
|
cert, err := generate.DefaultCertificate()
|
|
|
|
if err != nil {
|
2018-09-29 00:04:02 +02:00
|
|
|
log.Errorf("failed to generate default certificate: %v", err)
|
|
|
|
continue
|
2018-07-06 02:30:03 -06:00
|
|
|
}
|
|
|
|
serverEntryPoints[entryPointName].certs.DefaultCertificate = cert
|
|
|
|
}
|
|
|
|
if len(entryPoint.Configuration.TLS.Certificates) > 0 {
|
|
|
|
config, _ := entryPoint.Configuration.TLS.Certificates.CreateTLSConfig(entryPointName)
|
|
|
|
certMap := s.buildNameOrIPToCertificate(config.Certificates)
|
|
|
|
serverEntryPoints[entryPointName].certs.StaticCerts.Set(certMap)
|
|
|
|
|
|
|
|
}
|
2018-06-11 11:36:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return serverEntryPoints
|
|
|
|
}
|
|
|
|
|
2018-09-29 00:04:02 +02:00
|
|
|
func buildDefaultCertificate(defaultCertificate *traefiktls.Certificate) (*tls.Certificate, error) {
|
|
|
|
certFile, err := defaultCertificate.CertFile.Read()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get cert file content: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
keyFile, err := defaultCertificate.KeyFile.Read()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get key file content: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
cert, err := tls.X509KeyPair(certFile, keyFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to load X509 key pair: %v", err)
|
|
|
|
}
|
|
|
|
return &cert, nil
|
|
|
|
}
|
|
|
|
|
2018-06-11 11:36:03 +02:00
|
|
|
func (s *Server) buildDefaultHTTPRouter() *mux.Router {
|
|
|
|
rt := mux.NewRouter()
|
|
|
|
rt.NotFoundHandler = s.wrapHTTPHandlerWithAccessLog(http.HandlerFunc(http.NotFound), "backend not found")
|
|
|
|
rt.StrictSlash(true)
|
|
|
|
rt.SkipClean(true)
|
|
|
|
return rt
|
|
|
|
}
|
|
|
|
|
|
|
|
func sortedFrontendNamesForConfig(configuration *types.Configuration) []string {
|
|
|
|
var keys []string
|
|
|
|
for key := range configuration.Frontends {
|
|
|
|
keys = append(keys, key)
|
|
|
|
}
|
|
|
|
sort.Strings(keys)
|
|
|
|
return keys
|
|
|
|
}
|
2018-07-03 21:44:05 +07:00
|
|
|
|
|
|
|
func buildHostResolver(globalConfig configuration.GlobalConfiguration) *hostresolver.Resolver {
|
|
|
|
if globalConfig.HostResolver != nil {
|
|
|
|
return &hostresolver.Resolver{
|
|
|
|
CnameFlattening: globalConfig.HostResolver.CnameFlattening,
|
|
|
|
ResolvConfig: globalConfig.HostResolver.ResolvConfig,
|
|
|
|
ResolvDepth: globalConfig.HostResolver.ResolvDepth,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|