Skip to content

Commit

Permalink
feat(vpc): support ip_extra_set in address group (huaweicloud#5767)
Browse files Browse the repository at this point in the history
  • Loading branch information
Jason-Zhang9309 authored Oct 26, 2024
1 parent 7564da4 commit 2ea5147
Show file tree
Hide file tree
Showing 3 changed files with 207 additions and 5 deletions.
40 changes: 38 additions & 2 deletions docs/resources/vpc_address_group.md
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,29 @@ resource "huaweicloud_vpc_address_group" "ipv6" {
}
```

### Address Group with ip_extra_set

```hcl
resource "huaweicloud_vpc_address_group" "ipv6" {
name = "group-ipv4"
ip_version = 4
ip_extra_set {
ip = "192.168.3.2"
remarks = "terraform test 1"
}
ip_extra_set {
ip = "192.168.5.0/24"
remarks = "terraform test 2"
}
ip_extra_set {
ip = "192.168.3.20-192.168.3.100"
}
}
```

## Argument Reference

The following arguments are supported:
Expand All @@ -48,8 +71,13 @@ The following arguments are supported:
The value is a string of `1` to `64` characters that can contain letters, digits, underscores (_), hyphens (-) and
periods (.).

* `addresses` - (Required, List) Specifies an array of one or more IP addresses. The address can be a single IP
address, IP address range or IP address CIDR. The maximum length is 20.
* `addresses` - (Optional, List) Specifies an array of one or more IP addresses. The address can be a single IP
address, IP address range or IP address CIDR. The maximum length is 20. Only one of `addresses` and `ip_extra_set`
can be specified.

* `ip_extra_set` - (Optional, List) Specifies the IP addresses and their remarks in an IP address group.
The [ip_extra_set](#address_groups_ip_extra_set_struct) structure is documented below.
Only one of `addresses` and `ip_extra_set` can be specified.

* `ip_version` - (Optional, Int, ForceNew) Specifies the IP version, either `4` (default) or `6`.
Changing this creates a new address group.
Expand All @@ -66,6 +94,14 @@ The following arguments are supported:
* `force_destroy` - (Optional, Bool) Specifies whether to forcibly destroy the address group if it is associated with
a security group rule, the address group and the associated security group rule will be deleted together.
The default value is **false**.

<a name="address_groups_ip_extra_set_struct"></a>
The `ip_extra_set` block supports:

* `ip` - (Required, String) Specifies the IP address, IP address range, or CIDR block.

* `remarks` - (Optional, String) Specifies the supplementary information about the IP address,
IP address range, or CIDR block.

## Attribute Reference

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,56 @@ func TestAccVpcAddressGroup_basic(t *testing.T) {
})
}

func TestAccVpcAddressGroup_ipExtraSet(t *testing.T) {
var group vpc_model.ShowAddressGroupResponse

rName := acceptance.RandomAccResourceName()
rNameUpdate := rName + "_updated"
resourceName := "huaweicloud_vpc_address_group.test"

rc := acceptance.InitResourceCheck(
resourceName,
&group,
getVpcAddressGroupResourceFunc,
)

resource.ParallelTest(t, resource.TestCase{
PreCheck: func() { acceptance.TestAccPreCheck(t) },
ProviderFactories: acceptance.TestAccProviderFactories,
CheckDestroy: rc.CheckResourceDestroy(),
Steps: []resource.TestStep{
{
Config: testVpcAdressGroup_ipExtraSet(rName),
Check: resource.ComposeTestCheckFunc(
rc.CheckResourceExists(),
resource.TestCheckResourceAttr(resourceName, "name", rName),
resource.TestCheckResourceAttr(resourceName, "description", "created by acc test"),
resource.TestCheckResourceAttr(resourceName, "ip_version", "4"),
resource.TestCheckResourceAttr(resourceName, "ip_extra_set.#", "2"),
resource.TestCheckResourceAttr(resourceName, "max_capacity", "20"),
resource.TestCheckResourceAttr(resourceName, "enterprise_project_id", "0"),
),
},
{
Config: testVpcAdressGroup_ipExtraSetUpdate(rNameUpdate),
Check: resource.ComposeTestCheckFunc(
resource.TestCheckResourceAttr(resourceName, "name", rNameUpdate),
resource.TestCheckResourceAttr(resourceName, "description", "updated by acc test"),
resource.TestCheckResourceAttr(resourceName, "ip_extra_set.#", "3"),
resource.TestCheckResourceAttr(resourceName, "max_capacity", "10"),
resource.TestCheckResourceAttr(resourceName, "enterprise_project_id", "0"),
),
},
{
ResourceName: resourceName,
ImportState: true,
ImportStateVerify: true,
ImportStateVerifyIgnore: []string{"force_destroy"},
},
},
})
}

func TestAccVpcAddressGroup_ipv6(t *testing.T) {
var group vpc_model.ShowAddressGroupResponse

Expand Down Expand Up @@ -197,6 +247,50 @@ resource "huaweicloud_vpc_address_group" "test" {
`, rName)
}

func testVpcAdressGroup_ipExtraSet(rName string) string {
return fmt.Sprintf(`
resource "huaweicloud_vpc_address_group" "test" {
name = "%s"
description = "created by acc test"
ip_extra_set {
ip = "192.168.3.2"
remarks = "terraform test 1"
}
ip_extra_set {
ip = "192.168.3.20-192.168.3.100"
remarks = "terraform test 2"
}
}
`, rName)
}

