Skip to content

Commit

Permalink
Fix delete for VMSS flex
Browse files Browse the repository at this point in the history
  • Loading branch information
CecileRobertMichon authored and mboersma committed Mar 10, 2023
1 parent 0f497f8 commit 094fe9b
Show file tree
Hide file tree
Showing 4 changed files with 57 additions and 21 deletions.
6 changes: 1 addition & 5 deletions azure/services/scalesetvms/scalesetvms.go
Original file line number Diff line number Diff line change
Expand Up @@ -151,11 +151,7 @@ func (s *Service) deleteVMSSFlexVM(ctx context.Context, resourceID string) error
if err != nil {
return errors.Wrap(err, fmt.Sprintf("failed to parse resource id %q", resourceID))
}

resourceGroup := parsed.ResourceGroup
resourceName := strings.TrimPrefix(s.Scope.ProviderID(), azure.ProviderIDPrefix)
resourceNameSplits := strings.Split(resourceName, "/")
resourceName = resourceNameSplits[len(resourceNameSplits)-3] + "_" + resourceNameSplits[len(resourceNameSplits)-1]
resourceGroup, resourceName := parsed.ResourceGroup, parsed.ResourceName

log.V(4).Info("entering delete")
future := s.Scope.GetLongRunningOperationState(resourceName, serviceName, infrav1.DeleteFuture)
Expand Down
66 changes: 54 additions & 12 deletions azure/services/scalesetvms/scalesetvms_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ package scalesetvms

