| /* |
| Copyright 2017 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 proxy |
| |
| import ( |
| "fmt" |
| "reflect" |
| "testing" |
| "time" |
| |
| v1 "k8s.io/api/core/v1" |
| discovery "k8s.io/api/discovery/v1" |
| metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" |
| "k8s.io/apimachinery/pkg/types" |
| "k8s.io/apimachinery/pkg/util/sets" |
| "k8s.io/utils/ptr" |
| ) |
| |
| func (proxier *FakeProxier) addEndpointSlice(slice *discovery.EndpointSlice) { |
| proxier.endpointsChanges.EndpointSliceUpdate(slice, false) |
| } |
| |
| func (proxier *FakeProxier) updateEndpointSlice(oldSlice, slice *discovery.EndpointSlice) { |
| proxier.endpointsChanges.EndpointSliceUpdate(slice, false) |
| } |
| |
| func (proxier *FakeProxier) deleteEndpointSlice(slice *discovery.EndpointSlice) { |
| proxier.endpointsChanges.EndpointSliceUpdate(slice, true) |
| } |
| |
| func TestGetLocalEndpointIPs(t *testing.T) { |
| testCases := []struct { |
| endpointsMap EndpointsMap |
| expected map[types.NamespacedName]sets.Set[string] |
| }{{ |
| // Case[0]: nothing |
| endpointsMap: EndpointsMap{}, |
| expected: map[types.NamespacedName]sets.Set[string]{}, |
| }, { |
| // Case[1]: unnamed port |
| endpointsMap: EndpointsMap{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expected: map[types.NamespacedName]sets.Set[string]{}, |
| }, { |
| // Case[2]: unnamed port local |
| endpointsMap: EndpointsMap{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expected: map[types.NamespacedName]sets.Set[string]{ |
| {Namespace: "ns1", Name: "ep1"}: sets.New[string]("1.1.1.1"), |
| }, |
| }, { |
| // Case[3]: named local and non-local ports for the same IP. |
| endpointsMap: EndpointsMap{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "1.1.1.2", port: 11, endpoint: "1.1.1.2:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "1.1.1.1", port: 12, endpoint: "1.1.1.1:12", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "1.1.1.2", port: 12, endpoint: "1.1.1.2:12", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expected: map[types.NamespacedName]sets.Set[string]{ |
| {Namespace: "ns1", Name: "ep1"}: sets.New[string]("1.1.1.2"), |
| }, |
| }, { |
| // Case[4]: named local and non-local ports for different IPs. |
| endpointsMap: EndpointsMap{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns2", "ep2", "p22", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "2.2.2.2", port: 22, endpoint: "2.2.2.2:22", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "2.2.2.22", port: 22, endpoint: "2.2.2.22:22", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns2", "ep2", "p23", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "2.2.2.3", port: 23, endpoint: "2.2.2.3:23", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns4", "ep4", "p44", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "4.4.4.4", port: 44, endpoint: "4.4.4.4:44", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "4.4.4.5", port: 44, endpoint: "4.4.4.5:44", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns4", "ep4", "p45", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "4.4.4.6", port: 45, endpoint: "4.4.4.6:45", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expected: map[types.NamespacedName]sets.Set[string]{ |
| {Namespace: "ns2", Name: "ep2"}: sets.New[string]("2.2.2.2", "2.2.2.22", "2.2.2.3"), |
| {Namespace: "ns4", Name: "ep4"}: sets.New[string]("4.4.4.4", "4.4.4.6"), |
| }, |
| }, { |
| // Case[5]: named local and non-local ports for different IPs, some not ready. |
| endpointsMap: EndpointsMap{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns2", "ep2", "p22", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "2.2.2.2", port: 22, endpoint: "2.2.2.2:22", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "2.2.2.22", port: 22, endpoint: "2.2.2.22:22", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns2", "ep2", "p23", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "2.2.2.3", port: 23, endpoint: "2.2.2.3:23", isLocal: true, ready: false, serving: true, terminating: true}, |
| }, |
| makeServicePortName("ns4", "ep4", "p44", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "4.4.4.4", port: 44, endpoint: "4.4.4.4:44", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "4.4.4.5", port: 44, endpoint: "4.4.4.5:44", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns4", "ep4", "p45", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "4.4.4.6", port: 45, endpoint: "4.4.4.6:45", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expected: map[types.NamespacedName]sets.Set[string]{ |
| {Namespace: "ns2", Name: "ep2"}: sets.New[string]("2.2.2.2", "2.2.2.22"), |
| {Namespace: "ns4", Name: "ep4"}: sets.New[string]("4.4.4.4", "4.4.4.6"), |
| }, |
| }, { |
| // Case[6]: all endpoints are terminating,, so getLocalReadyEndpointIPs should return 0 ready endpoints |
| endpointsMap: EndpointsMap{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: false, serving: true, terminating: true}, |
| }, |
| makeServicePortName("ns2", "ep2", "p22", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "2.2.2.2", port: 22, endpoint: "2.2.2.2:22", isLocal: true, ready: false, serving: true, terminating: true}, |
| &BaseEndpointInfo{ip: "2.2.2.22", port: 22, endpoint: "2.2.2.22:22", isLocal: true, ready: false, serving: true, terminating: true}, |
| }, |
| makeServicePortName("ns2", "ep2", "p23", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "2.2.2.3", port: 23, endpoint: "2.2.2.3:23", isLocal: true, ready: false, serving: true, terminating: true}, |
| }, |
| makeServicePortName("ns4", "ep4", "p44", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "4.4.4.4", port: 44, endpoint: "4.4.4.4:44", isLocal: true, ready: false, serving: true, terminating: true}, |
| &BaseEndpointInfo{ip: "4.4.4.5", port: 44, endpoint: "4.4.4.5:44", isLocal: false, ready: false, serving: true, terminating: true}, |
| }, |
| makeServicePortName("ns4", "ep4", "p45", v1.ProtocolTCP): []Endpoint{ |
| &BaseEndpointInfo{ip: "4.4.4.6", port: 45, endpoint: "4.4.4.6:45", isLocal: true, ready: false, serving: true, terminating: true}, |
| }, |
| }, |
| expected: make(map[types.NamespacedName]sets.Set[string], 0), |
| }} |
| |
| for tci, tc := range testCases { |
| // outputs |
| localIPs := tc.endpointsMap.getLocalReadyEndpointIPs() |
| |
| if !reflect.DeepEqual(localIPs, tc.expected) { |
| t.Errorf("[%d] expected %#v, got %#v", tci, tc.expected, localIPs) |
| } |
| } |
| } |
| |
| func makeTestEndpointSlice(namespace, name string, slice int, epsFunc func(*discovery.EndpointSlice)) *discovery.EndpointSlice { |
| eps := &discovery.EndpointSlice{ |
| ObjectMeta: metav1.ObjectMeta{ |
| Name: fmt.Sprintf("%s-%d", name, slice), |
| Namespace: namespace, |
| Annotations: map[string]string{}, |
| Labels: map[string]string{ |
| discovery.LabelServiceName: name, |
| }, |
| }, |
| AddressType: discovery.AddressTypeIPv4, |
| } |
| epsFunc(eps) |
| return eps |
| } |
| |
| func TestUpdateEndpointsMap(t *testing.T) { |
| emptyEndpoint := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{} |
| } |
| unnamedPort := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To(""), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| unnamedPortReady := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| Conditions: discovery.EndpointConditions{ |
| Ready: ptr.To(true), |
| Serving: ptr.To(true), |
| Terminating: ptr.To(false), |
| }, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To(""), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| unnamedPortTerminating := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| Conditions: discovery.EndpointConditions{ |
| Ready: ptr.To(false), |
| Serving: ptr.To(true), |
| Terminating: ptr.To(true), |
| }, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To(""), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| unnamedPortLocal := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To(""), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| namedPortLocal := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| namedPort := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| namedPortRenamed := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11-2"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| namedPortRenumbered := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](22), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| namedPortsLocalNoLocal := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }, { |
| Addresses: []string{"1.1.1.2"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }, { |
| Name: ptr.To("p12"), |
| Port: ptr.To[int32](12), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| multipleSubsets_s1 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| multipleSubsets_s2 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.2"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p12"), |
| Port: ptr.To[int32](12), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| multipleSubsetsWithLocal_s1 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| multipleSubsetsWithLocal_s2 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.2"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p12"), |
| Port: ptr.To[int32](12), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| multipleSubsetsMultiplePortsLocal_s1 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }, { |
| Name: ptr.To("p12"), |
| Port: ptr.To[int32](12), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| multipleSubsetsMultiplePortsLocal_s2 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.3"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p13"), |
| Port: ptr.To[int32](13), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| multipleSubsetsIPsPorts1_s1 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }, { |
| Addresses: []string{"1.1.1.2"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }, { |
| Name: ptr.To("p12"), |
| Port: ptr.To[int32](12), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| multipleSubsetsIPsPorts1_s2 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.3"}, |
| }, { |
| Addresses: []string{"1.1.1.4"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p13"), |
| Port: ptr.To[int32](13), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }, { |
| Name: ptr.To("p14"), |
| Port: ptr.To[int32](14), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| multipleSubsetsIPsPorts2 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"2.2.2.1"}, |
| }, { |
| Addresses: []string{"2.2.2.2"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p21"), |
| Port: ptr.To[int32](21), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }, { |
| Name: ptr.To("p22"), |
| Port: ptr.To[int32](22), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| complexBefore1 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| complexBefore2_s1 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"2.2.2.2"}, |
| NodeName: ptr.To(testHostname), |
| }, { |
| Addresses: []string{"2.2.2.22"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p22"), |
| Port: ptr.To[int32](22), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| complexBefore2_s2 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"2.2.2.3"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p23"), |
| Port: ptr.To[int32](23), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| complexBefore4_s1 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"4.4.4.4"}, |
| NodeName: ptr.To(testHostname), |
| }, { |
| Addresses: []string{"4.4.4.5"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p44"), |
| Port: ptr.To[int32](44), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| complexBefore4_s2 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"4.4.4.6"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p45"), |
| Port: ptr.To[int32](45), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| complexAfter1_s1 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }, { |
| Addresses: []string{"1.1.1.11"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| complexAfter1_s2 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.2"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p12"), |
| Port: ptr.To[int32](12), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }, { |
| Name: ptr.To("p122"), |
| Port: ptr.To[int32](122), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| complexAfter3 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"3.3.3.3"}, |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p33"), |
| Port: ptr.To[int32](33), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| complexAfter4 := func(eps *discovery.EndpointSlice) { |
| eps.Endpoints = []discovery.Endpoint{{ |
| Addresses: []string{"4.4.4.4"}, |
| NodeName: ptr.To(testHostname), |
| }} |
| eps.Ports = []discovery.EndpointPort{{ |
| Name: ptr.To("p44"), |
| Port: ptr.To[int32](44), |
| Protocol: ptr.To(v1.ProtocolUDP), |
| }} |
| } |
| |
| testCases := []struct { |
| // previousEndpointSlices and currentEndpointSlices are used to call appropriate |
| // handlers OnEndpointSlice* (based on whether corresponding values are nil |
| // or non-nil) and must be of equal length. |
| name string |
| previousEndpointSlices []*discovery.EndpointSlice |
| currentEndpointSlices []*discovery.EndpointSlice |
| previousEndpointsMap map[ServicePortName][]*BaseEndpointInfo |
| expectedResult map[ServicePortName][]*BaseEndpointInfo |
| expectedDeletedUDPEndpoints []ServiceEndpoint |
| expectedNewlyActiveUDPServices map[ServicePortName]bool |
| expectedLocalEndpoints map[types.NamespacedName]int |
| expectedChangedEndpoints sets.Set[types.NamespacedName] |
| }{{ |
| name: "empty", |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{}, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{}, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New[types.NamespacedName](), |
| }, { |
| name: "no change, unnamed port", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, unnamedPort), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, unnamedPort), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New[types.NamespacedName](), |
| }, { |
| name: "no change, named port, local", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPortLocal), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPortLocal), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{ |
| makeNSN("ns1", "ep1"): 1, |
| }, |
| expectedChangedEndpoints: sets.New[types.NamespacedName](), |
| }, { |
| name: "no change, multiple slices", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, multipleSubsets_s1), |
| makeTestEndpointSlice("ns1", "ep1", 2, multipleSubsets_s2), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, multipleSubsets_s1), |
| makeTestEndpointSlice("ns1", "ep1", 2, multipleSubsets_s2), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.2", port: 12, endpoint: "1.1.1.2:12", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.2", port: 12, endpoint: "1.1.1.2:12", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New[types.NamespacedName](), |
| }, { |
| name: "no change, multiple slices, multiple ports, local", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, multipleSubsetsMultiplePortsLocal_s1), |
| makeTestEndpointSlice("ns1", "ep1", 2, multipleSubsetsMultiplePortsLocal_s2), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, multipleSubsetsMultiplePortsLocal_s1), |
| makeTestEndpointSlice("ns1", "ep1", 2, multipleSubsetsMultiplePortsLocal_s2), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 12, endpoint: "1.1.1.1:12", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p13", v1.ProtocolUDP): { |
| {ip: "1.1.1.3", port: 13, endpoint: "1.1.1.3:13", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 12, endpoint: "1.1.1.1:12", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p13", v1.ProtocolUDP): { |
| {ip: "1.1.1.3", port: 13, endpoint: "1.1.1.3:13", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{ |
| makeNSN("ns1", "ep1"): 1, |
| }, |
| expectedChangedEndpoints: sets.New[types.NamespacedName](), |
| }, { |
| name: "no change, multiple services, slices, IPs, and ports", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, multipleSubsetsIPsPorts1_s1), |
| makeTestEndpointSlice("ns1", "ep1", 2, multipleSubsetsIPsPorts1_s2), |
| makeTestEndpointSlice("ns2", "ep2", 1, multipleSubsetsIPsPorts2), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, multipleSubsetsIPsPorts1_s1), |
| makeTestEndpointSlice("ns1", "ep1", 2, multipleSubsetsIPsPorts1_s2), |
| makeTestEndpointSlice("ns2", "ep2", 1, multipleSubsetsIPsPorts2), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.2", port: 11, endpoint: "1.1.1.2:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 12, endpoint: "1.1.1.1:12", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.2", port: 12, endpoint: "1.1.1.2:12", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p13", v1.ProtocolUDP): { |
| {ip: "1.1.1.3", port: 13, endpoint: "1.1.1.3:13", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.4", port: 13, endpoint: "1.1.1.4:13", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p14", v1.ProtocolUDP): { |
| {ip: "1.1.1.3", port: 14, endpoint: "1.1.1.3:14", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.4", port: 14, endpoint: "1.1.1.4:14", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns2", "ep2", "p21", v1.ProtocolUDP): { |
| {ip: "2.2.2.1", port: 21, endpoint: "2.2.2.1:21", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "2.2.2.2", port: 21, endpoint: "2.2.2.2:21", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns2", "ep2", "p22", v1.ProtocolUDP): { |
| {ip: "2.2.2.1", port: 22, endpoint: "2.2.2.1:22", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "2.2.2.2", port: 22, endpoint: "2.2.2.2:22", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.2", port: 11, endpoint: "1.1.1.2:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 12, endpoint: "1.1.1.1:12", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.2", port: 12, endpoint: "1.1.1.2:12", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p13", v1.ProtocolUDP): { |
| {ip: "1.1.1.3", port: 13, endpoint: "1.1.1.3:13", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.4", port: 13, endpoint: "1.1.1.4:13", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p14", v1.ProtocolUDP): { |
| {ip: "1.1.1.3", port: 14, endpoint: "1.1.1.3:14", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.4", port: 14, endpoint: "1.1.1.4:14", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns2", "ep2", "p21", v1.ProtocolUDP): { |
| {ip: "2.2.2.1", port: 21, endpoint: "2.2.2.1:21", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "2.2.2.2", port: 21, endpoint: "2.2.2.2:21", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns2", "ep2", "p22", v1.ProtocolUDP): { |
| {ip: "2.2.2.1", port: 22, endpoint: "2.2.2.1:22", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "2.2.2.2", port: 22, endpoint: "2.2.2.2:22", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{ |
| makeNSN("ns1", "ep1"): 2, |
| makeNSN("ns2", "ep2"): 1, |
| }, |
| expectedChangedEndpoints: sets.New[types.NamespacedName](), |
| }, { |
| name: "add an EndpointSlice", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| nil, |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, unnamedPortLocal), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{}, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP): true, |
| }, |
| expectedLocalEndpoints: map[types.NamespacedName]int{ |
| makeNSN("ns1", "ep1"): 1, |
| }, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, { |
| name: "remove an EndpointSlice", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, unnamedPortLocal), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| nil, |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{}, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{{ |
| Endpoint: "1.1.1.1:11", |
| ServicePortName: makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP), |
| }}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, { |
| name: "add an IP and port", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPort), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPortsLocalNoLocal), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.2", port: 11, endpoint: "1.1.1.2:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 12, endpoint: "1.1.1.1:12", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.2", port: 12, endpoint: "1.1.1.2:12", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{ |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): true, |
| }, |
| expectedLocalEndpoints: map[types.NamespacedName]int{ |
| makeNSN("ns1", "ep1"): 1, |
| }, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, { |
| name: "remove an IP and port", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPortsLocalNoLocal), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPort), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.2", port: 11, endpoint: "1.1.1.2:11", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 12, endpoint: "1.1.1.1:12", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.2", port: 12, endpoint: "1.1.1.2:12", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{{ |
| Endpoint: "1.1.1.2:11", |
| ServicePortName: makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP), |
| }, { |
| Endpoint: "1.1.1.1:12", |
| ServicePortName: makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP), |
| }, { |
| Endpoint: "1.1.1.2:12", |
| ServicePortName: makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP), |
| }}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, { |
| name: "add a slice to an endpoint", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPort), |
| nil, |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, multipleSubsetsWithLocal_s1), |
| makeTestEndpointSlice("ns1", "ep1", 2, multipleSubsetsWithLocal_s2), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.2", port: 12, endpoint: "1.1.1.2:12", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{ |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): true, |
| }, |
| expectedLocalEndpoints: map[types.NamespacedName]int{ |
| makeNSN("ns1", "ep1"): 1, |
| }, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, { |
| name: "remove a slice from an endpoint", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, multipleSubsets_s1), |
| makeTestEndpointSlice("ns1", "ep1", 2, multipleSubsets_s2), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPort), |
| nil, |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.2", port: 12, endpoint: "1.1.1.2:12", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{{ |
| Endpoint: "1.1.1.2:12", |
| ServicePortName: makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP), |
| }}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, { |
| name: "rename a port", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPort), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPortRenamed), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11-2", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{{ |
| Endpoint: "1.1.1.1:11", |
| ServicePortName: makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP), |
| }}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{ |
| makeServicePortName("ns1", "ep1", "p11-2", v1.ProtocolUDP): true, |
| }, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, { |
| name: "renumber a port", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPort), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, namedPortRenumbered), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 22, endpoint: "1.1.1.1:22", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{{ |
| Endpoint: "1.1.1.1:11", |
| ServicePortName: makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP), |
| }}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, { |
| name: "complex add and remove", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, complexBefore1), |
| nil, |
| |
| makeTestEndpointSlice("ns2", "ep2", 1, complexBefore2_s1), |
| makeTestEndpointSlice("ns2", "ep2", 2, complexBefore2_s2), |
| |
| nil, |
| nil, |
| |
| makeTestEndpointSlice("ns4", "ep4", 1, complexBefore4_s1), |
| makeTestEndpointSlice("ns4", "ep4", 2, complexBefore4_s2), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, complexAfter1_s1), |
| makeTestEndpointSlice("ns1", "ep1", 2, complexAfter1_s2), |
| |
| nil, |
| nil, |
| |
| makeTestEndpointSlice("ns3", "ep3", 1, complexAfter3), |
| nil, |
| |
| makeTestEndpointSlice("ns4", "ep4", 1, complexAfter4), |
| nil, |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns2", "ep2", "p22", v1.ProtocolUDP): { |
| {ip: "2.2.2.22", port: 22, endpoint: "2.2.2.22:22", isLocal: true, ready: true, serving: true, terminating: false}, |
| {ip: "2.2.2.2", port: 22, endpoint: "2.2.2.2:22", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns2", "ep2", "p23", v1.ProtocolUDP): { |
| {ip: "2.2.2.3", port: 23, endpoint: "2.2.2.3:23", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns4", "ep4", "p44", v1.ProtocolUDP): { |
| {ip: "4.4.4.4", port: 44, endpoint: "4.4.4.4:44", isLocal: true, ready: true, serving: true, terminating: false}, |
| {ip: "4.4.4.5", port: 44, endpoint: "4.4.4.5:44", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns4", "ep4", "p45", v1.ProtocolUDP): { |
| {ip: "4.4.4.6", port: 45, endpoint: "4.4.4.6:45", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "p11", v1.ProtocolUDP): { |
| {ip: "1.1.1.11", port: 11, endpoint: "1.1.1.11:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): { |
| {ip: "1.1.1.2", port: 12, endpoint: "1.1.1.2:12", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "ep1", "p122", v1.ProtocolUDP): { |
| {ip: "1.1.1.2", port: 122, endpoint: "1.1.1.2:122", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns3", "ep3", "p33", v1.ProtocolUDP): { |
| {ip: "3.3.3.3", port: 33, endpoint: "3.3.3.3:33", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns4", "ep4", "p44", v1.ProtocolUDP): { |
| {ip: "4.4.4.4", port: 44, endpoint: "4.4.4.4:44", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{{ |
| Endpoint: "2.2.2.2:22", |
| ServicePortName: makeServicePortName("ns2", "ep2", "p22", v1.ProtocolUDP), |
| }, { |
| Endpoint: "2.2.2.22:22", |
| ServicePortName: makeServicePortName("ns2", "ep2", "p22", v1.ProtocolUDP), |
| }, { |
| Endpoint: "2.2.2.3:23", |
| ServicePortName: makeServicePortName("ns2", "ep2", "p23", v1.ProtocolUDP), |
| }, { |
| Endpoint: "4.4.4.5:44", |
| ServicePortName: makeServicePortName("ns4", "ep4", "p44", v1.ProtocolUDP), |
| }, { |
| Endpoint: "4.4.4.6:45", |
| ServicePortName: makeServicePortName("ns4", "ep4", "p45", v1.ProtocolUDP), |
| }}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{ |
| makeServicePortName("ns1", "ep1", "p12", v1.ProtocolUDP): true, |
| makeServicePortName("ns1", "ep1", "p122", v1.ProtocolUDP): true, |
| makeServicePortName("ns3", "ep3", "p33", v1.ProtocolUDP): true, |
| }, |
| expectedLocalEndpoints: map[types.NamespacedName]int{ |
| makeNSN("ns4", "ep4"): 1, |
| }, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1"), makeNSN("ns2", "ep2"), makeNSN("ns3", "ep3"), makeNSN("ns4", "ep4")), |
| }, { |
| name: "change from 0 endpoint address to 1 unnamed port", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, emptyEndpoint), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, unnamedPort), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{}, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP): true, |
| }, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, { |
| name: "change from ready to terminating pod", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, unnamedPortReady), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, unnamedPortTerminating), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: false, serving: true, terminating: true}, |
| }, |
| }, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, { |
| name: "change from terminating to empty pod", |
| previousEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, unnamedPortTerminating), |
| }, |
| currentEndpointSlices: []*discovery.EndpointSlice{ |
| makeTestEndpointSlice("ns1", "ep1", 1, emptyEndpoint), |
| }, |
| previousEndpointsMap: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP): { |
| {ip: "1.1.1.1", port: 11, endpoint: "1.1.1.1:11", isLocal: false, ready: false, serving: true, terminating: true}, |
| }, |
| }, |
| expectedResult: map[ServicePortName][]*BaseEndpointInfo{}, |
| expectedDeletedUDPEndpoints: []ServiceEndpoint{{ |
| Endpoint: "1.1.1.1:11", |
| ServicePortName: makeServicePortName("ns1", "ep1", "", v1.ProtocolUDP), |
| }}, |
| expectedNewlyActiveUDPServices: map[ServicePortName]bool{}, |
| expectedLocalEndpoints: map[types.NamespacedName]int{}, |
| expectedChangedEndpoints: sets.New(makeNSN("ns1", "ep1")), |
| }, |
| } |
| |
| for tci, tc := range testCases { |
| t.Run(tc.name, func(t *testing.T) { |
| fp := newFakeProxier(v1.IPv4Protocol, time.Time{}) |
| fp.hostname = testHostname |
| |
| // First check that after adding all previous versions of endpoints, |
| // the fp.previousEndpointsMap is as we expect. |
| for i := range tc.previousEndpointSlices { |
| if tc.previousEndpointSlices[i] != nil { |
| fp.addEndpointSlice(tc.previousEndpointSlices[i]) |
| } |
| } |
| fp.endpointsMap.Update(fp.endpointsChanges) |
| compareEndpointsMapsStr(t, fp.endpointsMap, tc.previousEndpointsMap) |
| |
| // Now let's call appropriate handlers to get to state we want to be. |
| if len(tc.previousEndpointSlices) != len(tc.currentEndpointSlices) { |
| t.Fatalf("[%d] different lengths of previous and current endpoints", tci) |
| return |
| } |
| |
| for i := range tc.previousEndpointSlices { |
| prev, curr := tc.previousEndpointSlices[i], tc.currentEndpointSlices[i] |
| switch { |
| case prev == nil && curr == nil: |
| continue |
| case prev == nil: |
| fp.addEndpointSlice(curr) |
| case curr == nil: |
| fp.deleteEndpointSlice(prev) |
| default: |
| fp.updateEndpointSlice(prev, curr) |
| } |
| } |
| |
| result := fp.endpointsMap.Update(fp.endpointsChanges) |
| newMap := fp.endpointsMap |
| compareEndpointsMapsStr(t, newMap, tc.expectedResult) |
| if !result.UpdatedServices.Equal(tc.expectedChangedEndpoints) { |
| t.Errorf("[%d] expected changed endpoints %q, got %q", tci, tc.expectedChangedEndpoints.UnsortedList(), result.UpdatedServices.UnsortedList()) |
| } |
| if len(result.DeletedUDPEndpoints) != len(tc.expectedDeletedUDPEndpoints) { |
| t.Errorf("[%d] expected %d staleEndpoints, got %d: %v", tci, len(tc.expectedDeletedUDPEndpoints), len(result.DeletedUDPEndpoints), result.DeletedUDPEndpoints) |
| } |
| for _, x := range tc.expectedDeletedUDPEndpoints { |
| found := false |
| for _, stale := range result.DeletedUDPEndpoints { |
| if stale == x { |
| found = true |
| break |
| } |
| } |
| if !found { |
| t.Errorf("[%d] expected staleEndpoints[%v], but didn't find it: %v", tci, x, result.DeletedUDPEndpoints) |
| } |
| } |
| if len(result.NewlyActiveUDPServices) != len(tc.expectedNewlyActiveUDPServices) { |
| t.Errorf("[%d] expected %d newlyActiveUDPServices, got %d: %v", tci, len(tc.expectedNewlyActiveUDPServices), len(result.NewlyActiveUDPServices), result.NewlyActiveUDPServices) |
| } |
| for svcName := range tc.expectedNewlyActiveUDPServices { |
| found := false |
| for _, newSvcName := range result.NewlyActiveUDPServices { |
| if newSvcName == svcName { |
| found = true |
| } |
| } |
| if !found { |
| t.Errorf("[%d] expected newlyActiveUDPServices[%v], but didn't find it: %v", tci, svcName, result.NewlyActiveUDPServices) |
| } |
| } |
| |
| localReadyEndpoints := fp.endpointsMap.LocalReadyEndpoints() |
| if !reflect.DeepEqual(localReadyEndpoints, tc.expectedLocalEndpoints) { |
| t.Errorf("[%d] expected local ready endpoints %v, got %v", tci, tc.expectedLocalEndpoints, localReadyEndpoints) |
| } |
| }) |
| } |
| } |
| |
| func TestLastChangeTriggerTime(t *testing.T) { |
| startTime := time.Date(2018, 01, 01, 0, 0, 0, 0, time.UTC) |
| t_1 := startTime.Add(-time.Second) |
| t0 := startTime.Add(time.Second) |
| t1 := t0.Add(time.Second) |
| t2 := t1.Add(time.Second) |
| t3 := t2.Add(time.Second) |
| |
| createEndpoints := func(namespace, name string, triggerTime time.Time) *discovery.EndpointSlice { |
| return &discovery.EndpointSlice{ |
| ObjectMeta: metav1.ObjectMeta{ |
| Name: name, |
| Namespace: namespace, |
| Annotations: map[string]string{ |
| v1.EndpointsLastChangeTriggerTime: triggerTime.Format(time.RFC3339Nano), |
| }, |
| Labels: map[string]string{ |
| discovery.LabelServiceName: name, |
| }, |
| }, |
| AddressType: discovery.AddressTypeIPv4, |
| Endpoints: []discovery.Endpoint{{ |
| Addresses: []string{"1.1.1.1"}, |
| }}, |
| Ports: []discovery.EndpointPort{{ |
| Name: ptr.To("p11"), |
| Port: ptr.To[int32](11), |
| Protocol: ptr.To(v1.ProtocolTCP), |
| }}, |
| } |
| } |
| |
| createName := func(namespace, name string) types.NamespacedName { |
| return types.NamespacedName{Namespace: namespace, Name: name} |
| } |
| |
| modifyEndpoints := func(slice *discovery.EndpointSlice, triggerTime time.Time) *discovery.EndpointSlice { |
| e := slice.DeepCopy() |
| (*e.Ports[0].Port)++ |
| e.Annotations[v1.EndpointsLastChangeTriggerTime] = triggerTime.Format(time.RFC3339Nano) |
| return e |
| } |
| |
| testCases := []struct { |
| name string |
| scenario func(fp *FakeProxier) |
| expected map[types.NamespacedName][]time.Time |
| }{ |
| { |
| name: "Single addEndpoints", |
| scenario: func(fp *FakeProxier) { |
| e := createEndpoints("ns", "ep1", t0) |
| fp.addEndpointSlice(e) |
| }, |
| expected: map[types.NamespacedName][]time.Time{createName("ns", "ep1"): {t0}}, |
| }, |
| { |
| name: "addEndpoints then updatedEndpoints", |
| scenario: func(fp *FakeProxier) { |
| e := createEndpoints("ns", "ep1", t0) |
| fp.addEndpointSlice(e) |
| |
| e1 := modifyEndpoints(e, t1) |
| fp.updateEndpointSlice(e, e1) |
| }, |
| expected: map[types.NamespacedName][]time.Time{createName("ns", "ep1"): {t0, t1}}, |
| }, |
| { |
| name: "Add two endpoints then modify one", |
| scenario: func(fp *FakeProxier) { |
| e1 := createEndpoints("ns", "ep1", t1) |
| fp.addEndpointSlice(e1) |
| |
| e2 := createEndpoints("ns", "ep2", t2) |
| fp.addEndpointSlice(e2) |
| |
| e11 := modifyEndpoints(e1, t3) |
| fp.updateEndpointSlice(e1, e11) |
| }, |
| expected: map[types.NamespacedName][]time.Time{createName("ns", "ep1"): {t1, t3}, createName("ns", "ep2"): {t2}}, |
| }, |
| { |
| name: "Endpoints without annotation set", |
| scenario: func(fp *FakeProxier) { |
| e := createEndpoints("ns", "ep1", t1) |
| delete(e.Annotations, v1.EndpointsLastChangeTriggerTime) |
| fp.addEndpointSlice(e) |
| }, |
| expected: map[types.NamespacedName][]time.Time{}, |
| }, |
| { |
| name: "Endpoints create before tracker started", |
| scenario: func(fp *FakeProxier) { |
| e := createEndpoints("ns", "ep1", t_1) |
| fp.addEndpointSlice(e) |
| }, |
| expected: map[types.NamespacedName][]time.Time{}, |
| }, |
| { |
| name: "addEndpoints then deleteEndpoints", |
| scenario: func(fp *FakeProxier) { |
| e := createEndpoints("ns", "ep1", t1) |
| fp.addEndpointSlice(e) |
| fp.deleteEndpointSlice(e) |
| }, |
| expected: map[types.NamespacedName][]time.Time{}, |
| }, |
| { |
| name: "add then delete then add again", |
| scenario: func(fp *FakeProxier) { |
| e := createEndpoints("ns", "ep1", t1) |
| fp.addEndpointSlice(e) |
| fp.deleteEndpointSlice(e) |
| e = modifyEndpoints(e, t2) |
| fp.addEndpointSlice(e) |
| }, |
| expected: map[types.NamespacedName][]time.Time{createName("ns", "ep1"): {t2}}, |
| }, |
| { |
| name: "delete", |
| scenario: func(fp *FakeProxier) { |
| e := createEndpoints("ns", "ep1", t1) |
| fp.deleteEndpointSlice(e) |
| }, |
| expected: map[types.NamespacedName][]time.Time{}, |
| }, |
| } |
| |
| for _, tc := range testCases { |
| fp := newFakeProxier(v1.IPv4Protocol, startTime) |
| |
| tc.scenario(fp) |
| |
| result := fp.endpointsMap.Update(fp.endpointsChanges) |
| got := result.LastChangeTriggerTimes |
| |
| if !reflect.DeepEqual(got, tc.expected) { |
| t.Errorf("%s: Invalid LastChangeTriggerTimes, expected: %v, got: %v", |
| tc.name, tc.expected, result.LastChangeTriggerTimes) |
| } |
| } |
| } |
| |
| func TestEndpointSliceUpdate(t *testing.T) { |
| fqdnSlice := generateEndpointSlice("svc1", "ns1", 2, 5, 999, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}) |
| fqdnSlice.AddressType = discovery.AddressTypeFQDN |
| |
| testCases := map[string]struct { |
| startingSlices []*discovery.EndpointSlice |
| endpointsChangeTracker *EndpointsChangeTracker |
| namespacedName types.NamespacedName |
| paramEndpointSlice *discovery.EndpointSlice |
| paramRemoveSlice bool |
| expectedReturnVal bool |
| expectedCurrentChange map[ServicePortName][]*BaseEndpointInfo |
| }{ |
| // test starting from an empty state |
| "add a simple slice that doesn't already exist": { |
| startingSlices: []*discovery.EndpointSlice{}, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: generateEndpointSlice("svc1", "ns1", 1, 3, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| paramRemoveSlice: false, |
| expectedReturnVal: true, |
| expectedCurrentChange: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "svc1", "port-0", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 80, endpoint: "10.0.1.1:80", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 80, endpoint: "10.0.1.2:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 80, endpoint: "10.0.1.3:80", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "svc1", "port-1", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 443, endpoint: "10.0.1.1:443", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 443, endpoint: "10.0.1.2:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 443, endpoint: "10.0.1.3:443", isLocal: false, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| }, |
| // test no modification to state - current change should be nil as nothing changes |
| "add the same slice that already exists": { |
| startingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: generateEndpointSlice("svc1", "ns1", 1, 3, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| paramRemoveSlice: false, |
| expectedReturnVal: false, |
| expectedCurrentChange: nil, |
| }, |
| // ensure that only valide address types are processed |
| "add an FQDN slice (invalid address type)": { |
| startingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: fqdnSlice, |
| paramRemoveSlice: false, |
| expectedReturnVal: false, |
| expectedCurrentChange: nil, |
| }, |
| // test additions to existing state |
| "add a slice that overlaps with existing state": { |
| startingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| generateEndpointSlice("svc1", "ns1", 2, 2, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: generateEndpointSlice("svc1", "ns1", 1, 5, 999, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| paramRemoveSlice: false, |
| expectedReturnVal: true, |
| expectedCurrentChange: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "svc1", "port-0", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 80, endpoint: "10.0.1.1:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 80, endpoint: "10.0.1.2:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 80, endpoint: "10.0.1.3:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.4", port: 80, endpoint: "10.0.1.4:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.5", port: 80, endpoint: "10.0.1.5:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.1", port: 80, endpoint: "10.0.2.1:80", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.2", port: 80, endpoint: "10.0.2.2:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "svc1", "port-1", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 443, endpoint: "10.0.1.1:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 443, endpoint: "10.0.1.2:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 443, endpoint: "10.0.1.3:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.4", port: 443, endpoint: "10.0.1.4:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.5", port: 443, endpoint: "10.0.1.5:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.1", port: 443, endpoint: "10.0.2.1:443", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.2", port: 443, endpoint: "10.0.2.2:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| }, |
| // test additions to existing state with partially overlapping slices and ports |
| "add a slice that overlaps with existing state and partial ports": { |
| startingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| generateEndpointSlice("svc1", "ns1", 2, 2, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: generateEndpointSliceWithOffset("svc1", "ns1", 3, 1, 5, 999, 999, []string{"host1"}, []*int32{ptr.To[int32](80)}), |
| paramRemoveSlice: false, |
| expectedReturnVal: true, |
| expectedCurrentChange: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "svc1", "port-0", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 80, endpoint: "10.0.1.1:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 80, endpoint: "10.0.1.2:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 80, endpoint: "10.0.1.3:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.4", port: 80, endpoint: "10.0.1.4:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.5", port: 80, endpoint: "10.0.1.5:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.1", port: 80, endpoint: "10.0.2.1:80", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.2", port: 80, endpoint: "10.0.2.2:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "svc1", "port-1", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 443, endpoint: "10.0.1.1:443", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 443, endpoint: "10.0.1.2:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 443, endpoint: "10.0.1.3:443", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.1", port: 443, endpoint: "10.0.2.1:443", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.2", port: 443, endpoint: "10.0.2.2:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| }, |
| // test deletions from existing state with partially overlapping slices and ports |
| "remove a slice that overlaps with existing state": { |
| startingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| generateEndpointSlice("svc1", "ns1", 2, 2, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: generateEndpointSlice("svc1", "ns1", 1, 5, 999, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| paramRemoveSlice: true, |
| expectedReturnVal: true, |
| expectedCurrentChange: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "svc1", "port-0", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.2.1", port: 80, endpoint: "10.0.2.1:80", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.2", port: 80, endpoint: "10.0.2.2:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "svc1", "port-1", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.2.1", port: 443, endpoint: "10.0.2.1:443", isLocal: false, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.2", port: 443, endpoint: "10.0.2.2:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| }, |
| // ensure a removal that has no effect turns into a no-op |
| "remove a slice that doesn't even exist in current state": { |
| startingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 5, 999, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| generateEndpointSlice("svc1", "ns1", 2, 2, 999, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: generateEndpointSlice("svc1", "ns1", 3, 5, 999, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| paramRemoveSlice: true, |
| expectedReturnVal: false, |
| expectedCurrentChange: nil, |
| }, |
| // start with all endpoints ready, transition to no endpoints ready |
| "transition all endpoints to unready state": { |
| startingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 999, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: generateEndpointSlice("svc1", "ns1", 1, 3, 1, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| paramRemoveSlice: false, |
| expectedReturnVal: true, |
| expectedCurrentChange: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "svc1", "port-0", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 80, endpoint: "10.0.1.1:80", isLocal: true, ready: false, serving: false, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 80, endpoint: "10.0.1.2:80", isLocal: true, ready: false, serving: false, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 80, endpoint: "10.0.1.3:80", isLocal: true, ready: false, serving: false, terminating: false}, |
| }, |
| makeServicePortName("ns1", "svc1", "port-1", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 443, endpoint: "10.0.1.1:443", isLocal: true, ready: false, serving: false, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 443, endpoint: "10.0.1.2:443", isLocal: true, ready: false, serving: false, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 443, endpoint: "10.0.1.3:443", isLocal: true, ready: false, serving: false, terminating: false}, |
| }, |
| }, |
| }, |
| // start with no endpoints ready, transition to all endpoints ready |
| "transition all endpoints to ready state": { |
| startingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 2, 1, 999, []string{"host1", "host2"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: generateEndpointSlice("svc1", "ns1", 1, 2, 999, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| paramRemoveSlice: false, |
| expectedReturnVal: true, |
| expectedCurrentChange: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "svc1", "port-0", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 80, endpoint: "10.0.1.1:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 80, endpoint: "10.0.1.2:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| makeServicePortName("ns1", "svc1", "port-1", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 443, endpoint: "10.0.1.1:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 443, endpoint: "10.0.1.2:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| }, |
| }, |
| }, |
| // start with some endpoints ready, transition to more endpoints ready |
| "transition some endpoints to ready state": { |
| startingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 2, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| generateEndpointSlice("svc1", "ns1", 2, 2, 2, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: generateEndpointSlice("svc1", "ns1", 1, 3, 3, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| paramRemoveSlice: false, |
| expectedReturnVal: true, |
| expectedCurrentChange: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "svc1", "port-0", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 80, endpoint: "10.0.1.1:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 80, endpoint: "10.0.1.2:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 80, endpoint: "10.0.1.3:80", isLocal: true, ready: false, serving: false, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.1", port: 80, endpoint: "10.0.2.1:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.2", port: 80, endpoint: "10.0.2.2:80", isLocal: true, ready: false, serving: false, terminating: false}, |
| }, |
| makeServicePortName("ns1", "svc1", "port-1", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 443, endpoint: "10.0.1.1:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 443, endpoint: "10.0.1.2:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 443, endpoint: "10.0.1.3:443", isLocal: true, ready: false, serving: false, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.1", port: 443, endpoint: "10.0.2.1:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.2", port: 443, endpoint: "10.0.2.2:443", isLocal: true, ready: false, serving: false, terminating: false}, |
| }, |
| }, |
| }, |
| // start with some endpoints ready, transition to some terminating |
| "transition some endpoints to terminating state": { |
| startingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 2, 2, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| generateEndpointSlice("svc1", "ns1", 2, 2, 2, 2, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| endpointsChangeTracker: NewEndpointsChangeTracker("host1", nil, v1.IPv4Protocol, nil, nil), |
| namespacedName: types.NamespacedName{Name: "svc1", Namespace: "ns1"}, |
| paramEndpointSlice: generateEndpointSlice("svc1", "ns1", 1, 3, 3, 2, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| paramRemoveSlice: false, |
| expectedReturnVal: true, |
| expectedCurrentChange: map[ServicePortName][]*BaseEndpointInfo{ |
| makeServicePortName("ns1", "svc1", "port-0", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 80, endpoint: "10.0.1.1:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 80, endpoint: "10.0.1.2:80", isLocal: true, ready: false, serving: true, terminating: true}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 80, endpoint: "10.0.1.3:80", isLocal: true, ready: false, serving: false, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.1", port: 80, endpoint: "10.0.2.1:80", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.2", port: 80, endpoint: "10.0.2.2:80", isLocal: true, ready: false, serving: false, terminating: true}, |
| }, |
| makeServicePortName("ns1", "svc1", "port-1", v1.ProtocolTCP): { |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 443, endpoint: "10.0.1.1:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 443, endpoint: "10.0.1.2:443", isLocal: true, ready: false, serving: true, terminating: true}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 443, endpoint: "10.0.1.3:443", isLocal: true, ready: false, serving: false, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.1", port: 443, endpoint: "10.0.2.1:443", isLocal: true, ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.2.2", port: 443, endpoint: "10.0.2.2:443", isLocal: true, ready: false, serving: false, terminating: true}, |
| }, |
| }, |
| }, |
| } |
| |
| for name, tc := range testCases { |
| t.Run(name, func(t *testing.T) { |
| initializeCache(tc.endpointsChangeTracker.endpointSliceCache, tc.startingSlices) |
| |
| got := tc.endpointsChangeTracker.EndpointSliceUpdate(tc.paramEndpointSlice, tc.paramRemoveSlice) |
| if !reflect.DeepEqual(got, tc.expectedReturnVal) { |
| t.Errorf("EndpointSliceUpdate return value got: %v, want %v", got, tc.expectedReturnVal) |
| } |
| |
| changes := tc.endpointsChangeTracker.checkoutChanges() |
| if tc.expectedCurrentChange == nil { |
| if len(changes) != 0 { |
| t.Errorf("Expected %s to have no changes", tc.namespacedName) |
| } |
| } else { |
| if _, exists := changes[tc.namespacedName]; !exists { |
| t.Fatalf("Expected %s to have changes", tc.namespacedName) |
| } |
| compareEndpointsMapsStr(t, changes[tc.namespacedName].current, tc.expectedCurrentChange) |
| } |
| }) |
| } |
| } |
| |
| func TestCheckoutChanges(t *testing.T) { |
| svcPortName0 := ServicePortName{types.NamespacedName{Namespace: "ns1", Name: "svc1"}, "port-0", v1.ProtocolTCP} |
| svcPortName1 := ServicePortName{types.NamespacedName{Namespace: "ns1", Name: "svc1"}, "port-1", v1.ProtocolTCP} |
| |
| testCases := map[string]struct { |
| endpointsChangeTracker *EndpointsChangeTracker |
| expectedChanges []*endpointsChange |
| items map[types.NamespacedName]*endpointsChange |
| appliedSlices []*discovery.EndpointSlice |
| pendingSlices []*discovery.EndpointSlice |
| }{ |
| "empty slices": { |
| endpointsChangeTracker: NewEndpointsChangeTracker("", nil, v1.IPv4Protocol, nil, nil), |
| expectedChanges: []*endpointsChange{}, |
| appliedSlices: []*discovery.EndpointSlice{}, |
| pendingSlices: []*discovery.EndpointSlice{}, |
| }, |
| "adding initial slice": { |
| endpointsChangeTracker: NewEndpointsChangeTracker("", nil, v1.IPv4Protocol, nil, nil), |
| expectedChanges: []*endpointsChange{{ |
| previous: EndpointsMap{}, |
| current: EndpointsMap{ |
| svcPortName0: []Endpoint{ |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 80, endpoint: "10.0.1.1:80", ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 80, endpoint: "10.0.1.2:80", ready: false, serving: true, terminating: true}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 80, endpoint: "10.0.1.3:80", ready: false, serving: false, terminating: false}, |
| }, |
| }, |
| }}, |
| appliedSlices: []*discovery.EndpointSlice{}, |
| pendingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 3, 2, []string{"host1"}, []*int32{ptr.To[int32](80)}), |
| }, |
| }, |
| "removing port in update": { |
| endpointsChangeTracker: NewEndpointsChangeTracker("", nil, v1.IPv4Protocol, nil, nil), |
| expectedChanges: []*endpointsChange{{ |
| previous: EndpointsMap{ |
| svcPortName0: []Endpoint{ |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 80, endpoint: "10.0.1.1:80", ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 80, endpoint: "10.0.1.2:80", ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 80, endpoint: "10.0.1.3:80", ready: false, serving: false, terminating: false}, |
| }, |
| svcPortName1: []Endpoint{ |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 443, endpoint: "10.0.1.1:443", ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 443, endpoint: "10.0.1.2:443", ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 443, endpoint: "10.0.1.3:443", ready: false, serving: false, terminating: false}, |
| }, |
| }, |
| current: EndpointsMap{ |
| svcPortName0: []Endpoint{ |
| &BaseEndpointInfo{ip: "10.0.1.1", port: 80, endpoint: "10.0.1.1:80", ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.2", port: 80, endpoint: "10.0.1.2:80", ready: true, serving: true, terminating: false}, |
| &BaseEndpointInfo{ip: "10.0.1.3", port: 80, endpoint: "10.0.1.3:80", ready: false, serving: false, terminating: false}, |
| }, |
| }, |
| }}, |
| appliedSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 3, 999, []string{"host1"}, []*int32{ptr.To[int32](80), ptr.To[int32](443)}), |
| }, |
| pendingSlices: []*discovery.EndpointSlice{ |
| generateEndpointSlice("svc1", "ns1", 1, 3, 3, 999, []string{"host1"}, []*int32{ptr.To[int32](80)}), |
| }, |
| }, |
| } |
| |
| for name, tc := range testCases { |
| t.Run(name, func(t *testing.T) { |
| for _, slice := range tc.appliedSlices { |
| tc.endpointsChangeTracker.EndpointSliceUpdate(slice, false) |
| } |
| tc.endpointsChangeTracker.checkoutChanges() |
| for _, slice := range tc.pendingSlices { |
| tc.endpointsChangeTracker.EndpointSliceUpdate(slice, false) |
| } |
| changes := tc.endpointsChangeTracker.checkoutChanges() |
| |
| if len(tc.expectedChanges) != len(changes) { |
| t.Fatalf("Expected %d changes, got %d", len(tc.expectedChanges), len(changes)) |
| } |
| |
| for _, change := range changes { |
| // All of the test cases have 0 or 1 changes, so if we're |
| // here, then expectedChanges[0] is what we expect. |
| expectedChange := tc.expectedChanges[0] |
| |
| if !reflect.DeepEqual(change.previous, expectedChange.previous) { |
| t.Errorf("Expected change.previous: %+v, got: %+v", expectedChange.previous, change.previous) |
| } |
| |
| if !reflect.DeepEqual(change.current, expectedChange.current) { |
| t.Errorf("Expected change.current: %+v, got: %+v", expectedChange.current, change.current) |
| } |
| } |
| }) |
| } |
| } |
| |
| // Test helpers |
| |
| func compareEndpointsMapsStr(t *testing.T, newMap EndpointsMap, expected map[ServicePortName][]*BaseEndpointInfo) { |
| t.Helper() |
| if len(newMap) != len(expected) { |
| t.Fatalf("expected %d results, got %d: %v", len(expected), len(newMap), newMap) |
| } |
| endpointEqual := func(a, b *BaseEndpointInfo) bool { |
| return a.endpoint == b.endpoint && a.isLocal == b.isLocal && a.ready == b.ready && a.serving == b.serving && a.terminating == b.terminating |
| } |
| for x := range expected { |
| if len(newMap[x]) != len(expected[x]) { |
| t.Logf("Endpoints %+v", newMap[x]) |
| t.Fatalf("expected %d endpoints for %v, got %d", len(expected[x]), x, len(newMap[x])) |
| } else { |
| for i := range expected[x] { |
| newEp, ok := newMap[x][i].(*BaseEndpointInfo) |
| if !ok { |
| t.Fatalf("Failed to cast endpointInfo") |
| } |
| if !endpointEqual(newEp, expected[x][i]) { |
| t.Fatalf("expected new[%v][%d] to be %v, got %v"+ |
| "(IsLocal expected %v, got %v) (Ready expected %v, got %v) (Serving expected %v, got %v) (Terminating expected %v got %v)", |
| x, i, expected[x][i], newEp, expected[x][i].isLocal, newEp.isLocal, expected[x][i].ready, newEp.ready, |
| expected[x][i].serving, newEp.serving, expected[x][i].terminating, newEp.terminating) |
| } |
| } |
| } |
| } |
| } |
| |
| func initializeCache(endpointSliceCache *EndpointSliceCache, endpointSlices []*discovery.EndpointSlice) { |
| for _, endpointSlice := range endpointSlices { |
| endpointSliceCache.updatePending(endpointSlice, false) |
| } |
| |
| for _, tracker := range endpointSliceCache.trackerByServiceMap { |
| tracker.applied = tracker.pending |
| tracker.pending = endpointSliceDataByName{} |
| } |
| } |