func testVpcAdressGroup_ipExtraSetUpdate(rName string) string {
return fmt.Sprintf(`
resource "huaweicloud_vpc_address_group" "test" {
name = "%s"
description = "updated by acc test"
ip_extra_set {
ip = "192.168.3.2"
remarks = "terraform test 1"
}
ip_extra_set {
ip = "192.168.5.0/24"
remarks = "terraform test 2"
}
ip_extra_set {
ip = "192.168.3.20-192.168.3.100"
}
max_capacity = 10
}
`, rName)
}

func testVpcAdressGroup_ipv6(rName string) string {
return fmt.Sprintf(`
resource "huaweicloud_vpc_address_group" "test" {
Expand Down
78 changes: 75 additions & 3 deletions huaweicloud/services/vpc/resource_huaweicloud_vpc_address_group.go
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,8 @@ func ResourceVpcAddressGroup() *schema.Resource {
"addresses": {
// the addresses will be sorted by cloud
Type: schema.TypeSet,
Required: true,
Optional: true,
Computed: true,
MaxItems: 20,
Elem: &schema.Schema{Type: schema.TypeString},
},
Expand All @@ -63,6 +64,25 @@ func ResourceVpcAddressGroup() *schema.Resource {
Default: 4,
ValidateFunc: validation.IntInSlice([]int{4, 6}),
},
"ip_extra_set": {
// the addresses will be sorted by cloud
Type: schema.TypeSet,
Optional: true,
Computed: true,
MaxItems: 20,
Elem: &schema.Resource{
Schema: map[string]*schema.Schema{
"ip": {
Type: schema.TypeString,
Required: true,
},
"remarks": {
Type: schema.TypeString,
Optional: true,
},
},
},
},
"description": {
Type: schema.TypeString,
Optional: true,
Expand Down Expand Up @@ -95,15 +115,17 @@ func resourceVpcAddressGroupCreate(ctx context.Context, d *schema.ResourceData,
return diag.Errorf("error creating VPC client: %s", err)
}

ipSet := utils.ExpandToStringListBySet(d.Get("addresses").(*schema.Set))
ipSet := buildIpSet(d)
ipExtraSet := buildIpExtraSet(d)

addressGroupBody := &vpc_model.CreateAddressGroupOption{
Name: d.Get("name").(string),
IpSet: &ipSet,
IpSet: ipSet,
IpVersion: int32(d.Get("ip_version").(int)),
Description: utils.StringIgnoreEmpty(d.Get("description").(string)),
EnterpriseProjectId: utils.StringIgnoreEmpty(cfg.GetEnterpriseProjectID(d)),
MaxCapacity: utils.Int32IgnoreEmpty(int32(d.Get("max_capacity").(int))),
IpExtraSet: ipExtraSet,
}

createOpts := &vpc_model.CreateAddressGroupRequest{
Expand All @@ -122,6 +144,34 @@ func resourceVpcAddressGroupCreate(ctx context.Context, d *schema.ResourceData,
return resourceVpcAddressGroupRead(ctx, d, meta)
}

func buildIpSet(d *schema.ResourceData) *[]string {
addresses := utils.ExpandToStringListBySet(d.Get("addresses").(*schema.Set))
if len(addresses) == 0 {
return nil
}

return &addresses
}

func buildIpExtraSet(d *schema.ResourceData) *[]vpc_model.IpExtraSetOption {
ipExtraSet := d.Get("ip_extra_set").(*schema.Set).List()
if len(ipExtraSet) == 0 {
return nil
}

res := make([]vpc_model.IpExtraSetOption, len(ipExtraSet))
for i, v := range ipExtraSet {
ip := v.(map[string]interface{})
remarks := ip["remarks"].(string)
res[i] = vpc_model.IpExtraSetOption{
Ip: ip["ip"].(string),
Remarks: &remarks,
}
}

return &res
}

func resourceVpcAddressGroupRead(_ context.Context, d *schema.ResourceData, meta interface{}) diag.Diagnostics {
c := meta.(*config.Config)
region := c.GetRegion(d)
Expand All @@ -147,11 +197,29 @@ func resourceVpcAddressGroupRead(_ context.Context, d *schema.ResourceData, meta
d.Set("ip_version", response.AddressGroup.IpVersion),
d.Set("max_capacity", response.AddressGroup.MaxCapacity),
d.Set("enterprise_project_id", response.AddressGroup.EnterpriseProjectId),
d.Set("ip_extra_set", flattenIpExtraSet(response.AddressGroup.IpExtraSet)),
)

return diag.FromErr(mErr.ErrorOrNil())
}

func flattenIpExtraSet(ipExtraSet []vpc_model.IpExtraSetRespOption) []map[string]interface{} {
if len(ipExtraSet) == 0 {
return nil
}

res := make([]map[string]interface{}, len(ipExtraSet))

for i, v := range ipExtraSet {
res[i] = map[string]interface{}{
"ip": v.Ip,
"remarks": v.Remarks,
}
}

return res
}

func resourceVpcAddressGroupUpdate(ctx context.Context, d *schema.ResourceData, meta interface{}) diag.Diagnostics {
c := meta.(*config.Config)
client, err := c.HcVpcV3Client(c.GetRegion(d))
Expand Down Expand Up @@ -180,6 +248,10 @@ func resourceVpcAddressGroupUpdate(ctx context.Context, d *schema.ResourceData,
addressGroupBody.IpSet = &ipSet
}

if d.HasChange("ip_extra_set") {
addressGroupBody.IpExtraSet = buildIpExtraSet(d)
}

if d.HasChange("max_capacity") {
addressGroupBody.MaxCapacity = utils.Int32IgnoreEmpty(int32(d.Get("max_capacity").(int)))
}
Expand Down

0 comments on commit 2ea5147

Please sign in to comment.