import (
"context"
"fmt"
"net/http"
"testing"
"time"
Expand All @@ -35,6 +36,7 @@ import (
"sigs.k8s.io/cluster-api-provider-azure/azure/converters"
"sigs.k8s.io/cluster-api-provider-azure/azure/scope"
"sigs.k8s.io/cluster-api-provider-azure/azure/services/scalesetvms/mock_scalesetvms"
"sigs.k8s.io/cluster-api-provider-azure/azure/services/virtualmachines/mock_virtualmachines"
infrav1exp "sigs.k8s.io/cluster-api-provider-azure/exp/api/v1beta1"
gomock2 "sigs.k8s.io/cluster-api-provider-azure/internal/test/matchers/gomock"
clusterv1 "sigs.k8s.io/cluster-api/api/v1beta1"
Expand Down Expand Up @@ -172,17 +174,18 @@ func TestService_Reconcile(t *testing.T) {
func TestService_Delete(t *testing.T) {
cases := []struct {
Name string
Setup func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder)
Setup func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder, v *mock_virtualmachines.MockClientMockRecorder)
Err error
CheckIsErr bool
}{
{
Name: "should start deleting successfully if no long running operation is active",
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder) {
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder, v *mock_virtualmachines.MockClientMockRecorder) {
s.ResourceGroup().Return("rg")
s.InstanceID().Return("0")
s.ProviderID().Return("foo")
s.ScaleSetName().Return("scaleset")
s.OrchestrationMode().Return(infrav1.UniformOrchestrationMode)
s.GetLongRunningOperationState("0", serviceName, infrav1.DeleteFuture).Return(nil)
future := &infrav1.Future{
Type: infrav1.DeleteFuture,
Expand All @@ -199,11 +202,12 @@ func TestService_Delete(t *testing.T) {
},
{
Name: "should finish deleting successfully when there's a long running operation that has completed",
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder) {
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder, v *mock_virtualmachines.MockClientMockRecorder) {
s.ResourceGroup().Return("rg")
s.InstanceID().Return("0")
s.ProviderID().Return("foo")
s.ScaleSetName().Return("scaleset")
s.OrchestrationMode().Return(infrav1.UniformOrchestrationMode)
future := &infrav1.Future{
Type: infrav1.DeleteFuture,
}
Expand All @@ -215,23 +219,25 @@ func TestService_Delete(t *testing.T) {
},
{
Name: "should not error when deleting, but resource is 404",
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder) {
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder, v *mock_virtualmachines.MockClientMockRecorder) {
s.ResourceGroup().Return("rg")
s.InstanceID().Return("0")
s.ProviderID().Return("foo")
s.ScaleSetName().Return("scaleset")
s.OrchestrationMode().Return(infrav1.UniformOrchestrationMode)
s.GetLongRunningOperationState("0", serviceName, infrav1.DeleteFuture).Return(nil)
m.DeleteAsync(gomock2.AContext(), "rg", "scaleset", "0").Return(nil, autorest404)
m.Get(gomock2.AContext(), "rg", "scaleset", "0").Return(compute.VirtualMachineScaleSetVM{}, nil)
},
},
{
Name: "should error when deleting, but a non-404 error is returned from DELETE call",
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder) {
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder, v *mock_virtualmachines.MockClientMockRecorder) {
s.ResourceGroup().Return("rg")
s.InstanceID().Return("0")
s.ProviderID().Return("foo")
s.ScaleSetName().Return("scaleset")
s.OrchestrationMode().Return(infrav1.UniformOrchestrationMode)
s.GetLongRunningOperationState("0", serviceName, infrav1.DeleteFuture).Return(nil)
m.DeleteAsync(gomock2.AContext(), "rg", "scaleset", "0").Return(nil, errors.New("boom"))
m.Get(gomock2.AContext(), "rg", "scaleset", "0").Return(compute.VirtualMachineScaleSetVM{}, nil)
Expand All @@ -240,11 +246,12 @@ func TestService_Delete(t *testing.T) {
},
{
Name: "should return error when a long running operation is active and getting the result returns an error",
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder) {
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder, v *mock_virtualmachines.MockClientMockRecorder) {
s.ResourceGroup().Return("rg")
s.InstanceID().Return("0")
s.ProviderID().Return("foo")
s.ScaleSetName().Return("scaleset")
s.OrchestrationMode().Return(infrav1.UniformOrchestrationMode)
future := &infrav1.Future{
Type: infrav1.DeleteFuture,
}
Expand All @@ -254,26 +261,61 @@ func TestService_Delete(t *testing.T) {
},
Err: errors.Wrap(errors.New("boom"), "failed to get result of long running operation"),
},
{
Name: "(flex) should delete successfully if no long-running operation is active",
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder, v *mock_virtualmachines.MockClientMockRecorder) {
s.ResourceGroup().Return("my-cluster")
s.ScaleSetName().Return("scaleset")
s.InstanceID().Return("0")
s.ProviderID().Return("azure:///subscriptions/1234-5678/resourceGroups/my-cluster/providers/Microsoft.Compute/virtualMachines/my-cluster_1234abcd")
s.OrchestrationMode().Return(infrav1.FlexibleOrchestrationMode)
s.GetLongRunningOperationState("my-cluster_1234abcd", serviceName, infrav1.DeleteFuture).Return(nil)
vmGetter := &VMSSFlexVMGetter{
Name: "my-cluster_1234abcd",
ResourceGroup: "my-cluster",
}
future := &infrav1.Future{
Type: infrav1.DeleteFuture,
}
sdkFuture, _ := converters.FutureToSDK(*future)
v.DeleteAsync(gomock2.AContext(), vmGetter).Return(sdkFuture, nil)
s.DeleteLongRunningOperationState("my-cluster_1234abcd", serviceName, infrav1.DeleteFuture)
v.GetByID(gomock2.AContext(), "/subscriptions/1234-5678/resourceGroups/my-cluster/providers/Microsoft.Compute/virtualMachines/my-cluster_1234abcd").Return(compute.VirtualMachine{}, nil)
},
},
{
Name: "(flex) should error if providerID is invalid",
Setup: func(s *mock_scalesetvms.MockScaleSetVMScopeMockRecorder, m *mock_scalesetvms.MockclientMockRecorder, v *mock_virtualmachines.MockClientMockRecorder) {
s.ResourceGroup().Return("my-cluster")
s.ScaleSetName().Return("scaleset")
s.InstanceID().Return("0")
s.ProviderID().Return("foo")
s.OrchestrationMode().Return(infrav1.FlexibleOrchestrationMode)
v.GetByID(gomock2.AContext(), "foo").Return(compute.VirtualMachine{}, nil)
},
Err: errors.Wrap(fmt.Errorf("parsing failed for %s. Invalid resource Id format", "foo"), fmt.Sprintf("failed to parse resource id %q", "foo")),
},
}

for _, c := range cases {
t.Run(c.Name, func(t *testing.T) {
var (
g = NewWithT(t)
mockCtrl = gomock.NewController(t)
scopeMock = mock_scalesetvms.NewMockScaleSetVMScope(mockCtrl)
clientMock = mock_scalesetvms.NewMockclient(mockCtrl)
g = NewWithT(t)
mockCtrl = gomock.NewController(t)
scopeMock = mock_scalesetvms.NewMockScaleSetVMScope(mockCtrl)
clientMock = mock_scalesetvms.NewMockclient(mockCtrl)
vmClientMock = mock_virtualmachines.NewMockClient(mockCtrl)
)
defer mockCtrl.Finish()

scopeMock.EXPECT().SubscriptionID().Return("subID").AnyTimes()
scopeMock.EXPECT().BaseURI().Return("https://localhost/").AnyTimes()
scopeMock.EXPECT().Authorizer().Return(nil).AnyTimes()
scopeMock.EXPECT().OrchestrationMode().Return(infrav1.UniformOrchestrationMode).AnyTimes()

service := NewService(scopeMock)
service.Client = clientMock
c.Setup(scopeMock.EXPECT(), clientMock.EXPECT())
service.VMClient = vmClientMock
c.Setup(scopeMock.EXPECT(), clientMock.EXPECT(), vmClientMock.EXPECT())

if err := service.Delete(context.TODO()); c.Err == nil {
g.Expect(err).To(Succeed())
Expand Down
2 changes: 0 additions & 2 deletions test/e2e/azure_machinepools.go
Original file line number Diff line number Diff line change
Expand Up @@ -117,7 +117,6 @@ func AzureMachinePoolsSpec(ctx context.Context, inputGetter func() AzureMachineP
}
wg.Wait()

/* TODO: uncomment with scale down fix for flexible mode
for _, mp := range machinepools {
goalReplicas := pointer.Int32Deref(mp.Spec.Replicas, 0) - 1
Byf("Scaling machine pool %s in from %d to %d", mp.Name, *mp.Spec.Replicas, goalReplicas)
Expand All @@ -135,7 +134,6 @@ func AzureMachinePoolsSpec(ctx context.Context, inputGetter func() AzureMachineP
}(mp)
}
wg.Wait()
*/

By("verifying that workload nodes are schedulable")
clientset := workloadClusterProxy.GetClientSet()
Expand Down
4 changes: 2 additions & 2 deletions test/e2e/azure_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -523,7 +523,7 @@ var _ = Describe("Workload cluster creation", func() {
withClusterName(clusterName),
withControlPlaneMachineCount(1),
withWorkerMachineCount(1),
withKubernetesVersion("v1.26.1"),
withKubernetesVersion("v1.26.2"),
withMachineDeploymentInterval(specName, ""),
withControlPlaneWaiters(clusterctl.ControlPlaneWaiters{
WaitForControlPlaneInitialized: EnsureControlPlaneInitialized,
Expand All @@ -532,7 +532,7 @@ var _ = Describe("Workload cluster creation", func() {
withControlPlaneInterval(specName, "wait-control-plane"),
), result)

By("Verifying machinepool can scale out", func() {
By("Verifying machinepool can scale out and in", func() {
AzureMachinePoolsSpec(ctx, func() AzureMachinePoolsSpecInput {
return AzureMachinePoolsSpecInput{
Cluster: result.Cluster,
Expand Down

0 comments on commit 094fe9b

Please sign in to comment.