Skip to content

Commit

Permalink
fix: Prevent NodeClaims from transitioning into Ready=Unknown durin…
Browse files Browse the repository at this point in the history
…g deletion (#2052)
  • Loading branch information
rschalo authored Mar 5, 2025
1 parent db4938c commit 91755d4
Show file tree
Hide file tree
Showing 2 changed files with 52 additions and 1 deletion.
20 changes: 19 additions & 1 deletion pkg/controllers/nodeclaim/lifecycle/controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -171,6 +171,24 @@ func (c *Controller) Reconcile(ctx context.Context, nodeClaim *v1.NodeClaim) (re

//nolint:gocyclo
func (c *Controller) finalize(ctx context.Context, nodeClaim *v1.NodeClaim) (reconcile.Result, error) {
// setting the deletion timestamp will bump the generation, so we need to
// perform a no-op for whatever the status condition is currently set to
// so that we bump the observed generation to the latest and prevent the nodeclaim
// root status from entering an `Unknown` state
stored := nodeClaim.DeepCopy()
for _, condition := range nodeClaim.Status.Conditions {
if nodeClaim.StatusConditions().IsDependentCondition(condition.Type) {
nodeClaim.StatusConditions().Set(condition)
}
}
if !equality.Semantic.DeepEqual(stored, nodeClaim) {
if err := c.kubeClient.Status().Patch(ctx, nodeClaim, client.MergeFromWithOptions(stored, client.MergeFromWithOptimisticLock{})); err != nil {
if errors.IsConflict(err) {
return reconcile.Result{Requeue: true}, nil
}
return reconcile.Result{}, err
}
}
if !controllerutil.ContainsFinalizer(nodeClaim, v1.TerminationFinalizer) {
return reconcile.Result{}, nil
}
Expand Down Expand Up @@ -226,7 +244,7 @@ func (c *Controller) finalize(ctx context.Context, nodeClaim *v1.NodeClaim) (rec
metrics.NodePoolLabel: nodeClaim.Labels[v1.NodePoolLabelKey],
})
}
stored := nodeClaim.DeepCopy() // The NodeClaim may have been modified in the EnsureTerminated function
stored = nodeClaim.DeepCopy() // The NodeClaim may have been modified in the EnsureTerminated function
controllerutil.RemoveFinalizer(nodeClaim, v1.TerminationFinalizer)
if !equality.Semantic.DeepEqual(stored, nodeClaim) {
// We use client.MergeFromWithOptimisticLock because patching a list with a JSON merge patch
Expand Down
33 changes: 33 additions & 0 deletions pkg/controllers/nodeclaim/lifecycle/suite_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -136,6 +136,39 @@ var _ = Describe("Finalizer", func() {
})
Expect(ok).To(BeFalse())
})
It("should update status conditions to the latest generation when finalizing", func() {
nodeClaim := test.NodeClaim(v1.NodeClaim{
ObjectMeta: metav1.ObjectMeta{
Labels: map[string]string{
v1.NodePoolLabelKey: nodePool.Name,
},
},
})
ExpectApplied(ctx, env.Client, nodePool, nodeClaim)
ExpectMakeNodeClaimsInitialized(ctx, env.Client, nodeClaim)
ExpectObjectReconciled(ctx, env.Client, nodeClaimController, nodeClaim)
nodeClaim = ExpectExists(ctx, env.Client, nodeClaim)
// add a finalizer so we can make assertions about all status conditions
ExpectDeletionTimestampSet(ctx, env.Client, nodeClaim)
ExpectObjectReconciled(ctx, env.Client, nodeClaimController, nodeClaim)
nodeClaim = ExpectExists(ctx, env.Client, nodeClaim)

Expect(nodeClaim.StatusConditions().Get(v1.ConditionTypeLaunched).IsTrue()).To(BeTrue())
Expect(nodeClaim.StatusConditions().Get(v1.ConditionTypeLaunched).ObservedGeneration).To(Equal(nodeClaim.Generation))
Expect(nodeClaim.StatusConditions().Get(v1.ConditionTypeRegistered).IsTrue()).To(BeTrue())
Expect(nodeClaim.StatusConditions().Get(v1.ConditionTypeRegistered).ObservedGeneration).To(Equal(nodeClaim.Generation))
Expect(nodeClaim.StatusConditions().Get(v1.ConditionTypeInitialized).IsTrue()).To(BeTrue())
Expect(nodeClaim.StatusConditions().Get(v1.ConditionTypeInitialized).ObservedGeneration).To(Equal(nodeClaim.Generation))
Expect(nodeClaim.StatusConditions().Get(status.ConditionReady).IsTrue()).To(BeTrue())
Expect(nodeClaim.StatusConditions().Get(status.ConditionReady).ObservedGeneration).To(Equal(nodeClaim.Generation))

ExpectObjectReconciled(ctx, env.Client, nodeClaimController, nodeClaim)
nodeClaim = ExpectExists(ctx, env.Client, nodeClaim)
Expect(nodeClaim.StatusConditions().Get(v1.ConditionTypeInstanceTerminating).IsTrue()).To(BeTrue())
Expect(nodeClaim.StatusConditions().Get(v1.ConditionTypeInstanceTerminating).ObservedGeneration).To(Equal(nodeClaim.Generation))
ExpectFinalizersRemoved(ctx, env.Client, nodeClaim)
ExpectDeleted(ctx, env.Client, nodeClaim)
})
})
It("should update observedGeneration if generation increases after all conditions are marked True", func() {
nodeClaim := test.NodeClaim(v1.NodeClaim{
Expand Down

0 comments on commit 91755d4

Please sign in to comment.