mirror of https://github.com/k3s-io/k3s
305 lines
9.8 KiB
Go
305 lines
9.8 KiB
Go
|
// +build !providerless
|
||
|
|
||
|
/*
|
||
|
Copyright 2016 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 azuredd
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
"runtime"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
"time"
|
||
|
|
||
|
"github.com/Azure/azure-sdk-for-go/services/compute/mgmt/2019-12-01/compute"
|
||
|
"k8s.io/klog/v2"
|
||
|
"k8s.io/mount-utils"
|
||
|
|
||
|
v1 "k8s.io/api/core/v1"
|
||
|
"k8s.io/apimachinery/pkg/types"
|
||
|
"k8s.io/apimachinery/pkg/util/wait"
|
||
|
cloudprovider "k8s.io/cloud-provider"
|
||
|
"k8s.io/kubernetes/pkg/volume"
|
||
|
"k8s.io/kubernetes/pkg/volume/util"
|
||
|
"k8s.io/legacy-cloud-providers/azure"
|
||
|
)
|
||
|
|
||
|
type azureDiskDetacher struct {
|
||
|
plugin *azureDataDiskPlugin
|
||
|
cloud *azure.Cloud
|
||
|
}
|
||
|
|
||
|
type azureDiskAttacher struct {
|
||
|
plugin *azureDataDiskPlugin
|
||
|
cloud *azure.Cloud
|
||
|
}
|
||
|
|
||
|
var _ volume.Attacher = &azureDiskAttacher{}
|
||
|
var _ volume.Detacher = &azureDiskDetacher{}
|
||
|
|
||
|
var _ volume.DeviceMounter = &azureDiskAttacher{}
|
||
|
var _ volume.DeviceUnmounter = &azureDiskDetacher{}
|
||
|
|
||
|
// Attach attaches a volume.Spec to an Azure VM referenced by NodeName, returning the disk's LUN
|
||
|
func (a *azureDiskAttacher) Attach(spec *volume.Spec, nodeName types.NodeName) (string, error) {
|
||
|
volumeSource, _, err := getVolumeSource(spec)
|
||
|
if err != nil {
|
||
|
klog.Warningf("failed to get azure disk spec (%v)", err)
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
diskController, err := getDiskController(a.plugin.host)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
lun, err := diskController.GetDiskLun(volumeSource.DiskName, volumeSource.DataDiskURI, nodeName)
|
||
|
if err == cloudprovider.InstanceNotFound {
|
||
|
// Log error and continue with attach
|
||
|
klog.Warningf(
|
||
|
"Error checking if volume is already attached to current node (%q). Will continue and try attach anyway. err=%v",
|
||
|
nodeName, err)
|
||
|
}
|
||
|
|
||
|
if err == nil {
|
||
|
// Volume is already attached to node.
|
||
|
klog.V(2).Infof("Attach operation is successful. volume %q is already attached to node %q at lun %d.", volumeSource.DiskName, nodeName, lun)
|
||
|
} else {
|
||
|
klog.V(2).Infof("GetDiskLun returned: %v. Initiating attaching volume %q to node %q.", err, volumeSource.DataDiskURI, nodeName)
|
||
|
|
||
|
isManagedDisk := (*volumeSource.Kind == v1.AzureManagedDisk)
|
||
|
lun, err = diskController.AttachDisk(isManagedDisk, volumeSource.DiskName, volumeSource.DataDiskURI, nodeName, compute.CachingTypes(*volumeSource.CachingMode))
|
||
|
if err == nil {
|
||
|
klog.V(2).Infof("Attach operation successful: volume %q attached to node %q.", volumeSource.DataDiskURI, nodeName)
|
||
|
} else {
|
||
|
klog.V(2).Infof("Attach volume %q to instance %q failed with %v", volumeSource.DataDiskURI, nodeName, err)
|
||
|
return "", err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return strconv.Itoa(int(lun)), err
|
||
|
}
|
||
|
|
||
|
func (a *azureDiskAttacher) VolumesAreAttached(specs []*volume.Spec, nodeName types.NodeName) (map[*volume.Spec]bool, error) {
|
||
|
volumesAttachedCheck := make(map[*volume.Spec]bool)
|
||
|
volumeSpecMap := make(map[string]*volume.Spec)
|
||
|
volumeIDList := []string{}
|
||
|
for _, spec := range specs {
|
||
|
volumeSource, _, err := getVolumeSource(spec)
|
||
|
if err != nil {
|
||
|
klog.Errorf("azureDisk - Error getting volume (%q) source : %v", spec.Name(), err)
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
volumeIDList = append(volumeIDList, volumeSource.DiskName)
|
||
|
volumesAttachedCheck[spec] = true
|
||
|
volumeSpecMap[volumeSource.DiskName] = spec
|
||
|
}
|
||
|
|
||
|
diskController, err := getDiskController(a.plugin.host)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
attachedResult, err := diskController.DisksAreAttached(volumeIDList, nodeName)
|
||
|
if err != nil {
|
||
|
// Log error and continue with attach
|
||
|
klog.Errorf(
|
||
|
"azureDisk - Error checking if volumes (%v) are attached to current node (%q). err=%v",
|
||
|
volumeIDList, nodeName, err)
|
||
|
return volumesAttachedCheck, err
|
||
|
}
|
||
|
|
||
|
for volumeID, attached := range attachedResult {
|
||
|
if !attached {
|
||
|
spec := volumeSpecMap[volumeID]
|
||
|
volumesAttachedCheck[spec] = false
|
||
|
klog.V(2).Infof("azureDisk - VolumesAreAttached: check volume %q (specName: %q) is no longer attached", volumeID, spec.Name())
|
||
|
}
|
||
|
}
|
||
|
return volumesAttachedCheck, nil
|
||
|
}
|
||
|
|
||
|
func (a *azureDiskAttacher) WaitForAttach(spec *volume.Spec, devicePath string, _ *v1.Pod, timeout time.Duration) (string, error) {
|
||
|
// devicePath could be a LUN number or
|
||
|
// "/dev/disk/azure/scsi1/lunx", "/dev/sdx" on Linux node
|
||
|
// "/dev/diskx" on Windows node
|
||
|
if strings.HasPrefix(devicePath, "/dev/") {
|
||
|
return devicePath, nil
|
||
|
}
|
||
|
|
||
|
volumeSource, _, err := getVolumeSource(spec)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
nodeName := types.NodeName(a.plugin.host.GetHostName())
|
||
|
diskName := volumeSource.DiskName
|
||
|
|
||
|
lun, err := strconv.Atoi(devicePath)
|
||
|
if err != nil {
|
||
|
return "", fmt.Errorf("parse %s failed with error: %v, diskName: %s, nodeName: %s", devicePath, err, diskName, nodeName)
|
||
|
}
|
||
|
|
||
|
exec := a.plugin.host.GetExec(a.plugin.GetPluginName())
|
||
|
|
||
|
io := &osIOHandler{}
|
||
|
scsiHostRescan(io, exec)
|
||
|
|
||
|
newDevicePath := ""
|
||
|
|
||
|
err = wait.PollImmediate(1*time.Second, timeout, func() (bool, error) {
|
||
|
if newDevicePath, err = findDiskByLun(int(lun), io, exec); err != nil {
|
||
|
return false, fmt.Errorf("azureDisk - WaitForAttach ticker failed node (%s) disk (%s) lun(%v) err(%s)", nodeName, diskName, lun, err)
|
||
|
}
|
||
|
|
||
|
// did we find it?
|
||
|
if newDevicePath != "" {
|
||
|
return true, nil
|
||
|
}
|
||
|
|
||
|
// wait until timeout
|
||
|
return false, nil
|
||
|
})
|
||
|
if err == nil && newDevicePath == "" {
|
||
|
err = fmt.Errorf("azureDisk - WaitForAttach failed within timeout node (%s) diskId:(%s) lun:(%v)", nodeName, diskName, lun)
|
||
|
}
|
||
|
|
||
|
return newDevicePath, err
|
||
|
}
|
||
|
|
||
|
// to avoid name conflicts (similar *.vhd name)
|
||
|
// we use hash diskUri and we use it as device mount target.
|
||
|
// this is generalized for both managed and blob disks
|
||
|
// we also prefix the hash with m/b based on disk kind
|
||
|
func (a *azureDiskAttacher) GetDeviceMountPath(spec *volume.Spec) (string, error) {
|
||
|
volumeSource, _, err := getVolumeSource(spec)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
if volumeSource.Kind == nil { // this spec was constructed from info on the node
|
||
|
pdPath := filepath.Join(a.plugin.host.GetPluginDir(azureDataDiskPluginName), util.MountsInGlobalPDPath, volumeSource.DataDiskURI)
|
||
|
return pdPath, nil
|
||
|
}
|
||
|
|
||
|
isManagedDisk := (*volumeSource.Kind == v1.AzureManagedDisk)
|
||
|
return makeGlobalPDPath(a.plugin.host, volumeSource.DataDiskURI, isManagedDisk)
|
||
|
}
|
||
|
|
||
|
func (a *azureDiskAttacher) MountDevice(spec *volume.Spec, devicePath string, deviceMountPath string) error {
|
||
|
mounter := a.plugin.host.GetMounter(azureDataDiskPluginName)
|
||
|
notMnt, err := mounter.IsLikelyNotMountPoint(deviceMountPath)
|
||
|
|
||
|
if err != nil {
|
||
|
if os.IsNotExist(err) {
|
||
|
dir := deviceMountPath
|
||
|
if runtime.GOOS == "windows" {
|
||
|
// in windows, as we use mklink, only need to MkdirAll for parent directory
|
||
|
dir = filepath.Dir(deviceMountPath)
|
||
|
}
|
||
|
if err := os.MkdirAll(dir, 0750); err != nil {
|
||
|
return fmt.Errorf("azureDisk - mountDevice:CreateDirectory failed with %s", err)
|
||
|
}
|
||
|
notMnt = true
|
||
|
} else {
|
||
|
return fmt.Errorf("azureDisk - mountDevice:IsLikelyNotMountPoint failed with %s", err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if !notMnt {
|
||
|
// testing original mount point, make sure the mount link is valid
|
||
|
if _, err := (&osIOHandler{}).ReadDir(deviceMountPath); err != nil {
|
||
|
// mount link is invalid, now unmount and remount later
|
||
|
klog.Warningf("azureDisk - ReadDir %s failed with %v, unmount this directory", deviceMountPath, err)
|
||
|
if err := mounter.Unmount(deviceMountPath); err != nil {
|
||
|
klog.Errorf("azureDisk - Unmount deviceMountPath %s failed with %v", deviceMountPath, err)
|
||
|
return err
|
||
|
}
|
||
|
notMnt = true
|
||
|
}
|
||
|
}
|
||
|
|
||
|
volumeSource, _, err := getVolumeSource(spec)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
options := []string{}
|
||
|
if notMnt {
|
||
|
diskMounter := util.NewSafeFormatAndMountFromHost(azureDataDiskPluginName, a.plugin.host)
|
||
|
mountOptions := util.MountOptionFromSpec(spec, options...)
|
||
|
if runtime.GOOS == "windows" {
|
||
|
// only parse devicePath on Windows node
|
||
|
diskNum, err := getDiskNum(devicePath)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
devicePath = diskNum
|
||
|
}
|
||
|
err = diskMounter.FormatAndMount(devicePath, deviceMountPath, *volumeSource.FSType, mountOptions)
|
||
|
if err != nil {
|
||
|
if cleanErr := os.Remove(deviceMountPath); cleanErr != nil {
|
||
|
return fmt.Errorf("azureDisk - mountDevice:FormatAndMount failed with %s and clean up failed with :%v", err, cleanErr)
|
||
|
}
|
||
|
return fmt.Errorf("azureDisk - mountDevice:FormatAndMount failed with %s", err)
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// Detach detaches disk from Azure VM.
|
||
|
func (d *azureDiskDetacher) Detach(diskURI string, nodeName types.NodeName) error {
|
||
|
if diskURI == "" {
|
||
|
return fmt.Errorf("invalid disk to detach: %q", diskURI)
|
||
|
}
|
||
|
|
||
|
diskController, err := getDiskController(d.plugin.host)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
err = diskController.DetachDisk("", diskURI, nodeName)
|
||
|
if err != nil {
|
||
|
klog.Errorf("failed to detach azure disk %q, err %v", diskURI, err)
|
||
|
}
|
||
|
|
||
|
klog.V(2).Infof("azureDisk - disk:%s was detached from node:%v", diskURI, nodeName)
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
// UnmountDevice unmounts the volume on the node
|
||
|
func (d *azureDiskDetacher) UnmountDevice(deviceMountPath string) error {
|
||
|
if runtime.GOOS == "windows" {
|
||
|
// Flush data cache for windows because it does not do so automatically during unmount device
|
||
|
exec := d.plugin.host.GetExec(d.plugin.GetPluginName())
|
||
|
err := util.WriteVolumeCache(deviceMountPath, exec)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
err := mount.CleanupMountPoint(deviceMountPath, d.plugin.host.GetMounter(d.plugin.GetPluginName()), false)
|
||
|
if err == nil {
|
||
|
klog.V(2).Infof("azureDisk - Device %s was unmounted", deviceMountPath)
|
||
|
} else {
|
||
|
klog.Warningf("azureDisk - Device %s failed to unmount with error: %s", deviceMountPath, err.Error())
|
||
|
}
|
||
|
return err
|
||
|
}
|