Revert assumed PVs and PVCs in unreserve extension point
This commit is contained in:
		| @@ -118,6 +118,9 @@ type SchedulerVolumeBinder interface { | |||||||
| 	// This function is called serially. | 	// This function is called serially. | ||||||
| 	AssumePodVolumes(assumedPod *v1.Pod, nodeName string) (allFullyBound bool, err error) | 	AssumePodVolumes(assumedPod *v1.Pod, nodeName string) (allFullyBound bool, err error) | ||||||
|  |  | ||||||
|  | 	// RevertAssumedPodVolumes will revert assumed PV and PVC cache. | ||||||
|  | 	RevertAssumedPodVolumes(assumedPod *v1.Pod, nodeName string) | ||||||
|  |  | ||||||
| 	// BindPodVolumes will: | 	// BindPodVolumes will: | ||||||
| 	// 1. Initiate the volume binding by making the API call to prebind the PV | 	// 1. Initiate the volume binding by making the API call to prebind the PV | ||||||
| 	// to its matching PVC. | 	// to its matching PVC. | ||||||
| @@ -387,6 +390,12 @@ func (b *volumeBinder) AssumePodVolumes(assumedPod *v1.Pod, nodeName string) (al | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // RevertAssumedPodVolumes will revert assumed PV and PVC cache. | ||||||
|  | func (b *volumeBinder) RevertAssumedPodVolumes(assumedPod *v1.Pod, nodeName string) { | ||||||
|  | 	b.revertAssumedPVs(b.podBindingCache.GetBindings(assumedPod, nodeName)) | ||||||
|  | 	b.revertAssumedPVCs(b.podBindingCache.GetProvisionedPVCs(assumedPod, nodeName)) | ||||||
|  | } | ||||||
|  |  | ||||||
| // BindPodVolumes gets the cached bindings and PVCs to provision in podBindingCache, | // BindPodVolumes gets the cached bindings and PVCs to provision in podBindingCache, | ||||||
| // makes the API update for those PVs/PVCs, and waits for the PVCs to be completely bound | // makes the API update for those PVs/PVCs, and waits for the PVCs to be completely bound | ||||||
| // by the PV controller. | // by the PV controller. | ||||||
|   | |||||||
| @@ -53,6 +53,9 @@ func (b *FakeVolumeBinder) AssumePodVolumes(assumedPod *v1.Pod, nodeName string) | |||||||
| 	return b.config.AllBound, b.config.AssumeErr | 	return b.config.AllBound, b.config.AssumeErr | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // RevertAssumedPodVolumes implements SchedulerVolumeBinder.RevertAssumedPodVolumes | ||||||
|  | func (b *FakeVolumeBinder) RevertAssumedPodVolumes(assumedPod *v1.Pod, nodeName string) {} | ||||||
|  |  | ||||||
| // BindPodVolumes implements SchedulerVolumeBinder.BindPodVolumes. | // BindPodVolumes implements SchedulerVolumeBinder.BindPodVolumes. | ||||||
| func (b *FakeVolumeBinder) BindPodVolumes(assumedPod *v1.Pod) error { | func (b *FakeVolumeBinder) BindPodVolumes(assumedPod *v1.Pod) error { | ||||||
| 	b.BindCalled = true | 	b.BindCalled = true | ||||||
|   | |||||||
| @@ -465,13 +465,14 @@ func (env *testEnv) validateAssume(t *testing.T, pod *v1.Pod, bindings []*bindin | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (env *testEnv) validateFailedAssume(t *testing.T, pod *v1.Pod, bindings []*bindingInfo, provisionings []*v1.PersistentVolumeClaim) { | func (env *testEnv) validateCacheRestored(t *testing.T, pod *v1.Pod, bindings []*bindingInfo, provisionings []*v1.PersistentVolumeClaim) { | ||||||
| 	// All PVs have been unmodified in cache | 	// All PVs have been unmodified in cache | ||||||
| 	pvCache := env.internalBinder.pvCache | 	pvCache := env.internalBinder.pvCache | ||||||
| 	for _, b := range bindings { | 	for _, b := range bindings { | ||||||
| 		pv, _ := pvCache.GetPV(b.pv.Name) | 		pv, _ := pvCache.GetPV(b.pv.Name) | ||||||
|  | 		apiPV, _ := pvCache.GetAPIPV(b.pv.Name) | ||||||
| 		// PV could be nil if it's missing from cache | 		// PV could be nil if it's missing from cache | ||||||
| 		if pv != nil && pv != b.pv { | 		if pv != nil && pv != apiPV { | ||||||
| 			t.Errorf("PV %q was modified in cache", b.pv.Name) | 			t.Errorf("PV %q was modified in cache", b.pv.Name) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -1225,7 +1226,7 @@ func TestAssumePodVolumes(t *testing.T) { | |||||||
| 			scenario.expectedProvisionings = scenario.provisionedPVCs | 			scenario.expectedProvisionings = scenario.provisionedPVCs | ||||||
| 		} | 		} | ||||||
| 		if scenario.shouldFail { | 		if scenario.shouldFail { | ||||||
| 			testEnv.validateFailedAssume(t, pod, scenario.expectedBindings, scenario.expectedProvisionings) | 			testEnv.validateCacheRestored(t, pod, scenario.bindings, scenario.provisionedPVCs) | ||||||
| 		} else { | 		} else { | ||||||
| 			testEnv.validateAssume(t, pod, scenario.expectedBindings, scenario.expectedProvisionings) | 			testEnv.validateAssume(t, pod, scenario.expectedBindings, scenario.expectedProvisionings) | ||||||
| 		} | 		} | ||||||
| @@ -1237,6 +1238,34 @@ func TestAssumePodVolumes(t *testing.T) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestRevertAssumedPodVolumes(t *testing.T) { | ||||||
|  | 	ctx, cancel := context.WithCancel(context.Background()) | ||||||
|  | 	defer cancel() | ||||||
|  |  | ||||||
|  | 	podPVCs := []*v1.PersistentVolumeClaim{unboundPVC, provisionedPVC} | ||||||
|  | 	bindings := []*bindingInfo{makeBinding(unboundPVC, pvNode1a)} | ||||||
|  | 	pvs := []*v1.PersistentVolume{pvNode1a} | ||||||
|  | 	provisionedPVCs := []*v1.PersistentVolumeClaim{provisionedPVC} | ||||||
|  | 	expectedBindings := []*bindingInfo{makeBinding(unboundPVC, pvNode1aBound)} | ||||||
|  | 	expectedProvisionings := []*v1.PersistentVolumeClaim{selectedNodePVC} | ||||||
|  |  | ||||||
|  | 	// Setup | ||||||
|  | 	testEnv := newTestBinder(t, ctx.Done()) | ||||||
|  | 	testEnv.initClaims(podPVCs, podPVCs) | ||||||
|  | 	pod := makePod(podPVCs) | ||||||
|  | 	testEnv.initPodCache(pod, "node1", bindings, provisionedPVCs) | ||||||
|  | 	testEnv.initVolumes(pvs, pvs) | ||||||
|  |  | ||||||
|  | 	allbound, err := testEnv.binder.AssumePodVolumes(pod, "node1") | ||||||
|  | 	if allbound || err != nil { | ||||||
|  | 		t.Errorf("No volumes are assumed") | ||||||
|  | 	} | ||||||
|  | 	testEnv.validateAssume(t, pod, expectedBindings, expectedProvisionings) | ||||||
|  |  | ||||||
|  | 	testEnv.binder.RevertAssumedPodVolumes(pod, "node1") | ||||||
|  | 	testEnv.validateCacheRestored(t, pod, bindings, provisionedPVCs) | ||||||
|  | } | ||||||
|  |  | ||||||
| func TestBindAPIUpdate(t *testing.T) { | func TestBindAPIUpdate(t *testing.T) { | ||||||
| 	type scenarioType struct { | 	type scenarioType struct { | ||||||
| 		// Inputs | 		// Inputs | ||||||
|   | |||||||
| @@ -157,9 +157,10 @@ func (pl *VolumeBinding) PreBind(ctx context.Context, cs *framework.CycleState, | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // Unreserve clears pod binding state. | // Unreserve clears assumed PV and PVC cache and pod binding state. | ||||||
| // TODO(#90962) Revert assumed PV/PVC cache | // It's idempotent, and does nothing if no cache or binding state found for the given pod. | ||||||
| func (pl *VolumeBinding) Unreserve(ctx context.Context, cs *framework.CycleState, pod *v1.Pod, nodeName string) { | func (pl *VolumeBinding) Unreserve(ctx context.Context, cs *framework.CycleState, pod *v1.Pod, nodeName string) { | ||||||
|  | 	pl.Binder.RevertAssumedPodVolumes(pod, nodeName) | ||||||
| 	pl.Binder.DeletePodBindings(pod) | 	pl.Binder.DeletePodBindings(pod) | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Shintaro Murakami
					Shintaro Murakami