mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-11-03 23:40:03 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			104 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			104 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
Copyright 2019 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 persistentvolume
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
 | 
						|
	"k8s.io/api/core/v1"
 | 
						|
	storage "k8s.io/api/storage/v1"
 | 
						|
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
						|
	"k8s.io/client-go/kubernetes/scheme"
 | 
						|
	storagelisters "k8s.io/client-go/listers/storage/v1"
 | 
						|
	"k8s.io/client-go/tools/reference"
 | 
						|
	v1helper "k8s.io/kubernetes/pkg/apis/core/v1/helper"
 | 
						|
)
 | 
						|
 | 
						|
// IsDelayBindingMode checks if claim is in delay binding mode.
 | 
						|
func IsDelayBindingMode(claim *v1.PersistentVolumeClaim, classLister storagelisters.StorageClassLister) (bool, error) {
 | 
						|
	className := v1helper.GetPersistentVolumeClaimClass(claim)
 | 
						|
	if className == "" {
 | 
						|
		return false, nil
 | 
						|
	}
 | 
						|
 | 
						|
	class, err := classLister.Get(className)
 | 
						|
	if err != nil {
 | 
						|
		return false, nil
 | 
						|
	}
 | 
						|
 | 
						|
	if class.VolumeBindingMode == nil {
 | 
						|
		return false, fmt.Errorf("VolumeBindingMode not set for StorageClass %q", className)
 | 
						|
	}
 | 
						|
 | 
						|
	return *class.VolumeBindingMode == storage.VolumeBindingWaitForFirstConsumer, nil
 | 
						|
}
 | 
						|
 | 
						|
// GetBindVolumeToClaim returns a new volume which is bound to given claim. In
 | 
						|
// addition, it returns a bool which indicates whether we made modification on
 | 
						|
// original volume.
 | 
						|
func GetBindVolumeToClaim(volume *v1.PersistentVolume, claim *v1.PersistentVolumeClaim) (*v1.PersistentVolume, bool, error) {
 | 
						|
	dirty := false
 | 
						|
 | 
						|
	// Check if the volume was already bound (either by user or by controller)
 | 
						|
	shouldSetBoundByController := false
 | 
						|
	if !IsVolumeBoundToClaim(volume, claim) {
 | 
						|
		shouldSetBoundByController = true
 | 
						|
	}
 | 
						|
 | 
						|
	// The volume from method args can be pointing to watcher cache. We must not
 | 
						|
	// modify these, therefore create a copy.
 | 
						|
	volumeClone := volume.DeepCopy()
 | 
						|
 | 
						|
	// Bind the volume to the claim if it is not bound yet
 | 
						|
	if volume.Spec.ClaimRef == nil ||
 | 
						|
		volume.Spec.ClaimRef.Name != claim.Name ||
 | 
						|
		volume.Spec.ClaimRef.Namespace != claim.Namespace ||
 | 
						|
		volume.Spec.ClaimRef.UID != claim.UID {
 | 
						|
 | 
						|
		claimRef, err := reference.GetReference(scheme.Scheme, claim)
 | 
						|
		if err != nil {
 | 
						|
			return nil, false, fmt.Errorf("Unexpected error getting claim reference: %v", err)
 | 
						|
		}
 | 
						|
		volumeClone.Spec.ClaimRef = claimRef
 | 
						|
		dirty = true
 | 
						|
	}
 | 
						|
 | 
						|
	// Set annBoundByController if it is not set yet
 | 
						|
	if shouldSetBoundByController && !metav1.HasAnnotation(volumeClone.ObjectMeta, annBoundByController) {
 | 
						|
		metav1.SetMetaDataAnnotation(&volumeClone.ObjectMeta, annBoundByController, "yes")
 | 
						|
		dirty = true
 | 
						|
	}
 | 
						|
 | 
						|
	return volumeClone, dirty, nil
 | 
						|
}
 | 
						|
 | 
						|
// IsVolumeBoundToClaim returns true, if given volume is pre-bound or bound
 | 
						|
// to specific claim. Both claim.Name and claim.Namespace must be equal.
 | 
						|
// If claim.UID is present in volume.Spec.ClaimRef, it must be equal too.
 | 
						|
func IsVolumeBoundToClaim(volume *v1.PersistentVolume, claim *v1.PersistentVolumeClaim) bool {
 | 
						|
	if volume.Spec.ClaimRef == nil {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
	if claim.Name != volume.Spec.ClaimRef.Name || claim.Namespace != volume.Spec.ClaimRef.Namespace {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
	if volume.Spec.ClaimRef.UID != "" && claim.UID != volume.Spec.ClaimRef.UID {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
	return true
 | 
						|
}
 |