mirror of
				https://github.com/traefik/traefik.git
				synced 2025-10-25 14:31:12 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			1180 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			1180 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package crd
 | |
| 
 | |
| import (
 | |
| 	"bufio"
 | |
| 	"bytes"
 | |
| 	"context"
 | |
| 	"crypto/sha1"
 | |
| 	"crypto/sha256"
 | |
| 	"encoding/base64"
 | |
| 	"encoding/json"
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"net"
 | |
| 	"os"
 | |
| 	"sort"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/cenkalti/backoff/v4"
 | |
| 	"github.com/mitchellh/hashstructure"
 | |
| 	ptypes "github.com/traefik/paerser/types"
 | |
| 	"github.com/traefik/traefik/v2/pkg/config/dynamic"
 | |
| 	"github.com/traefik/traefik/v2/pkg/job"
 | |
| 	"github.com/traefik/traefik/v2/pkg/log"
 | |
| 	"github.com/traefik/traefik/v2/pkg/provider"
 | |
| 	traefikv1alpha1 "github.com/traefik/traefik/v2/pkg/provider/kubernetes/crd/traefikio/v1alpha1"
 | |
| 	"github.com/traefik/traefik/v2/pkg/provider/kubernetes/k8s"
 | |
| 	"github.com/traefik/traefik/v2/pkg/safe"
 | |
| 	"github.com/traefik/traefik/v2/pkg/tls"
 | |
| 	"github.com/traefik/traefik/v2/pkg/types"
 | |
| 	corev1 "k8s.io/api/core/v1"
 | |
| 	apiextensionv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
 | |
| 	"k8s.io/apimachinery/pkg/labels"
 | |
| 	"k8s.io/apimachinery/pkg/util/intstr"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	annotationKubernetesIngressClass = "kubernetes.io/ingress.class"
 | |
| 	traefikDefaultIngressClass       = "traefik"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	providerName               = "kubernetescrd"
 | |
| 	providerNamespaceSeparator = "@"
 | |
| )
 | |
| 
 | |
| // Provider holds configurations of the provider.
 | |
| type Provider struct {
 | |
| 	Endpoint                  string          `description:"Kubernetes server endpoint (required for external cluster client)." json:"endpoint,omitempty" toml:"endpoint,omitempty" yaml:"endpoint,omitempty"`
 | |
| 	Token                     string          `description:"Kubernetes bearer token (not needed for in-cluster client)." json:"token,omitempty" toml:"token,omitempty" yaml:"token,omitempty" loggable:"false"`
 | |
| 	CertAuthFilePath          string          `description:"Kubernetes certificate authority file path (not needed for in-cluster client)." json:"certAuthFilePath,omitempty" toml:"certAuthFilePath,omitempty" yaml:"certAuthFilePath,omitempty"`
 | |
| 	Namespaces                []string        `description:"Kubernetes namespaces." json:"namespaces,omitempty" toml:"namespaces,omitempty" yaml:"namespaces,omitempty" export:"true"`
 | |
| 	AllowCrossNamespace       bool            `description:"Allow cross namespace resource reference." json:"allowCrossNamespace,omitempty" toml:"allowCrossNamespace,omitempty" yaml:"allowCrossNamespace,omitempty" export:"true"`
 | |
| 	AllowExternalNameServices bool            `description:"Allow ExternalName services." json:"allowExternalNameServices,omitempty" toml:"allowExternalNameServices,omitempty" yaml:"allowExternalNameServices,omitempty" export:"true"`
 | |
| 	LabelSelector             string          `description:"Kubernetes label selector to use." json:"labelSelector,omitempty" toml:"labelSelector,omitempty" yaml:"labelSelector,omitempty" export:"true"`
 | |
| 	IngressClass              string          `description:"Value of kubernetes.io/ingress.class annotation to watch for." json:"ingressClass,omitempty" toml:"ingressClass,omitempty" yaml:"ingressClass,omitempty" export:"true"`
 | |
| 	ThrottleDuration          ptypes.Duration `description:"Ingress refresh throttle duration" json:"throttleDuration,omitempty" toml:"throttleDuration,omitempty" yaml:"throttleDuration,omitempty" export:"true"`
 | |
| 	AllowEmptyServices        bool            `description:"Allow the creation of services without endpoints." json:"allowEmptyServices,omitempty" toml:"allowEmptyServices,omitempty" yaml:"allowEmptyServices,omitempty" export:"true"`
 | |
| 
 | |
| 	lastConfiguration safe.Safe
 | |
| 
 | |
| 	routerTransform k8s.RouterTransform
 | |
| }
 | |
| 
 | |
| func (p *Provider) SetRouterTransform(routerTransform k8s.RouterTransform) {
 | |
| 	p.routerTransform = routerTransform
 | |
| }
 | |
| 
 | |
