mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-31 13:50:01 +00:00 
			
		
		
		
	Move reporting of GetReliableMountRefs error to the volume plugins that have more context about severity of the error.
		
			
				
	
	
		
			505 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			505 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
| Copyright 2015 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 fc
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 
 | |
| 	"k8s.io/klog/v2"
 | |
| 	"k8s.io/mount-utils"
 | |
| 	utilexec "k8s.io/utils/exec"
 | |
| 	"k8s.io/utils/io"
 | |
| 	utilstrings "k8s.io/utils/strings"
 | |
| 
 | |
| 	v1 "k8s.io/api/core/v1"
 | |
| 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | |
| 	"k8s.io/apimachinery/pkg/types"
 | |
| 	"k8s.io/kubernetes/pkg/volume"
 | |
| 	"k8s.io/kubernetes/pkg/volume/util"
 | |
| 	"k8s.io/kubernetes/pkg/volume/util/volumepathhandler"
 | |
| )
 | |
| 
 | |
| // ProbeVolumePlugins is the primary entrypoint for volume plugins.
 | |
| func ProbeVolumePlugins() []volume.VolumePlugin {
 | |
| 	return []volume.VolumePlugin{&fcPlugin{nil}}
 | |
| }
 | |
| 
 | |
| type fcPlugin struct {
 | |
| 	host volume.VolumeHost
 | |
| }
 | |
| 
 | |
| var _ volume.VolumePlugin = &fcPlugin{}
 | |
| var _ volume.PersistentVolumePlugin = &fcPlugin{}
 | |
| var _ volume.BlockVolumePlugin = &fcPlugin{}
 | |
| 
 | |
| const (
 | |
| 	fcPluginName = "kubernetes.io/fc"
 | |
| )
 | |
| 
 | |
