From 5eb037bfc8014b741872a4208b3bee1f6a109a85 Mon Sep 17 00:00:00 2001 From: teowa <104055472+teowa@users.noreply.github.com> Date: Sun, 25 Aug 2024 04:01:27 +0000 Subject: [PATCH 1/6] new resource stack hci virtual hard disk --- .../services/azurestackhci/registration.go | 1 + .../stack_hci_virtual_hard_disk_resource.go | 326 ++++++++++++++++++ ...ack_hci_virtual_hard_disk_resource_test.go | 202 +++++++++++ .../stack_hci_virtual_hard_disk.html.markdown | 77 +++++ 4 files changed, 606 insertions(+) create mode 100644 internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go create mode 100644 internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go create mode 100644 website/docs/r/stack_hci_virtual_hard_disk.html.markdown diff --git a/internal/services/azurestackhci/registration.go b/internal/services/azurestackhci/registration.go index 49f65474ad48..44b2e552a83c 100644 --- a/internal/services/azurestackhci/registration.go +++ b/internal/services/azurestackhci/registration.go @@ -52,5 +52,6 @@ func (r Registration) DataSources() []sdk.DataSource { func (r Registration) Resources() []sdk.Resource { return []sdk.Resource{ StackHCILogicalNetworkResource{}, + StackHCIVirtualHardDiskResource{}, } } diff --git a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go new file mode 100644 index 000000000000..cf36cf8baa0e --- /dev/null +++ b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go @@ -0,0 +1,326 @@ +package azurestackhci + +import ( + "context" + "fmt" + "regexp" + "time" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/go-azure-helpers/lang/response" + "github.com/hashicorp/go-azure-helpers/resourcemanager/commonschema" + "github.com/hashicorp/go-azure-helpers/resourcemanager/location" + "github.com/hashicorp/go-azure-helpers/resourcemanager/tags" + "github.com/hashicorp/go-azure-sdk/resource-manager/azurestackhci/2024-01-01/storagecontainers" + "github.com/hashicorp/go-azure-sdk/resource-manager/azurestackhci/2024-01-01/virtualharddisks" + "github.com/hashicorp/go-azure-sdk/resource-manager/extendedlocation/2021-08-15/customlocations" + "github.com/hashicorp/terraform-provider-azurerm/internal/sdk" + "github.com/hashicorp/terraform-provider-azurerm/internal/tf/pluginsdk" + "github.com/hashicorp/terraform-provider-azurerm/internal/tf/validation" +) + +var ( + _ sdk.Resource = StackHCIVirtualHardDiskResource{} + _ sdk.ResourceWithUpdate = StackHCIVirtualHardDiskResource{} +) + +type StackHCIVirtualHardDiskResource struct{} + +func (StackHCIVirtualHardDiskResource) IDValidationFunc() pluginsdk.SchemaValidateFunc { + return virtualharddisks.ValidateVirtualHardDiskID +} + +func (StackHCIVirtualHardDiskResource) ResourceType() string { + return "azurerm_stack_hci_virtual_hard_disk" +} + +func (StackHCIVirtualHardDiskResource) ModelObject() interface{} { + return &StackHCIVirtualHardDiskResourceModel{} +} + +type StackHCIVirtualHardDiskResourceModel struct { + Name string `tfschema:"name"` + ResourceGroupName string `tfschema:"resource_group_name"` + Location string `tfschema:"location"` + CustomLocationId string `tfschema:"custom_location_id"` + BlockSizeInBytes int64 `tfschema:"block_size_in_bytes"` + DiskFileFormat string `tfschema:"disk_file_format"` + DiskSizeInGB int64 `tfschema:"disk_size_in_gb"` + DynamicEnabled bool `tfschema:"dynamic_enabled"` + HypervGeneration string `tfschema:"hyperv_generation"` + LogicalSectorInBytes int64 `tfschema:"logical_sector_in_bytes"` + PhysicalSectorInBytes int64 `tfschema:"physical_sector_in_bytes"` + StoragePathId string `tfschema:"storage_path_id"` + Tags map[string]interface{} `tfschema:"tags"` +} + +func (StackHCIVirtualHardDiskResource) Arguments() map[string]*pluginsdk.Schema { + return map[string]*pluginsdk.Schema{ + "name": { + Type: pluginsdk.TypeString, + Required: true, + ForceNew: true, + ValidateFunc: validation.StringMatch( + regexp.MustCompile(`^[a-zA-Z0-9][-._a-zA-Z0-9]{0,62}[a-zA-Z0-9]$`), + "name must be between 2 and 64 characters and can only contain alphanumberic characters, hyphen, dot and underline", + ), + }, + + "resource_group_name": commonschema.ResourceGroupName(), + + "location": commonschema.Location(), + + "custom_location_id": { + Type: pluginsdk.TypeString, + Required: true, + ForceNew: true, + ValidateFunc: customlocations.ValidateCustomLocationID, + }, + + "block_size_in_bytes": { + Type: pluginsdk.TypeInt, + Optional: true, + ForceNew: true, + ValidateFunc: validation.IntAtLeast(1), + }, + + "disk_file_format": { + Type: pluginsdk.TypeString, + Optional: true, + ForceNew: true, + ValidateFunc: validation.StringInSlice([]string{ + string(virtualharddisks.DiskFileFormatVhd), + string(virtualharddisks.DiskFileFormatVhdx), + }, false), + }, + + "disk_size_in_gb": { + Type: pluginsdk.TypeInt, + Optional: true, + ForceNew: true, + ValidateFunc: validation.IntAtLeast(1), + }, + + "dynamic_enabled": { + Type: pluginsdk.TypeBool, + Optional: true, + ForceNew: true, + Default: false, + }, + + "hyperv_generation": { + Type: pluginsdk.TypeString, + Optional: true, + ForceNew: true, + ValidateFunc: validation.StringInSlice([]string{ + string(virtualharddisks.HyperVGenerationVOne), + string(virtualharddisks.HyperVGenerationVTwo), + }, false), + }, + + "logical_sector_in_bytes": { + Type: pluginsdk.TypeInt, + Optional: true, + ForceNew: true, + ValidateFunc: validation.IntAtLeast(1), + }, + + "physical_sector_in_bytes": { + Type: pluginsdk.TypeInt, + Optional: true, + ForceNew: true, + ValidateFunc: validation.IntAtLeast(1), + }, + + "storage_path_id": { + Type: pluginsdk.TypeString, + Optional: true, + ForceNew: true, + ValidateFunc: storagecontainers.ValidateStorageContainerID, + }, + + "tags": commonschema.Tags(), + } +} + +func (StackHCIVirtualHardDiskResource) Attributes() map[string]*pluginsdk.Schema { + return map[string]*pluginsdk.Schema{} +} + +func (r StackHCIVirtualHardDiskResource) Create() sdk.ResourceFunc { + return sdk.ResourceFunc{ + Timeout: 30 * time.Minute, + Func: func(ctx context.Context, metadata sdk.ResourceMetaData) error { + client := metadata.Client.AzureStackHCI.VirtualHardDisks + + var config StackHCIVirtualHardDiskResourceModel + if err := metadata.Decode(&config); err != nil { + return fmt.Errorf("decoding: %+v", err) + } + + subscriptionId := metadata.Client.Account.SubscriptionId + id := virtualharddisks.NewVirtualHardDiskID(subscriptionId, config.ResourceGroupName, config.Name) + + existing, err := client.Get(ctx, id) + if err != nil && !response.WasNotFound(existing.HttpResponse) { + return fmt.Errorf("checking for presence of existing %s: %+v", id, err) + } + if !response.WasNotFound(existing.HttpResponse) { + return metadata.ResourceRequiresImport(r.ResourceType(), id) + } + + payload := virtualharddisks.VirtualHardDisks{ + Name: pointer.To(config.Name), + Location: location.Normalize(config.Location), + Tags: tags.Expand(config.Tags), + ExtendedLocation: &virtualharddisks.ExtendedLocation{ + Name: pointer.To(config.CustomLocationId), + Type: pointer.To(virtualharddisks.ExtendedLocationTypesCustomLocation), + }, + Properties: &virtualharddisks.VirtualHardDiskProperties{ + Dynamic: pointer.To(config.DynamicEnabled), + }, + } + + if config.BlockSizeInBytes != 0 { + payload.Properties.BlockSizeBytes = pointer.To(config.BlockSizeInBytes) + } + + if config.StoragePathId != "" { + payload.Properties.ContainerId = pointer.To(config.StoragePathId) + } + + if config.DiskFileFormat != "" { + payload.Properties.DiskFileFormat = pointer.To(virtualharddisks.DiskFileFormat(config.DiskFileFormat)) + } + + if config.DiskSizeInGB != 0 { + payload.Properties.DiskSizeGB = pointer.To(config.DiskSizeInGB) + } + + if config.HypervGeneration != "" { + payload.Properties.HyperVGeneration = pointer.To(virtualharddisks.HyperVGeneration(config.HypervGeneration)) + } + + if config.LogicalSectorInBytes != 0 { + payload.Properties.LogicalSectorBytes = pointer.To(config.LogicalSectorInBytes) + } + + if config.PhysicalSectorInBytes != 0 { + payload.Properties.PhysicalSectorBytes = pointer.To(config.PhysicalSectorInBytes) + } + + if err := client.CreateOrUpdateThenPoll(ctx, id, payload); err != nil { + return fmt.Errorf("performing create %s: %+v", id, err) + } + + metadata.SetID(id) + + return nil + }, + } +} + +func (r StackHCIVirtualHardDiskResource) Read() sdk.ResourceFunc { + return sdk.ResourceFunc{ + Timeout: 5 * time.Minute, + Func: func(ctx context.Context, metadata sdk.ResourceMetaData) error { + client := metadata.Client.AzureStackHCI.VirtualHardDisks + + id, err := virtualharddisks.ParseVirtualHardDiskID(metadata.ResourceData.Id()) + if err != nil { + return err + } + + resp, err := client.Get(ctx, *id) + if err != nil { + if response.WasNotFound(resp.HttpResponse) { + return metadata.MarkAsGone(id) + } + + return fmt.Errorf("retrieving %s: %+v", id, err) + } + + schema := StackHCIVirtualHardDiskResourceModel{ + Name: id.VirtualHardDiskName, + ResourceGroupName: id.ResourceGroupName, + } + + if model := resp.Model; model != nil { + schema.Location = location.Normalize(model.Location) + schema.Tags = tags.Flatten(model.Tags) + + if model.ExtendedLocation != nil && model.ExtendedLocation.Name != nil { + customLocationId, err := customlocations.ParseCustomLocationIDInsensitively(*model.ExtendedLocation.Name) + if err != nil { + return err + } + + schema.CustomLocationId = customLocationId.ID() + } + + if props := model.Properties; props != nil { + schema.BlockSizeInBytes = pointer.From(props.BlockSizeBytes) + schema.DiskFileFormat = string(pointer.From(props.DiskFileFormat)) + schema.DiskSizeInGB = pointer.From(props.DiskSizeGB) + schema.DynamicEnabled = pointer.From(props.Dynamic) + schema.HypervGeneration = string(pointer.From(props.HyperVGeneration)) + schema.LogicalSectorInBytes = pointer.From(props.LogicalSectorBytes) + schema.PhysicalSectorInBytes = pointer.From(props.PhysicalSectorBytes) + schema.StoragePathId = pointer.From(props.ContainerId) + } + } + + return metadata.Encode(&schema) + }, + } +} + +func (r StackHCIVirtualHardDiskResource) Update() sdk.ResourceFunc { + return sdk.ResourceFunc{ + Timeout: 30 * time.Minute, + Func: func(ctx context.Context, metadata sdk.ResourceMetaData) error { + client := metadata.Client.AzureStackHCI.VirtualHardDisks + + id, err := virtualharddisks.ParseVirtualHardDiskID(metadata.ResourceData.Id()) + if err != nil { + return err + } + + var model StackHCIVirtualHardDiskResourceModel + if err := metadata.Decode(&model); err != nil { + return fmt.Errorf("decoding: %+v", err) + } + + parameters := virtualharddisks.VirtualHardDisksUpdateRequest{} + if metadata.ResourceData.HasChange("tags") { + parameters.Tags = tags.Expand(model.Tags) + } + + if err := client.UpdateThenPoll(ctx, *id, parameters); err != nil { + return fmt.Errorf("updating %s: %+v", id, err) + } + return nil + }, + } +} + +func (r StackHCIVirtualHardDiskResource) Delete() sdk.ResourceFunc { + return sdk.ResourceFunc{ + Timeout: 30 * time.Minute, + Func: func(ctx context.Context, metadata sdk.ResourceMetaData) error { + client := metadata.Client.AzureStackHCI.VirtualHardDisks + + id, err := virtualharddisks.ParseVirtualHardDiskID(metadata.ResourceData.Id()) + if err != nil { + return err + } + + if err := client.DeleteThenPoll(ctx, *id); err != nil { + return fmt.Errorf("deleting %s: %+v", id, err) + } + + return nil + }, + } +} diff --git a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go new file mode 100644 index 000000000000..a45bb1e3615c --- /dev/null +++ b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go @@ -0,0 +1,202 @@ +package azurestackhci_test + +import ( + "context" + "fmt" + "os" + "testing" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/go-azure-sdk/resource-manager/azurestackhci/2024-01-01/virtualharddisks" + "github.com/hashicorp/terraform-provider-azurerm/internal/acceptance" + "github.com/hashicorp/terraform-provider-azurerm/internal/acceptance/check" + "github.com/hashicorp/terraform-provider-azurerm/internal/clients" + "github.com/hashicorp/terraform-provider-azurerm/internal/tf/pluginsdk" +) + +type StackHCIVirtualHardDiskResource struct{} + +func TestAccStackHCIVirtualHardDisk_basic(t *testing.T) { + data := acceptance.BuildTestData(t, "azurerm_stack_hci_virtual_hard_disk", "test") + r := StackHCIVirtualHardDiskResource{} + + if os.Getenv(customLocationIdEnv) == "" { + t.Skipf("skipping since %q has not been specified", customLocationIdEnv) + } + + data.ResourceTest(t, r, []acceptance.TestStep{ + { + Config: r.basic(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.ImportStep(), + }) +} + +func TestAccStackHCIVirtualHardDisk_complete(t *testing.T) { + if os.Getenv(customLocationIdEnv) == "" { + t.Skipf("skipping since %q has not been specified", customLocationIdEnv) + } + + data := acceptance.BuildTestData(t, "azurerm_stack_hci_virtual_hard_disk", "test") + r := StackHCIVirtualHardDiskResource{} + + data.ResourceTest(t, r, []acceptance.TestStep{ + { + Config: r.complete(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.ImportStep(), + }) +} + +func TestAccStackHCIVirtualHardDisk_update(t *testing.T) { + data := acceptance.BuildTestData(t, "azurerm_stack_hci_virtual_hard_disk", "test") + r := StackHCIVirtualHardDiskResource{} + + if os.Getenv(customLocationIdEnv) == "" { + t.Skipf("skipping since %q has not been specified", customLocationIdEnv) + } + + data.ResourceTest(t, r, []acceptance.TestStep{ + { + Config: r.basic(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.ImportStep(), + { + Config: r.complete(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.ImportStep(), + { + Config: r.basic(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.ImportStep(), + }) +} + +func TestAccStackHCIVirtualHardDisk_requiresImport(t *testing.T) { + if os.Getenv(customLocationIdEnv) == "" { + t.Skipf("skipping since %q has not been specified", customLocationIdEnv) + } + + data := acceptance.BuildTestData(t, "azurerm_stack_hci_virtual_hard_disk", "test") + r := StackHCIVirtualHardDiskResource{} + + data.ResourceTest(t, r, []acceptance.TestStep{ + { + Config: r.basic(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.RequiresImportErrorStep(r.requiresImport), + }) +} + +func (r StackHCIVirtualHardDiskResource) Exists(ctx context.Context, client *clients.Client, state *pluginsdk.InstanceState) (*bool, error) { + clusterClient := client.AzureStackHCI.VirtualHardDisks + id, err := virtualharddisks.ParseVirtualHardDiskID(state.ID) + if err != nil { + return nil, err + } + + resp, err := clusterClient.Get(ctx, *id) + if err != nil { + return nil, fmt.Errorf("retrieving %s: %+v", *id, err) + } + + return pointer.To(resp.Model != nil), nil +} + +func (r StackHCIVirtualHardDiskResource) requiresImport(data acceptance.TestData) string { + config := r.basic(data) + + return fmt.Sprintf(` +%s + +resource "azurerm_stack_hci_virtual_hard_disk" "import" { + name = azurerm_stack_hci_virtual_hard_disk.test.name + resource_group_name = azurerm_stack_hci_virtual_hard_disk.test.resource_group_name + location = azurerm_stack_hci_virtual_hard_disk.test.location + custom_location_id = azurerm_stack_hci_virtual_hard_disk.test.custom_location_id + disk_size_in_gb = azurerm_stack_hci_virtual_hard_disk.test.disk_size_in_gb +} +`, config) +} + +func (r StackHCIVirtualHardDiskResource) basic(data acceptance.TestData) string { + template := r.template(data) + return fmt.Sprintf(` +%s + +provider "azurerm" { + features {} +} + +resource "azurerm_stack_hci_virtual_hard_disk" "test" { + name = "acctest-ln-${var.random_string}" + resource_group_name = azurerm_resource_group.test.name + location = azurerm_resource_group.test.location + custom_location_id = %q + disk_size_in_gb = 2 + lifecycle { + ignore_changes = [storage_path_id] + } +} +`, template, os.Getenv(customLocationIdEnv)) +} + +func (r StackHCIVirtualHardDiskResource) complete(data acceptance.TestData) string { + template := r.template(data) + return fmt.Sprintf(` +%s + +provider "azurerm" { + features {} +} + +resource "azurerm_stack_hci_virtual_hard_disk" "test" { + name = "acctest-ln-${var.random_string}" + resource_group_name = azurerm_resource_group.test.name + location = azurerm_resource_group.test.location + custom_location_id = %q + disk_size_in_gb = 2 + + + tags = { + foo = "bar" + env = "test" + } +} +`, template, os.Getenv(customLocationIdEnv)) +} + +func (r StackHCIVirtualHardDiskResource) template(data acceptance.TestData) string { + return fmt.Sprintf(` +variable "primary_location" { + default = %q +} + +variable "random_string" { + default = %q +} + +resource "azurerm_resource_group" "test" { + name = "acctest-hci-ln-${var.random_string}" + location = var.primary_location +} +`, data.Locations.Primary, data.RandomString) +} diff --git a/website/docs/r/stack_hci_virtual_hard_disk.html.markdown b/website/docs/r/stack_hci_virtual_hard_disk.html.markdown new file mode 100644 index 000000000000..9e4a52a483b2 --- /dev/null +++ b/website/docs/r/stack_hci_virtual_hard_disk.html.markdown @@ -0,0 +1,77 @@ +--- +subcategory: "Azure Stack HCI" +layout: "azurerm" +page_title: "Azure Resource Manager: azurerm_stack_hci_virtual_hard_disk" +description: |- + Manages an Azure Stack HCI Virtual Hard Disk. +--- + +# azurerm_stack_hci_virtual_hard_disk + +Manages an Azure Stack HCI Virtual Hard Disk. + +## Example Usage + +```hcl +resource "azurerm_stack_hci_virtual_hard_disk" "example" { + name = "example" + resource_group_name = "example" + location = "West Europe" + custom_location_id = "TODO" +} +``` + +## Arguments Reference + +The following arguments are supported: + +* `custom_location_id` - (Required) The ID of the TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `location` - (Required) The Azure Region where the Azure Stack HCI Virtual Hard Disk should exist. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `name` - (Required) The name which should be used for this Azure Stack HCI Virtual Hard Disk. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `resource_group_name` - (Required) The name of the Resource Group where the Azure Stack HCI Virtual Hard Disk should exist. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +--- + +* `block_size_in_bytes` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `disk_file_format` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `disk_size_in_gb` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `dynamic_enabled` - (Optional) Should the TODO be enabled? Defaults to `false`. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `hyperv_generation` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `logical_sector_in_bytes` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `physical_sector_in_bytes` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `storage_path_id` - (Optional) The ID of the TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `tags` - (Optional) A mapping of tags which should be assigned to the Azure Stack HCI Virtual Hard Disk. + +## Attributes Reference + +In addition to the Arguments listed above - the following Attributes are exported: + +* `id` - The ID of the Azure Stack HCI Virtual Hard Disk. + +## Timeouts + +The `timeouts` block allows you to specify [timeouts](https://www.terraform.io/language/resources/syntax#operation-timeouts) for certain actions: + +* `create` - (Defaults to 30 minutes) Used when creating the Azure Stack HCI Virtual Hard Disk. +* `read` - (Defaults to 5 minutes) Used when retrieving the Azure Stack HCI Virtual Hard Disk. +* `update` - (Defaults to 30 minutes) Used when updating the Azure Stack HCI Virtual Hard Disk. +* `delete` - (Defaults to 30 minutes) Used when deleting the Azure Stack HCI Virtual Hard Disk. + +## Import + +Azure Stack HCI Virtual Hard Disks can be imported using the `resource id`, e.g. + +```shell +terraform import azurerm_stack_hci_virtual_hard_disk.example /subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/rg1/providers/Microsoft.AzureStackHCI/virtualHardDisks/disk1 +``` From b40bccdc9c9c83db80cc4710b39b5e56f506a87e Mon Sep 17 00:00:00 2001 From: teowa <104055472+teowa@users.noreply.github.com> Date: Tue, 24 Sep 2024 06:32:06 +0000 Subject: [PATCH 2/6] fix test --- .../stack_hci_virtual_hard_disk_resource.go | 2 +- ...ack_hci_virtual_hard_disk_resource_test.go | 95 ++++++++++++------- .../stack_hci_virtual_hard_disk.html.markdown | 50 +++++++--- 3 files changed, 99 insertions(+), 48 deletions(-) diff --git a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go index cf36cf8baa0e..de5d5ebb6daf 100644 --- a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go +++ b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go @@ -96,7 +96,7 @@ func (StackHCIVirtualHardDiskResource) Arguments() map[string]*pluginsdk.Schema "disk_size_in_gb": { Type: pluginsdk.TypeInt, - Optional: true, + Required: true, ForceNew: true, ValidateFunc: validation.IntAtLeast(1), }, diff --git a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go index a45bb1e3615c..97cc1c152e4d 100644 --- a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go +++ b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go @@ -17,13 +17,13 @@ import ( type StackHCIVirtualHardDiskResource struct{} func TestAccStackHCIVirtualHardDisk_basic(t *testing.T) { - data := acceptance.BuildTestData(t, "azurerm_stack_hci_virtual_hard_disk", "test") - r := StackHCIVirtualHardDiskResource{} - if os.Getenv(customLocationIdEnv) == "" { t.Skipf("skipping since %q has not been specified", customLocationIdEnv) } + data := acceptance.BuildTestData(t, "azurerm_stack_hci_virtual_hard_disk", "test") + r := StackHCIVirtualHardDiskResource{} + data.ResourceTest(t, r, []acceptance.TestStep{ { Config: r.basic(data), @@ -55,13 +55,13 @@ func TestAccStackHCIVirtualHardDisk_complete(t *testing.T) { } func TestAccStackHCIVirtualHardDisk_update(t *testing.T) { - data := acceptance.BuildTestData(t, "azurerm_stack_hci_virtual_hard_disk", "test") - r := StackHCIVirtualHardDiskResource{} - if os.Getenv(customLocationIdEnv) == "" { t.Skipf("skipping since %q has not been specified", customLocationIdEnv) } + data := acceptance.BuildTestData(t, "azurerm_stack_hci_virtual_hard_disk", "test") + r := StackHCIVirtualHardDiskResource{} + data.ResourceTest(t, r, []acceptance.TestStep{ { Config: r.basic(data), @@ -71,7 +71,7 @@ func TestAccStackHCIVirtualHardDisk_update(t *testing.T) { }, data.ImportStep(), { - Config: r.complete(data), + Config: r.update(data), Check: acceptance.ComposeTestCheckFunc( check.That(data.ResourceName).ExistsInAzure(r), ), @@ -121,23 +121,30 @@ func (r StackHCIVirtualHardDiskResource) Exists(ctx context.Context, client *cli return pointer.To(resp.Model != nil), nil } -func (r StackHCIVirtualHardDiskResource) requiresImport(data acceptance.TestData) string { - config := r.basic(data) - +func (r StackHCIVirtualHardDiskResource) basic(data acceptance.TestData) string { + template := r.template(data) return fmt.Sprintf(` %s -resource "azurerm_stack_hci_virtual_hard_disk" "import" { - name = azurerm_stack_hci_virtual_hard_disk.test.name - resource_group_name = azurerm_stack_hci_virtual_hard_disk.test.resource_group_name - location = azurerm_stack_hci_virtual_hard_disk.test.location - custom_location_id = azurerm_stack_hci_virtual_hard_disk.test.custom_location_id - disk_size_in_gb = azurerm_stack_hci_virtual_hard_disk.test.disk_size_in_gb +provider "azurerm" { + features {} } -`, config) + +resource "azurerm_stack_hci_virtual_hard_disk" "test" { + name = "acctest-vhd-%s" + resource_group_name = azurerm_resource_group.test.name + location = azurerm_resource_group.test.location + custom_location_id = %q + disk_size_in_gb = 2 + + lifecycle { + ignore_changes = [storage_path_id] + } +} +`, template, data.RandomString, os.Getenv(customLocationIdEnv)) } -func (r StackHCIVirtualHardDiskResource) basic(data acceptance.TestData) string { +func (r StackHCIVirtualHardDiskResource) update(data acceptance.TestData) string { template := r.template(data) return fmt.Sprintf(` %s @@ -147,56 +154,78 @@ provider "azurerm" { } resource "azurerm_stack_hci_virtual_hard_disk" "test" { - name = "acctest-ln-${var.random_string}" + name = "acctest-vhd-%s" resource_group_name = azurerm_resource_group.test.name location = azurerm_resource_group.test.location custom_location_id = %q disk_size_in_gb = 2 + + tags = { + foo = "bar" + env = "test" + } + lifecycle { ignore_changes = [storage_path_id] } } -`, template, os.Getenv(customLocationIdEnv)) +`, template, data.RandomString, os.Getenv(customLocationIdEnv)) } func (r StackHCIVirtualHardDiskResource) complete(data acceptance.TestData) string { template := r.template(data) return fmt.Sprintf(` -%s +%[1]s provider "azurerm" { features {} } +resource "azurerm_stack_hci_storage_path" "test" { + name = "acctest-sp-%[2]s" + resource_group_name = azurerm_resource_group.test.name + location = azurerm_resource_group.test.location + custom_location_id = %[3]q + path = "C:\\ClusterStorage\\UserStorage_2\\sp-%[2]s" +} + resource "azurerm_stack_hci_virtual_hard_disk" "test" { - name = "acctest-ln-${var.random_string}" + name = "acctest-vhd-%[2]s" resource_group_name = azurerm_resource_group.test.name location = azurerm_resource_group.test.location - custom_location_id = %q + custom_location_id = %[3]q disk_size_in_gb = 2 - + storage_path_id = azurerm_stack_hci_storage_path.test.id tags = { foo = "bar" env = "test" } } -`, template, os.Getenv(customLocationIdEnv)) +`, template, data.RandomString, os.Getenv(customLocationIdEnv)) } -func (r StackHCIVirtualHardDiskResource) template(data acceptance.TestData) string { +func (r StackHCIVirtualHardDiskResource) requiresImport(data acceptance.TestData) string { + config := r.basic(data) + return fmt.Sprintf(` -variable "primary_location" { - default = %q -} +%s -variable "random_string" { - default = %q +resource "azurerm_stack_hci_virtual_hard_disk" "import" { + name = azurerm_stack_hci_virtual_hard_disk.test.name + resource_group_name = azurerm_stack_hci_virtual_hard_disk.test.resource_group_name + location = azurerm_stack_hci_virtual_hard_disk.test.location + custom_location_id = azurerm_stack_hci_virtual_hard_disk.test.custom_location_id + disk_size_in_gb = azurerm_stack_hci_virtual_hard_disk.test.disk_size_in_gb +} +`, config) } +func (r StackHCIVirtualHardDiskResource) template(data acceptance.TestData) string { + return fmt.Sprintf(` resource "azurerm_resource_group" "test" { - name = "acctest-hci-ln-${var.random_string}" - location = var.primary_location + name = "acctest-hci-vhd-%[2]s" + location = %[1]q } `, data.Locations.Primary, data.RandomString) } diff --git a/website/docs/r/stack_hci_virtual_hard_disk.html.markdown b/website/docs/r/stack_hci_virtual_hard_disk.html.markdown index 9e4a52a483b2..f6bced75e1ad 100644 --- a/website/docs/r/stack_hci_virtual_hard_disk.html.markdown +++ b/website/docs/r/stack_hci_virtual_hard_disk.html.markdown @@ -13,11 +13,31 @@ Manages an Azure Stack HCI Virtual Hard Disk. ## Example Usage ```hcl +resource "azurerm_resource_group" "example" { + name = "example-rg" + location = "West Europe" +} + +resource "azurerm_stack_hci_storage_path" "example" { + name = "example-sp" + resource_group_name = azurerm_resource_group.example.name + location = azurerm_resource_group.example.location + custom_location_id = "/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/rg1/providers/Microsoft.ExtendedLocation/customLocations/cl1" + path = "C:\\ClusterStorage\\UserStorage_2\\sp-example" + tags = { + foo = "bar" + } +} + resource "azurerm_stack_hci_virtual_hard_disk" "example" { name = "example" - resource_group_name = "example" - location = "West Europe" - custom_location_id = "TODO" + resource_group_name = azurerm_resource_group.example.name + location = azurerm_resource_group.example.location + custom_location_id = "/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/rg1/providers/Microsoft.ExtendedLocation/customLocations/cl1" + storage_path_id = azurerm_stack_hci_storage_path.example.id + tags = { + foo = "bar" + } } ``` @@ -25,31 +45,33 @@ resource "azurerm_stack_hci_virtual_hard_disk" "example" { The following arguments are supported: -* `custom_location_id` - (Required) The ID of the TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +* `name` - (Required) The name which should be used for this Azure Stack HCI Virtual Hard Disk. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. + +* `resource_group_name` - (Required) The name of the Resource Group where the Azure Stack HCI Virtual Hard Disk should exist. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. * `location` - (Required) The Azure Region where the Azure Stack HCI Virtual Hard Disk should exist. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. -* `name` - (Required) The name which should be used for this Azure Stack HCI Virtual Hard Disk. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +* `custom_location_id` - (Required) The ID of the Custom Location where the Azure Stack HCI Virtual Hard Disk should exist. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. -* `resource_group_name` - (Required) The name of the Resource Group where the Azure Stack HCI Virtual Hard Disk should exist. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +* `disk_size_in_gb` - (Required) The size of the disk in GB. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. --- -* `block_size_in_bytes` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +* `block_size_in_bytes` - (Optional) The block size of the disk in bytes. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. -* `disk_file_format` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +* `disk_file_format` - (Optional) The format of the disk file. Possible values are `vhdx` and `vhd`. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. -* `disk_size_in_gb` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +* `dynamic_enabled` - (Optional) Whether to enable dynamic sizing for the Azure Stack HCI Virtual Hard Disk. Defaults to `false`. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. -* `dynamic_enabled` - (Optional) Should the TODO be enabled? Defaults to `false`. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +* `hyperv_generation` - (Optional) The hypervisor generation of the Azure Stack HCI Virtual Hard Disk. Possible values are `V1` and `V2`. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. -* `hyperv_generation` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +* `logical_sector_in_bytes` - (Optional) The logical sector size of the disk in bytes. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. -* `logical_sector_in_bytes` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +* `physical_sector_in_bytes` - (Optional) The physical sector size of the disk in bytes. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. -* `physical_sector_in_bytes` - (Optional) TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +* `storage_path_id` - (Optional) The ID of the Azure Stack HCI Storage Path used for this Virtual Hard Disk. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. -* `storage_path_id` - (Optional) The ID of the TODO. Changing this forces a new Azure Stack HCI Virtual Hard Disk to be created. +-> **Note:** If `storage_path_id` is not specified, the Virtual Hard Disk will be placed in a high availability Storage Path. If you experience a diff you may need to add this to `ignore_changes`. * `tags` - (Optional) A mapping of tags which should be assigned to the Azure Stack HCI Virtual Hard Disk. From aee7c956a738550296441b5e6a44438f87f7efb2 Mon Sep 17 00:00:00 2001 From: teowa <104055472+teowa@users.noreply.github.com> Date: Tue, 24 Sep 2024 06:39:48 +0000 Subject: [PATCH 3/6] terrafmt --- .../stack_hci_virtual_hard_disk_resource_test.go | 4 ++-- website/docs/r/stack_hci_virtual_hard_disk.html.markdown | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go index 97cc1c152e4d..ced2ada5a3c6 100644 --- a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go +++ b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go @@ -195,7 +195,7 @@ resource "azurerm_stack_hci_virtual_hard_disk" "test" { location = azurerm_resource_group.test.location custom_location_id = %[3]q disk_size_in_gb = 2 - storage_path_id = azurerm_stack_hci_storage_path.test.id + storage_path_id = azurerm_stack_hci_storage_path.test.id tags = { foo = "bar" @@ -216,7 +216,7 @@ resource "azurerm_stack_hci_virtual_hard_disk" "import" { resource_group_name = azurerm_stack_hci_virtual_hard_disk.test.resource_group_name location = azurerm_stack_hci_virtual_hard_disk.test.location custom_location_id = azurerm_stack_hci_virtual_hard_disk.test.custom_location_id - disk_size_in_gb = azurerm_stack_hci_virtual_hard_disk.test.disk_size_in_gb + disk_size_in_gb = azurerm_stack_hci_virtual_hard_disk.test.disk_size_in_gb } `, config) } diff --git a/website/docs/r/stack_hci_virtual_hard_disk.html.markdown b/website/docs/r/stack_hci_virtual_hard_disk.html.markdown index f6bced75e1ad..78aeddc8085c 100644 --- a/website/docs/r/stack_hci_virtual_hard_disk.html.markdown +++ b/website/docs/r/stack_hci_virtual_hard_disk.html.markdown @@ -30,7 +30,7 @@ resource "azurerm_stack_hci_storage_path" "example" { } resource "azurerm_stack_hci_virtual_hard_disk" "example" { - name = "example" + name = "example" resource_group_name = azurerm_resource_group.example.name location = azurerm_resource_group.example.location custom_location_id = "/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/rg1/providers/Microsoft.ExtendedLocation/customLocations/cl1" From 22e7fc502863f061177a4a1b1c0c809803bc5e5d Mon Sep 17 00:00:00 2001 From: teowa <104055472+teowa@users.noreply.github.com> Date: Sun, 29 Sep 2024 07:47:18 +0000 Subject: [PATCH 4/6] update schema;update test --- .../stack_hci_virtual_hard_disk_resource.go | 7 +--- ...ack_hci_virtual_hard_disk_resource_test.go | 34 +++++++++++++++++++ 2 files changed, 35 insertions(+), 6 deletions(-) diff --git a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go index de5d5ebb6daf..4c4c7ed1e95f 100644 --- a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go +++ b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go @@ -70,12 +70,7 @@ func (StackHCIVirtualHardDiskResource) Arguments() map[string]*pluginsdk.Schema "location": commonschema.Location(), - "custom_location_id": { - Type: pluginsdk.TypeString, - Required: true, - ForceNew: true, - ValidateFunc: customlocations.ValidateCustomLocationID, - }, + "custom_location_id": commonschema.ResourceIDReferenceRequiredForceNew(&customlocations.CustomLocationId{}), "block_size_in_bytes": { Type: pluginsdk.TypeInt, diff --git a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go index ced2ada5a3c6..c4480a43aec3 100644 --- a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go +++ b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go @@ -77,6 +77,13 @@ func TestAccStackHCIVirtualHardDisk_update(t *testing.T) { ), }, data.ImportStep(), + { + Config: r.updateTag(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.ImportStep(), { Config: r.basic(data), Check: acceptance.ComposeTestCheckFunc( @@ -162,7 +169,34 @@ resource "azurerm_stack_hci_virtual_hard_disk" "test" { tags = { foo = "bar" + } + + lifecycle { + ignore_changes = [storage_path_id] + } +} +`, template, data.RandomString, os.Getenv(customLocationIdEnv)) +} + +func (r StackHCIVirtualHardDiskResource) updateTag(data acceptance.TestData) string { + template := r.template(data) + return fmt.Sprintf(` +%s + +provider "azurerm" { + features {} +} + +resource "azurerm_stack_hci_virtual_hard_disk" "test" { + name = "acctest-vhd-%s" + resource_group_name = azurerm_resource_group.test.name + location = azurerm_resource_group.test.location + custom_location_id = %q + disk_size_in_gb = 2 + + tags = { env = "test" + foo = "bar" } lifecycle { From 1037c3e32bb1c6cd2497c4fcbb5f94fa336965a6 Mon Sep 17 00:00:00 2001 From: teowa <104055472+teowa@users.noreply.github.com> Date: Tue, 8 Oct 2024 09:19:23 +0000 Subject: [PATCH 5/6] add disk_size_in_gb to example --- website/docs/r/stack_hci_virtual_hard_disk.html.markdown | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/website/docs/r/stack_hci_virtual_hard_disk.html.markdown b/website/docs/r/stack_hci_virtual_hard_disk.html.markdown index 78aeddc8085c..a25d4f037603 100644 --- a/website/docs/r/stack_hci_virtual_hard_disk.html.markdown +++ b/website/docs/r/stack_hci_virtual_hard_disk.html.markdown @@ -30,10 +30,11 @@ resource "azurerm_stack_hci_storage_path" "example" { } resource "azurerm_stack_hci_virtual_hard_disk" "example" { - name = "example" + name = "example-vhd" resource_group_name = azurerm_resource_group.example.name location = azurerm_resource_group.example.location custom_location_id = "/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/rg1/providers/Microsoft.ExtendedLocation/customLocations/cl1" + disk_size_in_gb = 2 storage_path_id = azurerm_stack_hci_storage_path.example.id tags = { foo = "bar" From 1261c2ed47376dbf4f8a6ee915b7738059b4f15b Mon Sep 17 00:00:00 2001 From: teowa <104055472+teowa@users.noreply.github.com> Date: Tue, 8 Oct 2024 09:55:53 +0000 Subject: [PATCH 6/6] fix complete test --- .../stack_hci_virtual_hard_disk_resource.go | 21 ++++++++----------- ...ack_hci_virtual_hard_disk_resource_test.go | 18 ++++++++++------ 2 files changed, 21 insertions(+), 18 deletions(-) diff --git a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go index 4c4c7ed1e95f..1aee9a66396a 100644 --- a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go +++ b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource.go @@ -72,6 +72,13 @@ func (StackHCIVirtualHardDiskResource) Arguments() map[string]*pluginsdk.Schema "custom_location_id": commonschema.ResourceIDReferenceRequiredForceNew(&customlocations.CustomLocationId{}), + "disk_size_in_gb": { + Type: pluginsdk.TypeInt, + Required: true, + ForceNew: true, + ValidateFunc: validation.IntAtLeast(1), + }, + "block_size_in_bytes": { Type: pluginsdk.TypeInt, Optional: true, @@ -89,13 +96,6 @@ func (StackHCIVirtualHardDiskResource) Arguments() map[string]*pluginsdk.Schema }, false), }, - "disk_size_in_gb": { - Type: pluginsdk.TypeInt, - Required: true, - ForceNew: true, - ValidateFunc: validation.IntAtLeast(1), - }, - "dynamic_enabled": { Type: pluginsdk.TypeBool, Optional: true, @@ -173,7 +173,8 @@ func (r StackHCIVirtualHardDiskResource) Create() sdk.ResourceFunc { Type: pointer.To(virtualharddisks.ExtendedLocationTypesCustomLocation), }, Properties: &virtualharddisks.VirtualHardDiskProperties{ - Dynamic: pointer.To(config.DynamicEnabled), + Dynamic: pointer.To(config.DynamicEnabled), + DiskSizeGB: pointer.To(config.DiskSizeInGB), }, } @@ -189,10 +190,6 @@ func (r StackHCIVirtualHardDiskResource) Create() sdk.ResourceFunc { payload.Properties.DiskFileFormat = pointer.To(virtualharddisks.DiskFileFormat(config.DiskFileFormat)) } - if config.DiskSizeInGB != 0 { - payload.Properties.DiskSizeGB = pointer.To(config.DiskSizeInGB) - } - if config.HypervGeneration != "" { payload.Properties.HyperVGeneration = pointer.To(virtualharddisks.HyperVGeneration(config.HypervGeneration)) } diff --git a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go index c4480a43aec3..aaa4d1bf9bb0 100644 --- a/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go +++ b/internal/services/azurestackhci/stack_hci_virtual_hard_disk_resource_test.go @@ -224,12 +224,18 @@ resource "azurerm_stack_hci_storage_path" "test" { } resource "azurerm_stack_hci_virtual_hard_disk" "test" { - name = "acctest-vhd-%[2]s" - resource_group_name = azurerm_resource_group.test.name - location = azurerm_resource_group.test.location - custom_location_id = %[3]q - disk_size_in_gb = 2 - storage_path_id = azurerm_stack_hci_storage_path.test.id + name = "acctest-vhd-%[2]s" + resource_group_name = azurerm_resource_group.test.name + location = azurerm_resource_group.test.location + custom_location_id = %[3]q + disk_size_in_gb = 2 + dynamic_enabled = false + hyperv_generation = "V2" + physical_sector_in_bytes = 4096 + logical_sector_in_bytes = 512 + block_size_in_bytes = 1024 + disk_file_format = "vhdx" + storage_path_id = azurerm_stack_hci_storage_path.test.id tags = { foo = "bar"