mirror of
https://github.com/traefik/traefik.git
synced 2026-06-17 19:09:29 +03:00
429 lines
14 KiB
Go
429 lines
14 KiB
Go
package server
|
|
|
|
import (
|
|
"context"
|
|
"slices"
|
|
"strings"
|
|
|
|
"github.com/go-acme/lego/v4/challenge/tlsalpn01"
|
|
"github.com/rs/zerolog/log"
|
|
"github.com/traefik/traefik/v3/pkg/config/dynamic"
|
|
httpmuxer "github.com/traefik/traefik/v3/pkg/muxer/http"
|
|
"github.com/traefik/traefik/v3/pkg/observability/logs"
|
|
otypes "github.com/traefik/traefik/v3/pkg/observability/types"
|
|
"github.com/traefik/traefik/v3/pkg/server/provider"
|
|
traefiktls "github.com/traefik/traefik/v3/pkg/tls"
|
|
)
|
|
|
|
func mergeConfiguration(configurations dynamic.Configurations, defaultEntryPoints []string) dynamic.Configuration {
|
|
// TODO: see if we can use DeepCopies inside, so that the given argument is left
|
|
// untouched, and the modified copy is returned.
|
|
conf := dynamic.Configuration{
|
|
HTTP: &dynamic.HTTPConfiguration{
|
|
Routers: make(map[string]*dynamic.Router),
|
|
Middlewares: make(map[string]*dynamic.Middleware),
|
|
Services: make(map[string]*dynamic.Service),
|
|
Models: make(map[string]*dynamic.Model),
|
|
ServersTransports: make(map[string]*dynamic.ServersTransport),
|
|
},
|
|
TCP: &dynamic.TCPConfiguration{
|
|
Routers: make(map[string]*dynamic.TCPRouter),
|
|
Services: make(map[string]*dynamic.TCPService),
|
|
Middlewares: make(map[string]*dynamic.TCPMiddleware),
|
|
Models: make(map[string]*dynamic.TCPModel),
|
|
ServersTransports: make(map[string]*dynamic.TCPServersTransport),
|
|
},
|
|
UDP: &dynamic.UDPConfiguration{
|
|
Routers: make(map[string]*dynamic.UDPRouter),
|
|
Services: make(map[string]*dynamic.UDPService),
|
|
},
|
|
TLS: &dynamic.TLSConfiguration{
|
|
Stores: make(map[string]traefiktls.Store),
|
|
Options: make(map[string]traefiktls.Options),
|
|
},
|
|
}
|
|
|
|
var defaultTLSOptionProviders []string
|
|
var defaultTLSStoreProviders []string
|
|
for pvd, configuration := range configurations {
|
|
if configuration.HTTP != nil {
|
|
for routerName, router := range configuration.HTTP.Routers {
|
|
// If no entrypoint is defined, and the router has no parentRefs (i.e. is not a child router),
|
|
// we set the default entrypoints.
|
|
if len(router.EntryPoints) == 0 && router.ParentRefs == nil {
|
|
log.Debug().
|
|
Str(logs.RouterName, routerName).
|
|
Strs(logs.EntryPointName, defaultEntryPoints).
|
|
Msg("No entryPoint defined for this router, using the default one(s) instead")
|
|
router.EntryPoints = defaultEntryPoints
|
|
}
|
|
|
|
// The `ruleSyntax` option is deprecated.
|
|
// We exclude the "default" value to avoid logging it,
|
|
// as it is the value used for internal models and computed rules.
|
|
if router.RuleSyntax != "" && router.RuleSyntax != "default" {
|
|
log.Warn().
|
|
Str(logs.RouterName, routerName).
|
|
Msg("Router's `ruleSyntax` option is deprecated, please remove any usage of this option.")
|
|
}
|
|
|
|
var qualifiedParentRefs []string
|
|
for _, parentRef := range router.ParentRefs {
|
|
if parts := strings.Split(parentRef, "@"); len(parts) == 1 {
|
|
parentRef = provider.MakeQualifiedName(pvd, parentRef)
|
|
}
|
|
|
|
qualifiedParentRefs = append(qualifiedParentRefs, parentRef)
|
|
}
|
|
router.ParentRefs = qualifiedParentRefs
|
|
|
|
conf.HTTP.Routers[provider.MakeQualifiedName(pvd, routerName)] = router
|
|
}
|
|
for middlewareName, middleware := range configuration.HTTP.Middlewares {
|
|
conf.HTTP.Middlewares[provider.MakeQualifiedName(pvd, middlewareName)] = middleware
|
|
}
|
|
for serviceName, service := range configuration.HTTP.Services {
|
|
conf.HTTP.Services[provider.MakeQualifiedName(pvd, serviceName)] = service
|
|
}
|
|
for modelName, model := range configuration.HTTP.Models {
|
|
conf.HTTP.Models[provider.MakeQualifiedName(pvd, modelName)] = model
|
|
}
|
|
for serversTransportName, serversTransport := range configuration.HTTP.ServersTransports {
|
|
conf.HTTP.ServersTransports[provider.MakeQualifiedName(pvd, serversTransportName)] = serversTransport
|
|
}
|
|
}
|
|
|
|
if configuration.TCP != nil {
|
|
for routerName, router := range configuration.TCP.Routers {
|
|
if len(router.EntryPoints) == 0 {
|
|
log.Debug().
|
|
Str(logs.RouterName, routerName).
|
|
Msgf("No entryPoint defined for this TCP router, using the default one(s) instead: %+v", defaultEntryPoints)
|
|
router.EntryPoints = defaultEntryPoints
|
|
}
|
|
conf.TCP.Routers[provider.MakeQualifiedName(pvd, routerName)] = router
|
|
}
|
|
for middlewareName, middleware := range configuration.TCP.Middlewares {
|
|
conf.TCP.Middlewares[provider.MakeQualifiedName(pvd, middlewareName)] = middleware
|
|
}
|
|
for serviceName, service := range configuration.TCP.Services {
|
|
conf.TCP.Services[provider.MakeQualifiedName(pvd, serviceName)] = service
|
|
}
|
|
for modelName, model := range configuration.TCP.Models {
|
|
conf.TCP.Models[provider.MakeQualifiedName(pvd, modelName)] = model
|
|
}
|
|
for serversTransportName, serversTransport := range configuration.TCP.ServersTransports {
|
|
conf.TCP.ServersTransports[provider.MakeQualifiedName(pvd, serversTransportName)] = serversTransport
|
|
}
|
|
}
|
|
|
|
if configuration.UDP != nil {
|
|
for routerName, router := range configuration.UDP.Routers {
|
|
conf.UDP.Routers[provider.MakeQualifiedName(pvd, routerName)] = router
|
|
}
|
|
for serviceName, service := range configuration.UDP.Services {
|
|
conf.UDP.Services[provider.MakeQualifiedName(pvd, serviceName)] = service
|
|
}
|
|
}
|
|
|
|
if configuration.TLS != nil {
|
|
for _, cert := range configuration.TLS.Certificates {
|
|
if slices.Contains(cert.Stores, tlsalpn01.ACMETLS1Protocol) && pvd != "tlsalpn.acme" {
|
|
continue
|
|
}
|
|
|
|
conf.TLS.Certificates = append(conf.TLS.Certificates, cert)
|
|
}
|
|
|
|
for key, store := range configuration.TLS.Stores {
|
|
if key != traefiktls.DefaultTLSStoreName {
|
|
key = provider.MakeQualifiedName(pvd, key)
|
|
} else {
|
|
defaultTLSStoreProviders = append(defaultTLSStoreProviders, pvd)
|
|
}
|
|
conf.TLS.Stores[key] = store
|
|
}
|
|
|
|
for tlsOptionsName, options := range configuration.TLS.Options {
|
|
if tlsOptionsName != "default" {
|
|
tlsOptionsName = provider.MakeQualifiedName(pvd, tlsOptionsName)
|
|
} else {
|
|
defaultTLSOptionProviders = append(defaultTLSOptionProviders, pvd)
|
|
}
|
|
|
|
conf.TLS.Options[tlsOptionsName] = options
|
|
}
|
|
}
|
|
}
|
|
|
|
if len(defaultTLSStoreProviders) > 1 {
|
|
log.Error().Msgf("Default TLS Store defined in multiple providers: %v", defaultTLSStoreProviders)
|
|
delete(conf.TLS.Stores, traefiktls.DefaultTLSStoreName)
|
|
}
|
|
|
|
if len(defaultTLSOptionProviders) == 0 {
|
|
conf.TLS.Options[traefiktls.DefaultTLSConfigName] = traefiktls.DefaultTLSOptions
|
|
} else if len(defaultTLSOptionProviders) > 1 {
|
|
log.Error().Msgf("Default TLS Options defined in multiple providers %v", defaultTLSOptionProviders)
|
|
// We do not set an empty tls.TLS{} as above so that we actually get a "cascading failure" later on,
|
|
// i.e. routers depending on this missing TLS option will fail to initialize as well.
|
|
delete(conf.TLS.Options, traefiktls.DefaultTLSConfigName)
|
|
}
|
|
|
|
return resolveHTTPTLSOptions(conf)
|
|
}
|
|
|
|
func resolveHTTPTLSOptions(cfg dynamic.Configuration) dynamic.Configuration {
|
|
if cfg.HTTP == nil || len(cfg.HTTP.Routers) == 0 {
|
|
return cfg
|
|
}
|
|
|
|
rts := make(map[string]*dynamic.Router)
|
|
|
|
// Keyed by domain, then by options reference.
|
|
// The actual source of truth for what TLS options will actually be used for the connection.
|
|
// As opposed to tlsOptionsForHost, it keeps track of all the (different) TLS
|
|
// options that occur for a given host name, so that later on we can set relevant
|
|
// errors and logging for all the routers concerned (i.e. wrongly configured).
|
|
tlsOptionsForHostSNI := map[string]map[string][]string{}
|
|
|
|
for routerHTTPName, routerHTTPConfig := range cfg.HTTP.Routers {
|
|
rts[routerHTTPName] = routerHTTPConfig.DeepCopy()
|
|
|
|
if routerHTTPConfig.TLS == nil {
|
|
continue
|
|
}
|
|
|
|
ctxRouter := provider.AddInContext(context.Background(), routerHTTPName)
|
|
logger := log.Ctx(ctxRouter).With().Str(logs.RouterName, routerHTTPName).Logger()
|
|
|
|
tlsOptionsName := traefiktls.DefaultTLSConfigName
|
|
if len(routerHTTPConfig.TLS.Options) > 0 && routerHTTPConfig.TLS.Options != traefiktls.DefaultTLSConfigName {
|
|
tlsOptionsName = provider.GetQualifiedName(ctxRouter, routerHTTPConfig.TLS.Options)
|
|
}
|
|
|
|
domains, err := httpmuxer.ParseDomains(routerHTTPConfig.Rule)
|
|
if err != nil {
|
|
logger.Error().Err(err).Msgf("Invalid rule %s", routerHTTPConfig.Rule)
|
|
continue
|
|
}
|
|
|
|
if len(domains) == 0 {
|
|
rts[routerHTTPName].TLS.ResolvedOptions = "default"
|
|
logger.Warn().Msgf("No domain found in rule %v, the TLS options applied for this router will depend on the SNI of each request", routerHTTPConfig.Rule)
|
|
}
|
|
|
|
for _, domain := range domains {
|
|
// domain is already in lower case thanks to the domain parsing
|
|
if tlsOptionsForHostSNI[domain] == nil {
|
|
tlsOptionsForHostSNI[domain] = make(map[string][]string)
|
|
}
|
|
tlsOptionsForHostSNI[domain][tlsOptionsName] = append(tlsOptionsForHostSNI[domain][tlsOptionsName], routerHTTPName)
|
|
}
|
|
}
|
|
|
|
for hostSNI, tlsConfigs := range tlsOptionsForHostSNI {
|
|
if len(tlsConfigs) == 1 {
|
|
for optionsName, v := range tlsConfigs {
|
|
log.Debug().Msgf("Adding route for %s with TLS options %s", hostSNI, optionsName)
|
|
for _, s := range v {
|
|
rts[s].TLS.ResolvedOptions = optionsName
|
|
}
|
|
}
|
|
continue
|
|
}
|
|
|
|
// multiple tlsConfigs
|
|
routers := make([]string, 0, len(tlsConfigs))
|
|
for _, v := range tlsConfigs {
|
|
for _, s := range v {
|
|
rts[s].TLS.ResolvedOptions = traefiktls.DefaultTLSConfigName
|
|
routers = append(routers, s)
|
|
}
|
|
}
|
|
|
|
log.Warn().Msgf("Found different TLS options for routers on the same host %v, so using the default TLS options instead for these routers: %#v", hostSNI, routers)
|
|
}
|
|
|
|
cfg.HTTP.Routers = rts
|
|
return cfg
|
|
}
|
|
|
|
func applyModel(cfg dynamic.Configuration) dynamic.Configuration {
|
|
if cfg.HTTP != nil && len(cfg.HTTP.Models) > 0 {
|
|
rts := make(map[string]*dynamic.Router)
|
|
|
|
modelRouterNames := make(map[string][]string)
|
|
for name, rt := range cfg.HTTP.Routers {
|
|
// Only root routers can have models applied.
|
|
if rt.ParentRefs != nil {
|
|
rts[name] = rt
|
|
continue
|
|
}
|
|
|
|
router := rt.DeepCopy()
|
|
|
|
if !router.DefaultRule && router.RuleSyntax == "" {
|
|
for modelName, model := range cfg.HTTP.Models {
|
|
// models cannot be provided by another provider than the internal one.
|
|
if !strings.HasSuffix(modelName, "@internal") {
|
|
continue
|
|
}
|
|
router.RuleSyntax = model.DefaultRuleSyntax
|
|
break
|
|
}
|
|
}
|
|
|
|
eps := router.EntryPoints
|
|
router.EntryPoints = nil
|
|
|
|
for _, epName := range eps {
|
|
m, ok := cfg.HTTP.Models[epName+"@internal"]
|
|
if ok {
|
|
cp := router.DeepCopy()
|
|
|
|
cp.EntryPoints = []string{epName}
|
|
|
|
if cp.TLS == nil {
|
|
cp.TLS = m.TLS
|
|
}
|
|
|
|
cp.Middlewares = append(m.Middlewares, cp.Middlewares...)
|
|
|
|
if m.DeniedEncodedPathCharacters != nil {
|
|
// As the denied encoded path characters option is not configurable at the router level,
|
|
// we can simply copy the whole structure to override the router's default config.
|
|
cp.DeniedEncodedPathCharacters = m.DeniedEncodedPathCharacters
|
|
}
|
|
|
|
if cp.Observability == nil {
|
|
cp.Observability = &dynamic.RouterObservabilityConfig{}
|
|
}
|
|
|
|
if cp.Observability.AccessLogs == nil {
|
|
cp.Observability.AccessLogs = m.Observability.AccessLogs
|
|
}
|
|
|
|
if cp.Observability.Metrics == nil {
|
|
cp.Observability.Metrics = m.Observability.Metrics
|
|
}
|
|
|
|
if cp.Observability.Tracing == nil {
|
|
cp.Observability.Tracing = m.Observability.Tracing
|
|
}
|
|
|
|
if cp.Observability.TraceVerbosity == "" {
|
|
cp.Observability.TraceVerbosity = m.Observability.TraceVerbosity
|
|
}
|
|
|
|
rtName := name
|
|
if len(eps) > 1 {
|
|
rtName = epName + "-" + name
|
|
modelRouterNames[name] = append(modelRouterNames[name], rtName)
|
|
}
|
|
|
|
rts[rtName] = cp
|
|
} else {
|
|
router.EntryPoints = append(router.EntryPoints, epName)
|
|
|
|
rts[name] = router
|
|
}
|
|
}
|
|
}
|
|
|
|
for _, rt := range rts {
|
|
if rt.ParentRefs == nil {
|
|
continue
|
|
}
|
|
|
|
var parentRefs []string
|
|
for _, ref := range rt.ParentRefs {
|
|
// Only add the initial parent ref if it still exists.
|
|
if _, ok := rts[ref]; ok {
|
|
parentRefs = append(parentRefs, ref)
|
|
}
|
|
|
|
if names, ok := modelRouterNames[ref]; ok {
|
|
parentRefs = append(parentRefs, names...)
|
|
}
|
|
}
|
|
|
|
rt.ParentRefs = parentRefs
|
|
}
|
|
|
|
cfg.HTTP.Routers = rts
|
|
}
|
|
|
|
// Apply the default observability model to HTTP routers.
|
|
applyDefaultObservabilityModel(cfg)
|
|
|
|
if cfg.TCP == nil || len(cfg.TCP.Models) == 0 {
|
|
return cfg
|
|
}
|
|
|
|
tcpRouters := make(map[string]*dynamic.TCPRouter)
|
|
|
|
for name, rt := range cfg.TCP.Routers {
|
|
router := rt.DeepCopy()
|
|
|
|
if router.RuleSyntax == "" {
|
|
for _, model := range cfg.TCP.Models {
|
|
router.RuleSyntax = model.DefaultRuleSyntax
|
|
break
|
|
}
|
|
}
|
|
|
|
tcpRouters[name] = router
|
|
}
|
|
|
|
cfg.TCP.Routers = tcpRouters
|
|
|
|
return cfg
|
|
}
|
|
|
|
// applyDefaultObservabilityModel applies the default observability model to the configuration.
|
|
// This function is used to ensure that the observability configuration is set for all routers,
|
|
// and make sure it is serialized and available in the API.
|
|
// We could have introduced a "default" model, but it would have been more complex to manage for now.
|
|
// This could be generalized in the future.
|
|
// TODO: check if we can remove this and rely on the SetDefaults instead.
|
|
func applyDefaultObservabilityModel(cfg dynamic.Configuration) {
|
|
if cfg.HTTP != nil {
|
|
for _, router := range cfg.HTTP.Routers {
|
|
// Only root routers can have models applied.
|
|
if router.ParentRefs != nil {
|
|
continue
|
|
}
|
|
|
|
if router.Observability == nil {
|
|
router.Observability = &dynamic.RouterObservabilityConfig{
|
|
AccessLogs: pointer(true),
|
|
Metrics: pointer(true),
|
|
Tracing: pointer(true),
|
|
TraceVerbosity: otypes.MinimalVerbosity,
|
|
}
|
|
|
|
continue
|
|
}
|
|
|
|
if router.Observability.AccessLogs == nil {
|
|
router.Observability.AccessLogs = pointer(true)
|
|
}
|
|
|
|
if router.Observability.Metrics == nil {
|
|
router.Observability.Metrics = pointer(true)
|
|
}
|
|
|
|
if router.Observability.Tracing == nil {
|
|
router.Observability.Tracing = pointer(true)
|
|
}
|
|
|
|
if router.Observability.TraceVerbosity == "" {
|
|
router.Observability.TraceVerbosity = otypes.MinimalVerbosity
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func pointer[T any](v T) *T { return &v }
|