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
|
|
|
}
|
|
|
|
}
|