| func (plugin *fcPlugin) Init(host volume.VolumeHost) error {
 | |
| 	plugin.host = host
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) GetPluginName() string {
 | |
| 	return fcPluginName
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) GetVolumeName(spec *volume.Spec) (string, error) {
 | |
| 	volumeSource, _, err := getVolumeSource(spec)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	// API server validates these parameters beforehand but attach/detach
 | |
| 	// controller creates volumespec without validation. They may be nil
 | |
| 	// or zero length. We should check again to avoid unexpected conditions.
 | |
| 	if len(volumeSource.TargetWWNs) != 0 && volumeSource.Lun != nil {
 | |
| 		// TargetWWNs are the FibreChannel target worldwide names
 | |
| 		return fmt.Sprintf("%v:%v", volumeSource.TargetWWNs, *volumeSource.Lun), nil
 | |
| 	} else if len(volumeSource.WWIDs) != 0 {
 | |
| 		// WWIDs are the FibreChannel World Wide Identifiers
 | |
| 		return fmt.Sprintf("%v", volumeSource.WWIDs), nil
 | |
| 	}
 | |
| 
 | |
| 	return "", err
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) CanSupport(spec *volume.Spec) bool {
 | |
| 	return (spec.Volume != nil && spec.Volume.FC != nil) || (spec.PersistentVolume != nil && spec.PersistentVolume.Spec.FC != nil)
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) RequiresRemount(spec *volume.Spec) bool {
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) SupportsMountOption() bool {
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) SupportsBulkVolumeVerification() bool {
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) GetAccessModes() []v1.PersistentVolumeAccessMode {
 | |
| 	return []v1.PersistentVolumeAccessMode{
 | |
| 		v1.ReadWriteOnce,
 | |
| 		v1.ReadOnlyMany,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) NewMounter(spec *volume.Spec, pod *v1.Pod, _ volume.VolumeOptions) (volume.Mounter, error) {
 | |
| 	// Inject real implementations here, test through the internal function.
 | |
| 	return plugin.newMounterInternal(spec, pod.UID, &fcUtil{}, plugin.host.GetMounter(plugin.GetPluginName()), plugin.host.GetExec(plugin.GetPluginName()))
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) newMounterInternal(spec *volume.Spec, podUID types.UID, manager diskManager, mounter mount.Interface, exec utilexec.Interface) (volume.Mounter, error) {
 | |
| 	// fc volumes used directly in a pod have a ReadOnly flag set by the pod author.
 | |
| 	// fc volumes used as a PersistentVolume gets the ReadOnly flag indirectly through the persistent-claim volume used to mount the PV
 | |
| 	fc, readOnly, err := getVolumeSource(spec)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	wwns, lun, wwids, err := getWwnsLunWwids(fc)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("fc: no fc disk information found. failed to make a new mounter")
 | |
| 	}
 | |
| 	fcDisk := &fcDisk{
 | |
| 		podUID:  podUID,
 | |
| 		volName: spec.Name(),
 | |
| 		wwns:    wwns,
 | |
| 		lun:     lun,
 | |
| 		wwids:   wwids,
 | |
| 		manager: manager,
 | |
| 		io:      &osIOHandler{},
 | |
| 		plugin:  plugin,
 | |
| 	}
 | |
| 
 | |
| 	volumeMode, err := util.GetVolumeMode(spec)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	klog.V(5).Infof("fc: newMounterInternal volumeMode %s", volumeMode)
 | |
| 	return &fcDiskMounter{
 | |
| 		fcDisk:       fcDisk,
 | |
| 		fsType:       fc.FSType,
 | |
| 		volumeMode:   volumeMode,
 | |
| 		readOnly:     readOnly,
 | |
| 		mounter:      &mount.SafeFormatAndMount{Interface: mounter, Exec: exec},
 | |
| 		deviceUtil:   util.NewDeviceHandler(util.NewIOHandler()),
 | |
| 		mountOptions: util.MountOptionFromSpec(spec),
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) NewBlockVolumeMapper(spec *volume.Spec, pod *v1.Pod, _ volume.VolumeOptions) (volume.BlockVolumeMapper, error) {
 | |
| 	// If this called via GenerateUnmapDeviceFunc(), pod is nil.
 | |
| 	// Pass empty string as dummy uid since uid isn't used in the case.
 | |
| 	var uid types.UID
 | |
| 	if pod != nil {
 | |
| 		uid = pod.UID
 | |
| 	}
 | |
| 	return plugin.newBlockVolumeMapperInternal(spec, uid, &fcUtil{}, plugin.host.GetMounter(plugin.GetPluginName()), plugin.host.GetExec(plugin.GetPluginName()))
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) newBlockVolumeMapperInternal(spec *volume.Spec, podUID types.UID, manager diskManager, mounter mount.Interface, exec utilexec.Interface) (volume.BlockVolumeMapper, error) {
 | |
| 	fc, readOnly, err := getVolumeSource(spec)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	wwns, lun, wwids, err := getWwnsLunWwids(fc)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("fc: no fc disk information found. failed to make a new mapper")
 | |
| 	}
 | |
| 
 | |
| 	mapper := &fcDiskMapper{
 | |
| 		fcDisk: &fcDisk{
 | |
| 			podUID:  podUID,
 | |
| 			volName: spec.Name(),
 | |
| 			wwns:    wwns,
 | |
| 			lun:     lun,
 | |
| 			wwids:   wwids,
 | |
| 			manager: manager,
 | |
| 			io:      &osIOHandler{},
 | |
| 			plugin:  plugin},
 | |
| 		readOnly:   readOnly,
 | |
| 		mounter:    &mount.SafeFormatAndMount{Interface: mounter, Exec: exec},
 | |
| 		deviceUtil: util.NewDeviceHandler(util.NewIOHandler()),
 | |
| 	}
 | |
| 
 | |
| 	blockPath, err := mapper.GetGlobalMapPath(spec)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("failed to get device path: %v", err)
 | |
| 	}
 | |
| 	mapper.MetricsProvider = volume.NewMetricsBlock(filepath.Join(blockPath, string(podUID)))
 | |
| 
 | |
