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_healthcare_dicom_service resource & data source - support for new properties #27375

Conversation

hqhqhqhqhqhqhqhqhqhqhq
Copy link
Contributor

Community Note

  • Please vote on this PR by adding a 👍 reaction to the original PR to help the community and maintainers prioritize for review
  • Please do not leave comments along the lines of "+1", "me too" or "any updates", they generate extra noise for PR followers and do not help prioritize for review

Description

For azurerm_healthcare_dicom_service resource, added properties:

  • cors_configuration
  • encryption_key_url
  • storage_configuration
  • data_partitions_enabled

For azurerm_healthcare_dicom_service data source, added properties:

  • cors_configuration
  • encryption_key_url
  • storage_configuration
  • data_partitions_enabled
  • provision_state
  • event_state

PR Checklist

  • I have followed the guidelines in our Contributing Documentation.
  • I have checked to ensure there aren't other open Pull Requests for the same update/change.
  • I have checked if my changes close any open issues. If so please include appropriate closing keywords below.
  • I have updated/added Documentation as required written in a helpful and kind way to assist users that may be unfamiliar with the resource / data source.
  • I have used a meaningful PR title to help maintainers and other users understand this change and help prevent duplicate work.
    For example: “resource_name_here - description of change e.g. adding property new_property_name_here

Changes to existing Resource / Data Source

  • I have added an explanation of what my changes do and why I'd like you to include them (This may be covered by linking to an issue above, but may benefit from additional explanation).
  • I have written new tests for my resource or datasource changes & updated any relevent documentation.
  • I have successfully run tests with my changes locally. If not, please provide details on testing challenges that prevented you running the tests.
  • (For changes that include a state migration only). I have manually tested the migration path between relevant versions of the provider.

Testing

  • My submission includes Test coverage as described in the Contribution Guide and the tests pass. (if this is not possible for any reason, please include details of why you did or could not add test coverage)
image image image image image

Change Log

Below please provide what should go into the changelog (if anything) conforming to the Changelog Format documented here.

  • azurerm_healthcare_dicom_service resource & data source - support for new properties

This is a (please select all that apply):

  • Bug Fix
  • New Feature (ie adding a service, resource, or data source)
  • Enhancement
  • Breaking Change

Note

If this PR changes meaningfully during the course of review please update the title and description as required.

Computed: true,
},

