external-dns/source/pod_indexer_test.go
Ivan Ka 4fd5596601
feat(source/pods): support for annotation and label filter (#5583)
* feat(source): pods added support for annotation filter and label selectors

* feat(source/pods): support for annotation and label filter

Signed-off-by: ivan katliarchuk <ivan.katliarchuk@gmail.com>

---------

Signed-off-by: ivan katliarchuk <ivan.katliarchuk@gmail.com>
2025-07-03 09:15:34 -07:00

233 lines
6.3 KiB
Go

/*
Copyright 2025 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 (
"fmt"
"math/rand/v2"
"net"
"strconv"
"testing"
"github.com/google/uuid"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/client-go/kubernetes/fake"
"sigs.k8s.io/external-dns/source/annotations"
)
type podSpec struct {
namespace string
labels map[string]string
annotations map[string]string
// with labels and annotations
totalTarget int
// without provided labels and annotations
totalRandom int
}
func fixtureCreatePodsWithNodes(input []podSpec) []*corev1.Pod {
var pods []*corev1.Pod
var createPod = func(index int, spec podSpec) *corev1.Pod {
return &corev1.Pod{
ObjectMeta: metav1.ObjectMeta{
Name: fmt.Sprintf("pod-%d-%s", index, uuid.NewString()),
Namespace: spec.namespace,
Labels: func() map[string]string {
if spec.totalTarget > index {
return spec.labels
}
return map[string]string{
"app": fmt.Sprintf("my-app-%d", rand.IntN(10)),
"index": strconv.Itoa(index),
}
}(),
Annotations: func() map[string]string {
if spec.totalTarget > index {
return spec.annotations
}
return map[string]string{
"key1": fmt.Sprintf("value-%d", rand.IntN(10)),
}
}(),
},
Spec: corev1.PodSpec{},
Status: corev1.PodStatus{
Phase: corev1.PodRunning,
PodIPs: []corev1.PodIP{
{IP: net.IPv4(192, byte(rand.IntN(250)), byte(rand.IntN(250)), byte(index)).String()},
},
},
}
}
for _, el := range input {
totalPods := el.totalTarget + el.totalRandom
for i := 0; i < totalPods; i++ {
pods = append(pods, createPod(i, el))
}
}
for i := 0; i < 3; i++ {
rand.Shuffle(len(pods), func(i, j int) {
pods[i], pods[j] = pods[j], pods[i]
})
}
// assign nodes to pods
for i, pod := range pods {
pod.Spec.NodeName = fmt.Sprintf("node-%d", i/5) // Assign 5 pods per node
}
return pods
}
func TestPodsWithAnnotationsAndLabels(t *testing.T) {
// total target pods 700
// total random pods 3950
pods := fixtureCreatePodsWithNodes([]podSpec{
{
namespace: "dev",
labels: map[string]string{"app": "nginx", "env": "dev", "agent": "enabled"},
annotations: map[string]string{"arch": "amd64"},
totalTarget: 300,
totalRandom: 700,
},
{
namespace: "prod",
labels: map[string]string{"app": "nginx", "env": "prod", "agent": "enabled"},
annotations: map[string]string{"arch": "amd64"},
totalTarget: 150,
totalRandom: 2700,
},
{
namespace: "default",
labels: map[string]string{"app": "nginx", "agent": "disabled"},
annotations: map[string]string{"arch": "amd64"},
totalTarget: 250,
totalRandom: 450,
},
{
namespace: "kube-system",
labels: map[string]string{},
annotations: map[string]string{},
totalTarget: 0,
totalRandom: 100,
},
})
client := fake.NewClientset()
nodes := map[string]bool{}
for _, pod := range pods {
if _, exists := nodes[pod.Spec.NodeName]; !exists {
nodes[pod.Spec.NodeName] = true
node := &corev1.Node{
ObjectMeta: metav1.ObjectMeta{
Name: pod.Spec.NodeName,
},
}
if _, err := client.CoreV1().Nodes().Create(t.Context(), node, metav1.CreateOptions{}); err != nil {
assert.NoError(t, err)
}
}
if _, err := client.CoreV1().Pods(pod.Namespace).Create(t.Context(), pod, metav1.CreateOptions{}); err != nil {
assert.NoError(t, err)
}
}
tests := []struct {
name string
namespace string
labelSelector string
annotationFilter string
expectedEndpointCount int
}{
{
name: "prod namespace with labels",
namespace: "prod",
labelSelector: "app=nginx",
expectedEndpointCount: 150,
},
{
name: "prod namespace with annotations",
namespace: "prod",
annotationFilter: "arch=amd64",
expectedEndpointCount: 150,
},
{
name: "prod namespace with annotations and labels not exists",
namespace: "prod",
labelSelector: "app=not-exists",
annotationFilter: "arch=amd64",
expectedEndpointCount: 0,
},
{
name: "all namespaces with correct annotations and labels",
namespace: "",
labelSelector: "app=nginx,agent=enabled",
annotationFilter: "arch=amd64",
expectedEndpointCount: 450, // 300 from dev + 150 from prod
},
{
name: "all namespaces with loose annotations and labels",
namespace: "",
labelSelector: "app=nginx",
annotationFilter: "arch=amd64",
expectedEndpointCount: 700, // 300 from dev + 150 from prod + 250 from default
},
{
name: "all namespaces with loose annotations and labels",
namespace: "",
labelSelector: "agent",
annotationFilter: "arch",
expectedEndpointCount: 700,
},
{
name: "all namespaces without filters",
namespace: "",
labelSelector: "",
annotationFilter: "",
expectedEndpointCount: 4650,
},
{
name: "single namespace without filters",
namespace: "default",
labelSelector: "",
annotationFilter: "",
expectedEndpointCount: 700,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
selector, _ := annotations.ParseFilter(tt.labelSelector)
pSource, err := NewPodSource(
t.Context(), client,
tt.namespace, "",
false, "",
"{{ .Name }}.tld.org", false,
tt.annotationFilter, selector)
require.NoError(t, err)
endpoints, err := pSource.Endpoints(t.Context())
require.NoError(t, err)
assert.Len(t, endpoints, tt.expectedEndpointCount)
})
}
}