k3s/vendor/k8s.io/kubernetes/pkg/util/tolerations/tolerations.go

108 lines
2.7 KiB
Go
Raw Normal View History

2019-08-30 18:33:25 +00:00
/*
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 tolerations
import (
2019-09-27 21:51:53 +00:00
apiequality "k8s.io/apimachinery/pkg/api/equality"
2020-08-10 17:43:49 +00:00
"k8s.io/klog/v2"
2019-08-30 18:33:25 +00:00
api "k8s.io/kubernetes/pkg/apis/core"
)
// VerifyAgainstWhitelist checks if the provided tolerations
// satisfy the provided whitelist and returns true, otherwise returns false
2019-09-27 21:51:53 +00:00
func VerifyAgainstWhitelist(tolerations, whitelist []api.Toleration) bool {
if len(whitelist) == 0 || len(tolerations) == 0 {
2019-08-30 18:33:25 +00:00
return true
}
2019-09-27 21:51:53 +00:00
next:
for _, t := range tolerations {
for _, w := range whitelist {
if isSuperset(w, t) {
continue next
}
2019-08-30 18:33:25 +00:00
}
2019-09-27 21:51:53 +00:00
return false
2019-08-30 18:33:25 +00:00
}
2019-09-27 21:51:53 +00:00
2019-08-30 18:33:25 +00:00
return true
}
2019-09-27 21:51:53 +00:00
// MergeTolerations merges two sets of tolerations into one. If one toleration is a superset of
// another, only the superset is kept.
func MergeTolerations(first, second []api.Toleration) []api.Toleration {
all := append(first, second...)
var merged []api.Toleration
next:
for i, t := range all {
for _, t2 := range merged {
if isSuperset(t2, t) {
continue next // t is redundant; ignore it
}
2019-08-30 18:33:25 +00:00
}
2019-09-27 21:51:53 +00:00
if i+1 < len(all) {
for _, t2 := range all[i+1:] {
// If the tolerations are equal, prefer the first.
if !apiequality.Semantic.DeepEqual(&t, &t2) && isSuperset(t2, t) {
continue next // t is redundant; ignore it
}
}
2019-08-30 18:33:25 +00:00
}
2019-09-27 21:51:53 +00:00
merged = append(merged, t)
2019-08-30 18:33:25 +00:00
}
2019-09-27 21:51:53 +00:00
return merged
2019-08-30 18:33:25 +00:00
}
2019-09-27 21:51:53 +00:00
// isSuperset checks whether ss tolerates a superset of t.
func isSuperset(ss, t api.Toleration) bool {
if apiequality.Semantic.DeepEqual(&t, &ss) {
return true
2019-08-30 18:33:25 +00:00
}
2019-09-27 21:51:53 +00:00
if t.Key != ss.Key &&
// An empty key with Exists operator means match all keys & values.
(ss.Key != "" || ss.Operator != api.TolerationOpExists) {
return false
2019-08-30 18:33:25 +00:00
}
2019-09-27 21:51:53 +00:00
// An empty effect means match all effects.
if t.Effect != ss.Effect && ss.Effect != "" {
return false
2019-08-30 18:33:25 +00:00
}
2019-09-27 21:51:53 +00:00
if ss.Effect == api.TaintEffectNoExecute {
if ss.TolerationSeconds != nil {
if t.TolerationSeconds == nil ||
*t.TolerationSeconds > *ss.TolerationSeconds {
return false
}
}
2019-08-30 18:33:25 +00:00
}
2019-09-27 21:51:53 +00:00
switch ss.Operator {
case api.TolerationOpEqual, "": // empty operator means Equal
return t.Operator == api.TolerationOpEqual && t.Value == ss.Value
case api.TolerationOpExists:
2019-08-30 18:33:25 +00:00
return true
2019-09-27 21:51:53 +00:00
default:
klog.Errorf("Unknown toleration operator: %s", ss.Operator)
return false
2019-08-30 18:33:25 +00:00
}
}