mirror of
				https://github.com/kubernetes-sigs/external-dns.git
				synced 2025-11-04 12:41:00 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			241 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			241 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
Copyright 2022 The Kubernetes Authors.
 | 
						|
 | 
						|
Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
you may not use this file except in compliance with the License.
 | 
						|
You may obtain a copy of the License at
 | 
						|
 | 
						|
    http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 | 
						|
Unless required by applicable law or agreed to in writing, software
 | 
						|
distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
See the License for the specific language governing permissions and
 | 
						|
limitations under the License.
 | 
						|
*/
 | 
						|
 | 
						|
package source
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
	"sort"
 | 
						|
 | 
						|
	"github.com/pkg/errors"
 | 
						|
	log "github.com/sirupsen/logrus"
 | 
						|
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
						|
	"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
 | 
						|
	"k8s.io/apimachinery/pkg/labels"
 | 
						|
	"k8s.io/apimachinery/pkg/runtime"
 | 
						|
	"k8s.io/apimachinery/pkg/runtime/schema"
 | 
						|
	"k8s.io/client-go/dynamic"
 | 
						|
	"k8s.io/client-go/dynamic/dynamicinformer"
 | 
						|
	"k8s.io/client-go/informers"
 | 
						|
	"k8s.io/client-go/kubernetes"
 | 
						|
	"k8s.io/client-go/kubernetes/scheme"
 | 
						|
	"k8s.io/client-go/tools/cache"
 | 
						|
 | 
						|
	f5 "github.com/F5Networks/k8s-bigip-ctlr/v2/config/apis/cis/v1"
 | 
						|
 | 
						|
	"sigs.k8s.io/external-dns/endpoint"
 | 
						|
)
 | 
						|
 | 
						|
var f5VirtualServerGVR = schema.GroupVersionResource{
 | 
						|
	Group:    "cis.f5.com",
 | 
						|
	Version:  "v1",
 | 
						|
	Resource: "virtualservers",
 | 
						|
}
 | 
						|
 | 
						|
// virtualServerSource is an implementation of Source for F5 VirtualServer objects.
 | 
						|
type f5VirtualServerSource struct {
 | 
						|
	dynamicKubeClient     dynamic.Interface
 | 
						|
	virtualServerInformer informers.GenericInformer
 | 
						|
	kubeClient            kubernetes.Interface
 | 
						|
	annotationFilter      string
 | 
						|
	namespace             string
 | 
						|
	unstructuredConverter *unstructuredConverter
 | 
						|
}
 | 
						|
 | 
						|