| func (p *Provider) applyRouterTransform(ctx context.Context, rt *dynamic.Router, ingress *traefikv1alpha1.IngressRoute) {
 | |
| 	if p.routerTransform == nil {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	err := p.routerTransform.Apply(ctx, rt, ingress.Annotations)
 | |
| 	if err != nil {
 | |
| 		log.FromContext(ctx).WithError(err).Error("Apply router transform")
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p *Provider) newK8sClient(ctx context.Context) (*clientWrapper, error) {
 | |
| 	_, err := labels.Parse(p.LabelSelector)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("invalid label selector: %q", p.LabelSelector)
 | |
| 	}
 | |
| 	log.FromContext(ctx).Infof("label selector is: %q", p.LabelSelector)
 | |
| 
 | |
| 	withEndpoint := ""
 | |
| 	if p.Endpoint != "" {
 | |
| 		withEndpoint = fmt.Sprintf(" with endpoint %s", p.Endpoint)
 | |
| 	}
 | |
| 
 | |
| 	var client *clientWrapper
 | |
| 	switch {
 | |
| 	case os.Getenv("KUBERNETES_SERVICE_HOST") != "" && os.Getenv("KUBERNETES_SERVICE_PORT") != "":
 | |
| 		log.FromContext(ctx).Infof("Creating in-cluster Provider client%s", withEndpoint)
 | |
| 		client, err = newInClusterClient(p.Endpoint)
 | |
| 	case os.Getenv("KUBECONFIG") != "":
 | |
| 		log.FromContext(ctx).Infof("Creating cluster-external Provider client from KUBECONFIG %s", os.Getenv("KUBECONFIG"))
 | |
| 		client, err = newExternalClusterClientFromFile(os.Getenv("KUBECONFIG"))
 | |
| 	default:
 | |
| 		log.FromContext(ctx).Infof("Creating cluster-external Provider client%s", withEndpoint)
 | |
| 		client, err = newExternalClusterClient(p.Endpoint, p.Token, p.CertAuthFilePath)
 | |
| 	}
 | |
| 
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	client.labelSelector = p.LabelSelector
 | |
| 	return client, nil
 | |
| }
 | |
| 
 | |
| // Init the provider.
 | |
| func (p *Provider) Init() error {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // Provide allows the k8s provider to provide configurations to traefik
 | |
| // using the given configuration channel.
 | |
| func (p *Provider) Provide(configurationChan chan<- dynamic.Message, pool *safe.Pool) error {
 | |
| 	ctxLog := log.With(context.Background(), log.Str(log.ProviderName, providerName))
 | |
| 	logger := log.FromContext(ctxLog)
 | |
| 
 | |
| 	logger.Warn("CRDs API Group \"traefik.containo.us\" is deprecated, and its support will end starting with Traefik v3. Please use the API Group \"traefik.io\" instead.")
 | |
| 	logger.Warn("CRDs API Version \"traefik.io/v1alpha1\" will not be supported in Traefik v3 itself. However, an automatic migration path to the next version will be available.")
 | |
| 
 | |
| 	k8sClient, err := p.newK8sClient(ctxLog)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	if p.AllowCrossNamespace {
 | |
| 		logger.Warn("Cross-namespace reference between IngressRoutes and resources is enabled, please ensure that this is expected (see AllowCrossNamespace option)")
 | |
| 	}
 | |
| 
 | |
| 	if p.AllowExternalNameServices {
 | |
| 		logger.Warn("ExternalName service loading is enabled, please ensure that this is expected (see AllowExternalNameServices option)")
 | |
| 	}
 | |
| 
 | |
| 	pool.GoCtx(func(ctxPool context.Context) {
 | |
| 		operation := func() error {
 | |
| 			eventsChan, err := k8sClient.WatchAll(p.Namespaces, ctxPool.Done())
 | |
| 			if err != nil {
 | |
| 				logger.Errorf("Error watching kubernetes events: %v", err)
 | |
| 				timer := time.NewTimer(1 * time.Second)
 | |
| 				select {
 | |
| 				case <-timer.C:
 | |
| 					return err
 | |
| 				case <-ctxPool.Done():
 | |
| 					return nil
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			throttleDuration := time.Duration(p.ThrottleDuration)
 | |
| 			throttledChan := throttleEvents(ctxLog, throttleDuration, pool, eventsChan)
 | |
| 			if throttledChan != nil {
 | |
| 				eventsChan = throttledChan
 | |
| 			}
 | |
| 
 | |
| 			for {
 | |
| 				select {
 | |
| 				case <-ctxPool.Done():
 | |
| 					return nil
 | |
| 				case event := <-eventsChan:
 | |
| 					// Note that event is the *first* event that came in during this throttling interval -- if we're hitting our throttle, we may have dropped events.
 | |
| 					// This is fine, because we don't treat different event types differently.
 | |
| 					// But if we do in the future, we'll need to track more information about the dropped events.
 | |
| 					conf := p.loadConfigurationFromCRD(ctxLog, k8sClient)
 | |
| 
 | |
| 					confHash, err := hashstructure.Hash(conf, nil)
 | |
| 					switch {
 | |
| 					case err != nil:
 | |
| 						logger.Error("Unable to hash the configuration")
 | |
| 					case p.lastConfiguration.Get() == confHash:
 | |
| 						logger.Debugf("Skipping Kubernetes event kind %T", event)
 | |
| 					default:
 | |
| 						p.lastConfiguration.Set(confHash)
 | |
| 						configurationChan <- dynamic.Message{
 | |
| 							ProviderName:  providerName,
 | |
| 							Configuration: conf,
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					// If we're throttling,
 | |
| 					// we sleep here for the throttle duration to enforce that we don't refresh faster than our throttle.
 | |
| 					// time.Sleep returns immediately if p.ThrottleDuration is 0 (no throttle).
 | |
| 					time.Sleep(throttleDuration)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		notify := func(err error, time time.Duration) {
 | |
| 			logger.Errorf("Provider connection error: %v; retrying in %s", err, time)
 | |
| 		}
 | |
| 		err := backoff.RetryNotify(safe.OperationWithRecover(operation), backoff.WithContext(job.NewBackOff(backoff.NewExponentialBackOff()), ctxPool), notify)
 | |
| 		if err != nil {
 | |
| 			logger.Errorf("Cannot connect to Provider: %v", err)
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (p *Provider) loadConfigurationFromCRD(ctx context.Context, client Client) *dynamic.Configuration {
 | |
| 	stores, tlsConfigs := buildTLSStores(ctx, client)
 | |
| 	if tlsConfigs == nil {
 | |
| 		tlsConfigs = make(map[string]*tls.CertAndStores)
 | |
| 	}
 | |
| 
 | |
| 	conf := &dynamic.Configuration{
 | |
| 		// TODO: choose between mutating and returning tlsConfigs
 | |
| 		HTTP: p.loadIngressRouteConfiguration(ctx, client, tlsConfigs),
 | |
| 		TCP:  p.loadIngressRouteTCPConfiguration(ctx, client, tlsConfigs),
 | |
| 		UDP:  p.loadIngressRouteUDPConfiguration(ctx, client),
 | |
| 		TLS: &dynamic.TLSConfiguration{
 | |
| 			Options: buildTLSOptions(ctx, client),
 | |
| 			Stores:  stores,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	// Done after because tlsConfigs is mutated by the others above.
 | |
| 	conf.TLS.Certificates = getTLSConfig(tlsConfigs)
 | |
| 
 | |
| 	for _, middleware := range client.GetMiddlewares() {
 | |
| 		id := provider.Normalize(makeID(middleware.Namespace, middleware.Name))
 | |
| 		ctxMid := log.With(ctx, log.Str(log.MiddlewareName, id))
 | |
| 
 | |
| 		basicAuth, err := createBasicAuthMiddleware(client, middleware.Namespace, middleware.Spec.BasicAuth)
 | |
| 		if err != nil {
 | |
| 			log.FromContext(ctxMid).Errorf("Error while reading basic auth middleware: %v", err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		digestAuth, err := createDigestAuthMiddleware(client, middleware.Namespace, middleware.Spec.DigestAuth)
 | |
| 		if err != nil {
 | |
| 			log.FromContext(ctxMid).Errorf("Error while reading digest auth middleware: %v", err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		forwardAuth, err := createForwardAuthMiddleware(client, middleware.Namespace, middleware.Spec.ForwardAuth)
 | |
| 		if err != nil {
 | |
| 			log.FromContext(ctxMid).Errorf("Error while reading forward auth middleware: %v", err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		errorPage, errorPageService, err := p.createErrorPageMiddleware(client, middleware.Namespace, middleware.Spec.Errors)
 | |
| 		if err != nil {
 | |
| 			log.FromContext(ctxMid).Errorf("Error while reading error page middleware: %v", err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		if errorPage != nil && errorPageService != nil {
 | |
| 			serviceName := id + "-errorpage-service"
 | |
| 			errorPage.Service = serviceName
 | |
| 			conf.HTTP.Services[serviceName] = errorPageService
 | |
| 		}
 | |
| 
 | |
| 		plugin, err := createPluginMiddleware(client, middleware.Namespace, middleware.Spec.Plugin)
 | |
| 		if err != nil {
 | |
| 			log.FromContext(ctxMid).Errorf("Error while reading plugins middleware: %v", err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		rateLimit, err := createRateLimitMiddleware(middleware.Spec.RateLimit)
 | |
| 		if err != nil {
 | |
| 			log.FromContext(ctxMid).Errorf("Error while reading rateLimit middleware: %v", err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		retry, err := createRetryMiddleware(middleware.Spec.Retry)
 | |
| 		if err != nil {
 | |
| 			log.FromContext(ctxMid).Errorf("Error while reading retry middleware: %v", err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		circuitBreaker, err := createCircuitBreakerMiddleware(middleware.Spec.CircuitBreaker)
 | |
| 		if err != nil {
 | |
| 			log.FromContext(ctxMid).Errorf("Error while reading circuit breaker middleware: %v", err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		conf.HTTP.Middlewares[id] = &dynamic.Middleware{
 | |
| 			AddPrefix:         middleware.Spec.AddPrefix,
 | |
| 			StripPrefix:       middleware.Spec.StripPrefix,
 | |
| 			StripPrefixRegex:  middleware.Spec.StripPrefixRegex,
 | |
| 			ReplacePath:       middleware.Spec.ReplacePath,
 | |
| 			ReplacePathRegex:  middleware.Spec.ReplacePathRegex,
 | |
| 			Chain:             createChainMiddleware(ctxMid, middleware.Namespace, middleware.Spec.Chain),
 | |
| 			IPWhiteList:       middleware.Spec.IPWhiteList,
 | |
| 			Headers:           middleware.Spec.Headers,
 | |
| 			Errors:            errorPage,
 | |
| 			RateLimit:         rateLimit,
 | |
| 			RedirectRegex:     middleware.Spec.RedirectRegex,
 | |
| 			RedirectScheme:    middleware.Spec.RedirectScheme,
 | |
| 			BasicAuth:         basicAuth,
 | |
| 			DigestAuth:        digestAuth,
 | |
| 			ForwardAuth:       forwardAuth,
 | |
| 			InFlightReq:       middleware.Spec.InFlightReq,
 | |
| 			Buffering:         middleware.Spec.Buffering,
 | |
| 			CircuitBreaker:    circuitBreaker,
 | |
| 			Compress:          middleware.Spec.Compress,
 | |
| 			PassTLSClientCert: middleware.Spec.PassTLSClientCert,
 | |
| 			Retry:             retry,
 | |
| 			ContentType:       middleware.Spec.ContentType,
 | |
| 			Plugin:            plugin,
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for _, middlewareTCP := range client.GetMiddlewareTCPs() {
 | |
| 		id := provider.Normalize(makeID(middlewareTCP.Namespace, middlewareTCP.Name))
 | |
| 
 | |
| 		conf.TCP.Middlewares[id] = &dynamic.TCPMiddleware{
 | |
| 			InFlightConn: middlewareTCP.Spec.InFlightConn,
 | |
| 			IPWhiteList:  middlewareTCP.Spec.IPWhiteList,
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	cb := configBuilder{
 | |
| 		client:                    client,
 | |
| 		allowCrossNamespace:       p.AllowCrossNamespace,
 | |
| 		allowExternalNameServices: p.AllowExternalNameServices,
 | |
| 		allowEmptyServices:        p.AllowEmptyServices,
 | |
| 	}
 | |
| 
 | |
| 	for _, service := range client.GetTraefikServices() {
 | |
| 		err := cb.buildTraefikService(ctx, service, conf.HTTP.Services)
 | |
| 		if err != nil {
 | |
| 			log.FromContext(ctx).WithField(log.ServiceName, service.Name).
 | |
| 				Errorf("Error while building TraefikService: %v", err)
 | |
| 			continue
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for _, serversTransport := range client.GetServersTransports() {
 | |
| 		logger := log.FromContext(ctx).WithField(log.ServersTransportName, serversTransport.Name)
 | |
| 
 | |
| 		var rootCAs []tls.FileOrContent
 | |
| 		for _, secret := range serversTransport.Spec.RootCAsSecrets {
 | |
| 			caSecret, err := loadCASecret(serversTransport.Namespace, secret, client)
 | |
| 			if err != nil {
 | |
| 				logger.Errorf("Error while loading rootCAs %s: %v", secret, err)
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			rootCAs = append(rootCAs, tls.FileOrContent(caSecret))
 | |
| 		}
 | |
| 
 | |
| 		var certs tls.Certificates
 | |
| 		for _, secret := range serversTransport.Spec.CertificatesSecrets {
 | |
| 			tlsSecret, tlsKey, err := loadAuthTLSSecret(serversTransport.Namespace, secret, client)
 | |
| 			if err != nil {
 | |
| 				logger.Errorf("Error while loading certificates %s: %v", secret, err)
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			certs = append(certs, tls.Certificate{
 | |
| 				CertFile: tls.FileOrContent(tlsSecret),
 | |
| 				KeyFile:  tls.FileOrContent(tlsKey),
 | |
| 			})
 | |
| 		}
 | |
| 
 | |
| 		forwardingTimeout := &dynamic.ForwardingTimeouts{}
 | |
| 		forwardingTimeout.SetDefaults()
 | |
| 
 | |
| 		if serversTransport.Spec.ForwardingTimeouts != nil {
 | |
| 			if serversTransport.Spec.ForwardingTimeouts.DialTimeout != nil {
 | |
| 				err := forwardingTimeout.DialTimeout.Set(serversTransport.Spec.ForwardingTimeouts.DialTimeout.String())
 | |
| 				if err != nil {
 | |
| 					logger.Errorf("Error while reading DialTimeout: %v", err)
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if serversTransport.Spec.ForwardingTimeouts.ResponseHeaderTimeout != nil {
 | |
| 				err := forwardingTimeout.ResponseHeaderTimeout.Set(serversTransport.Spec.ForwardingTimeouts.ResponseHeaderTimeout.String())
 | |
| 				if err != nil {
 | |
| 					logger.Errorf("Error while reading ResponseHeaderTimeout: %v", err)
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if serversTransport.Spec.ForwardingTimeouts.IdleConnTimeout != nil {
 | |
| 				err := forwardingTimeout.IdleConnTimeout.Set(serversTransport.Spec.ForwardingTimeouts.IdleConnTimeout.String())
 | |
| 				if err != nil {
 | |
| 					logger.Errorf("Error while reading IdleConnTimeout: %v", err)
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if serversTransport.Spec.ForwardingTimeouts.ReadIdleTimeout != nil {
 | |
| 				err := forwardingTimeout.ReadIdleTimeout.Set(serversTransport.Spec.ForwardingTimeouts.ReadIdleTimeout.String())
 | |
| 				if err != nil {
 | |
| 					logger.Errorf("Error while reading ReadIdleTimeout: %v", err)
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if serversTransport.Spec.ForwardingTimeouts.PingTimeout != nil {
 | |
| 				err := forwardingTimeout.PingTimeout.Set(serversTransport.Spec.ForwardingTimeouts.PingTimeout.String())
 | |
| 				if err != nil {
 | |
| 					logger.Errorf("Error while reading PingTimeout: %v", err)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		id := provider.Normalize(makeID(serversTransport.Namespace, serversTransport.Name))
 | |
| 		conf.HTTP.ServersTransports[id] = &dynamic.ServersTransport{
 | |
| 			ServerName:          serversTransport.Spec.ServerName,
 | |
| 			InsecureSkipVerify:  serversTransport.Spec.InsecureSkipVerify,
 | |
| 			RootCAs:             rootCAs,
 | |
| 			Certificates:        certs,
 | |
| 			DisableHTTP2:        serversTransport.Spec.DisableHTTP2,
 | |
| 			MaxIdleConnsPerHost: serversTransport.Spec.MaxIdleConnsPerHost,
 | |
| 			ForwardingTimeouts:  forwardingTimeout,
 | |
| 			PeerCertURI:         serversTransport.Spec.PeerCertURI,
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return conf
 | |
| }
 | |
| 
 | |
| // getServicePort always returns a valid port, an error otherwise.
 | |
| func getServicePort(svc *corev1.Service, port intstr.IntOrString) (*corev1.ServicePort, error) {
 | |
| 	if svc == nil {
 | |
| 		return nil, errors.New("service is not defined")
 | |
| 	}
 | |
| 
 | |
| 	if (port.Type == intstr.Int && port.IntVal == 0) || (port.Type == intstr.String && port.StrVal == "") {
 | |
| 		return nil, errors.New("ingressRoute service port not defined")
 | |
| 	}
 | |
| 
 | |
| 	hasValidPort := false
 | |
| 	for _, p := range svc.Spec.Ports {
 | |
| 		if (port.Type == intstr.Int && port.IntVal == p.Port) || (port.Type == intstr.String && port.StrVal == p.Name) {
 | |
| 			return &p, nil
 | |
| 		}
 | |
| 
 | |
| 		if p.Port != 0 {
 | |
| 			hasValidPort = true
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if svc.Spec.Type != corev1.ServiceTypeExternalName || port.Type == intstr.String {
 | |
| 		return nil, fmt.Errorf("service port not found: %s", &port)
 | |
| 	}
 | |
| 
 | |
| 	if hasValidPort {
 | |
| 		log.WithoutContext().
 | |
| 			Warnf("The port %s from IngressRoute doesn't match with ports defined in the ExternalName service %s/%s.", port, svc.Namespace, svc.Name)
 | |
| 	}
 | |
| 
 | |
| 	return &corev1.ServicePort{Port: port.IntVal}, nil
 | |
| }
 | |
| 
 | |
| func getNativeServiceAddress(service corev1.Service, svcPort corev1.ServicePort) (string, error) {
 | |
| 	if service.Spec.ClusterIP == "None" {
 | |
| 		return "", fmt.Errorf("no clusterIP on headless service: %s/%s", service.Namespace, service.Name)
 | |
| 	}
 | |
| 
 | |
| 	if service.Spec.ClusterIP == "" {
 | |
| 		return "", fmt.Errorf("no clusterIP found for service: %s/%s", service.Namespace, service.Name)
 | |
| 	}
 | |
| 
 | |
| 	return net.JoinHostPort(service.Spec.ClusterIP, strconv.Itoa(int(svcPort.Port))), nil
 | |
| }
 | |
| 
 | |
| func createPluginMiddleware(k8sClient Client, ns string, plugins map[string]apiextensionv1.JSON) (map[string]dynamic.PluginConf, error) {
 | |
| 	if plugins == nil {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	data, err := json.Marshal(plugins)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	pcMap := map[string]dynamic.PluginConf{}
 | |
| 	if err = json.Unmarshal(data, &pcMap); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	for _, pc := range pcMap {
 | |
| 		for key := range pc {
 | |
| 			if pc[key], err = loadSecretKeys(k8sClient, ns, pc[key]); err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return pcMap, nil
 | |
| }
 | |
| 
 | |
| func loadSecretKeys(k8sClient Client, ns string, i interface{}) (interface{}, error) {
 | |
| 	var err error
 | |
| 	switch iv := i.(type) {
 | |
| 	case string:
 | |
| 		if !strings.HasPrefix(iv, "urn:k8s:secret:") {
 | |
| 			return iv, nil
 | |
| 		}
 | |
| 
 | |
| 		return getSecretValue(k8sClient, ns, iv)
 | |
| 
 | |
| 	case []interface{}:
 | |
| 		for i := range iv {
 | |
| 			if iv[i], err = loadSecretKeys(k8sClient, ns, iv[i]); err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 	case map[string]interface{}:
 | |
| 		for k := range iv {
 | |
| 			if iv[k], err = loadSecretKeys(k8sClient, ns, iv[k]); err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return i, nil
 | |
| }
 | |
| 
 | |
| func getSecretValue(c Client, ns, urn string) (string, error) {
 | |
| 	parts := strings.Split(urn, ":")
 | |
| 	if len(parts) != 5 {
 | |
| 		return "", fmt.Errorf("malformed secret URN %q", urn)
 | |
| 	}
 | |
| 
 | |
| 	secretName := parts[3]
 | |
| 	secret, ok, err := c.GetSecret(ns, secretName)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	if !ok {
 | |
| 		return "", fmt.Errorf("secret %s/%s is not found", ns, secretName)
 | |
| 	}
 | |
| 
 | |
| 	secretKey := parts[4]
 | |
| 	secretValue, ok := secret.Data[secretKey]
 | |
| 	if !ok {
 | |
| 		return "", fmt.Errorf("key %q not found in secret %s/%s", secretKey, ns, secretName)
 | |
| 	}
 | |
| 
 | |
| 	return string(secretValue), nil
 | |
| }
 | |
| 
 | |
| func createCircuitBreakerMiddleware(circuitBreaker *traefikv1alpha1.CircuitBreaker) (*dynamic.CircuitBreaker, error) {
 | |
| 	if circuitBreaker == nil {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	cb := &dynamic.CircuitBreaker{Expression: circuitBreaker.Expression}
 | |
| 	cb.SetDefaults()
 | |
| 
 | |
| 	if circuitBreaker.CheckPeriod != nil {
 | |
| 		if err := cb.CheckPeriod.Set(circuitBreaker.CheckPeriod.String()); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if circuitBreaker.FallbackDuration != nil {
 | |
| 		if err := cb.FallbackDuration.Set(circuitBreaker.FallbackDuration.String()); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if circuitBreaker.RecoveryDuration != nil {
 | |
| 		if err := cb.RecoveryDuration.Set(circuitBreaker.RecoveryDuration.String()); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return cb, nil
 | |
| }
 | |
| 
 | |
| func createRateLimitMiddleware(rateLimit *traefikv1alpha1.RateLimit) (*dynamic.RateLimit, error) {
 | |
| 	if rateLimit == nil {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	rl := &dynamic.RateLimit{Average: rateLimit.Average}
 | |
| 	rl.SetDefaults()
 | |
| 
 | |
| 	if rateLimit.Burst != nil {
 | |
| 		rl.Burst = *rateLimit.Burst
 | |
| 	}
 | |
| 
 | |
| 	if rateLimit.Period != nil {
 | |
| 		err := rl.Period.Set(rateLimit.Period.String())
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if rateLimit.SourceCriterion != nil {
 | |
| 		rl.SourceCriterion = rateLimit.SourceCriterion
 | |
| 	}
 | |
| 
 | |
| 	return rl, nil
 | |
| }
 | |
| 
 | |
| func createRetryMiddleware(retry *traefikv1alpha1.Retry) (*dynamic.Retry, error) {
 | |
| 	if retry == nil {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	r := &dynamic.Retry{Attempts: retry.Attempts}
 | |
| 
 | |
| 	err := r.InitialInterval.Set(retry.InitialInterval.String())
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	return r, nil
 | |
| }
 | |
| 
 | |
| func (p *Provider) createErrorPageMiddleware(client Client, namespace string, errorPage *traefikv1alpha1.ErrorPage) (*dynamic.ErrorPage, *dynamic.Service, error) {
 | |
| 	if errorPage == nil {
 | |
| 		return nil, nil, nil
 | |
| 	}
 | |
| 
 | |
| 	errorPageMiddleware := &dynamic.ErrorPage{
 | |
| 		Status: errorPage.Status,
 | |
| 		Query:  errorPage.Query,
 | |
| 	}
 | |
| 
 | |
| 	cb := configBuilder{
 | |
| 		client:                    client,
 | |
| 		allowCrossNamespace:       p.AllowCrossNamespace,
 | |
| 		allowExternalNameServices: p.AllowExternalNameServices,
 | |
| 		allowEmptyServices:        p.AllowEmptyServices,
 | |
| 	}
 | |
| 
 | |
| 	balancerServerHTTP, err := cb.buildServersLB(namespace, errorPage.Service.LoadBalancerSpec)
 | |
| 	if err != nil {
 | |
| 		return nil, nil, err
 | |
| 	}
 | |
| 
 | |
| 	return errorPageMiddleware, balancerServerHTTP, nil
 | |
| }
 | |
| 
 | |
| func createForwardAuthMiddleware(k8sClient Client, namespace string, auth *traefikv1alpha1.ForwardAuth) (*dynamic.ForwardAuth, error) {
 | |
| 	if auth == nil {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 	if len(auth.Address) == 0 {
 | |
| 		return nil, fmt.Errorf("forward authentication requires an address")
 | |
| 	}
 | |
| 
 | |
| 	forwardAuth := &dynamic.ForwardAuth{
 | |
| 		Address:                  auth.Address,
 | |
| 		TrustForwardHeader:       auth.TrustForwardHeader,
 | |
| 		AuthResponseHeaders:      auth.AuthResponseHeaders,
 | |
| 		AuthResponseHeadersRegex: auth.AuthResponseHeadersRegex,
 | |
| 		AuthRequestHeaders:       auth.AuthRequestHeaders,
 | |
| 	}
 | |
| 
 | |
| 	if auth.TLS == nil {
 | |
| 		return forwardAuth, nil
 | |
| 	}
 | |
| 
 | |
| 	forwardAuth.TLS = &types.ClientTLS{
 | |
| 		CAOptional:         auth.TLS.CAOptional,
 | |
| 		InsecureSkipVerify: auth.TLS.InsecureSkipVerify,
 | |
| 	}
 | |
| 
 | |
| 	if len(auth.TLS.CASecret) > 0 {
 | |
| 		caSecret, err := loadCASecret(namespace, auth.TLS.CASecret, k8sClient)
 | |
| 		if err != nil {
 | |
| 			return nil, fmt.Errorf("failed to load auth ca secret: %w", err)
 | |
| 		}
 | |
| 		forwardAuth.TLS.CA = caSecret
 | |
| 	}
 | |
| 
 | |
| 	if len(auth.TLS.CertSecret) > 0 {
 | |
| 		authSecretCert, authSecretKey, err := loadAuthTLSSecret(namespace, auth.TLS.CertSecret, k8sClient)
 | |
| 		if err != nil {
 | |
| 			return nil, fmt.Errorf("failed to load auth secret: %w", err)
 | |
| 		}
 | |
| 		forwardAuth.TLS.Cert = authSecretCert
 | |
| 		forwardAuth.TLS.Key = authSecretKey
 | |
| 	}
 | |
| 
 | |
| 	return forwardAuth, nil
 | |
| }
 | |
| 
 | |
| func loadCASecret(namespace, secretName string, k8sClient Client) (string, error) {
 | |
| 	secret, ok, err := k8sClient.GetSecret(namespace, secretName)
 | |
| 	if err != nil {
 | |
| 		return "", fmt.Errorf("failed to fetch secret '%s/%s': %w", namespace, secretName, err)
 | |
| 	}
 | |
| 
 | |
| 	if !ok {
 | |
| 		return "", fmt.Errorf("secret '%s/%s' not found", namespace, secretName)
 | |
| 	}
 | |
| 
 | |
| 	if secret == nil {
 | |
| 		return "", fmt.Errorf("data for secret '%s/%s' must not be nil", namespace, secretName)
 | |
| 	}
 | |
| 
 | |
| 	tlsCAData, err := getCABlocks(secret, namespace, secretName)
 | |
| 	if err == nil {
 | |
| 		return tlsCAData, nil
 | |
| 	}
 | |
| 
 | |
| 	// TODO: remove this behavior in the next major version (v3)
 | |
| 	if len(secret.Data) == 1 {
 | |
| 		// For backwards compatibility, use the only available secret data as CA if both 'ca.crt' and 'tls.ca' are missing.
 | |
| 		for _, v := range secret.Data {
 | |
| 			return string(v), nil
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return "", fmt.Errorf("could not find CA block: %w", err)
 | |
| }
 | |
| 
 | |
| func loadAuthTLSSecret(namespace, secretName string, k8sClient Client) (string, string, error) {
 | |
| 	secret, exists, err := k8sClient.GetSecret(namespace, secretName)
 | |
| 	if err != nil {
 | |
| 		return "", "", fmt.Errorf("failed to fetch secret '%s/%s': %w", namespace, secretName, err)
 | |
| 	}
 | |
| 
 | |
| 	if !exists {
 | |
| 		return "", "", fmt.Errorf("secret '%s/%s' does not exist", namespace, secretName)
 | |
| 	}
 | |
| 
 | |
| 	if secret == nil {
 | |
| 		return "", "", fmt.Errorf("data for secret '%s/%s' must not be nil", namespace, secretName)
 | |
| 	}
 | |
| 
 | |
| 	return getCertificateBlocks(secret, namespace, secretName)
 | |
| }
 | |
| 
 | |
| func createBasicAuthMiddleware(client Client, namespace string, basicAuth *traefikv1alpha1.BasicAuth) (*dynamic.BasicAuth, error) {
 | |
| 	if basicAuth == nil {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	if basicAuth.Secret == "" {
 | |
| 		return nil, fmt.Errorf("auth secret must be set")
 | |
| 	}
 | |
| 
 | |
| 	secret, ok, err := client.GetSecret(namespace, basicAuth.Secret)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("failed to fetch secret '%s/%s': %w", namespace, basicAuth.Secret, err)
 | |
| 	}
 | |
| 	if !ok {
 | |
| 		return nil, fmt.Errorf("secret '%s/%s' not found", namespace, basicAuth.Secret)
 | |
| 	}
 | |
| 	if secret == nil {
 | |
| 		return nil, fmt.Errorf("data for secret '%s/%s' must not be nil", namespace, basicAuth.Secret)
 | |
| 	}
 | |
| 
 | |
| 	if secret.Type == corev1.SecretTypeBasicAuth {
 | |
| 		credentials, err := loadBasicAuthCredentials(secret)
 | |
| 		if err != nil {
 | |
| 			return nil, fmt.Errorf("failed to load basic auth credentials: %w", err)
 | |
| 		}
 | |
| 
 | |
| 		return &dynamic.BasicAuth{
 | |
| 			Users:        credentials,
 | |
| 			Realm:        basicAuth.Realm,
 | |
| 			RemoveHeader: basicAuth.RemoveHeader,
 | |
| 			HeaderField:  basicAuth.HeaderField,
 | |
| 		}, nil
 | |
| 	}
 | |
| 
 | |
| 	credentials, err := loadAuthCredentials(secret)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("failed to load basic auth credentials: %w", err)
 | |
| 	}
 | |
| 
 | |
| 	return &dynamic.BasicAuth{
 | |
| 		Users:        credentials,
 | |
| 		Realm:        basicAuth.Realm,
 | |
| 		RemoveHeader: basicAuth.RemoveHeader,
 | |
| 		HeaderField:  basicAuth.HeaderField,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func createDigestAuthMiddleware(client Client, namespace string, digestAuth *traefikv1alpha1.DigestAuth) (*dynamic.DigestAuth, error) {
 | |
| 	if digestAuth == nil {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	if digestAuth.Secret == "" {
 | |
| 		return nil, fmt.Errorf("auth secret must be set")
 | |
| 	}
 | |
| 
 | |
| 	secret, ok, err := client.GetSecret(namespace, digestAuth.Secret)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("failed to fetch secret '%s/%s': %w", namespace, digestAuth.Secret, err)
 | |
| 	}
 | |
| 	if !ok {
 | |
| 		return nil, fmt.Errorf("secret '%s/%s' not found", namespace, digestAuth.Secret)
 | |
| 	}
 | |
| 	if secret == nil {
 | |
| 		return nil, fmt.Errorf("data for secret '%s/%s' must not be nil", namespace, digestAuth.Secret)
 | |
| 	}
 | |
| 
 | |
| 	credentials, err := loadAuthCredentials(secret)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("failed to load digest auth credentials: %w", err)
 | |
| 	}
 | |
| 
 | |
| 	return &dynamic.DigestAuth{
 | |
| 		Users:        credentials,
 | |
| 		Realm:        digestAuth.Realm,
 | |
| 		RemoveHeader: digestAuth.RemoveHeader,
 | |
| 		HeaderField:  digestAuth.HeaderField,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func loadBasicAuthCredentials(secret *corev1.Secret) ([]string, error) {
 | |
| 	username, usernameExists := secret.Data["username"]
 | |
| 	password, passwordExists := secret.Data["password"]
 | |
| 	if !(usernameExists && passwordExists) {
 | |
| 		return nil, fmt.Errorf("secret '%s/%s' must contain both username and password keys", secret.Namespace, secret.Name)
 | |
| 	}
 | |
| 
 | |
| 	hash := sha1.New()
 | |
| 	hash.Write(password)
 | |
| 	passwordSum := base64.StdEncoding.EncodeToString(hash.Sum(nil))
 | |
| 
 | |
| 	return []string{fmt.Sprintf("%s:{SHA}%s", username, passwordSum)}, nil
 | |
| }
 | |
| 
 | |
| func loadAuthCredentials(secret *corev1.Secret) ([]string, error) {
 | |
| 	if len(secret.Data) != 1 {
 | |
| 		return nil, fmt.Errorf("found %d elements for secret '%s/%s', must be single element exactly", len(secret.Data), secret.Namespace, secret.Name)
 | |
| 	}
 | |
| 
 | |
| 	var firstSecret []byte
 | |
| 	for _, v := range secret.Data {
 | |
| 		firstSecret = v
 | |
| 		break
 | |
| 	}
 | |
| 
 | |
| 	var credentials []string
 | |
| 	scanner := bufio.NewScanner(bytes.NewReader(firstSecret))
 | |
| 	for scanner.Scan() {
 | |
| 		if cred := scanner.Text(); len(cred) > 0 {
 | |
| 			credentials = append(credentials, cred)
 | |
| 		}
 | |
| 	}
 | |
| 	if err := scanner.Err(); err != nil {
 | |
| 		return nil, fmt.Errorf("error reading secret for %s/%s: %w", secret.Namespace, secret.Name, err)
 | |
| 	}
 | |
| 	if len(credentials) == 0 {
 | |
| 		return nil, fmt.Errorf("secret '%s/%s' does not contain any credentials", secret.Namespace, secret.Name)
 | |
| 	}
 | |
| 
 | |
| 	return credentials, nil
 | |
| }
 | |
| 
 | |
| func createChainMiddleware(ctx context.Context, namespace string, chain *traefikv1alpha1.Chain) *dynamic.Chain {
 | |
| 	if chain == nil {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	var mds []string
 | |
| 	for _, mi := range chain.Middlewares {
 | |
| 		if strings.Contains(mi.Name, providerNamespaceSeparator) {
 | |
| 			if len(mi.Namespace) > 0 {
 | |
| 				log.FromContext(ctx).
 | |
| 					Warnf("namespace %q is ignored in cross-provider context", mi.Namespace)
 | |
| 			}
 | |
| 			mds = append(mds, mi.Name)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		ns := mi.Namespace
 | |
| 		if len(ns) == 0 {
 | |
| 			ns = namespace
 | |
| 		}
 | |
| 		mds = append(mds, makeID(ns, mi.Name))
 | |
| 	}
 | |
| 	return &dynamic.Chain{Middlewares: mds}
 | |
| }
 | |
| 
 | |
| func buildTLSOptions(ctx context.Context, client Client) map[string]tls.Options {
 | |
| 	tlsOptionsCRD := client.GetTLSOptions()
 | |
| 	var tlsOptions map[string]tls.Options
 | |
| 
 | |
| 	if len(tlsOptionsCRD) == 0 {
 | |
| 		return tlsOptions
 | |
| 	}
 | |
| 	tlsOptions = make(map[string]tls.Options)
 | |
| 	var nsDefault []string
 | |
| 
 | |
| 	for _, tlsOption := range tlsOptionsCRD {
 | |
| 		logger := log.FromContext(log.With(ctx, log.Str("tlsOption", tlsOption.Name), log.Str("namespace", tlsOption.Namespace)))
 | |
| 		var clientCAs []tls.FileOrContent
 | |
| 
 | |
| 		for _, secretName := range tlsOption.Spec.ClientAuth.SecretNames {
 | |
| 			secret, exists, err := client.GetSecret(tlsOption.Namespace, secretName)
 | |
| 			if err != nil {
 | |
| 				logger.Errorf("Failed to fetch secret %s/%s: %v", tlsOption.Namespace, secretName, err)
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			if !exists {
 | |
| 				logger.Warnf("Secret %s/%s does not exist", tlsOption.Namespace, secretName)
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			cert, err := getCABlocks(secret, tlsOption.Namespace, secretName)
 | |
| 			if err != nil {
 | |
| 				logger.Errorf("Failed to extract CA from secret %s/%s: %v", tlsOption.Namespace, secretName, err)
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			clientCAs = append(clientCAs, tls.FileOrContent(cert))
 | |
| 		}
 | |
| 
 | |
| 		id := makeID(tlsOption.Namespace, tlsOption.Name)
 | |
| 		// If the name is default, we override the default config.
 | |
| 		if tlsOption.Name == tls.DefaultTLSConfigName {
 | |
| 			id = tlsOption.Name
 | |
| 			nsDefault = append(nsDefault, tlsOption.Namespace)
 | |
| 		}
 | |
| 
 | |
| 		alpnProtocols := tls.DefaultTLSOptions.ALPNProtocols
 | |
| 		if len(tlsOption.Spec.ALPNProtocols) > 0 {
 | |
| 			alpnProtocols = tlsOption.Spec.ALPNProtocols
 | |
| 		}
 | |
| 
 | |
| 		tlsOptions[id] = tls.Options{
 | |
| 			MinVersion:       tlsOption.Spec.MinVersion,
 | |
| 			MaxVersion:       tlsOption.Spec.MaxVersion,
 | |
| 			CipherSuites:     tlsOption.Spec.CipherSuites,
 | |
| 			CurvePreferences: tlsOption.Spec.CurvePreferences,
 | |
| 			ClientAuth: tls.ClientAuth{
 | |
| 				CAFiles:        clientCAs,
 | |
| 				ClientAuthType: tlsOption.Spec.ClientAuth.ClientAuthType,
 | |
| 			},
 | |
| 			SniStrict:     tlsOption.Spec.SniStrict,
 | |
| 			ALPNProtocols: alpnProtocols,
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if len(nsDefault) > 1 {
 | |
| 		delete(tlsOptions, tls.DefaultTLSConfigName)
 | |
| 		log.FromContext(ctx).Errorf("Default TLS Options defined in multiple namespaces: %v", nsDefault)
 | |
| 	}
 | |
| 
 | |
| 	return tlsOptions
 | |
| }
 | |
| 
 | |
| func buildTLSStores(ctx context.Context, client Client) (map[string]tls.Store, map[string]*tls.CertAndStores) {
 | |
| 	tlsStoreCRD := client.GetTLSStores()
 | |
| 	if len(tlsStoreCRD) == 0 {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 
 | |
| 	var nsDefault []string
 | |
| 	tlsStores := make(map[string]tls.Store)
 | |
| 	tlsConfigs := make(map[string]*tls.CertAndStores)
 | |
| 
 | |
| 	for _, t := range tlsStoreCRD {
 | |
| 		logger := log.FromContext(log.With(ctx, log.Str("TLSStore", t.Name), log.Str("namespace", t.Namespace)))
 | |
| 
 | |
| 		id := makeID(t.Namespace, t.Name)
 | |
| 
 | |
| 		// If the name is default, we override the default config.
 | |
| 		if t.Name == tls.DefaultTLSStoreName {
 | |
| 			id = t.Name
 | |
| 			nsDefault = append(nsDefault, t.Namespace)
 | |
| 		}
 | |
| 
 | |
| 		var tlsStore tls.Store
 | |
| 
 | |
| 		if t.Spec.DefaultCertificate != nil {
 | |
| 			secretName := t.Spec.DefaultCertificate.SecretName
 | |
| 
 | |
| 			secret, exists, err := client.GetSecret(t.Namespace, secretName)
 | |
| 			if err != nil {
 | |
| 				logger.Errorf("Failed to fetch secret %s/%s: %v", t.Namespace, secretName, err)
 | |
| 				continue
 | |
| 			}
 | |
| 			if !exists {
 | |
| 				logger.Errorf("Secret %s/%s does not exist", t.Namespace, secretName)
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			cert, key, err := getCertificateBlocks(secret, t.Namespace, secretName)
 | |
| 			if err != nil {
 | |
| 				logger.Errorf("Could not get certificate blocks: %v", err)
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			tlsStore.DefaultCertificate = &tls.Certificate{
 | |
| 				CertFile: tls.FileOrContent(cert),
 | |
| 				KeyFile:  tls.FileOrContent(key),
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if t.Spec.DefaultGeneratedCert != nil {
 | |
| 			tlsStore.DefaultGeneratedCert = &tls.GeneratedCert{
 | |
| 				Resolver: t.Spec.DefaultGeneratedCert.Resolver,
 | |
| 				Domain:   t.Spec.DefaultGeneratedCert.Domain,
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if err := buildCertificates(client, id, t.Namespace, t.Spec.Certificates, tlsConfigs); err != nil {
 | |
| 			logger.Errorf("Failed to load certificates: %v", err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		tlsStores[id] = tlsStore
 | |
| 	}
 | |
| 
 | |
| 	if len(nsDefault) > 1 {
 | |
| 		delete(tlsStores, tls.DefaultTLSStoreName)
 | |
| 		log.FromContext(ctx).Errorf("Default TLS Stores defined in multiple namespaces: %v", nsDefault)
 | |
| 	}
 | |
| 
 | |
| 	return tlsStores, tlsConfigs
 | |
| }
 | |
| 
 | |
| // buildCertificates loads TLSStore certificates from secrets and sets them into tlsConfigs.
 | |
| func buildCertificates(client Client, tlsStore, namespace string, certificates []traefikv1alpha1.Certificate, tlsConfigs map[string]*tls.CertAndStores) error {
 | |
| 	for _, c := range certificates {
 | |
| 		configKey := namespace + "/" + c.SecretName
 | |
| 		if _, tlsExists := tlsConfigs[configKey]; !tlsExists {
 | |
| 			certAndStores, err := getTLS(client, c.SecretName, namespace)
 | |
| 			if err != nil {
 | |
| 				return fmt.Errorf("unable to read secret %s: %w", configKey, err)
 | |
| 			}
 | |
| 
 | |
| 			certAndStores.Stores = []string{tlsStore}
 | |
| 			tlsConfigs[configKey] = certAndStores
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func makeServiceKey(rule, ingressName string) (string, error) {
 | |
| 	h := sha256.New()
 | |
| 	if _, err := h.Write([]byte(rule)); err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	key := fmt.Sprintf("%s-%.10x", ingressName, h.Sum(nil))
 | |
| 
 | |
| 	return key, nil
 | |
| }
 | |
| 
 | |
| func makeID(namespace, name string) string {
 | |
| 	if namespace == "" {
 | |
| 		return name
 | |
| 	}
 | |
| 
 | |
| 	return namespace + "-" + name
 | |
| }
 | |
| 
 | |
| func shouldProcessIngress(ingressClass, ingressClassAnnotation string) bool {
 | |
| 	return ingressClass == ingressClassAnnotation ||
 | |
| 		(len(ingressClass) == 0 && ingressClassAnnotation == traefikDefaultIngressClass)
 | |
| }
 | |
| 
 | |
| func getTLS(k8sClient Client, secretName, namespace string) (*tls.CertAndStores, error) {
 | |
| 	secret, exists, err := k8sClient.GetSecret(namespace, secretName)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("failed to fetch secret %s/%s: %w", namespace, secretName, err)
 | |
| 	}
 | |
| 	if !exists {
 | |
| 		return nil, fmt.Errorf("secret %s/%s does not exist", namespace, secretName)
 | |
| 	}
 | |
| 
 | |
| 	cert, key, err := getCertificateBlocks(secret, namespace, secretName)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	return &tls.CertAndStores{
 | |
| 		Certificate: tls.Certificate{
 | |
| 			CertFile: tls.FileOrContent(cert),
 | |
| 			KeyFile:  tls.FileOrContent(key),
 | |
| 		},
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func getTLSConfig(tlsConfigs map[string]*tls.CertAndStores) []*tls.CertAndStores {
 | |
| 	var secretNames []string
 | |
| 	for secretName := range tlsConfigs {
 | |
| 		secretNames = append(secretNames, secretName)
 | |
| 	}
 | |
| 	sort.Strings(secretNames)
 | |
| 
 | |
| 	var configs []*tls.CertAndStores
 | |
| 	for _, secretName := range secretNames {
 | |
| 		configs = append(configs, tlsConfigs[secretName])
 | |
| 	}
 | |
| 
 | |
| 	return configs
 | |
| }
 | |
| 
 | |
| func getCertificateBlocks(secret *corev1.Secret, namespace, secretName string) (string, string, error) {
 | |
| 	var missingEntries []string
 | |
| 
 | |
| 	tlsCrtData, tlsCrtExists := secret.Data["tls.crt"]
 | |
| 	if !tlsCrtExists {
 | |
| 		missingEntries = append(missingEntries, "tls.crt")
 | |
| 	}
 | |
| 
 | |
| 	tlsKeyData, tlsKeyExists := secret.Data["tls.key"]
 | |
| 	if !tlsKeyExists {
 | |
| 		missingEntries = append(missingEntries, "tls.key")
 | |
| 	}
 | |
| 
 | |
| 	if len(missingEntries) > 0 {
 | |
| 		return "", "", fmt.Errorf("secret %s/%s is missing the following TLS data entries: %s",
 | |
| 			namespace, secretName, strings.Join(missingEntries, ", "))
 | |
| 	}
 | |
| 
 | |
| 	cert := string(tlsCrtData)
 | |
| 	if cert == "" {
 | |
| 		missingEntries = append(missingEntries, "tls.crt")
 | |
| 	}
 | |
| 
 | |
| 	key := string(tlsKeyData)
 | |
| 	if key == "" {
 | |
| 		missingEntries = append(missingEntries, "tls.key")
 | |
| 	}
 | |
| 
 | |
| 	if len(missingEntries) > 0 {
 | |
| 		return "", "", fmt.Errorf("secret %s/%s contains the following empty TLS data entries: %s",
 | |
| 			namespace, secretName, strings.Join(missingEntries, ", "))
 | |
| 	}
 | |
| 
 | |
| 	return cert, key, nil
 | |
| }
 | |
| 
 | |
| func getCABlocks(secret *corev1.Secret, namespace, secretName string) (string, error) {
 | |
| 	tlsCrtData, tlsCrtExists := secret.Data["tls.ca"]
 | |
| 	if tlsCrtExists {
 | |
| 		return string(tlsCrtData), nil
 | |
| 	}
 | |
| 
 | |
| 	tlsCrtData, tlsCrtExists = secret.Data["ca.crt"]
 | |
| 	if tlsCrtExists {
 | |
| 		return string(tlsCrtData), nil
 | |
| 	}
 | |
| 
 | |
| 	return "", fmt.Errorf("secret %s/%s contains neither tls.ca nor ca.crt", namespace, secretName)
 | |
| }
 | |
| 
 | |
| func throttleEvents(ctx context.Context, throttleDuration time.Duration, pool *safe.Pool, eventsChan <-chan interface{}) chan interface{} {
 | |
| 	if throttleDuration == 0 {
 | |
| 		return nil
 | |
| 	}
 | |
| 	// Create a buffered channel to hold the pending event (if we're delaying processing the event due to throttling)
 | |
| 	eventsChanBuffered := make(chan interface{}, 1)
 | |
| 
 | |
| 	// Run a goroutine that reads events from eventChan and does a non-blocking write to pendingEvent.
 | |
| 	// This guarantees that writing to eventChan will never block,
 | |
| 	// and that pendingEvent will have something in it if there's been an event since we read from that channel.
 | |
| 	pool.GoCtx(func(ctxPool context.Context) {
 | |
| 		for {
 | |
| 			select {
 | |
| 			case <-ctxPool.Done():
 | |
| 				return
 | |
| 			case nextEvent := <-eventsChan:
 | |
| 				select {
 | |
| 				case eventsChanBuffered <- nextEvent:
 | |
| 				default:
 | |
| 					// We already have an event in eventsChanBuffered, so we'll do a refresh as soon as our throttle allows us to.
 | |
| 					// It's fine to drop the event and keep whatever's in the buffer -- we don't do different things for different events
 | |
| 					log.FromContext(ctx).Debugf("Dropping event kind %T due to throttling", nextEvent)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	return eventsChanBuffered
 | |
| }
 | |
| 
 | |
| func isNamespaceAllowed(allowCrossNamespace bool, parentNamespace, namespace string) bool {
 | |
| 	// If allowCrossNamespace option is not defined the default behavior is to allow cross namespace references.
 | |
| 	return allowCrossNamespace || parentNamespace == namespace
 | |
| }
 |