| 	return mapper, nil
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) NewUnmounter(volName string, podUID types.UID) (volume.Unmounter, error) {
 | |
| 	// Inject real implementations here, test through the internal function.
 | |
| 	return plugin.newUnmounterInternal(volName, podUID, &fcUtil{}, plugin.host.GetMounter(plugin.GetPluginName()), plugin.host.GetExec(plugin.GetPluginName()))
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) newUnmounterInternal(volName string, podUID types.UID, manager diskManager, mounter mount.Interface, exec utilexec.Interface) (volume.Unmounter, error) {
 | |
| 	return &fcDiskUnmounter{
 | |
| 		fcDisk: &fcDisk{
 | |
| 			podUID:  podUID,
 | |
| 			volName: volName,
 | |
| 			manager: manager,
 | |
| 			plugin:  plugin,
 | |
| 			io:      &osIOHandler{},
 | |
| 		},
 | |
| 		mounter:    mounter,
 | |
| 		deviceUtil: util.NewDeviceHandler(util.NewIOHandler()),
 | |
| 		exec:       exec,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) NewBlockVolumeUnmapper(volName string, podUID types.UID) (volume.BlockVolumeUnmapper, error) {
 | |
| 	return plugin.newUnmapperInternal(volName, podUID, &fcUtil{}, plugin.host.GetExec(plugin.GetPluginName()))
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) newUnmapperInternal(volName string, podUID types.UID, manager diskManager, exec utilexec.Interface) (volume.BlockVolumeUnmapper, error) {
 | |
| 	return &fcDiskUnmapper{
 | |
| 		fcDisk: &fcDisk{
 | |
| 			podUID:  podUID,
 | |
| 			volName: volName,
 | |
| 			manager: manager,
 | |
| 			plugin:  plugin,
 | |
| 			io:      &osIOHandler{},
 | |
| 		},
 | |
| 		exec:       exec,
 | |
| 		deviceUtil: util.NewDeviceHandler(util.NewIOHandler()),
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func (plugin *fcPlugin) ConstructVolumeSpec(volumeName, mountPath string) (*volume.Spec, error) {
 | |
| 	// Find globalPDPath from pod volume directory(mountPath)
 | |
| 	// examples:
 | |
| 	//   mountPath:     pods/{podUid}/volumes/kubernetes.io~fc/{volumeName}
 | |
| 	//   globalPDPath : plugins/kubernetes.io/fc/50060e801049cfd1-lun-0
 | |
| 	var globalPDPath string
 | |
| 
 | |
| 	mounter := plugin.host.GetMounter(plugin.GetPluginName())
 | |
| 	// Try really hard to get the global mount of the volume, an error returned from here would
 | |
| 	// leave the global mount still mounted, while marking the volume as unused.
 | |
| 	// The volume can then be mounted on several nodes, resulting in volume
 | |
| 	// corruption.
 | |
| 	paths, err := util.GetReliableMountRefs(mounter, mountPath)
 | |
| 	if io.IsInconsistentReadError(err) {
 | |
| 		klog.Errorf("Failed to read mount refs from /proc/mounts for %s: %s", mountPath, err)
 | |
| 		klog.Errorf("Kubelet cannot unmount volume at %s, please unmount it manually", mountPath)
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	for _, path := range paths {
 | |
| 		if strings.Contains(path, plugin.host.GetPluginDir(fcPluginName)) {
 | |
| 			globalPDPath = path
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| 	// Couldn't fetch globalPDPath
 | |
| 	if len(globalPDPath) == 0 {
 | |
| 		return nil, fmt.Errorf("couldn't fetch globalPDPath. failed to obtain volume spec")
 | |
| 	}
 | |
| 
 | |
| 	wwns, lun, wwids, err := parsePDName(globalPDPath)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("failed to retrieve volume plugin information from globalPDPath: %s", err)
 | |
| 	}
 | |
| 	// Create volume from wwn+lun or wwid
 | |
| 	fcVolume := &v1.Volume{
 | |
| 		Name: volumeName,
 | |
| 		VolumeSource: v1.VolumeSource{
 | |
| 			FC: &v1.FCVolumeSource{WWIDs: wwids, Lun: &lun, TargetWWNs: wwns},
 | |
| 		},
 | |
| 	}
 | |
| 	klog.V(5).Infof("ConstructVolumeSpec: TargetWWNs: %v, Lun: %v, WWIDs: %v",
 | |
| 		fcVolume.VolumeSource.FC.TargetWWNs, *fcVolume.VolumeSource.FC.Lun, fcVolume.VolumeSource.FC.WWIDs)
 | |
| 	return volume.NewSpecFromVolume(fcVolume), nil
 | |
| }
 | |
| 
 | |
| // ConstructBlockVolumeSpec creates a new volume.Spec with following steps.
 | |
| //   - Searches a file whose name is {pod uuid} under volume plugin directory.
 | |
| //   - If a file is found, then retreives volumePluginDependentPath from globalMapPathUUID.
 | |
| //   - Once volumePluginDependentPath is obtained, store volume information to VolumeSource
 | |
| // examples:
 | |
| //   mapPath: pods/{podUid}/{DefaultKubeletVolumeDevicesDirName}/{escapeQualifiedPluginName}/{volumeName}
 | |
| //   globalMapPathUUID : plugins/kubernetes.io/{PluginName}/{DefaultKubeletVolumeDevicesDirName}/{volumePluginDependentPath}/{pod uuid}
 | |
| func (plugin *fcPlugin) ConstructBlockVolumeSpec(podUID types.UID, volumeName, mapPath string) (*volume.Spec, error) {
 | |
| 	pluginDir := plugin.host.GetVolumeDevicePluginDir(fcPluginName)
 | |
| 	blkutil := volumepathhandler.NewBlockVolumePathHandler()
 | |
| 	globalMapPathUUID, err := blkutil.FindGlobalMapPathUUIDFromPod(pluginDir, mapPath, podUID)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	klog.V(5).Infof("globalMapPathUUID: %v, err: %v", globalMapPathUUID, err)
 | |
| 
 | |
| 	// Retrieve globalPDPath from globalMapPathUUID
 | |
| 	// globalMapPathUUID examples:
 | |
| 	//   wwn+lun: plugins/kubernetes.io/fc/volumeDevices/50060e801049cfd1-lun-0/{pod uuid}
 | |
| 	//   wwid: plugins/kubernetes.io/fc/volumeDevices/3600508b400105e210000900000490000/{pod uuid}
 | |
| 	globalPDPath := filepath.Dir(globalMapPathUUID)
 | |
| 	// Create volume from wwn+lun or wwid
 | |
| 	wwns, lun, wwids, err := parsePDName(globalPDPath)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("failed to retrieve volume plugin information from globalPDPath: %s", err)
 | |
| 	}
 | |
| 	fcPV := createPersistentVolumeFromFCVolumeSource(volumeName,
 | |
| 		v1.FCVolumeSource{TargetWWNs: wwns, Lun: &lun, WWIDs: wwids})
 | |
| 	klog.V(5).Infof("ConstructBlockVolumeSpec: TargetWWNs: %v, Lun: %v, WWIDs: %v",
 | |
| 		fcPV.Spec.PersistentVolumeSource.FC.TargetWWNs,
 | |
| 		*fcPV.Spec.PersistentVolumeSource.FC.Lun,
 | |
| 		fcPV.Spec.PersistentVolumeSource.FC.WWIDs)
 | |
| 
 | |
| 	return volume.NewSpecFromPersistentVolume(fcPV, false), nil
 | |
| }
 | |
| 
 | |
| type fcDisk struct {
 | |
| 	volName string
 | |
| 	podUID  types.UID
 | |
| 	wwns    []string
 | |
| 	lun     string
 | |
| 	wwids   []string
 | |
| 	plugin  *fcPlugin
 | |
| 	// Utility interface that provides API calls to the provider to attach/detach disks.
 | |
| 	manager diskManager
 | |
| 	// io handler interface
 | |
| 	io ioHandler
 | |
| 	volume.MetricsNil
 | |
| }
 | |
| 
 | |
| func (fc *fcDisk) GetPath() string {
 | |
| 	// safe to use PodVolumeDir now: volume teardown occurs before pod is cleaned up
 | |
| 	return fc.plugin.host.GetPodVolumeDir(fc.podUID, utilstrings.EscapeQualifiedName(fcPluginName), fc.volName)
 | |
| }
 | |
| 
 | |
| func (fc *fcDisk) fcGlobalMapPath(spec *volume.Spec) (string, error) {
 | |
| 	mounter, err := volumeSpecToMounter(spec, fc.plugin.host)
 | |
| 	if err != nil {
 | |
| 		klog.Warningf("failed to get fc mounter: %v", err)
 | |
| 		return "", err
 | |
| 	}
 | |
| 	return fc.manager.MakeGlobalVDPDName(*mounter.fcDisk), nil
 | |
| }
 | |
| 
 | |
| func (fc *fcDisk) fcPodDeviceMapPath() (string, string) {
 | |
| 	return fc.plugin.host.GetPodVolumeDeviceDir(fc.podUID, utilstrings.EscapeQualifiedName(fcPluginName)), fc.volName
 | |
| }
 | |
| 
 | |
| type fcDiskMounter struct {
 | |
| 	*fcDisk
 | |
| 	readOnly     bool
 | |
| 	fsType       string
 | |
| 	volumeMode   v1.PersistentVolumeMode
 | |
| 	mounter      *mount.SafeFormatAndMount
 | |
| 	deviceUtil   util.DeviceUtil
 | |
| 	mountOptions []string
 | |
| }
 | |
| 
 | |
| var _ volume.Mounter = &fcDiskMounter{}
 | |
| 
 | |
| func (b *fcDiskMounter) GetAttributes() volume.Attributes {
 | |
| 	return volume.Attributes{
 | |
| 		ReadOnly:        b.readOnly,
 | |
| 		Managed:         !b.readOnly,
 | |
| 		SupportsSELinux: true,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Checks prior to mount operations to verify that the required components (binaries, etc.)
 | |
| // to mount the volume are available on the underlying node.
 | |
| // If not, it returns an error
 | |
| func (b *fcDiskMounter) CanMount() error {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (b *fcDiskMounter) SetUp(mounterArgs volume.MounterArgs) error {
 | |
| 	return b.SetUpAt(b.GetPath(), mounterArgs)
 | |
| }
 | |
| 
 | |
| func (b *fcDiskMounter) SetUpAt(dir string, mounterArgs volume.MounterArgs) error {
 | |
| 	// diskSetUp checks mountpoints and prevent repeated calls
 | |
| 	err := diskSetUp(b.manager, *b, dir, b.mounter, mounterArgs.FsGroup, mounterArgs.FSGroupChangePolicy)
 | |
| 	if err != nil {
 | |
| 		klog.Errorf("fc: failed to setup")
 | |
| 	}
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| type fcDiskUnmounter struct {
 | |
| 	*fcDisk
 | |
| 	mounter    mount.Interface
 | |
| 	deviceUtil util.DeviceUtil
 | |
| 	exec       utilexec.Interface
 | |
| }
 | |
| 
 | |
| var _ volume.Unmounter = &fcDiskUnmounter{}
 | |
| 
 | |
| // Unmounts the bind mount, and detaches the disk only if the disk
 | |
| // resource was the last reference to that disk on the kubelet.
 | |
| func (c *fcDiskUnmounter) TearDown() error {
 | |
| 	return c.TearDownAt(c.GetPath())
 | |
| }
 | |
| 
 | |
| func (c *fcDiskUnmounter) TearDownAt(dir string) error {
 | |
| 	return mount.CleanupMountPoint(dir, c.mounter, false)
 | |
| }
 | |
| 
 | |
| // Block Volumes Support
 | |
| type fcDiskMapper struct {
 | |
| 	*fcDisk
 | |
| 	volume.MetricsProvider
 | |
| 	readOnly   bool
 | |
| 	mounter    mount.Interface
 | |
| 	deviceUtil util.DeviceUtil
 | |
| }
 | |
| 
 | |
| var _ volume.BlockVolumeMapper = &fcDiskMapper{}
 | |
| 
 | |
| type fcDiskUnmapper struct {
 | |
| 	*fcDisk
 | |
| 	deviceUtil util.DeviceUtil
 | |
| 	exec       utilexec.Interface
 | |
| }
 | |
| 
 | |
| var _ volume.BlockVolumeUnmapper = &fcDiskUnmapper{}
 | |
| var _ volume.CustomBlockVolumeUnmapper = &fcDiskUnmapper{}
 | |
| 
 | |
| func (c *fcDiskUnmapper) TearDownDevice(mapPath, devicePath string) error {
 | |
| 	err := c.manager.DetachBlockFCDisk(*c, mapPath, devicePath)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("fc: failed to detach disk: %s\nError: %v", mapPath, err)
 | |
| 	}
 | |
| 	klog.V(4).Infof("fc: %s is unmounted, deleting the directory", mapPath)
 | |
| 	if err = os.RemoveAll(mapPath); err != nil {
 | |
| 		return fmt.Errorf("fc: failed to delete the directory: %s\nError: %v", mapPath, err)
 | |
| 	}
 | |
| 	klog.V(4).Infof("fc: successfully detached disk: %s", mapPath)
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (c *fcDiskUnmapper) UnmapPodDevice() error {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // GetGlobalMapPath returns global map path and error
 | |
| // path: plugins/kubernetes.io/{PluginName}/volumeDevices/{WWID}/{podUid}
 | |
| func (fc *fcDisk) GetGlobalMapPath(spec *volume.Spec) (string, error) {
 | |
| 	return fc.fcGlobalMapPath(spec)
 | |
| }
 | |
| 
 | |
| // GetPodDeviceMapPath returns pod device map path and volume name
 | |
| // path: pods/{podUid}/volumeDevices/kubernetes.io~fc
 | |
| // volumeName: pv0001
 | |
| func (fc *fcDisk) GetPodDeviceMapPath() (string, string) {
 | |
| 	return fc.fcPodDeviceMapPath()
 | |
| }
 | |
| 
 | |
| func getVolumeSource(spec *volume.Spec) (*v1.FCVolumeSource, bool, error) {
 | |
| 	// fc volumes used directly in a pod have a ReadOnly flag set by the pod author.
 | |
| 	// fc volumes used as a PersistentVolume gets the ReadOnly flag indirectly through the persistent-claim volume used to mount the PV
 | |
| 	if spec.Volume != nil && spec.Volume.FC != nil {
 | |
| 		return spec.Volume.FC, spec.Volume.FC.ReadOnly, nil
 | |
| 	} else if spec.PersistentVolume != nil &&
 | |
| 		spec.PersistentVolume.Spec.FC != nil {
 | |
| 		return spec.PersistentVolume.Spec.FC, spec.ReadOnly, nil
 | |
| 	}
 | |
| 
 | |
| 	return nil, false, fmt.Errorf("Spec does not reference a FibreChannel volume type")
 | |
| }
 | |
| 
 | |
| func createPersistentVolumeFromFCVolumeSource(volumeName string, fc v1.FCVolumeSource) *v1.PersistentVolume {
 | |
| 	block := v1.PersistentVolumeBlock
 | |
| 	return &v1.PersistentVolume{
 | |
| 		ObjectMeta: metav1.ObjectMeta{
 | |
| 			Name: volumeName,
 | |
| 		},
 | |
| 		Spec: v1.PersistentVolumeSpec{
 | |
| 			PersistentVolumeSource: v1.PersistentVolumeSource{
 | |
| 				FC: &fc,
 | |
| 			},
 | |
| 			VolumeMode: &block,
 | |
| 		},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func getWwnsLunWwids(fc *v1.FCVolumeSource) ([]string, string, []string, error) {
 | |
| 	var lun string
 | |
| 	var wwids []string
 | |
| 	if fc.Lun != nil && len(fc.TargetWWNs) != 0 {
 | |
| 		lun = strconv.Itoa(int(*fc.Lun))
 | |
| 		return fc.TargetWWNs, lun, wwids, nil
 | |
| 	}
 | |
| 	if len(fc.WWIDs) != 0 {
 | |
| 		for _, wwid := range fc.WWIDs {
 | |
| 			wwids = append(wwids, strings.Replace(wwid, " ", "_", -1))
 | |
| 		}
 | |
| 		return fc.TargetWWNs, lun, wwids, nil
 | |
| 	}
 | |
| 	return nil, "", nil, fmt.Errorf("fc: no fc disk information found")
 | |
| }
 |