func NewF5VirtualServerSource(
 | 
						|
	ctx context.Context,
 | 
						|
	dynamicKubeClient dynamic.Interface,
 | 
						|
	kubeClient kubernetes.Interface,
 | 
						|
	namespace string,
 | 
						|
	annotationFilter string,
 | 
						|
) (Source, error) {
 | 
						|
	informerFactory := dynamicinformer.NewFilteredDynamicSharedInformerFactory(dynamicKubeClient, 0, namespace, nil)
 | 
						|
	virtualServerInformer := informerFactory.ForResource(f5VirtualServerGVR)
 | 
						|
 | 
						|
	virtualServerInformer.Informer().AddEventHandler(
 | 
						|
		cache.ResourceEventHandlerFuncs{
 | 
						|
			AddFunc: func(obj interface{}) {
 | 
						|
			},
 | 
						|
		},
 | 
						|
	)
 | 
						|
 | 
						|
	informerFactory.Start(ctx.Done())
 | 
						|
 | 
						|
	// wait for the local cache to be populated.
 | 
						|
	if err := waitForDynamicCacheSync(context.Background(), informerFactory); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	uc, err := newVSUnstructuredConverter()
 | 
						|
	if err != nil {
 | 
						|
		return nil, errors.Wrapf(err, "failed to setup unstructured converter")
 | 
						|
	}
 | 
						|
 | 
						|
	return &f5VirtualServerSource{
 | 
						|
		dynamicKubeClient:     dynamicKubeClient,
 | 
						|
		virtualServerInformer: virtualServerInformer,
 | 
						|
		kubeClient:            kubeClient,
 | 
						|
		namespace:             namespace,
 | 
						|
		annotationFilter:      annotationFilter,
 | 
						|
		unstructuredConverter: uc,
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
// Endpoints returns endpoint objects for each host-target combination that should be processed.
 | 
						|
// Retrieves all VirtualServers in the source's namespace(s).
 | 
						|
func (vs *f5VirtualServerSource) Endpoints(ctx context.Context) ([]*endpoint.Endpoint, error) {
 | 
						|
	virtualServerObjects, err := vs.virtualServerInformer.Lister().ByNamespace(vs.namespace).List(labels.Everything())
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	var virtualServers []*f5.VirtualServer
 | 
						|
	for _, vsObj := range virtualServerObjects {
 | 
						|
		unstructuredHost, ok := vsObj.(*unstructured.Unstructured)
 | 
						|
		if !ok {
 | 
						|
			return nil, errors.New("could not convert")
 | 
						|
		}
 | 
						|
 | 
						|
		virtualServer := &f5.VirtualServer{}
 | 
						|
		err := vs.unstructuredConverter.scheme.Convert(unstructuredHost, virtualServer, nil)
 | 
						|
		if err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
		virtualServers = append(virtualServers, virtualServer)
 | 
						|
	}
 | 
						|
 | 
						|
	virtualServers, err = vs.filterByAnnotations(virtualServers)
 | 
						|
	if err != nil {
 | 
						|
		return nil, errors.Wrap(err, "failed to filter VirtualServers")
 | 
						|
	}
 | 
						|
 | 
						|
	endpoints, err := vs.endpointsFromVirtualServers(virtualServers)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	// Sort endpoints
 | 
						|
	for _, ep := range endpoints {
 | 
						|
		sort.Sort(ep.Targets)
 | 
						|
	}
 | 
						|
 | 
						|
	return endpoints, nil
 | 
						|
}
 | 
						|
 | 
						|
func (vs *f5VirtualServerSource) AddEventHandler(ctx context.Context, handler func()) {
 | 
						|
	log.Debug("Adding event handler for VirtualServer")
 | 
						|
 | 
						|
	vs.virtualServerInformer.Informer().AddEventHandler(eventHandlerFunc(handler))
 | 
						|
}
 | 
						|
 | 
						|
// endpointsFromVirtualServers extracts the endpoints from a slice of VirtualServers
 | 
						|
func (vs *f5VirtualServerSource) endpointsFromVirtualServers(virtualServers []*f5.VirtualServer) ([]*endpoint.Endpoint, error) {
 | 
						|
	var endpoints []*endpoint.Endpoint
 | 
						|
 | 
						|
	for _, virtualServer := range virtualServers {
 | 
						|
		ttl, err := getTTLFromAnnotations(virtualServer.Annotations)
 | 
						|
		if err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
 | 
						|
		if virtualServer.Spec.VirtualServerAddress != "" {
 | 
						|
			ep := &endpoint.Endpoint{
 | 
						|
				Targets: endpoint.Targets{
 | 
						|
					virtualServer.Spec.VirtualServerAddress,
 | 
						|
				},
 | 
						|
				RecordType: "A",
 | 
						|
				DNSName:    virtualServer.Spec.Host,
 | 
						|
				Labels:     endpoint.NewLabels(),
 | 
						|
				RecordTTL:  ttl,
 | 
						|
			}
 | 
						|
 | 
						|
			vs.setResourceLabel(virtualServer, ep)
 | 
						|
			endpoints = append(endpoints, ep)
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		if virtualServer.Status.VSAddress != "" {
 | 
						|
			ep := &endpoint.Endpoint{
 | 
						|
				Targets: endpoint.Targets{
 | 
						|
					virtualServer.Status.VSAddress,
 | 
						|
				},
 | 
						|
				RecordType: "A",
 | 
						|
				DNSName:    virtualServer.Spec.Host,
 | 
						|
				Labels:     endpoint.NewLabels(),
 | 
						|
				RecordTTL:  ttl,
 | 
						|
			}
 | 
						|
 | 
						|
			vs.setResourceLabel(virtualServer, ep)
 | 
						|
			endpoints = append(endpoints, ep)
 | 
						|
			continue
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return endpoints, nil
 | 
						|
}
 | 
						|
 | 
						|
// newUnstructuredConverter returns a new unstructuredConverter initialized
 | 
						|
func newVSUnstructuredConverter() (*unstructuredConverter, error) {
 | 
						|
	uc := &unstructuredConverter{
 | 
						|
		scheme: runtime.NewScheme(),
 | 
						|
	}
 | 
						|
 | 
						|
	// Add the core types we need
 | 
						|
	uc.scheme.AddKnownTypes(f5VirtualServerGVR.GroupVersion(), &f5.VirtualServer{}, &f5.VirtualServerList{})
 | 
						|
	if err := scheme.AddToScheme(uc.scheme); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	return uc, nil
 | 
						|
}
 | 
						|
 | 
						|
// filterByAnnotations filters a list of VirtualServers by a given annotation selector.
 | 
						|
func (vs *f5VirtualServerSource) filterByAnnotations(virtualServers []*f5.VirtualServer) ([]*f5.VirtualServer, error) {
 | 
						|
	labelSelector, err := metav1.ParseToLabelSelector(vs.annotationFilter)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	selector, err := metav1.LabelSelectorAsSelector(labelSelector)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	// empty filter returns original list
 | 
						|
	if selector.Empty() {
 | 
						|
		return virtualServers, nil
 | 
						|
	}
 | 
						|
 | 
						|
	filteredList := []*f5.VirtualServer{}
 | 
						|
 | 
						|
	for _, vs := range virtualServers {
 | 
						|
		// convert the VirtualServer's annotations to an equivalent label selector
 | 
						|
		annotations := labels.Set(vs.Annotations)
 | 
						|
 | 
						|
		// include VirtualServer if its annotations match the selector
 | 
						|
		if selector.Matches(annotations) {
 | 
						|
			filteredList = append(filteredList, vs)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return filteredList, nil
 | 
						|
}
 | 
						|
 | 
						|
func (vs *f5VirtualServerSource) setResourceLabel(virtualServer *f5.VirtualServer, ep *endpoint.Endpoint) {
 | 
						|
	ep.Labels[endpoint.ResourceLabelKey] = fmt.Sprintf("f5-virtualserver/%s/%s", virtualServer.Namespace, virtualServer.Name)
 | 
						|
}
 |