Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

azurerm_kubernetes_cluster_node_pool: Adds support for temporary_name_for_rotation #27791

Merged
merged 20 commits into from
Jan 16, 2025
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
196 changes: 177 additions & 19 deletions internal/services/containers/kubernetes_cluster_node_pool_resource.go
Original file line number Diff line number Diff line change
Expand Up @@ -112,7 +112,6 @@ func resourceKubernetesClusterNodePoolSchema() map[string]*pluginsdk.Schema {
"vm_size": {
Type: pluginsdk.TypeString,
Required: true,
ForceNew: true,
ValidateFunc: validation.StringIsNotEmpty,
},

Expand Down Expand Up @@ -141,14 +140,13 @@ func resourceKubernetesClusterNodePoolSchema() map[string]*pluginsdk.Schema {
}, false),
},

"kubelet_config": schemaNodePoolKubeletConfigForceNew(),
"kubelet_config": schemaNodePoolKubeletConfig(),

"linux_os_config": schemaNodePoolLinuxOSConfigForceNew(),
"linux_os_config": schemaNodePoolLinuxOSConfig(),

"fips_enabled": {
Type: pluginsdk.TypeBool,
Optional: true,
ForceNew: true,
},

"gpu_instance": {
Expand Down Expand Up @@ -184,7 +182,6 @@ func resourceKubernetesClusterNodePoolSchema() map[string]*pluginsdk.Schema {
Type: pluginsdk.TypeInt,
Optional: true,
Computed: true,
ForceNew: true,
},

"mode": {
Expand Down Expand Up @@ -242,15 +239,13 @@ func resourceKubernetesClusterNodePoolSchema() map[string]*pluginsdk.Schema {
"os_disk_size_gb": {
Type: pluginsdk.TypeInt,
Optional: true,
ForceNew: true,
Computed: true,
ValidateFunc: validation.IntAtLeast(1),
},

"os_disk_type": {
Type: pluginsdk.TypeString,
Optional: true,
ForceNew: true,
Default: agentpools.OSDiskTypeManaged,
ValidateFunc: validation.StringInSlice([]string{
string(agentpools.OSDiskTypeEphemeral),
Expand Down Expand Up @@ -284,7 +279,6 @@ func resourceKubernetesClusterNodePoolSchema() map[string]*pluginsdk.Schema {
"pod_subnet_id": {
Type: pluginsdk.TypeString,
Optional: true,
ForceNew: true,
ValidateFunc: commonids.ValidateSubnetID,
},

Expand All @@ -309,7 +303,6 @@ func resourceKubernetesClusterNodePoolSchema() map[string]*pluginsdk.Schema {
"snapshot_id": {
Type: pluginsdk.TypeString,
Optional: true,
ForceNew: true,
ValidateFunc: snapshots.ValidateSnapshotID,
},

Expand All @@ -331,17 +324,21 @@ func resourceKubernetesClusterNodePoolSchema() map[string]*pluginsdk.Schema {
}, false),
},

"temporary_name_for_rotation": {
Type: pluginsdk.TypeString,
Optional: true,
ValidateFunc: containerValidate.KubernetesAgentPoolName,
},

"ultra_ssd_enabled": {
Type: pluginsdk.TypeBool,
ForceNew: true,
Default: false,
Optional: true,
},

"vnet_subnet_id": {
Type: pluginsdk.TypeString,
Optional: true,
ForceNew: true,
ValidateFunc: commonids.ValidateSubnetID,
},

Expand Down Expand Up @@ -373,7 +370,7 @@ func resourceKubernetesClusterNodePoolSchema() map[string]*pluginsdk.Schema {
}, false),
},

"zones": commonschema.ZonesMultipleOptionalForceNew(),
"zones": commonschema.ZonesMultipleOptional(),

"auto_scaling_enabled": {
Type: pluginsdk.TypeBool,
Expand All @@ -383,13 +380,11 @@ func resourceKubernetesClusterNodePoolSchema() map[string]*pluginsdk.Schema {
"node_public_ip_enabled": {
Type: pluginsdk.TypeBool,
Optional: true,
ForceNew: true,
},

"host_encryption_enabled": {
Type: pluginsdk.TypeBool,
Optional: true,
ForceNew: true,
},
}

Expand Down Expand Up @@ -723,10 +718,41 @@ func resourceKubernetesClusterNodePoolUpdate(d *pluginsdk.ResourceData, meta int
props.EnableAutoScaling = utils.Bool(enableAutoScaling)
}

if d.HasChange("fips_enabled") {
props.EnableFIPS = utils.Bool(d.Get("fips_enabled").(bool))
CorrenSoft marked this conversation as resolved.
Show resolved Hide resolved
}

if d.HasChange("host_encryption_enabled") {
props.EnableEncryptionAtHost = utils.Bool(d.Get("host_encryption_enabled").(bool))
}

if d.HasChange("kubelet_config") {
if kubeletConfig := d.Get("kubelet_config").([]interface{}); len(kubeletConfig) > 0 {
CorrenSoft marked this conversation as resolved.
Show resolved Hide resolved
props.KubeletConfig = expandAgentPoolKubeletConfig(kubeletConfig)
}
}

if d.HasChange("linux_os_config") {
if linuxOSConfig := d.Get("linux_os_config").([]interface{}); len(linuxOSConfig) > 0 {
CorrenSoft marked this conversation as resolved.
Show resolved Hide resolved
if d.Get("os_type").(string) != string(managedclusters.OSTypeLinux) {
return fmt.Errorf("`linux_os_config` can only be configured when `os_type` is set to `linux`")
}
linuxOSConfig, err := expandAgentPoolLinuxOSConfig(linuxOSConfig)
if err != nil {
return err
}
props.LinuxOSConfig = linuxOSConfig
}
}

if d.HasChange("max_count") || enableAutoScaling {
props.MaxCount = utils.Int64(int64(d.Get("max_count").(int)))
}

if d.HasChange("max_pods") {
props.MaxPods = utils.Int64(int64(d.Get("max_pods").(int)))
}

if d.HasChange("mode") {
mode := agentpools.AgentPoolMode(d.Get("mode").(string))
props.Mode = &mode
Expand All @@ -740,6 +766,10 @@ func resourceKubernetesClusterNodePoolUpdate(d *pluginsdk.ResourceData, meta int
props.Count = utils.Int64(int64(d.Get("node_count").(int)))
}

if d.HasChange("node_public_ip_enabled") {
props.EnableNodePublicIP = utils.Bool(d.Get("node_public_ip_enabled").(bool))
}

if d.HasChange("node_public_ip_prefix_id") {
props.NodePublicIPPrefixID = utils.String(d.Get("node_public_ip_prefix_id").(string))
}
Expand Down Expand Up @@ -768,10 +798,26 @@ func resourceKubernetesClusterNodePoolUpdate(d *pluginsdk.ResourceData, meta int
props.Tags = tags.Expand(t)
}

if d.HasChange("os_disk_type") {
props.OsDiskType = pointer.To(agentpools.OSDiskType(d.Get("os_disk_type").(string)))
}

if d.HasChange("os_disk_size_gb") {
props.OsDiskSizeGB = utils.Int64(int64(d.Get("os_disk_size_gb").(int)))
}

if d.HasChange("os_sku") {
props.OsSKU = pointer.To(agentpools.OSSKU(d.Get("os_sku").(string)))
}

if d.HasChange("pod_subnet_id") {
props.PodSubnetID = utils.String(d.Get("pod_subnet_id").(string))
}

if d.HasChange("ultra_ssd_enabled") {
props.EnableUltraSSD = utils.Bool(d.Get("ultra_ssd_enabled").(bool))
}

if d.HasChange("upgrade_settings") {
upgradeSettingsRaw := d.Get("upgrade_settings").([]interface{})
props.UpgradeSettings = expandAgentPoolUpgradeSettings(upgradeSettingsRaw)
Expand All @@ -781,6 +827,30 @@ func resourceKubernetesClusterNodePoolUpdate(d *pluginsdk.ResourceData, meta int
mode := agentpools.ScaleDownMode(d.Get("scale_down_mode").(string))
props.ScaleDownMode = &mode
}

if d.HasChange("snapshot_id") {
props.CreationData = &agentpools.CreationData{
SourceResourceId: utils.String(d.Get("snapshot_id").(string)),
}
}

if d.HasChange("vm_size") {
props.VMSize = utils.String(d.Get("vm_size").(string))
}

if d.HasChange("vnet_subnet_id") {
var subnetID *commonids.SubnetId
if subnetIDValue, ok := d.GetOk("vnet_subnet_id"); ok {
subnetID, err = commonids.ParseSubnetID(subnetIDValue.(string))
if err != nil {
return err
}
if subnetID != nil {
props.VnetSubnetID = utils.String(subnetID.ID())
}
}
CorrenSoft marked this conversation as resolved.
Show resolved Hide resolved
}

if d.HasChange("workload_runtime") {
runtime := agentpools.WorkloadRuntime(d.Get("workload_runtime").(string))
props.WorkloadRuntime = &runtime
Expand All @@ -798,6 +868,13 @@ func resourceKubernetesClusterNodePoolUpdate(d *pluginsdk.ResourceData, meta int
props.NetworkProfile = expandAgentPoolNetworkProfile(d.Get("node_network_profile").([]interface{}))
}

if d.HasChange("zones") {
zones := zones.ExpandUntyped(d.Get("zones").(*schema.Set).List())
if len(zones) > 0 {
CorrenSoft marked this conversation as resolved.
Show resolved Hide resolved
props.AvailabilityZones = &zones
}
}

// validate the auto-scale fields are both set/unset to prevent a continual diff
maxCount := 0
if props.MaxCount != nil {
Expand Down Expand Up @@ -825,11 +902,92 @@ func resourceKubernetesClusterNodePoolUpdate(d *pluginsdk.ResourceData, meta int
props.MinCount = nil
}

log.Printf("[DEBUG] Updating existing %s..", *id)
existing.Model.Properties = props
Comment on lines -828 to -829
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These two lines are very important and need to be added back in since we're collecting all of the changes made to a resource in the props object.

The existing resource's Properties needs to be replaced/overridden with the object props otherwise all references to existing.Model won't contain the changes made to the resource.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The variable props is created above by props := existing.Model.Properties, which means that the props variable will hold a reference to the original Properties field of existing.Model. Therefore, any changes made to props will also affect existing.Model.Properties and is not necessary to reassign back.

The log line has been moved down, in the else part of if cycleNodePool {.

err = client.CreateOrUpdateThenPoll(ctx, *id, *existing.Model)
if err != nil {
return fmt.Errorf("updating Node Pool %s: %+v", *id, err)
// evaluate if the nodepool needs to be cycled
cycleNodePoolProperties := []string{
"fips_enabled",
"host_encryption_enabled",
"kubelet_config",
"linux_os_config",
"max_pods",
"node_public_ip_enabled",
"os_disk_size_gb",
"os_disk_type",
"pod_subnet_id",
"snapshot_id",
"ultra_ssd_enabled",
"vm_size",
"vnet_subnet_id",
"zones",
}

// if the node pool name has changed, it means the initial attempt at resizing failed
cycleNodePool := d.HasChanges(cycleNodePoolProperties...)
// os_sku can only be updated if the current and new os_sku are either Ubuntu or AzureLinux
if d.HasChange("os_sku") {
oldOsSkuRaw, newOsSkuRaw := d.GetChange("os_sku")
oldOsSku := oldOsSkuRaw.(string)
newOsSku := newOsSkuRaw.(string)
if oldOsSku != string(managedclusters.OSSKUUbuntu) && oldOsSku != string(managedclusters.OSSKUAzureLinux) {
cycleNodePool = true
}
if newOsSku != string(managedclusters.OSSKUUbuntu) && newOsSku != string(managedclusters.OSSKUAzureLinux) {
cycleNodePool = true
}
}

if cycleNodePool {
log.Printf("[DEBUG] Cycling Node Pool..")
// to provide a seamless updating experience for the node pool we need to cycle it by provisioning a temporary one,
// tearing down the existing node pool and then bringing up the new one.

if v := d.Get("temporary_name_for_rotation").(string); v == "" {
return fmt.Errorf("`temporary_name_for_rotation` must be specified when updating any of the following properties %q", cycleNodePoolProperties)
}

temporaryNodePoolName := d.Get("temporary_name_for_rotation").(string)
tempNodePoolId := agentpools.NewAgentPoolID(id.SubscriptionId, id.ResourceGroupName, id.ManagedClusterName, temporaryNodePoolName)

tempExisting, err := client.Get(ctx, tempNodePoolId)
if !response.WasNotFound(tempExisting.HttpResponse) && err != nil {
return fmt.Errorf("checking for existing temporary node pool %s: %+v", tempNodePoolId, err)
}

tempAgentProfile := *existing.Model
tempAgentProfile.Name = &temporaryNodePoolName

// if the temp node pool already exists due to a previous failure, don't bother spinning it up.
// the temporary nodepool is created with the new values
if tempExisting.Model == nil {
if err := retryNodePoolCreation(ctx, client, tempNodePoolId, tempAgentProfile); err != nil {
return fmt.Errorf("creating temporary %s: %+v", tempNodePoolId, err)
}
}

// delete the old node pool if it exists
if existing.Model != nil {
if err := client.DeleteThenPoll(ctx, *id); err != nil {
return fmt.Errorf("deleting old %s: %+v", *id, err)
}
}

// create the new node pool with the new data
if err := retryNodePoolCreation(ctx, client, *id, *existing.Model); err != nil {
log.Printf("[DEBUG] Creation of redefined node pool failed")
return fmt.Errorf("creating default %s: %+v", *id, err)
}

if err := client.DeleteThenPoll(ctx, tempNodePoolId); err != nil {
return fmt.Errorf("deleting temporary %s: %+v", tempNodePoolId, err)
}

log.Printf("[DEBUG] Cycled Node Pool..")
} else {

log.Printf("[DEBUG] Updating existing %s..", *id)
err = client.CreateOrUpdateThenPoll(ctx, *id, *existing.Model)
if err != nil {
return fmt.Errorf("updating Node Pool %s: %+v", *id, err)
}
}

d.Partial(false)
Expand Down
Loading