"cors_configuration": {
Copy link
Collaborator

Choose a reason for hiding this comment

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

configuration is redundant,

Suggested change
"cors_configuration": {
"cors": {

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks, updated.

Computed: true,
},

"storage_configuration": {
Copy link
Collaborator

Choose a reason for hiding this comment

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

ditto

Suggested change
"storage_configuration": {
"storage": {

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks, updated.

@@ -115,6 +118,86 @@ func resourceHealthcareApisDicomService() *pluginsdk.Resource {
Default: true,
},

"cors_configuration": {
Copy link
Collaborator

Choose a reason for hiding this comment

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

Suggested change
"cors_configuration": {
"cors": {

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks, updated.

Comment on lines 134 to 135
},
"allowed_headers": {
Copy link
Collaborator

Choose a reason for hiding this comment

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

Suggested change
},
"allowed_headers": {
},
"allowed_headers": {

please add blank lines between properties

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks, updated.

},
"max_age_in_seconds": {
Type: pluginsdk.TypeInt,
Optional: true,
Copy link
Collaborator

Choose a reason for hiding this comment

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

can you validate this? i presume -3 is not a valid value? is 0? is 99999999999 ?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Trid it out, can support 0 to 99998 inclusive, added the validatefunc

ValidateFunc: validation.IsURLWithHTTPS,
},

"storage_configuration": {
Copy link
Collaborator

Choose a reason for hiding this comment

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

Suggested change
"storage_configuration": {
"storage": {

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Updated

Copy link
Member

@stephybun stephybun left a comment

Choose a reason for hiding this comment

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

Thanks @hqhqhqhqhqhqhqhqhqhqhq. Could you take a look through and fix up the. comments left in-line? Once that's done we can take another look through.

Comment on lines 161 to 169
"provision_state": {
Type: pluginsdk.TypeString,
Computed: true,
},

"event_state": {
Type: pluginsdk.TypeString,
Computed: true,
},
Copy link
Member

Choose a reason for hiding this comment

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

These fields contain transient information that isn't meaningful to users, we don't usually expose these in resources or data sources

Suggested change
"provision_state": {
Type: pluginsdk.TypeString,
Computed: true,
},
"event_state": {
Type: pluginsdk.TypeString,
Computed: true,
},

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks, this has been removed from the data source and the docs

Comment on lines 330 to 334
enableDataPartitions := false
if props.EnableDataPartitions != nil {
enableDataPartitions = pointer.From(props.EnableDataPartitions)
}
d.Set("data_partitions_enabled", enableDataPartitions)
Copy link
Member

Choose a reason for hiding this comment

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

pointer.From can do the nil check for us, so this can be reduced to

Suggested change
enableDataPartitions := false
if props.EnableDataPartitions != nil {
enableDataPartitions = pointer.From(props.EnableDataPartitions)
}
d.Set("data_partitions_enabled", enableDataPartitions)
d.Set("data_partitions_enabled", pointer.From(props.EnableDataPartitions))

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks, updated in the resource & data source

Comment on lines 373 to 395
var enableDataPartitions *bool
if v, ok := d.GetOk("data_partitions_enabled"); ok {
enableDataPartitions = pointer.To(v.(bool))
}

var corsConfiguration *dicomservices.CorsConfiguration
if v, ok := d.GetOk("cors"); ok {
corsConfiguration = expandDicomServiceCorsConfiguration(v.([]interface{}))
}

var encryption *dicomservices.Encryption
if v, ok := d.GetOk("encryption_key_url"); ok {
encryption = &dicomservices.Encryption{
CustomerManagedKeyEncryption: &dicomservices.EncryptionCustomerManagedKeyEncryption{
KeyEncryptionKeyUrl: pointer.To(v.(string)),
},
}
}

var storageConfiguration *dicomservices.StorageConfiguration
if v, ok := d.GetOk("storage"); ok {
storageConfiguration = expandStorageConfiguration(v.([]interface{}))
}
Copy link
Member

Choose a reason for hiding this comment

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

We should be using d.HasChanges here so we can support the use of ignore_changes, you may need to rewrite this to retrieve the existing resource and patch in changes with d.HasChanges. The update method in our guide on adding new resources has an explanation on how this should be done:

https://github.com/hashicorp/terraform-provider-azurerm/blob/main/contributing/topics/guide-new-resource.md

Here is also a practical example from within the provider:

func resourcePublicIpUpdate(d *pluginsdk.ResourceData, meta interface{}) error {
client := meta.(*clients.Client).Network.PublicIPAddresses
ctx, cancel := timeouts.ForUpdate(meta.(*clients.Client).StopContext, d)
defer cancel()
log.Printf("[INFO] preparing arguments for AzureRM Public IP update.")
id, err := commonids.ParsePublicIPAddressID(d.Id())
if err != nil {
return err
}
existing, err := client.Get(ctx, *id, publicipaddresses.DefaultGetOperationOptions())
if err != nil {
return fmt.Errorf("retrieving %s: %+v", id, err)
}
if existing.Model == nil {
return fmt.Errorf("retrieving %s: `model` was nil", id)
}
if existing.Model.Properties == nil {
return fmt.Errorf("retrieving %s: `properties` was nil", id)
}
payload := existing.Model
if d.HasChange("allocation_method") {
payload.Properties.PublicIPAllocationMethod = pointer.To(publicipaddresses.IPAllocationMethod(d.Get("allocation_method").(string)))
}
if d.HasChange("ddos_protection_mode") {
if payload.Properties.DdosSettings == nil {
payload.Properties.DdosSettings = &publicipaddresses.DdosSettings{}
}
payload.Properties.DdosSettings.ProtectionMode = pointer.To(publicipaddresses.DdosSettingsProtectionMode(d.Get("ddos_protection_mode").(string)))
}
if d.HasChange("ddos_protection_plan_id") {
if !strings.EqualFold(string(*payload.Properties.DdosSettings.ProtectionMode), "enabled") {
return fmt.Errorf("ddos protection plan id can only be set when ddos protection is enabled")
}
if payload.Properties.DdosSettings == nil {
payload.Properties.DdosSettings = &publicipaddresses.DdosSettings{}
}
payload.Properties.DdosSettings.DdosProtectionPlan = &publicipaddresses.SubResource{
Id: pointer.To(d.Get("ddos_protection_plan_id").(string)),
}
}
if d.HasChange("idle_timeout_in_minutes") {
payload.Properties.IdleTimeoutInMinutes = utils.Int64(int64(d.Get("idle_timeout_in_minutes").(int)))
}
if d.HasChange("domain_name_label") {
if payload.Properties.DnsSettings == nil {
payload.Properties.DnsSettings = &publicipaddresses.PublicIPAddressDnsSettings{}
}
payload.Properties.DnsSettings.DomainNameLabel = utils.String(d.Get("domain_name_label").(string))
}
if d.HasChange("reverse_fqdn") {
if payload.Properties.DnsSettings == nil {
payload.Properties.DnsSettings = &publicipaddresses.PublicIPAddressDnsSettings{}
}
payload.Properties.DnsSettings.ReverseFqdn = utils.String(d.Get("reverse_fqdn").(string))
}
if d.HasChanges("tags") {
payload.Tags = tags.Expand(d.Get("tags").(map[string]interface{}))
}
if err = client.CreateOrUpdateThenPoll(ctx, *id, *payload); err != nil {

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Got it, have updated the update function in the resource to this format.

Comment on lines 564 to 566
if configuration.FileSystemName != nil {
result["file_system_name"] = *configuration.FileSystemName
}
Copy link
Member

Choose a reason for hiding this comment

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

We could use pointer.From here for the nil check

Suggested change
if configuration.FileSystemName != nil {
result["file_system_name"] = *configuration.FileSystemName
}
result["file_system_name"] = pointer.From(configuration.FileSystemName)

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks, updated, the flatten function (used by the resource & data source)

Comment on lines 568 to 570
if configuration.StorageResourceId != nil {
result["storage_account_id"] = *configuration.StorageResourceId
}
Copy link
Member

Choose a reason for hiding this comment

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

We should parse the Storage account ID that we're getting back from the API so that we can catch any casing inconsistencies and potentially prevent them from creeping into user's state

Suggested change
if configuration.StorageResourceId != nil {
result["storage_account_id"] = *configuration.StorageResourceId
}
if v := pointer.From(configuration.StorageResourceId); v != "" {
id, err := commonids.ParseStorageAccountID(v)
if err != nil {
return nil, err
}
result["storage_account_id"] = id.ID()
}

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Thanks, updated the flatten function (used by the resource & data source), the function now also returns error to the calling function

@@ -51,6 +63,29 @@ An `authentication` supports the following:

* `audience` - The intended audience to receive authentication tokens for the service. The default value is <https://dicom.azurehealthcareapis.azure.com>

---

A `cors` supports the following:
Copy link
Member

Choose a reason for hiding this comment

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

For data sources we say a block exports rather than supports since these aren't arguments that can be supplied

Suggested change
A `cors` supports the following:
A `cors` exports the following:

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Updated


---

A `storage` block supports the following:
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
A `storage` block supports the following:
A `storage` block exports the following:

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Updated


* `file_system_name` - The filesystem name of connected storage account.

* `storage_account_id` - The resource id of connected storage account.
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
* `storage_account_id` - The resource id of connected storage account.
* `storage_account_id` - The resource ID of connected storage account.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Updated the doc with required changes & other parts where it is id rather than ID


* `file_system_name` - (Required) The filesystem name of connected storage account. Changing this forces a new Healthcare DICOM Service to be created.

* `storage_account_id` - (Required) The resource id of connected storage account. Changing this forces a new Healthcare DICOM Service to be created.
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
* `storage_account_id` - (Required) The resource id of connected storage account. Changing this forces a new Healthcare DICOM Service to be created.
* `storage_account_id` - (Required) The resource ID of connected storage account. Changing this forces a new Healthcare DICOM Service to be created.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Updated

website/docs/r/healthcare_dicom.html.markdown Show resolved Hide resolved
@hqhqhqhqhqhqhqhqhqhqhq
Copy link
Contributor Author

Thanks @hqhqhqhqhqhqhqhqhqhqhq. Could you take a look through and fix up the. comments left in-line? Once that's done we can take another look through.

@stephybun Thanks for all the reviews🙏, I have addressed all of them and reran the tests:
image
image

Please help give another review when available, thank you!

Copy link
Member

@stephybun stephybun left a comment

Choose a reason for hiding this comment

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

Sorry for the delayed re-review @hqhqhqhqhqhqhqhqhqhqhq. This looks mostly fine, I posted a response to one of the comments regarding the default values for the cors and storage block, please take a look at the detail in the response.

@hqhqhqhqhqhqhqhqhqhqhq hqhqhqhqhqhqhqhqhqhqhq requested a review from a team as a code owner November 6, 2024 04:11
Copy link
Member

@stephybun stephybun left a comment

Choose a reason for hiding this comment

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

One final comment @hqhqhqhqhqhqhqhqhqhqhq then this is probably good to go.

Comment on lines 249 to 283
var enableDataPartitions *bool
if v, ok := d.GetOk("data_partitions_enabled"); ok {
enableDataPartitions = pointer.To(v.(bool))
}

var corsConfiguration *dicomservices.CorsConfiguration
if v, ok := d.GetOk("cors"); ok {
corsConfiguration = expandDicomServiceCorsConfiguration(v.([]interface{}))
}

var encryption *dicomservices.Encryption
if v, ok := d.GetOk("encryption_key_url"); ok {
encryption = &dicomservices.Encryption{
CustomerManagedKeyEncryption: &dicomservices.EncryptionCustomerManagedKeyEncryption{
KeyEncryptionKeyURL: pointer.To(v.(string)),
},
}
}

var storageConfiguration *dicomservices.StorageConfiguration
if v, ok := d.GetOk("storage"); ok {
storageConfiguration = expandStorageConfiguration(v.([]interface{}))
}

parameters := dicomservices.DicomService{
Identity: i,
Properties: &dicomservices.DicomServiceProperties{
PublicNetworkAccess: pointer.To(dicomservices.PublicNetworkAccessEnabled),
CorsConfiguration: corsConfiguration,
EnableDataPartitions: enableDataPartitions,
Encryption: encryption,
PublicNetworkAccess: pointer.To(dicomservices.PublicNetworkAccessEnabled),
StorageConfiguration: storageConfiguration,
},
Location: pointer.To(location.Normalize(d.Get("location").(string))),
Tags: tags.Expand(t),
Copy link
Member

Choose a reason for hiding this comment

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

We typically don't set optional properties in the create payload if they user hasn't specified them, so this would look more like:

	parameters := dicomservices.DicomService{
		Identity: i,
		Properties: &dicomservices.DicomServiceProperties{
			PublicNetworkAccess:  pointer.To(dicomservices.PublicNetworkAccessEnabled),
		},
		Location: pointer.To(location.Normalize(d.Get("location").(string))),
		Tags:     tags.Expand(t),
	}

	if v, ok := d.GetOk("data_partitions_enabled"); ok {
		parameters.Properties.EnableDataPartitions = pointer.To(v.(bool))
	}

	cors := expandDicomServiceCorsConfiguration(d.Get("cors").([]interface{}))
	if cors != nil {
		parameters.Properties.CorsConfiguration = cors
	}

	if v, ok := d.GetOk("encryption_key_url"); ok && v != "" {
		parameters.Properties.Encryption = &dicomservices.Encryption{
			CustomerManagedKeyEncryption: &dicomservices.EncryptionCustomerManagedKeyEncryption{
				KeyEncryptionKeyURL: pointer.To(v.(string)),
			},
		}
	}

	storage := expandStorageConfiguration(d.Get("storage").([]interface{}))
	if storage != nil  {
		parameters.Properties.StorageConfiguration = storage
	}

@hqhqhqhqhqhqhqhqhqhqhq
Copy link
Contributor Author

Hi @stephybun, updated as requested, please help give this another review when you have time, thanks!

Copy link
Member

@stephybun stephybun left a comment

Choose a reason for hiding this comment

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

Thanks @hqhqhqhqhqhqhqhqhqhqhq LGTM 👍

@stephybun stephybun merged commit 538b4f5 into hashicorp:main Jan 9, 2025
34 checks passed
@github-actions github-actions bot added this to the v4.15.0 milestone Jan 9, 2025
stephybun added a commit that referenced this pull request Jan 9, 2025
jackofallops pushed a commit that referenced this pull request Jan 10, 2025
jackofallops added a commit that referenced this pull request Jan 10, 2025
* Update CHANGELOG.md for #28233

* Update for #28215

* Update CHANGELOG.md for #28279

* Update CHANGELOG.md #28269

* Update CHANGELOG.md #27876

* Update CHANGELOG.md #28069

* Update CHANGELOG.md for #28312

* Update CHANGELOG.md for #28278

* Update CHANGELOG.md #28311

* Update CHANGELOG.md undo 28311

* Update CHANGELOG.md #27874

* Update CHANGELOG.md

* Update CHANGELOG for #28352

* Update CHANGELOG.md for #28390

* Update CHANGELOG.md for #28398

* Update CHANGELOG.md for #28425

* Update CHANGELOG.md #28427

* Update CHANGELOG.md #28280

* Update CHANGELOG.md for #28319

* Update CHANGELOG.md #24801

* Update for #28360 #28216 #27830 #28404 #27401 #27122 #27931 #28442

* Update for #28379

* Update CHANGELOG.md for #28281

* Update for #28380

* Update for #27375

* Update for #25695

* Update CHANGELOG.md #27985

* Update CHANGELOG.md - update release date manually until can be scripted

* Update CHANGELOG.md revert date change as script available

* pre-release script updates

---------

Co-authored-by: stephybun <steph@hashicorp.com>
Co-authored-by: catriona-m <86247157+catriona-m@users.noreply.github.com>
Co-authored-by: Wyatt Fry <wyattfry@gmail.com>
Co-authored-by: sreallymatt <106555974+sreallymatt@users.noreply.github.com>
Co-authored-by: Matthew Frahry <mbfrahry@gmail.com>
Co-authored-by: kt <kt@katbyte.me>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants