Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add helpers to handle startup taints #126975

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions staging/src/k8s.io/api/core/v1/taint.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,11 @@ func (t *Taint) MatchTaint(taintToMatch *Taint) bool {
return t.Key == taintToMatch.Key && t.Effect == taintToMatch.Effect
}

// MatchTaintByKey checks if the taint key matches taintToMatch key
func (t *Taint) MatchTaintByKey(taintToMatch *Taint) bool {
return t.Key == taintToMatch.Key
}

// taint.ToString() converts taint struct to string in format '<key>=<value>:<effect>', '<key>=<value>:', '<key>:<effect>', or '<key>'.
func (t *Taint) ToString() string {
if len(t.Effect) == 0 {
Expand Down
68 changes: 68 additions & 0 deletions staging/src/k8s.io/api/core/v1/taint_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -133,3 +133,71 @@ func TestMatchTaint(t *testing.T) {
}
}
}

func TestMatchTaintByKey(t *testing.T) {
testCases := []struct {
description string
taint *Taint
taintToMatch Taint
expectMatch bool
}{
{
description: "two taints with the same key should match",
taint: &Taint{
Key: "foo",
},
taintToMatch: Taint{
Key: "foo",
},
expectMatch: true,
},
{
description: "two taints with the same key but different value should match",
taint: &Taint{
Key: "foo",
Value: "bar",
Effect: TaintEffectNoSchedule,
},
taintToMatch: Taint{
Key: "foo",
Value: "different-value",
Effect: TaintEffectNoSchedule,
},
expectMatch: true,
},
{
description: "two taints with the different key cannot match",
taint: &Taint{
Key: "foo",
Value: "bar",
Effect: TaintEffectNoSchedule,
},
taintToMatch: Taint{
Key: "different-key",
Value: "bar",
Effect: TaintEffectNoSchedule,
},
expectMatch: false,
},
{
description: "two taints with the same key but different effect should match",
taint: &Taint{
Key: "foo",
Value: "bar",
Effect: TaintEffectNoSchedule,
},
taintToMatch: Taint{
Key: "foo",
Value: "bar",
Effect: TaintEffectPreferNoSchedule,
},
expectMatch: true,
},
}

for _, tc := range testCases {
if tc.expectMatch != tc.taint.MatchTaintByKey(&tc.taintToMatch) {
t.Errorf("[%s] expect taint %s match taint %s", tc.description, tc.taint.ToString(), tc.taintToMatch.ToString())
}
}
}
79 changes: 62 additions & 17 deletions staging/src/k8s.io/cloud-provider/node/helpers/taints.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ import (
"context"
"encoding/json"
"fmt"
"k8s.io/klog/v2"
"time"

"k8s.io/api/core/v1"
Expand Down Expand Up @@ -145,19 +146,15 @@ func addOrUpdateTaint(node *v1.Node, taint *v1.Taint) (*v1.Node, bool, error) {
return newNode, true, nil
}

// RemoveTaintOffNode is for cleaning up taints temporarily added to node,
// won't fail if target taint doesn't exist or has been removed.
// If passed a node it'll check if there's anything to be done, if taint is not present it won't issue
// any API calls.
func RemoveTaintOffNode(c clientset.Interface, nodeName string, node *v1.Node, taints ...*v1.Taint) error {
func removeTaintOffNode(c clientset.Interface, nodeName string, node *v1.Node, fn matchTaintFunc, taints ...*v1.Taint) error {
if len(taints) == 0 {
return nil
}
// Short circuit for limiting amount of API calls.
if node != nil {
match := false
for _, taint := range taints {
if taintExists(node.Spec.Taints, taint) {
if taintExists(node.Spec.Taints, taint, fn) {
match = true
break
}
Expand Down Expand Up @@ -187,7 +184,7 @@ func RemoveTaintOffNode(c clientset.Interface, nodeName string, node *v1.Node, t
oldNodeCopy := oldNode
updated := false
for _, taint := range taints {
curNewNode, ok, err := removeTaint(oldNodeCopy, taint)
curNewNode, ok, err := removeMatchingTaint(oldNodeCopy, taint, fn)
if err != nil {
return fmt.Errorf("failed to remove taint of node")
}
Expand All @@ -202,44 +199,92 @@ func RemoveTaintOffNode(c clientset.Interface, nodeName string, node *v1.Node, t
})
}

// taintExists checks if the given taint exists in list of taints. Returns true if exists false otherwise.
func taintExists(taints []v1.Taint, taintToFind *v1.Taint) bool {
// RemoveTaintOffNode is for cleaning up taints temporarily added to node,
// won't fail if target taint doesn't exist or has been removed.
// If passed a node it'll check if there's anything to be done, if taint is not present it won't issue
// any API calls.
func RemoveTaintOffNode(c clientset.Interface, nodeName string, node *v1.Node, taints ...*v1.Taint) error {
return removeTaintOffNode(c, nodeName, node, (*v1.Taint).MatchTaint, taints...)
}

// RemoveTaintOffNodeByKey is for cleaning up taints temporarily added to node matching by key only,
// won't fail if target taint doesn't exist or has been removed.
// If passed a node it'll check if there's anything to be done, if taint is not present it won't issue
// any API calls.
func RemoveTaintOffNodeByKey(c clientset.Interface, nodeName string, node *v1.Node, taints ...*v1.Taint) error {
return removeTaintOffNode(c, nodeName, node, (*v1.Taint).MatchTaintByKey, taints...)
}

type matchTaintFunc func(taint *v1.Taint, taintToMatch *v1.Taint) bool

// taintExists checks if matchFn evaluates to true on the list of taints. Returns true if there's a match, false otherwise
func taintExists(taints []v1.Taint, taintToFind *v1.Taint, matchFn matchTaintFunc) bool {
for _, taint := range taints {
if taint.MatchTaint(taintToFind) {
if matchFn(&taint, taintToFind) {
return true
}
}
return false
}

// removeTaint tries to remove a taint from annotations list. Returns a new copy of updated Node and true if something was updated
// removeTaint tries to remove a taint from annotations list that satisfies matchFn predicate. Returns a new copy of updated Node and true if something was updated
// false otherwise.
func removeTaint(node *v1.Node, taint *v1.Taint) (*v1.Node, bool, error) {
func removeMatchingTaint(node *v1.Node, taint *v1.Taint, matchFn matchTaintFunc) (*v1.Node, bool, error) {
newNode := node.DeepCopy()
nodeTaints := newNode.Spec.Taints
if len(nodeTaints) == 0 {
return newNode, false, nil
}

if !taintExists(nodeTaints, taint) {
if !taintExists(nodeTaints, taint, matchFn) {
return newNode, false, nil
}

newTaints, _ := deleteTaint(nodeTaints, taint)
newTaints, _ := deleteMatchingTaint(nodeTaints, taint, matchFn)
newNode.Spec.Taints = newTaints
return newNode, true, nil
}

// deleteTaint removes all the taints that have the same key and effect to given taintToDelete.
func deleteTaint(taints []v1.Taint, taintToDelete *v1.Taint) ([]v1.Taint, bool) {
// deleteTaint removes all the taints that match using matchFn
func deleteMatchingTaint(taints []v1.Taint, taintToDelete *v1.Taint, matchFn matchTaintFunc) ([]v1.Taint, bool) {
newTaints := []v1.Taint{}
deleted := false
for i := range taints {
if taintToDelete.MatchTaint(&taints[i]) {
if matchFn(taintToDelete, &taints[i]) {
deleted = true
continue
}
newTaints = append(newTaints, taints[i])
}
return newTaints, deleted
}

type ComponentReadyFunc func(ctx context.Context, c clientset.Interface, nodeName, driverName string) error

// RemoveNotReadyTaint removes the taint componentName/agent-not-ready from the local node
// This taint can be optionally applied by users to prevent startup race conditions such as
// https://github.com/kubernetes/kubernetes/issues/95911
func RemoveNotReadyTaint(c clientset.Interface, nodeName, componentName string, readyFn ComponentReadyFunc) error {
ctx := context.Background()
node, err := c.CoreV1().Nodes().Get(ctx, nodeName, metav1.GetOptions{})
if err != nil {
return err
}

if err := readyFn(ctx, c, nodeName, componentName); err != nil {
return err
}

const AgentNotReadyNodeTaintKeySuffix = "/agent-not-ready"

taintKeyToRemove := componentName + AgentNotReadyNodeTaintKeySuffix
klog.V(2).Infof("removing taint with key %s from local node %s", taintKeyToRemove, nodeName)

err = RemoveTaintOffNodeByKey(c, nodeName, node, &v1.Taint{Key: taintKeyToRemove})
if err != nil {
return err
}

klog.V(2).Infof("removed taint with key %s from local node %s successfully", taintKeyToRemove, nodeName)
return nil
}
Loading