mirror of
https://github.com/kubernetes-sigs/external-dns.git
synced 2025-08-05 09:06:58 +02:00
* Add annotation filter to Ambassador Host Source This change makes the Ambassador Host source respect the External-DNS annotationFilter allowing for an Ambassador Host resource to specify what External-DNS deployment to use when there are multiple External-DNS deployments within the same cluster. Before this change if you had two External-DNS deployments within the cluster and used the Ambassador Host source the first External-DNS to process the resource will create the record and not the one that was specified in the filter annotation. I added the `filterByAnnotations` function so that it matched the same way the other sources have implemented annotation filtering. I didn't add the controller check only because I wanted to keep this change to implementing the annotationFilter. Example: Create two External-DNS deployments 1 public and 1 private and set the Ambassador Host to use the public External-DNS using the annotation filter. ``` --- apiVersion: apps/v1 kind: Deployment metadata: name: external-dns-private spec: strategy: type: Recreate selector: matchLabels: app: external-dns-private template: metadata: labels: app: external-dns-private annotations: iam.amazonaws.com/role: {ARN} # AWS ARN role spec: serviceAccountName: external-dns containers: - name: external-dns image: k8s.gcr.io/external-dns/external-dns:latest args: - --source=ambassador-host - --domain-filter=example.net # will make ExternalDNS see only the hosted zones matching provided domain, omit to process all available hosted zones - --provider=aws - --policy=upsert-only # would prevent ExternalDNS from deleting any records, omit to enable full synchronization - --aws-zone-type=private # only look at public hosted zones (valid values are public, private or no value for both) - --registry=txt - --txt-owner-id= {Hosted Zone ID} # Insert Route53 Hosted Zone ID here - --annotation-filter=kubernetes.io/ingress.class in (private) --- apiVersion: apps/v1 kind: Deployment metadata: name: external-dns-public spec: strategy: type: Recreate selector: matchLabels: app: external-dns-public template: metadata: labels: app: external-dns-public annotations: iam.amazonaws.com/role: {ARN} # AWS ARN role spec: serviceAccountName: external-dns containers: - name: external-dns image: k8s.gcr.io/external-dns/external-dns:latest args: - --source=ambassador-host - --domain-filter=example.net # will make ExternalDNS see only the hosted zones matching provided domain, omit to process all available hosted zones - --provider=aws - --policy=upsert-only # would prevent ExternalDNS from deleting any records, omit to enable full synchronization - --aws-zone-type= # only look at public hosted zones (valid values are public, private or no value for both) - --registry=txt - --txt-owner-id= {Hosted Zone ID} # Insert Route53 Hosted Zone ID here - --annotation-filter=kubernetes.io/ingress.class in (public) --- apiVersion: getambassador.io/v3alpha1 kind: Host metadata: name: your-hostname annotations: external-dns.ambassador-service: emissary-ingress/emissary kubernetes.io/ingress.class: public spec: acmeProvider: authority: none hostname: your-hostname.example.com ``` Fixes kubernetes-sigs/external-dns#2632 * Add Label filltering for Ambassador Host source Currently the `--label-filter` flag can only be used to filter CRDs, Ingress, Service and Openshift Route objects which match the label selector passed through that flag. This change extends the functionality to the Ambassador Host type object. When the flag is not specified the default value is `labels.Everything()` which is an empty string, the same as before. An annotation based filter is inefficient because the filtering has to be done in the controller instead of the API server like with label filtering. The Annotation based filtering has been left in for legacy reasons so the Ambassador Host source can be used inconjunction with the other sources that don't yet support label filltering. It is possible to use label based filltering with annotation based filltering so you can initially filter by label then filter the returned hosts by annotation. This is not recomended * Update Ambassador Host source docs Add that the Ambassador Host source now supports both annotation and label filltering.
323 lines
10 KiB
Go
323 lines
10 KiB
Go
/*
|
|
Copyright 2020 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"
|
|
"strings"
|
|
|
|
ambassador "github.com/datawire/ambassador/pkg/api/getambassador.io/v2"
|
|
"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"
|
|
|
|
"sigs.k8s.io/external-dns/endpoint"
|
|
)
|
|
|
|
// ambHostAnnotation is the annotation in the Host that maps to a Service
|
|
const ambHostAnnotation = "external-dns.ambassador-service"
|
|
|
|
// groupName is the group name for the Ambassador API
|
|
const groupName = "getambassador.io"
|
|
|
|
var schemeGroupVersion = schema.GroupVersion{Group: groupName, Version: "v2"}
|
|
|
|
var ambHostGVR = schemeGroupVersion.WithResource("hosts")
|
|
|
|
// ambassadorHostSource is an implementation of Source for Ambassador Host objects.
|
|
// The IngressRoute implementation uses the spec.virtualHost.fqdn value for the hostname.
|
|
// Use targetAnnotationKey to explicitly set Endpoint.
|
|
type ambassadorHostSource struct {
|
|
dynamicKubeClient dynamic.Interface
|
|
kubeClient kubernetes.Interface
|
|
namespace string
|
|
annotationFilter string
|
|
ambassadorHostInformer informers.GenericInformer
|
|
unstructuredConverter *unstructuredConverter
|
|
labelSelector labels.Selector
|
|
}
|
|
|
|
// NewAmbassadorHostSource creates a new ambassadorHostSource with the given config.
|
|
func NewAmbassadorHostSource(
|
|
ctx context.Context,
|
|
dynamicKubeClient dynamic.Interface,
|
|
kubeClient kubernetes.Interface,
|
|
namespace string,
|
|
annotationFilter string,
|
|
labelSelector labels.Selector,
|
|
) (Source, error) {
|
|
var err error
|
|
|
|
// Use shared informer to listen for add/update/delete of Host in the specified namespace.
|
|
// Set resync period to 0, to prevent processing when nothing has changed.
|
|
informerFactory := dynamicinformer.NewFilteredDynamicSharedInformerFactory(dynamicKubeClient, 0, namespace, nil)
|
|
ambassadorHostInformer := informerFactory.ForResource(ambHostGVR)
|
|
|
|
// Add default resource event handlers to properly initialize informer.
|
|
ambassadorHostInformer.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 := newUnstructuredConverter()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to setup Unstructured Converter")
|
|
}
|
|
|
|
return &ambassadorHostSource{
|
|
dynamicKubeClient: dynamicKubeClient,
|
|
kubeClient: kubeClient,
|
|
namespace: namespace,
|
|
annotationFilter: annotationFilter,
|
|
ambassadorHostInformer: ambassadorHostInformer,
|
|
unstructuredConverter: uc,
|
|
labelSelector: labelSelector,
|
|
}, nil
|
|
}
|
|
|
|
// Endpoints returns endpoint objects for each host-target combination that should be processed.
|
|
// Retrieves all Hosts in the source's namespace(s).
|
|
func (sc *ambassadorHostSource) Endpoints(ctx context.Context) ([]*endpoint.Endpoint, error) {
|
|
hosts, err := sc.ambassadorHostInformer.Lister().ByNamespace(sc.namespace).List(sc.labelSelector)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Get a list of Ambassador Host resources
|
|
ambassadorHosts := []*ambassador.Host{}
|
|
for _, hostObj := range hosts {
|
|
unstructuredHost, ok := hostObj.(*unstructured.Unstructured)
|
|
if !ok {
|
|
return nil, errors.New("could not convert")
|
|
}
|
|
|
|
host := &ambassador.Host{}
|
|
err := sc.unstructuredConverter.scheme.Convert(unstructuredHost, host, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
ambassadorHosts = append(ambassadorHosts, host)
|
|
}
|
|
|
|
// Filter Ambassador Hosts
|
|
ambassadorHosts, err = sc.filterByAnnotations(ambassadorHosts)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to filter Ambassador Hosts by annotation")
|
|
}
|
|
|
|
endpoints := []*endpoint.Endpoint{}
|
|
|
|
for _, host := range ambassadorHosts {
|
|
fullname := fmt.Sprintf("%s/%s", host.Namespace, host.Name)
|
|
|
|
// look for the "exernal-dns.ambassador-service" annotation. If it is not there then just ignore this `Host`
|
|
service, found := host.Annotations[ambHostAnnotation]
|
|
if !found {
|
|
log.Debugf("Host %s ignored: no annotation %q found", fullname, ambHostAnnotation)
|
|
continue
|
|
}
|
|
|
|
targets := getTargetsFromTargetAnnotation(host.Annotations)
|
|
if len(targets) == 0 {
|
|
targets, err = sc.targetsFromAmbassadorLoadBalancer(ctx, service)
|
|
if err != nil {
|
|
log.Warningf("Could not find targets for service %s for Host %s: %v", service, fullname, err)
|
|
continue
|
|
}
|
|
}
|
|
|
|
hostEndpoints, err := sc.endpointsFromHost(host, targets)
|
|
if err != nil {
|
|
log.Warningf("Could not get endpoints for Host %s", err)
|
|
continue
|
|
}
|
|
if len(hostEndpoints) == 0 {
|
|
log.Debugf("No endpoints could be generated from Host %s", fullname)
|
|
continue
|
|
}
|
|
|
|
log.Debugf("Endpoints generated from Host: %s: %v", fullname, hostEndpoints)
|
|
endpoints = append(endpoints, hostEndpoints...)
|
|
}
|
|
|
|
for _, ep := range endpoints {
|
|
sort.Sort(ep.Targets)
|
|
}
|
|
|
|
return endpoints, nil
|
|
}
|
|
|
|
// endpointsFromHost extracts the endpoints from a Host object
|
|
func (sc *ambassadorHostSource) endpointsFromHost(host *ambassador.Host, targets endpoint.Targets) ([]*endpoint.Endpoint, error) {
|
|
var endpoints []*endpoint.Endpoint
|
|
annotations := host.Annotations
|
|
|
|
resource := fmt.Sprintf("host/%s/%s", host.Namespace, host.Name)
|
|
providerSpecific, setIdentifier := getProviderSpecificAnnotations(annotations)
|
|
ttl := getTTLFromAnnotations(annotations, resource)
|
|
|
|
if host.Spec != nil {
|
|
hostname := host.Spec.Hostname
|
|
if hostname != "" {
|
|
endpoints = append(endpoints, endpointsForHostname(hostname, targets, ttl, providerSpecific, setIdentifier, resource)...)
|
|
}
|
|
}
|
|
|
|
return endpoints, nil
|
|
}
|
|
|
|
func (sc *ambassadorHostSource) targetsFromAmbassadorLoadBalancer(ctx context.Context, service string) (endpoint.Targets, error) {
|
|
lbNamespace, lbName, err := parseAmbLoadBalancerService(service)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
svc, err := sc.kubeClient.CoreV1().Services(lbNamespace).Get(ctx, lbName, metav1.GetOptions{})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
targets := extractLoadBalancerTargets(svc, false)
|
|
|
|
return targets, nil
|
|
}
|
|
|
|
// parseAmbLoadBalancerService returns a name/namespace tuple from the annotation in
|
|
// an Ambassador Host CRD
|
|
//
|
|
// This is a thing because Ambassador has historically supported cross-namespace
|
|
// references using a name.namespace syntax, but here we want to also support
|
|
// namespace/name.
|
|
//
|
|
// Returns namespace, name, error.
|
|
|
|
func parseAmbLoadBalancerService(service string) (namespace, name string, err error) {
|
|
// Start by assuming that we have namespace/name.
|
|
parts := strings.Split(service, "/")
|
|
|
|
if len(parts) == 1 {
|
|
// No "/" at all, so let's try for name.namespace. To be consistent with the
|
|
// rest of Ambassador, use SplitN to limit this to one split, so that e.g.
|
|
// svc.foo.bar uses service "svc" in namespace "foo.bar".
|
|
parts = strings.SplitN(service, ".", 2)
|
|
|
|
if len(parts) == 2 {
|
|
// We got a namespace, great.
|
|
name := parts[0]
|
|
namespace := parts[1]
|
|
|
|
return namespace, name, nil
|
|
}
|
|
|
|
// If here, we have no separator, so the whole string is the service, and
|
|
// we can assume the default namespace.
|
|
name := service
|
|
namespace := "default"
|
|
|
|
return namespace, name, nil
|
|
} else if len(parts) == 2 {
|
|
// This is "namespace/name". Note that the name could be qualified,
|
|
// which is fine.
|
|
namespace := parts[0]
|
|
name := parts[1]
|
|
|
|
return namespace, name, nil
|
|
}
|
|
|
|
// If we got here, this string is simply ill-formatted. Return an error.
|
|
return "", "", errors.New(fmt.Sprintf("invalid external-dns service: %s", service))
|
|
}
|
|
|
|
func (sc *ambassadorHostSource) AddEventHandler(ctx context.Context, handler func()) {
|
|
}
|
|
|
|
// unstructuredConverter handles conversions between unstructured.Unstructured and Ambassador types
|
|
type unstructuredConverter struct {
|
|
// scheme holds an initializer for converting Unstructured to a type
|
|
scheme *runtime.Scheme
|
|
}
|
|
|
|
// newUnstructuredConverter returns a new unstructuredConverter initialized
|
|
func newUnstructuredConverter() (*unstructuredConverter, error) {
|
|
uc := &unstructuredConverter{
|
|
scheme: runtime.NewScheme(),
|
|
}
|
|
|
|
// Setup converter to understand custom CRD types
|
|
ambassador.AddToScheme(uc.scheme)
|
|
|
|
// Add the core types we need
|
|
if err := scheme.AddToScheme(uc.scheme); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return uc, nil
|
|
}
|
|
|
|
// Filter a list of Ambassador Host Resources to only return the ones that
|
|
// contain the required External-DNS annotation filter
|
|
func (sc *ambassadorHostSource) filterByAnnotations(ambassadorHosts []*ambassador.Host) ([]*ambassador.Host, error) {
|
|
// External-DNS Annotation Filter
|
|
labelSelector, err := metav1.ParseToLabelSelector(sc.annotationFilter)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
selector, err := metav1.LabelSelectorAsSelector(labelSelector)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// empty filter returns original list of Ambassador Hosts
|
|
if selector.Empty() {
|
|
return ambassadorHosts, nil
|
|
}
|
|
|
|
// Return a filtered list of Ambassador Hosts
|
|
filteredList := []*ambassador.Host{}
|
|
for _, host := range ambassadorHosts {
|
|
annotations := labels.Set(host.Annotations)
|
|
// include Ambassador Host if its annotations match the annotation filter
|
|
if selector.Matches(annotations) {
|
|
filteredList = append(filteredList, host)
|
|
}
|
|
}
|
|
|
|
return filteredList, nil
|
|
}
|