Skip to content

Commit

Permalink
Added multi disk support tests
Browse files Browse the repository at this point in the history
  • Loading branch information
vr4manta committed Dec 5, 2024
1 parent eb41bfe commit 8848cff
Show file tree
Hide file tree
Showing 6 changed files with 393 additions and 17 deletions.
1 change: 1 addition & 0 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -374,6 +374,7 @@ generate-e2e-templates-main: $(KUSTOMIZE) ## Generate test templates for the mai
cp "$(RELEASE_DIR)/main/cluster-template-ignition.yaml" "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/base/cluster-template-ignition.yaml"
"$(KUSTOMIZE)" --load-restrictor LoadRestrictionsNone build "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/base" > "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/cluster-template.yaml"
"$(KUSTOMIZE)" --load-restrictor LoadRestrictionsNone build "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/hw-upgrade" > "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/cluster-template-hw-upgrade.yaml"
"$(KUSTOMIZE)" --load-restrictor LoadRestrictionsNone build "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/multi-disk" > "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/cluster-template-multi-disk.yaml"
"$(KUSTOMIZE)" --load-restrictor LoadRestrictionsNone build "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/storage-policy" > "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/cluster-template-storage-policy.yaml"
"$(KUSTOMIZE)" --load-restrictor LoadRestrictionsNone build "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/conformance" > "$(E2E_GOVMOMI_TEMPLATE_DIR)/main/cluster-template-conformance.yaml"
# Since CAPI uses different flavor names for KCP and MD remediation using MHC
Expand Down
277 changes: 260 additions & 17 deletions pkg/services/govmomi/vcenter/clone_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ package vcenter
import (
ctx "context"
"crypto/tls"
"fmt"
"testing"

"github.com/vmware/govmomi/object"
Expand Down Expand Up @@ -60,25 +61,30 @@ func TestGetDiskSpec(t *testing.T) {
additionalCloneDiskSizes []int32
name string
disks object.VirtualDeviceList
dataDisks []infrav1.VSphereDisk
expectedDiskCount int
err string
}{
{
name: "Successfully clone template with correct disk requirements",
disks: defaultDisks,
cloneDiskSize: defaultSizeGiB,
expectDevice: true,
name: "Successfully clone template with correct disk requirements",
disks: defaultDisks,
cloneDiskSize: defaultSizeGiB,
expectDevice: true,
expectedDiskCount: 1,
},
{
name: "Successfully clone template and increase disk requirements",
disks: defaultDisks,
cloneDiskSize: defaultSizeGiB + 1,
expectDevice: true,
name: "Successfully clone template and increase disk requirements",
disks: defaultDisks,
cloneDiskSize: defaultSizeGiB + 1,
expectDevice: true,
expectedDiskCount: 1,
},
{
name: "Successfully clone template with no explicit disk requirements",
disks: defaultDisks,
cloneDiskSize: 0,
expectDevice: true,
name: "Successfully clone template with no explicit disk requirements",
disks: defaultDisks,
cloneDiskSize: 0,
expectDevice: true,
expectedDiskCount: 1,
},
{
name: "Fail to clone template with lower disk requirements then on template",
Expand All @@ -98,6 +104,7 @@ func TestGetDiskSpec(t *testing.T) {
cloneDiskSize: defaultSizeGiB + 1,
additionalCloneDiskSizes: []int32{defaultSizeGiB + 1},
expectDevice: true,
expectedDiskCount: 2,
},
{
name: "Fails to clone template and decrease second disk size",
Expand All @@ -114,32 +121,268 @@ func TestGetDiskSpec(t *testing.T) {
cloneSpec := infrav1.VirtualMachineCloneSpec{
DiskGiB: tc.cloneDiskSize,
AdditionalDisksGiB: tc.additionalCloneDiskSizes,
DataDisks: tc.dataDisks,
}
vsphereVM := &infrav1.VSphereVM{
Spec: infrav1.VSphereVMSpec{
VirtualMachineCloneSpec: cloneSpec,
},
}
vmContext := &capvcontext.VMContext{VSphereVM: vsphereVM}
devices, err := getDiskSpec(vmContext, tc.disks)
deviceResults, err := getDiskSpec(vmContext, tc.disks)
if (tc.err != "" && err == nil) || (tc.err == "" && err != nil) || (err != nil && tc.err != err.Error()) {
t.Fatalf("Expected to get '%v' error from getDiskSpec, got: '%v'", tc.err, err)
}
if deviceFound := len(devices) != 0; tc.expectDevice != deviceFound {
t.Fatalf("Expected to get a device: %v, but got: '%#v'", tc.expectDevice, devices)
if deviceFound := len(deviceResults) != 0; tc.expectDevice != deviceFound {
t.Fatalf("Expected to get a device: %v, but got: '%#v'", tc.expectDevice, deviceResults)
}
if tc.expectDevice {
primaryDevice := devices[0]
primaryDevice := deviceResults[0]
validateDiskSpec(t, primaryDevice, tc.cloneDiskSize)
if len(tc.additionalCloneDiskSizes) != 0 {
secondaryDevice := devices[1]
secondaryDevice := deviceResults[1]
validateDiskSpec(t, secondaryDevice, tc.additionalCloneDiskSizes[0])
}

// Check number of disks present
if len(deviceResults) != tc.expectedDiskCount {
t.Fatalf("Expected device count to be %v, but found %v", tc.expectedDiskCount, len(deviceResults))
}
}
})
}
}

func TestCreateDiskSpec(t *testing.T) {
model, session, server := initSimulator(t)
t.Cleanup(model.Remove)
t.Cleanup(server.Close)
vm := simulator.Map.Any("VirtualMachine").(*simulator.VirtualMachine)
machine := object.NewVirtualMachine(session.Client.Client, vm.Reference())

devices, err := machine.Device(ctx.TODO())
if err != nil {
t.Fatalf("Failed to obtain vm devices: %v", err)
}
defaultDisks := devices.SelectByType((*types.VirtualDisk)(nil))
if len(defaultDisks) < 1 {
t.Fatal("Unable to find attached disk for data disk testing")
}

testCases := []struct {
expectDevice bool
cloneDiskSize int32
additionalCloneDiskSizes []int32
name string
disks object.VirtualDeviceList
dataDisks []infrav1.VSphereDisk
expectedDiskCount int
err string
}{
{
name: "Successfully add data disk",
disks: devices,
dataDisks: []infrav1.VSphereDisk{
{
SizeGiB: 10,
},
},
expectDevice: true,
},
{
name: "Successfully add multiple data disks",
disks: devices,
dataDisks: []infrav1.VSphereDisk{
{
SizeGiB: 10,
},
{
SizeGiB: 20,
},
},
expectDevice: true,
},
{
name: "Successfully add multiple data disks when template has multiple disks",
disks: append(devices, defaultDisks...),
dataDisks: []infrav1.VSphereDisk{
{
SizeGiB: 10,
},
{
SizeGiB: 20,
},
},
expectDevice: true,
},
}

for _, test := range testCases {
tc := test
t.Run(tc.name, func(t *testing.T) {
deviceResults, err := createDataDisks(ctx.TODO(), tc.dataDisks, tc.disks)
if (tc.err != "" && err == nil) || (tc.err == "" && err != nil) || (err != nil && tc.err != err.Error()) {
t.Fatalf("Expected to get '%v' error from createDataDisks, got: '%v'", tc.err, err)
}
if deviceFound := len(deviceResults) != 0; tc.expectDevice != deviceFound {
t.Fatalf("Expected to get a device: %v, but got: '%#v'", tc.expectDevice, deviceResults)
}
if tc.expectDevice {
// Check number of disks present
if len(deviceResults) != len(tc.dataDisks) {
t.Fatalf("Expected device count to be %v, but found %v", tc.expectedDiskCount, len(deviceResults))
}
}
})
}
}

func TestAddingDataDisks(t *testing.T) {
model, session, server := initSimulator(t)
t.Cleanup(model.Remove)
t.Cleanup(server.Close)
vm := simulator.Map.Any("VirtualMachine").(*simulator.VirtualMachine)
machine := object.NewVirtualMachine(session.Client.Client, vm.Reference())

deviceList, err := machine.Device(ctx.TODO())
if err != nil {
t.Fatalf("Failed to obtain vm devices: %v", err)
}

// Find primary disk and get controller
disks := deviceList.SelectByType((*types.VirtualDisk)(nil))
primaryDisk := disks[0].(*types.VirtualDisk)
controller, ok := deviceList.FindByKey(primaryDisk.ControllerKey).(types.BaseVirtualController)
if !ok {
t.Fatalf("unable to get controller for test")
}

testCases := []struct {
name string
devices object.VirtualDeviceList
controller types.BaseVirtualController
dataDisks []infrav1.VSphereDisk
expectedUnitNumber []int
err string
}{
{
name: "Add data disk with 1 ova disk",
devices: deviceList,
controller: controller,
dataDisks: createDataDiskDefinitions(1),
expectedUnitNumber: []int{1},
},
{
name: "Add data disk with 2 ova disk",
devices: createAdditionalDisks(deviceList, controller, 1),
controller: controller,
dataDisks: createDataDiskDefinitions(1),
expectedUnitNumber: []int{2},
},
{
name: "Add multiple data disk with 1 ova disk",
devices: deviceList,
controller: controller,
dataDisks: createDataDiskDefinitions(2),
expectedUnitNumber: []int{1, 2},
},
{
name: "Add too many data disks with 1 ova disk",
devices: deviceList,
controller: controller,
dataDisks: createDataDiskDefinitions(30),
err: "all unit numbers are already in-use",
},
{
name: "Add data disk with no ova disk",
devices: nil,
controller: nil,
dataDisks: createDataDiskDefinitions(1),
err: "Invalid disk count: 0",
},
{
name: "Add too many data disks with 1 ova disk",
devices: deviceList,
controller: controller,
dataDisks: createDataDiskDefinitions(40),
err: "all unit numbers are already in-use",
},
}

for _, test := range testCases {
tc := test
t.Run(tc.name, func(t *testing.T) {
var funcError error

// Create the data disks
newDisks, funcError := createDataDisks(ctx.TODO(), tc.dataDisks, tc.devices)
if (tc.err != "" && funcError == nil) || (tc.err == "" && funcError != nil) || (funcError != nil && tc.err != funcError.Error()) {
t.Fatalf("Expected to get '%v' error from assignUnitNumber, got: '%v'", tc.err, funcError)
}

// Validate the configs of new data disks
for index, disk := range newDisks {
if funcError != nil {
break
}

unitNumber := *disk.GetVirtualDeviceConfigSpec().Device.GetVirtualDevice().UnitNumber
if tc.err == "" && unitNumber != int32(tc.expectedUnitNumber[index]) {
t.Fatalf("Expected to get unitNumber '%d' error from assignUnitNumber, got: '%d'", tc.expectedUnitNumber[index], unitNumber)
}
}
})
}
}

func createAdditionalDisks(devices object.VirtualDeviceList, controller types.BaseVirtualController, numOfDisks int) object.VirtualDeviceList {
deviceList := devices
disks := devices.SelectByType((*types.VirtualDisk)(nil))
primaryDisk := disks[0].(*types.VirtualDisk)

for i := 0; i < numOfDisks; i++ {
newDevice := createVirtualDisk(primaryDisk.ControllerKey+1, controller, 10)
newUnitNumber := *primaryDisk.UnitNumber + int32(i+1)
newDevice.UnitNumber = &newUnitNumber
deviceList = append(deviceList, newDevice)
}
return deviceList
}

func createVirtualDisk(key int32, controller types.BaseVirtualController, diskSize int32) *types.VirtualDisk {
dev := &types.VirtualDisk{
VirtualDevice: types.VirtualDevice{
Key: key,
Backing: &types.VirtualDiskFlatVer2BackingInfo{
DiskMode: string(types.VirtualDiskModePersistent),
ThinProvisioned: types.NewBool(true),
VirtualDeviceFileBackingInfo: types.VirtualDeviceFileBackingInfo{
FileName: "",
},
},
},
CapacityInKB: int64(diskSize) * 1024 * 1024,
}

if controller != nil {
dev.VirtualDevice.ControllerKey = controller.GetVirtualController().Key
}
return dev
}

func createDataDiskDefinitions(numOfDataDisks int) []infrav1.VSphereDisk {
disks := []infrav1.VSphereDisk{}

for i := 0; i < numOfDataDisks; i++ {
disk := infrav1.VSphereDisk{
Name: fmt.Sprintf("disk_%d", i),
SizeGiB: 10 * int32(i),
}
disks = append(disks, disk)
}
return disks
}

func validateDiskSpec(t *testing.T, device types.BaseVirtualDeviceConfigSpec, cloneDiskSize int32) {
t.Helper()
disk := device.GetVirtualDeviceConfigSpec().Device.(*types.VirtualDisk)
Expand Down
1 change: 1 addition & 0 deletions test/e2e/config/vsphere.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -173,6 +173,7 @@ providers:
- sourcePath: "../../../test/e2e/data/infrastructure-vsphere-govmomi/main/cluster-template-ipam.yaml"
- sourcePath: "../../../test/e2e/data/infrastructure-vsphere-govmomi/main/cluster-template-kcp-remediation.yaml"
- sourcePath: "../../../test/e2e/data/infrastructure-vsphere-govmomi/main/cluster-template-md-remediation.yaml"
- sourcePath: "../../../test/e2e/data/infrastructure-vsphere-govmomi/main/cluster-template-multi-disk.yaml"
- sourcePath: "../../../test/e2e/data/infrastructure-vsphere-govmomi/main/cluster-template-node-drain.yaml"
- sourcePath: "../../../test/e2e/data/infrastructure-vsphere-govmomi/main/cluster-template-ownerrefs-finalizers.yaml"
- sourcePath: "../../../test/e2e/data/infrastructure-vsphere-govmomi/main/cluster-template-pci.yaml"
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
apiVersion: infrastructure.cluster.x-k8s.io/v1beta1
kind: VSphereMachineTemplate
metadata:
name: '${CLUSTER_NAME}'
namespace: '${NAMESPACE}'
spec:
template:
spec:
dataDisks:
- name: "disk_1"
sizeGiB: 10
- name: "disk_2"
sizeGiB: 20
---
apiVersion: infrastructure.cluster.x-k8s.io/v1beta1
kind: VSphereMachineTemplate
metadata:
name: '${CLUSTER_NAME}-worker'
namespace: '${NAMESPACE}'
spec:
template:
spec:
dataDisks:
- name: "disk_1"
sizeGiB: 20
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
apiVersion: kustomize.config.k8s.io/v1beta1
kind: Kustomization
resources:
- ../base
patchesStrategicMerge:
- data-disks-patch.yaml
Loading

0 comments on commit 8848cff

Please sign in to comment.