mirror of
https://github.com/k3s-io/kubernetes.git
synced 2025-08-03 01:06:27 +00:00
Extend validation for ReplicationController
Provide type safe checks for empty sets of selectors.
This commit is contained in:
parent
fbd71c9c02
commit
d32024870a
@ -20,6 +20,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/labels"
|
||||||
"github.com/GoogleCloudPlatform/kubernetes/pkg/util"
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/util"
|
||||||
"github.com/golang/glog"
|
"github.com/golang/glog"
|
||||||
)
|
)
|
||||||
@ -286,8 +287,21 @@ func ValidateService(service *Service) []error {
|
|||||||
if service.ID == "" {
|
if service.ID == "" {
|
||||||
allErrs.Append(makeInvalidError("Service.ID", service.ID))
|
allErrs.Append(makeInvalidError("Service.ID", service.ID))
|
||||||
}
|
}
|
||||||
if len(service.Selector) == 0 {
|
if labels.Set(service.Selector).AsSelector().Empty() {
|
||||||
allErrs.Append(makeInvalidError("Service.Selector", service.Selector))
|
allErrs.Append(makeInvalidError("Service.Selector", service.Selector))
|
||||||
}
|
}
|
||||||
return []error(allErrs)
|
return []error(allErrs)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ValidateReplicationController tests if required fields in the replication controller are set.
|
||||||
|
func ValidateReplicationController(controller *ReplicationController) []error {
|
||||||
|
errors := []error{}
|
||||||
|
if controller.ID == "" {
|
||||||
|
errors = append(errors, makeInvalidError("ReplicationController.ID", controller.ID))
|
||||||
|
}
|
||||||
|
if labels.Set(controller.DesiredState.ReplicaSelector).AsSelector().Empty() {
|
||||||
|
errors = append(errors, makeInvalidError("ReplicationController.ReplicaSelector", controller.DesiredState.ReplicaSelector))
|
||||||
|
}
|
||||||
|
errors = append(errors, ValidateManifest(&controller.DesiredState.PodTemplate.DesiredState.Manifest)...)
|
||||||
|
return errors
|
||||||
|
}
|
||||||
|
@ -294,3 +294,63 @@ func TestValidateService(t *testing.T) {
|
|||||||
t.Errorf("Unexpected error list: %#v", errs)
|
t.Errorf("Unexpected error list: %#v", errs)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestValidateReplicationController(t *testing.T) {
|
||||||
|
validSelector := map[string]string{"a": "b"}
|
||||||
|
validPodTemplate := PodTemplate{
|
||||||
|
DesiredState: PodState{
|
||||||
|
Manifest: ContainerManifest{
|
||||||
|
Version: "v1beta1",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
successCases := []ReplicationController{
|
||||||
|
{
|
||||||
|
JSONBase: JSONBase{ID: "abc"},
|
||||||
|
DesiredState: ReplicationControllerState{
|
||||||
|
ReplicaSelector: validSelector,
|
||||||
|
PodTemplate: validPodTemplate,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
JSONBase: JSONBase{ID: "abc-123"},
|
||||||
|
DesiredState: ReplicationControllerState{
|
||||||
|
ReplicaSelector: validSelector,
|
||||||
|
PodTemplate: validPodTemplate,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, successCase := range successCases {
|
||||||
|
if errs := ValidateReplicationController(&successCase); len(errs) != 0 {
|
||||||
|
t.Errorf("expected success: %v", errs)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
errorCases := map[string]ReplicationController{
|
||||||
|
"zero-length ID": {
|
||||||
|
JSONBase: JSONBase{ID: ""},
|
||||||
|
DesiredState: ReplicationControllerState{
|
||||||
|
ReplicaSelector: validSelector,
|
||||||
|
PodTemplate: validPodTemplate,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"empty selector": {
|
||||||
|
JSONBase: JSONBase{ID: "abc"},
|
||||||
|
DesiredState: ReplicationControllerState{
|
||||||
|
PodTemplate: validPodTemplate,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"invalid manifest": {
|
||||||
|
JSONBase: JSONBase{ID: "abc"},
|
||||||
|
DesiredState: ReplicationControllerState{
|
||||||
|
ReplicaSelector: validSelector,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for k, v := range errorCases {
|
||||||
|
if errs := ValidateReplicationController(&v); len(errs) == 0 {
|
||||||
|
t.Errorf("expected failure for %s", k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -27,6 +27,9 @@ type Selector interface {
|
|||||||
// Matches returns true if this selector matches the given set of labels.
|
// Matches returns true if this selector matches the given set of labels.
|
||||||
Matches(Labels) bool
|
Matches(Labels) bool
|
||||||
|
|
||||||
|
// Empty returns true if this selector does not restrict the selection space.
|
||||||
|
Empty() bool
|
||||||
|
|
||||||
// String returns a human readable string that represents this selector.
|
// String returns a human readable string that represents this selector.
|
||||||
String() string
|
String() string
|
||||||
}
|
}
|
||||||
@ -44,6 +47,10 @@ func (t *hasTerm) Matches(ls Labels) bool {
|
|||||||
return ls.Get(t.label) == t.value
|
return ls.Get(t.label) == t.value
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (t *hasTerm) Empty() bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
func (t *hasTerm) String() string {
|
func (t *hasTerm) String() string {
|
||||||
return fmt.Sprintf("%v=%v", t.label, t.value)
|
return fmt.Sprintf("%v=%v", t.label, t.value)
|
||||||
}
|
}
|
||||||
@ -56,6 +63,10 @@ func (t *notHasTerm) Matches(ls Labels) bool {
|
|||||||
return ls.Get(t.label) != t.value
|
return ls.Get(t.label) != t.value
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (t *notHasTerm) Empty() bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
func (t *notHasTerm) String() string {
|
func (t *notHasTerm) String() string {
|
||||||
return fmt.Sprintf("%v!=%v", t.label, t.value)
|
return fmt.Sprintf("%v!=%v", t.label, t.value)
|
||||||
}
|
}
|
||||||
@ -71,6 +82,21 @@ func (t andTerm) Matches(ls Labels) bool {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (t andTerm) Empty() bool {
|
||||||
|
if t == nil {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if len([]Selector(t)) == 0 {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
for i := range t {
|
||||||
|
if !t[i].Empty() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
func (t andTerm) String() string {
|
func (t andTerm) String() string {
|
||||||
var terms []string
|
var terms []string
|
||||||
for _, q := range t {
|
for _, q := range t {
|
||||||
@ -87,8 +113,12 @@ func try(selectorPiece, op string) (lhs, rhs string, ok bool) {
|
|||||||
return "", "", false
|
return "", "", false
|
||||||
}
|
}
|
||||||
|
|
||||||
// SelectorFromSet returns a Selector which will match exactly the given Set.
|
// SelectorFromSet returns a Selector which will match exactly the given Set. A
|
||||||
|
// nil Set is considered equivalent to Everything().
|
||||||
func SelectorFromSet(ls Set) Selector {
|
func SelectorFromSet(ls Set) Selector {
|
||||||
|
if ls == nil {
|
||||||
|
return Everything()
|
||||||
|
}
|
||||||
items := make([]Selector, 0, len(ls))
|
items := make([]Selector, 0, len(ls))
|
||||||
for label, value := range ls {
|
for label, value := range ls {
|
||||||
items = append(items, &hasTerm{label: label, value: value})
|
items = append(items, &hasTerm{label: label, value: value})
|
||||||
@ -124,3 +154,12 @@ func ParseSelector(selector string) (Selector, error) {
|
|||||||
}
|
}
|
||||||
return andTerm(items), nil
|
return andTerm(items), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MustParseSelector parses the selection or panics.
|
||||||
|
func MustParseSelector(selector string) Selector {
|
||||||
|
s, err := ParseSelector(selector)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
@ -84,6 +84,9 @@ func TestEverything(t *testing.T) {
|
|||||||
if !Everything().Matches(Set{"x": "y"}) {
|
if !Everything().Matches(Set{"x": "y"}) {
|
||||||
t.Errorf("Nil selector didn't match")
|
t.Errorf("Nil selector didn't match")
|
||||||
}
|
}
|
||||||
|
if !Everything().Empty() {
|
||||||
|
t.Errorf("Everything was not empty")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestSelectorMatches(t *testing.T) {
|
func TestSelectorMatches(t *testing.T) {
|
||||||
@ -132,3 +135,31 @@ func TestSetMatches(t *testing.T) {
|
|||||||
expectNoMatchDirect(t, Set{"baz": "=bar"}, labelset)
|
expectNoMatchDirect(t, Set{"baz": "=bar"}, labelset)
|
||||||
expectNoMatchDirect(t, Set{"foo": "=bar", "foobar": "bar", "baz": "blah"}, labelset)
|
expectNoMatchDirect(t, Set{"foo": "=bar", "foobar": "bar", "baz": "blah"}, labelset)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestNilMapIsValid(t *testing.T) {
|
||||||
|
selector := Set(nil).AsSelector()
|
||||||
|
if selector == nil {
|
||||||
|
t.Errorf("Selector for nil set should be Everything")
|
||||||
|
}
|
||||||
|
if !selector.Empty() {
|
||||||
|
t.Errorf("Selector for nil set should be Empty")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSetIsEmpty(t *testing.T) {
|
||||||
|
if !(Set{}).AsSelector().Empty() {
|
||||||
|
t.Errorf("Empty set should be empty")
|
||||||
|
}
|
||||||
|
if !(andTerm(nil)).Empty() {
|
||||||
|
t.Errorf("Nil andTerm should be empty")
|
||||||
|
}
|
||||||
|
if (&hasTerm{}).Empty() {
|
||||||
|
t.Errorf("hasTerm should not be empty")
|
||||||
|
}
|
||||||
|
if !(andTerm{andTerm{}}).Empty() {
|
||||||
|
t.Errorf("Nested andTerm should be empty")
|
||||||
|
}
|
||||||
|
if (andTerm{&hasTerm{"a", "b"}}).Empty() {
|
||||||
|
t.Errorf("Nested andTerm should not be empty")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user