diff --git a/internal/services/servicenetworking/application_load_balancer_association_resource.go b/internal/services/servicenetworking/application_load_balancer_association_resource.go new file mode 100644 index 0000000000000..4debcd409293a --- /dev/null +++ b/internal/services/servicenetworking/application_load_balancer_association_resource.go @@ -0,0 +1,237 @@ +package servicenetworking + +import ( + "context" + "fmt" + "time" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/go-azure-helpers/lang/response" + "github.com/hashicorp/go-azure-helpers/resourcemanager/commonids" + "github.com/hashicorp/go-azure-helpers/resourcemanager/commonschema" + "github.com/hashicorp/go-azure-helpers/resourcemanager/location" + "github.com/hashicorp/go-azure-sdk/resource-manager/servicenetworking/2023-05-01-preview/associationsinterface" + "github.com/hashicorp/go-azure-sdk/resource-manager/servicenetworking/2023-05-01-preview/trafficcontrollerinterface" + "github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" + "github.com/hashicorp/terraform-provider-azurerm/helpers/tf" + "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" +) + +type AssociationResource struct{} + +type AssociationModel struct { + Name string `tfschema:"name"` + ApplicationLoadBalancerId string `tfschema:"application_load_balancer_id"` + SubnetId string `tfschema:"subnet_id"` + Tags map[string]string `tfschema:"tags"` +} + +var _ sdk.ResourceWithUpdate = AssociationResource{} + +func (t AssociationResource) Arguments() map[string]*schema.Schema { + return map[string]*schema.Schema{ + "name": { + Type: pluginsdk.TypeString, + Required: true, + ForceNew: true, + ValidateFunc: validation.StringIsNotEmpty, + }, + + "application_load_balancer_id": { + Type: pluginsdk.TypeString, + Required: true, + ForceNew: true, + ValidateFunc: associationsinterface.ValidateTrafficControllerID, + }, + + "subnet_id": { + Type: pluginsdk.TypeString, + Required: true, + ValidateFunc: commonids.ValidateSubnetID, + }, + + "tags": commonschema.Tags(), + } +} + +func (t AssociationResource) Attributes() map[string]*schema.Schema { + return map[string]*schema.Schema{} +} + +func (t AssociationResource) ModelObject() interface{} { + return &AssociationModel{} +} + +func (t AssociationResource) ResourceType() string { + return "azurerm_application_load_balancer_association" +} + +func (t AssociationResource) IDValidationFunc() pluginsdk.SchemaValidateFunc { + return associationsinterface.ValidateAssociationID +} +func (t AssociationResource) Create() sdk.ResourceFunc { + return sdk.ResourceFunc{ + Timeout: 30 * time.Minute, + Func: func(ctx context.Context, metadata sdk.ResourceMetaData) error { + trafficControllerClient := metadata.Client.ServiceNetworking.TrafficControllerInterface + client := metadata.Client.ServiceNetworking.AssociationsInterface + + var config AssociationModel + if err := metadata.Decode(&config); err != nil { + return fmt.Errorf("decoding %v", err) + } + + parsedTrafficControllerId, err := associationsinterface.ParseTrafficControllerID(config.ApplicationLoadBalancerId) + if err != nil { + return err + } + + controllerId := trafficcontrollerinterface.NewTrafficControllerID(parsedTrafficControllerId.SubscriptionId, parsedTrafficControllerId.ResourceGroupName, parsedTrafficControllerId.TrafficControllerName) + controller, err := trafficControllerClient.Get(ctx, controllerId) + if err != nil { + return fmt.Errorf("retrieving %s: %+v", controllerId, err) + } + + if controller.Model == nil { + return fmt.Errorf("retrieving %s: Model was nil", controllerId) + } + + loc := controller.Model.Location + + id := associationsinterface.NewAssociationID(parsedTrafficControllerId.SubscriptionId, parsedTrafficControllerId.ResourceGroupName, parsedTrafficControllerId.TrafficControllerName, config.Name) + existing, err := client.Get(ctx, id) + if err != nil { + if !response.WasNotFound(existing.HttpResponse) { + return fmt.Errorf("checking for presence of exisiting %s: %+v", id, err) + } + } + + if !response.WasNotFound(existing.HttpResponse) { + return tf.ImportAsExistsError(t.ResourceType(), id.ID()) + } + + association := associationsinterface.Association{ + Location: location.Normalize(loc), + Properties: &associationsinterface.AssociationProperties{ + Subnet: &associationsinterface.AssociationSubnet{ + Id: config.SubnetId, + }, + AssociationType: associationsinterface.AssociationTypeSubnets, + }, + } + + if len(config.Tags) > 0 { + association.Tags = &config.Tags + } + + if err := client.CreateOrUpdateThenPoll(ctx, id, association); err != nil { + return fmt.Errorf("creating %s: %+v", id, err) + } + + metadata.SetID(id) + return nil + }, + } +} + +func (t AssociationResource) Read() sdk.ResourceFunc { + return sdk.ResourceFunc{ + Timeout: 5 * time.Minute, + Func: func(ctx context.Context, metadata sdk.ResourceMetaData) error { + client := metadata.Client.ServiceNetworking.AssociationsInterface + + id, err := associationsinterface.ParseAssociationID(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("retreiving %s: %v", id.ID(), err) + } + + trafficControllerId := associationsinterface.NewTrafficControllerID(id.SubscriptionId, id.ResourceGroupName, id.TrafficControllerName) + state := AssociationModel{ + Name: id.AssociationName, + ApplicationLoadBalancerId: trafficControllerId.ID(), + } + + if model := resp.Model; model != nil { + state.Tags = pointer.From(model.Tags) + + if prop := model.Properties; prop != nil { + if prop.Subnet != nil { + state.SubnetId = prop.Subnet.Id + } + } + } + + return metadata.Encode(&state) + }, + } +} + +func (t AssociationResource) Update() sdk.ResourceFunc { + return sdk.ResourceFunc{ + Timeout: 30 * time.Minute, + Func: func(ctx context.Context, metadata sdk.ResourceMetaData) error { + client := metadata.Client.ServiceNetworking.AssociationsInterface + + var plan AssociationModel + if err := metadata.Decode(&plan); err != nil { + return fmt.Errorf("decoding %v", err) + } + + id, err := associationsinterface.ParseAssociationID(metadata.ResourceData.Id()) + if err != nil { + return fmt.Errorf("parsing id %v", err) + } + + associationUpdate := associationsinterface.AssociationUpdate{} + + if metadata.ResourceData.HasChange("tags") { + associationUpdate.Tags = &plan.Tags + } + + if metadata.ResourceData.HasChange("subnet_id") { + associationUpdate.Properties = &associationsinterface.AssociationUpdateProperties{ + Subnet: &associationsinterface.AssociationSubnetUpdate{ + Id: &plan.SubnetId, + }, + AssociationType: pointer.To(associationsinterface.AssociationTypeSubnets), + } + } + + if _, err = client.Update(ctx, *id, associationUpdate); err != nil { + return fmt.Errorf("updating `azurerm_application_load_balancer_association` %s: %v", id.ID(), err) + } + + return nil + }, + } +} + +func (t AssociationResource) Delete() sdk.ResourceFunc { + return sdk.ResourceFunc{ + Timeout: 30 * time.Minute, + Func: func(ctx context.Context, metadata sdk.ResourceMetaData) error { + client := metadata.Client.ServiceNetworking.AssociationsInterface + + id, err := associationsinterface.ParseAssociationID(metadata.ResourceData.Id()) + if err != nil { + return err + } + + if err = client.DeleteThenPoll(ctx, *id); err != nil { + return fmt.Errorf("deleting %s: %v", id.ID(), err) + } + + return nil + }, + } +} diff --git a/internal/services/servicenetworking/application_load_balancer_association_resource_test.go b/internal/services/servicenetworking/application_load_balancer_association_resource_test.go new file mode 100644 index 0000000000000..5032d19ebd95f --- /dev/null +++ b/internal/services/servicenetworking/application_load_balancer_association_resource_test.go @@ -0,0 +1,224 @@ +package servicenetworking_test + +import ( + "context" + "fmt" + "testing" + + "github.com/hashicorp/go-azure-helpers/lang/pointer" + "github.com/hashicorp/go-azure-helpers/lang/response" + "github.com/hashicorp/go-azure-sdk/resource-manager/servicenetworking/2023-05-01-preview/associationsinterface" + "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 AssociationResource struct{} + +func (r AssociationResource) Exists(ctx context.Context, clients *clients.Client, state *pluginsdk.InstanceState) (*bool, error) { + id, err := associationsinterface.ParseAssociationID(state.ID) + if err != nil { + return nil, fmt.Errorf("while parsing resource ID: %+v", err) + } + + resp, err := clients.ServiceNetworking.AssociationsInterface.Get(ctx, *id) + if err != nil { + if response.WasNotFound(resp.HttpResponse) { + return pointer.To(false), nil + } + return nil, fmt.Errorf("while checking existence for %q: %+v", id.String(), err) + } + return pointer.To(resp.Model != nil), nil +} + +func TestAccApplicationLoadBalancerAssociation_basic(t *testing.T) { + data := acceptance.BuildTestData(t, "azurerm_application_load_balancer_association", "test") + + r := AssociationResource{} + data.ResourceTest(t, r, []acceptance.TestStep{ + { + Config: r.basic(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.ImportStep(), + }) +} + +func TestAccApplicationLoadBalancerAssociation_update(t *testing.T) { + data := acceptance.BuildTestData(t, "azurerm_application_load_balancer_association", "test") + + r := AssociationResource{} + data.ResourceTest(t, r, []acceptance.TestStep{ + { + Config: r.basic(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.ImportStep(), + { + Config: r.updateSubnet(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.ImportStep(), + }) +} + +func TestAccApplicationLoadBalancerAssociation_complete(t *testing.T) { + data := acceptance.BuildTestData(t, "azurerm_application_load_balancer_association", "test") + + r := AssociationResource{} + data.ResourceTest(t, r, []acceptance.TestStep{ + { + Config: r.complete(data), + Check: acceptance.ComposeTestCheckFunc( + check.That(data.ResourceName).ExistsInAzure(r), + ), + }, + data.ImportStep(), + }) +} + +func TestAccApplicationLoadBalancerAssociation_requiresImport(t *testing.T) { + data := acceptance.BuildTestData(t, "azurerm_application_load_balancer_association", "test") + + r := AssociationResource{} + 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 AssociationResource) template(data acceptance.TestData) string { + return fmt.Sprintf(` +resource "azurerm_resource_group" "test" { + name = "acctestrg-alb-%[1]d" + location = "%[2]s" +} + +resource "azurerm_application_load_balancer" "test" { + name = "acctestalb-%[1]d" + location = azurerm_resource_group.test.location + resource_group_name = azurerm_resource_group.test.name +} + +resource "azurerm_virtual_network" "test" { + name = "acctestvnet%[1]d" + address_space = ["10.0.0.0/16"] + location = azurerm_resource_group.test.location + resource_group_name = azurerm_resource_group.test.name +} + +resource "azurerm_subnet" "test" { + name = "acctestsubnet%[1]d" + resource_group_name = azurerm_resource_group.test.name + virtual_network_name = azurerm_virtual_network.test.name + address_prefixes = ["10.0.1.0/24"] + + delegation { + name = "delegation" + + service_delegation { + name = "Microsoft.ServiceNetworking/trafficControllers" + actions = ["Microsoft.Network/virtualNetworks/subnets/join/action"] + } + } +} + +`, data.RandomInteger, data.Locations.Primary) +} + +func (r AssociationResource) basic(data acceptance.TestData) string { + return fmt.Sprintf(` +provider "azurerm" { + features { + } +} + +%s + +resource "azurerm_application_load_balancer_association" "test" { + name = "acct-%d" + application_load_balancer_id = azurerm_application_load_balancer.test.id + subnet_id = azurerm_subnet.test.id +} +`, r.template(data), data.RandomInteger) +} + +func (r AssociationResource) updateSubnet(data acceptance.TestData) string { + return fmt.Sprintf(` +provider "azurerm" { + features { + } +} + +resource "azurerm_subnet" "test2" { + name = "subnet2-%[2]d" + resource_group_name = azurerm_resource_group.test.name + virtual_network_name = azurerm_virtual_network.test.name + address_prefixes = ["10.0.2.0/24"] + + delegation { + name = "delegation" + + service_delegation { + name = "Microsoft.ServiceNetworking/trafficControllers" + actions = ["Microsoft.Network/virtualNetworks/subnets/join/action"] + } + } +} + +%[1]s + +resource "azurerm_application_load_balancer_association" "test" { + name = "acct-%[2]d" + application_load_balancer_id = azurerm_application_load_balancer.test.id + subnet_id = azurerm_subnet.test2.id + tags = { + key = "value" + } +} +`, r.template(data), data.RandomInteger) +} + +func (r AssociationResource) complete(data acceptance.TestData) string { + return fmt.Sprintf(` +provider "azurerm" { + features { + } +} + +%s + +resource "azurerm_application_load_balancer_association" "test" { + name = "acct-%d" + application_load_balancer_id = azurerm_application_load_balancer.test.id + subnet_id = azurerm_subnet.test.id + tags = { + key = "value" + } +} +`, r.template(data), data.RandomInteger) +} + +func (r AssociationResource) requiresImport(data acceptance.TestData) string { + return fmt.Sprintf(` + %s + +resource "azurerm_application_load_balancer_association" "import" { + name = azurerm_application_load_balancer_association.test.name + application_load_balancer_id = azurerm_application_load_balancer_association.test.application_load_balancer_id + subnet_id = azurerm_application_load_balancer_association.test.subnet_id +} +`, r.basic(data)) +} diff --git a/internal/services/servicenetworking/registration.go b/internal/services/servicenetworking/registration.go index 52f9f267fee95..ef4e5039db618 100644 --- a/internal/services/servicenetworking/registration.go +++ b/internal/services/servicenetworking/registration.go @@ -20,6 +20,7 @@ func (r Registration) Resources() []sdk.Resource { return []sdk.Resource{ ApplicationLoadBalancerResource{}, FrontendsResource{}, + AssociationResource{}, } } diff --git a/website/docs/r/application_load_balancer_association.html.markdown b/website/docs/r/application_load_balancer_association.html.markdown new file mode 100644 index 0000000000000..44f20115df589 --- /dev/null +++ b/website/docs/r/application_load_balancer_association.html.markdown @@ -0,0 +1,94 @@ +--- +subcategory: "Service Networking" +layout: "azurerm" +page_title: "Azure Resource Manager: azurerm_application_load_balancer_association" +description: |- + Manages an Application Gateway for Containers Association. +--- + +# azurerm_application_load_balancer_association + +Manages an Application Gateway for Containers Association. + +## Example Usage + +```hcl +resource "azurerm_resource_group" "example" { + name = "example-rg" + location = "northeurope" +} + +resource "azurerm_application_load_balancer" "example" { + name = "example-alb" + location = azurerm_resource_group.example.location + resource_group_name = azurerm_resource_group.example.name +} + +resource "azurerm_virtual_network" "example" { + name = "example-vnet" + address_space = ["10.0.0.0/16"] + location = azurerm_resource_group.example.location + resource_group_name = azurerm_resource_group.example.name +} + +resource "azurerm_subnet" "example" { + name = "example-subnet" + resource_group_name = azurerm_resource_group.example.name + virtual_network_name = azurerm_virtual_network.example.name + address_prefixes = ["10.0.1.0/24"] + + delegation { + name = "delegation" + + service_delegation { + name = "Microsoft.ServiceNetworking/trafficControllers" + actions = ["Microsoft.Network/virtualNetworks/subnets/join/action"] + } + } +} + +resource "azurerm_application_load_balancer_association" "example" { + name = "example" + application_load_balancer_id = azurerm_application_load_balancer.example.id + subnet_id = azurerm_subnet_example.id +} +``` + +## Arguments Reference + +The following arguments are supported: + +* `name` - (Required) The name which should be used for this Application Gateway for Containers Association. Changing this forces a new resource to be created. + +* `application_load_balancer_id` - (Required) The ID of the Application Gateway for Containers. Changing this forces a new resource to be created. + +* `subnet_id` - (Required) The ID of the subnet which the Application Gateway for Containers associated to. Changing this forces a new resource to be created. + +**Note:** The subnet should be delegated by `Microsoft.ServiceNetworking/trafficControllers` as the example. + +--- + +* `tags` - (Optional) A mapping of tags which should be assigned to the Application Gateway for Containers Association. + +## Attributes Reference + +In addition to the Arguments listed above - the following Attributes are exported: + +* `id` - The ID of the Application Gateway for Containers Association. + +## 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 Application Gateway for Containers Association. +* `read` - (Defaults to 5 minutes) Used when retrieving the Application Gateway for Containers Association. +* `update` - (Defaults to 30 minutes) Used when updating the Application Gateway for Containers Association. +* `delete` - (Defaults to 30 minutes) Used when deleting the Application Gateway for Containers Association. + +## Import + +Application Gateway for Containers Associations can be imported using the `resource id`, e.g. + +```shell +terraform import azurerm_application_load_balancer_association.example /subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/rg1/providers/Microsoft.ServiceNetworking/trafficControllers/alb1/associations/association1 +```