2022-09-05 03:06:08 -04:00
|
|
|
/*
|
|
|
|
Copyright (c) Edgeless Systems GmbH
|
|
|
|
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
2022-07-18 04:18:29 -04:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
2022-07-29 09:00:15 -04:00
|
|
|
"strings"
|
2022-07-18 04:18:29 -04:00
|
|
|
|
2022-08-30 09:15:51 -04:00
|
|
|
"github.com/Azure/azure-sdk-for-go/sdk/azcore/to"
|
2022-07-18 04:18:29 -04:00
|
|
|
"github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/compute/armcompute/v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
// GetScalingGroupImage returns the image URI of the scaling group.
|
|
|
|
func (c *Client) GetScalingGroupImage(ctx context.Context, scalingGroupID string) (string, error) {
|
|
|
|
_, resourceGroup, scaleSet, err := splitVMSSID(scalingGroupID)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
res, err := c.scaleSetsAPI.Get(ctx, resourceGroup, scaleSet, nil)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if res.Properties == nil ||
|
|
|
|
res.Properties.VirtualMachineProfile == nil ||
|
|
|
|
res.Properties.VirtualMachineProfile.StorageProfile == nil ||
|
|
|
|
res.Properties.VirtualMachineProfile.StorageProfile.ImageReference == nil ||
|
2022-08-30 09:15:51 -04:00
|
|
|
res.Properties.VirtualMachineProfile.StorageProfile.ImageReference.ID == nil && res.Properties.VirtualMachineProfile.StorageProfile.ImageReference.CommunityGalleryImageID == nil {
|
2022-07-18 04:18:29 -04:00
|
|
|
return "", fmt.Errorf("scalet set %q does not have valid image reference", scalingGroupID)
|
|
|
|
}
|
2022-08-30 09:15:51 -04:00
|
|
|
if res.Properties.VirtualMachineProfile.StorageProfile.ImageReference.ID != nil {
|
|
|
|
return *res.Properties.VirtualMachineProfile.StorageProfile.ImageReference.ID, nil
|
|
|
|
}
|
2022-10-05 09:02:46 -04:00
|
|
|
return *res.Properties.VirtualMachineProfile.StorageProfile.ImageReference.CommunityGalleryImageID, nil
|
2022-07-18 04:18:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// SetScalingGroupImage sets the image URI of the scaling group.
|
|
|
|
func (c *Client) SetScalingGroupImage(ctx context.Context, scalingGroupID, imageURI string) error {
|
|
|
|
_, resourceGroup, scaleSet, err := splitVMSSID(scalingGroupID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
poller, err := c.scaleSetsAPI.BeginUpdate(ctx, resourceGroup, scaleSet, armcompute.VirtualMachineScaleSetUpdate{
|
|
|
|
Properties: &armcompute.VirtualMachineScaleSetUpdateProperties{
|
|
|
|
VirtualMachineProfile: &armcompute.VirtualMachineScaleSetUpdateVMProfile{
|
|
|
|
StorageProfile: &armcompute.VirtualMachineScaleSetUpdateStorageProfile{
|
2022-08-30 09:15:51 -04:00
|
|
|
ImageReference: imageReferenceFromImage(imageURI),
|
2022-07-18 04:18:29 -04:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}, nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if _, err := poller.PollUntilDone(ctx, nil); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2022-07-29 09:00:15 -04:00
|
|
|
|
2022-09-15 10:48:32 -04:00
|
|
|
// GetScalingGroupName retrieves the name of a scaling group, as expected by Kubernetes.
|
|
|
|
// This keeps the casing of the original name, but Kubernetes requires the name to be lowercase,
|
|
|
|
// so use strings.ToLower() on the result if using the name in a Kubernetes context.
|
|
|
|
func (c *Client) GetScalingGroupName(scalingGroupID string) (string, error) {
|
2022-07-29 09:00:15 -04:00
|
|
|
_, _, scaleSet, err := splitVMSSID(scalingGroupID)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("getting scaling group name: %w", err)
|
|
|
|
}
|
2022-09-15 10:48:32 -04:00
|
|
|
return scaleSet, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetScalingGroupName retrieves the name of a scaling group as needed by the cluster-autoscaler.
|
|
|
|
func (c *Client) GetAutoscalingGroupName(scalingGroupID string) (string, error) {
|
|
|
|
return c.GetScalingGroupName(scalingGroupID)
|
2022-07-29 09:00:15 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// ListScalingGroups retrieves a list of scaling groups for the cluster.
|
|
|
|
func (c *Client) ListScalingGroups(ctx context.Context, uid string) (controlPlaneGroupIDs []string, workerGroupIDs []string, err error) {
|
|
|
|
scaleSetIDs, err := c.getScaleSets(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, fmt.Errorf("listing scaling groups: %w", err)
|
|
|
|
}
|
|
|
|
for _, scaleSetID := range scaleSetIDs {
|
|
|
|
_, _, scaleSet, err := splitVMSSID(scaleSetID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, fmt.Errorf("getting scaling group name: %w", err)
|
|
|
|
}
|
2022-10-06 06:14:56 -04:00
|
|
|
if isControlPlaneInstanceGroup(scaleSet) {
|
2022-07-29 09:00:15 -04:00
|
|
|
controlPlaneGroupIDs = append(controlPlaneGroupIDs, scaleSetID)
|
2022-10-06 06:14:56 -04:00
|
|
|
} else if isWorkerInstanceGroup(scaleSet) {
|
2022-07-29 09:00:15 -04:00
|
|
|
workerGroupIDs = append(workerGroupIDs, scaleSetID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return controlPlaneGroupIDs, workerGroupIDs, nil
|
|
|
|
}
|
2022-08-30 09:15:51 -04:00
|
|
|
|
2022-10-06 06:14:56 -04:00
|
|
|
// isControlPlaneInstanceGroup returns true if the instance group is a control plane instance group.
|
|
|
|
func isControlPlaneInstanceGroup(instanceGroupName string) bool {
|
|
|
|
return strings.Contains(instanceGroupName, "control-plane")
|
|
|
|
}
|
|
|
|
|
|
|
|
// isWorkerInstanceGroup returns true if the instance group is a worker instance group.
|
|
|
|
func isWorkerInstanceGroup(instanceGroupName string) bool {
|
|
|
|
return strings.Contains(instanceGroupName, "worker")
|
|
|
|
}
|
|
|
|
|
2022-08-30 09:15:51 -04:00
|
|
|
func imageReferenceFromImage(img string) *armcompute.ImageReference {
|
|
|
|
ref := &armcompute.ImageReference{}
|
|
|
|
|
|
|
|
if strings.HasPrefix(img, "/CommunityGalleries") {
|
|
|
|
ref.CommunityGalleryImageID = to.Ptr(img)
|
|
|
|
} else {
|
|
|
|
ref.ID = to.Ptr(img)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ref
|
|
|
|
}
|