
In many cases clients may wish to view not ready addresses for endpoints in order to do set membership prior to a pod being ready. For instance, a pod that uses the service endpoints to connect to other pods under the same service, but does not want to signal ready before it has contacted at least a minimal number of other pods. This is backwards compatible with old servers and clients. There is an additional cost in size of endpoints before services ramp up, which will add minor CPU and memory use for services that have a significant number of pods which have not become ready.
445 lines
16 KiB
Go
445 lines
16 KiB
Go
/*
|
|
Copyright 2015 The Kubernetes Authors All rights reserved.
|
|
|
|
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 endpoints
|
|
|
|
import (
|
|
"reflect"
|
|
"testing"
|
|
|
|
"github.com/davecgh/go-spew/spew"
|
|
"k8s.io/kubernetes/pkg/api"
|
|
"k8s.io/kubernetes/pkg/types"
|
|
)
|
|
|
|
func podRef(uid string) *api.ObjectReference {
|
|
ref := api.ObjectReference{UID: types.UID(uid)}
|
|
return &ref
|
|
}
|
|
|
|
func TestPackSubsets(t *testing.T) {
|
|
// The downside of table-driven tests is that some things have to live outside the table.
|
|
fooObjRef := api.ObjectReference{Name: "foo"}
|
|
barObjRef := api.ObjectReference{Name: "bar"}
|
|
|
|
testCases := []struct {
|
|
name string
|
|
given []api.EndpointSubset
|
|
expect []api.EndpointSubset
|
|
}{
|
|
{
|
|
name: "empty everything",
|
|
given: []api.EndpointSubset{{Addresses: []api.EndpointAddress{}, Ports: []api.EndpointPort{}}},
|
|
expect: []api.EndpointSubset{},
|
|
}, {
|
|
name: "empty addresses",
|
|
given: []api.EndpointSubset{{Addresses: []api.EndpointAddress{}, Ports: []api.EndpointPort{{Port: 111}}}},
|
|
expect: []api.EndpointSubset{},
|
|
}, {
|
|
name: "empty ports",
|
|
given: []api.EndpointSubset{{Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}}, Ports: []api.EndpointPort{}}},
|
|
expect: []api.EndpointSubset{},
|
|
}, {
|
|
name: "empty ports",
|
|
given: []api.EndpointSubset{{NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4"}}, Ports: []api.EndpointPort{}}},
|
|
expect: []api.EndpointSubset{},
|
|
}, {
|
|
name: "one set, one ip, one port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, one notReady ip, one port",
|
|
given: []api.EndpointSubset{{
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, one ip, one UID, one port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, one notReady ip, one UID, one port",
|
|
given: []api.EndpointSubset{{
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, one ip, empty UID, one port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, one notReady ip, empty UID, one port",
|
|
given: []api.EndpointSubset{{
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, two ips, one port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}, {IP: "5.6.7.8"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}, {IP: "5.6.7.8"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, two mixed ips, one port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "5.6.7.8"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "5.6.7.8"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, two duplicate ips, one port, notReady is covered by ready",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, one ip, two ports",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}, {Port: 222}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}, {Port: 222}},
|
|
}},
|
|
}, {
|
|
name: "one set, dup ips, one port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}, {IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, dup ips with target-refs, one port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{
|
|
{IP: "1.2.3.4", TargetRef: &fooObjRef},
|
|
{IP: "1.2.3.4", TargetRef: &barObjRef},
|
|
},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: &fooObjRef}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, dup mixed ips with target-refs, one port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{
|
|
{IP: "1.2.3.4", TargetRef: &fooObjRef},
|
|
},
|
|
NotReadyAddresses: []api.EndpointAddress{
|
|
{IP: "1.2.3.4", TargetRef: &barObjRef},
|
|
},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
// finding the same address twice is considered an error on input, only the first address+port
|
|
// reference is preserved
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: &fooObjRef}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "one set, one ip, dup ports",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}, {Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "two sets, dup ip, dup port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "two sets, dup mixed ip, dup port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "two sets, dup ip, two ports",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 222}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}, {Port: 222}},
|
|
}},
|
|
}, {
|
|
name: "two sets, dup ip, dup uids, two ports",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 222}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}, {Port: 222}},
|
|
}},
|
|
}, {
|
|
name: "two sets, dup mixed ip, dup uids, two ports",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 222}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 222}},
|
|
}},
|
|
}, {
|
|
name: "two sets, two ips, dup port",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "5.6.7.8"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}, {IP: "5.6.7.8"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "two set, dup ip, two uids, dup ports",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-2")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{
|
|
{IP: "1.2.3.4", TargetRef: podRef("uid-1")},
|
|
{IP: "1.2.3.4", TargetRef: podRef("uid-2")},
|
|
},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "two set, dup ip, with and without uid, dup ports",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-2")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{
|
|
{IP: "1.2.3.4"},
|
|
{IP: "1.2.3.4", TargetRef: podRef("uid-2")},
|
|
},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "two sets, two ips, two dup ip with uid, dup port, wrong order",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "5.6.7.8"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "5.6.7.8", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{
|
|
{IP: "1.2.3.4"},
|
|
{IP: "1.2.3.4", TargetRef: podRef("uid-1")},
|
|
{IP: "5.6.7.8"},
|
|
{IP: "5.6.7.8", TargetRef: podRef("uid-1")},
|
|
},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "two sets, two mixed ips, two dup ip with uid, dup port, wrong order, ends up with split addresses",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "5.6.7.8"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "5.6.7.8", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4", TargetRef: podRef("uid-1")}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{
|
|
{IP: "5.6.7.8"},
|
|
},
|
|
NotReadyAddresses: []api.EndpointAddress{
|
|
{IP: "1.2.3.4"},
|
|
{IP: "1.2.3.4", TargetRef: podRef("uid-1")},
|
|
{IP: "5.6.7.8", TargetRef: podRef("uid-1")},
|
|
},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}},
|
|
}, {
|
|
name: "two sets, two ips, two ports",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "5.6.7.8"}},
|
|
Ports: []api.EndpointPort{{Port: 222}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "5.6.7.8"}},
|
|
Ports: []api.EndpointPort{{Port: 222}},
|
|
}},
|
|
}, {
|
|
name: "four sets, three ips, three ports, jumbled",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.5"}},
|
|
Ports: []api.EndpointPort{{Port: 222}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.6"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.5"}},
|
|
Ports: []api.EndpointPort{{Port: 333}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}, {IP: "1.2.3.6"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.5"}},
|
|
Ports: []api.EndpointPort{{Port: 222}, {Port: 333}},
|
|
}},
|
|
}, {
|
|
name: "four sets, three mixed ips, three ports, jumbled",
|
|
given: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.5"}},
|
|
Ports: []api.EndpointPort{{Port: 222}},
|
|
}, {
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.6"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.5"}},
|
|
Ports: []api.EndpointPort{{Port: 333}},
|
|
}},
|
|
expect: []api.EndpointSubset{{
|
|
Addresses: []api.EndpointAddress{{IP: "1.2.3.4"}, {IP: "1.2.3.6"}},
|
|
Ports: []api.EndpointPort{{Port: 111}},
|
|
}, {
|
|
NotReadyAddresses: []api.EndpointAddress{{IP: "1.2.3.5"}},
|
|
Ports: []api.EndpointPort{{Port: 222}, {Port: 333}},
|
|
}},
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
result := RepackSubsets(tc.given)
|
|
if !reflect.DeepEqual(result, SortSubsets(tc.expect)) {
|
|
t.Errorf("case %q: expected %s, got %s", tc.name, spew.Sprintf("%#v", SortSubsets(tc.expect)), spew.Sprintf("%#v", result))
|
|
}
|
|
}
|
|
}
|