mirror of https://github.com/k3s-io/k3s
255 lines
9.8 KiB
Go
255 lines
9.8 KiB
Go
/*
|
|
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 diskmanagers
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"hash/fnv"
|
|
"strings"
|
|
|
|
"github.com/vmware/govmomi/object"
|
|
"github.com/vmware/govmomi/vim25/types"
|
|
|
|
"k8s.io/klog/v2"
|
|
"k8s.io/legacy-cloud-providers/vsphere/vclib"
|
|
)
|
|
|
|
// vmDiskManager implements VirtualDiskProvider interface for creating volume using Virtual Machine Reconfigure approach
|
|
type vmDiskManager struct {
|
|
diskPath string
|
|
volumeOptions *vclib.VolumeOptions
|
|
vmOptions *vclib.VMOptions
|
|
}
|
|
|
|
// Create implements Disk's Create interface
|
|
// Contains implementation of VM based Provisioning to provision disk with SPBM Policy or VSANStorageProfileData
|
|
func (vmdisk vmDiskManager) Create(ctx context.Context, datastore *vclib.Datastore) (canonicalDiskPath string, err error) {
|
|
if vmdisk.volumeOptions.SCSIControllerType == "" {
|
|
vmdisk.volumeOptions.SCSIControllerType = vclib.PVSCSIControllerType
|
|
}
|
|
pbmClient, err := vclib.NewPbmClient(ctx, datastore.Client())
|
|
if err != nil {
|
|
klog.Errorf("error occurred while creating new pbmClient, err: %+v", err)
|
|
return "", err
|
|
}
|
|
|
|
if vmdisk.volumeOptions.StoragePolicyID == "" && vmdisk.volumeOptions.StoragePolicyName != "" {
|
|
vmdisk.volumeOptions.StoragePolicyID, err = pbmClient.ProfileIDByName(ctx, vmdisk.volumeOptions.StoragePolicyName)
|
|
if err != nil {
|
|
klog.Errorf("error occurred while getting Profile Id from Profile Name: %s, err: %+v", vmdisk.volumeOptions.StoragePolicyName, err)
|
|
return "", err
|
|
}
|
|
}
|
|
if vmdisk.volumeOptions.StoragePolicyID != "" {
|
|
compatible, faultMessage, err := datastore.IsCompatibleWithStoragePolicy(ctx, vmdisk.volumeOptions.StoragePolicyID)
|
|
if err != nil {
|
|
klog.Errorf("error occurred while checking datastore compatibility with storage policy id: %s, err: %+v", vmdisk.volumeOptions.StoragePolicyID, err)
|
|
return "", err
|
|
}
|
|
|
|
if !compatible {
|
|
klog.Errorf("Datastore: %s is not compatible with Policy: %s", datastore.Name(), vmdisk.volumeOptions.StoragePolicyName)
|
|
return "", fmt.Errorf("user specified datastore is not compatible with the storagePolicy: %q. Failed with faults: %+q", vmdisk.volumeOptions.StoragePolicyName, faultMessage)
|
|
}
|
|
}
|
|
|
|
storageProfileSpec := &types.VirtualMachineDefinedProfileSpec{}
|
|
// Is PBM storage policy ID is present, set the storage spec profile ID,
|
|
// else, set raw the VSAN policy string.
|
|
if vmdisk.volumeOptions.StoragePolicyID != "" {
|
|
storageProfileSpec.ProfileId = vmdisk.volumeOptions.StoragePolicyID
|
|
} else if vmdisk.volumeOptions.VSANStorageProfileData != "" {
|
|
// Check Datastore type - VSANStorageProfileData is only applicable to vSAN Datastore
|
|
dsType, err := datastore.GetType(ctx)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
if dsType != vclib.VSANDatastoreType {
|
|
klog.Errorf("The specified datastore: %q is not a VSAN datastore", datastore.Name())
|
|
return "", fmt.Errorf("the specified datastore: %q is not a VSAN datastore."+
|
|
" the policy parameters will work only with VSAN Datastore."+
|
|
" so, please specify a valid VSAN datastore in Storage class definition", datastore.Name())
|
|
}
|
|
storageProfileSpec.ProfileId = ""
|
|
storageProfileSpec.ProfileData = &types.VirtualMachineProfileRawData{
|
|
ExtensionKey: "com.vmware.vim.sps",
|
|
ObjectData: vmdisk.volumeOptions.VSANStorageProfileData,
|
|
}
|
|
} else {
|
|
klog.Errorf("Both volumeOptions.StoragePolicyID and volumeOptions.VSANStorageProfileData are not set. One of them should be set")
|
|
return "", fmt.Errorf("both volumeOptions.StoragePolicyID and volumeOptions.VSANStorageProfileData are not set. One of them should be set")
|
|
}
|
|
var dummyVM *vclib.VirtualMachine
|
|
// Check if VM already exist in the folder.
|
|
// If VM is already present, use it, else create a new dummy VM.
|
|
fnvHash := fnv.New32a()
|
|
fnvHash.Write([]byte(vmdisk.volumeOptions.Name))
|
|
dummyVMFullName := vclib.DummyVMPrefixName + "-" + fmt.Sprint(fnvHash.Sum32())
|
|
dummyVM, err = datastore.Datacenter.GetVMByPath(ctx, vmdisk.vmOptions.VMFolder.InventoryPath+"/"+dummyVMFullName)
|
|
if err != nil {
|
|
// Create a dummy VM
|
|
klog.V(1).Infof("Creating Dummy VM: %q", dummyVMFullName)
|
|
dummyVM, err = vmdisk.createDummyVM(ctx, datastore.Datacenter, dummyVMFullName)
|
|
if err != nil {
|
|
klog.Errorf("failed to create Dummy VM. err: %v", err)
|
|
return "", err
|
|
}
|
|
}
|
|
|
|
// Reconfigure the VM to attach the disk with the VSAN policy configured
|
|
virtualMachineConfigSpec := types.VirtualMachineConfigSpec{}
|
|
disk, _, err := dummyVM.CreateDiskSpec(ctx, vmdisk.diskPath, datastore, vmdisk.volumeOptions)
|
|
if err != nil {
|
|
klog.Errorf("failed to create Disk Spec. err: %v", err)
|
|
return "", err
|
|
}
|
|
deviceConfigSpec := &types.VirtualDeviceConfigSpec{
|
|
Device: disk,
|
|
Operation: types.VirtualDeviceConfigSpecOperationAdd,
|
|
FileOperation: types.VirtualDeviceConfigSpecFileOperationCreate,
|
|
}
|
|
|
|
deviceConfigSpec.Profile = append(deviceConfigSpec.Profile, storageProfileSpec)
|
|
virtualMachineConfigSpec.DeviceChange = append(virtualMachineConfigSpec.DeviceChange, deviceConfigSpec)
|
|
fileAlreadyExist := false
|
|
task, err := dummyVM.Reconfigure(ctx, virtualMachineConfigSpec)
|
|
if err != nil {
|
|
klog.Errorf("Failed to reconfig. err: %v", err)
|
|
return "", err
|
|
}
|
|
err = task.Wait(ctx)
|
|
if err != nil {
|
|
fileAlreadyExist = isAlreadyExists(vmdisk.diskPath, err)
|
|
if fileAlreadyExist {
|
|
//Skip error and continue to detach the disk as the disk was already created on the datastore.
|
|
klog.V(vclib.LogLevel).Infof("File: %v already exists", vmdisk.diskPath)
|
|
} else {
|
|
klog.Errorf("Failed to attach the disk to VM: %q with err: %+v", dummyVMFullName, err)
|
|
return "", err
|
|
}
|
|
}
|
|
// Detach the disk from the dummy VM.
|
|
err = dummyVM.DetachDisk(ctx, vmdisk.diskPath)
|
|
if err != nil {
|
|
if vclib.DiskNotFoundErrMsg == err.Error() && fileAlreadyExist {
|
|
// Skip error if disk was already detached from the dummy VM but still present on the datastore.
|
|
klog.V(vclib.LogLevel).Infof("File: %v is already detached", vmdisk.diskPath)
|
|
} else {
|
|
klog.Errorf("Failed to detach the disk: %q from VM: %q with err: %+v", vmdisk.diskPath, dummyVMFullName, err)
|
|
return "", err
|
|
}
|
|
}
|
|
// Delete the dummy VM
|
|
err = dummyVM.DeleteVM(ctx)
|
|
if err != nil {
|
|
klog.Errorf("Failed to destroy the vm: %q with err: %+v", dummyVMFullName, err)
|
|
}
|
|
return vmdisk.diskPath, nil
|
|
}
|
|
|
|
func (vmdisk vmDiskManager) Delete(ctx context.Context, datacenter *vclib.Datacenter) error {
|
|
return fmt.Errorf("vmDiskManager.Delete is not supported")
|
|
}
|
|
|
|
// CreateDummyVM create a Dummy VM at specified location with given name.
|
|
func (vmdisk vmDiskManager) createDummyVM(ctx context.Context, datacenter *vclib.Datacenter, vmName string) (*vclib.VirtualMachine, error) {
|
|
// Create a virtual machine config spec with 1 SCSI adapter.
|
|
virtualMachineConfigSpec := types.VirtualMachineConfigSpec{
|
|
Name: vmName,
|
|
Files: &types.VirtualMachineFileInfo{
|
|
VmPathName: "[" + vmdisk.volumeOptions.Datastore + "]",
|
|
},
|
|
NumCPUs: 1,
|
|
MemoryMB: 4,
|
|
DeviceChange: []types.BaseVirtualDeviceConfigSpec{
|
|
&types.VirtualDeviceConfigSpec{
|
|
Operation: types.VirtualDeviceConfigSpecOperationAdd,
|
|
Device: &types.ParaVirtualSCSIController{
|
|
VirtualSCSIController: types.VirtualSCSIController{
|
|
SharedBus: types.VirtualSCSISharingNoSharing,
|
|
VirtualController: types.VirtualController{
|
|
BusNumber: 0,
|
|
VirtualDevice: types.VirtualDevice{
|
|
Key: 1000,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
task, err := vmdisk.vmOptions.VMFolder.CreateVM(ctx, virtualMachineConfigSpec, vmdisk.vmOptions.VMResourcePool, nil)
|
|
if err != nil {
|
|
klog.Errorf("Failed to create VM. err: %+v", err)
|
|
return nil, err
|
|
}
|
|
|
|
dummyVMTaskInfo, err := task.WaitForResult(ctx, nil)
|
|
if err != nil {
|
|
klog.Errorf("Error occurred while waiting for create VM task result. err: %+v", err)
|
|
return nil, err
|
|
}
|
|
|
|
vmRef := dummyVMTaskInfo.Result.(object.Reference)
|
|
dummyVM := object.NewVirtualMachine(datacenter.Client(), vmRef.Reference())
|
|
return &vclib.VirtualMachine{VirtualMachine: dummyVM, Datacenter: datacenter}, nil
|
|
}
|
|
|
|
// CleanUpDummyVMs deletes stale dummyVM's
|
|
func CleanUpDummyVMs(ctx context.Context, folder *vclib.Folder) error {
|
|
vmList, err := folder.GetVirtualMachines(ctx)
|
|
if err != nil {
|
|
klog.V(4).Infof("Failed to get virtual machines in the kubernetes cluster: %s, err: %+v", folder.InventoryPath, err)
|
|
return err
|
|
}
|
|
if vmList == nil || len(vmList) == 0 {
|
|
klog.Errorf("No virtual machines found in the kubernetes cluster: %s", folder.InventoryPath)
|
|
return fmt.Errorf("no virtual machines found in the kubernetes cluster: %s", folder.InventoryPath)
|
|
}
|
|
var dummyVMList []*vclib.VirtualMachine
|
|
// Loop through VM's in the Kubernetes cluster to find dummy VM's
|
|
for _, vm := range vmList {
|
|
vmName, err := vm.ObjectName(ctx)
|
|
if err != nil {
|
|
klog.V(4).Infof("Unable to get name from VM with err: %+v", err)
|
|
continue
|
|
}
|
|
if strings.HasPrefix(vmName, vclib.DummyVMPrefixName) {
|
|
vmObj := vclib.VirtualMachine{VirtualMachine: object.NewVirtualMachine(folder.Client(), vm.Reference())}
|
|
dummyVMList = append(dummyVMList, &vmObj)
|
|
}
|
|
}
|
|
for _, vm := range dummyVMList {
|
|
err = vm.DeleteVM(ctx)
|
|
if err != nil {
|
|
klog.V(4).Infof("Unable to delete dummy VM with err: %+v", err)
|
|
continue
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func isAlreadyExists(path string, err error) bool {
|
|
errorMessage := fmt.Sprintf("Cannot complete the operation because the file or folder %s already exists", path)
|
|
if errorMessage == err.Error() {
|
|
return true
|
|
}
|
|
return false
|
|
}
|