From 4948098aafaecf50d0879ddb6fb3244496e6863b Mon Sep 17 00:00:00 2001 From: Chris Marget Date: Mon, 29 Jul 2024 18:28:42 -0400 Subject: [PATCH 1/3] WIP: add testing code --- apstra/resource_ipv4_pool.go | 5 +++ apstra/resource_ipv6_pool.go | 5 +++ apstra/resources/ipv4_pool_subnet.go | 2 +- apstra/resources/ipv6_pool_subnet.go | 2 +- apstra/test_helpers_test.go | 55 ++++++++++++++++++++++++++++ go.mod | 1 - go.sum | 2 - 7 files changed, 67 insertions(+), 5 deletions(-) diff --git a/apstra/resource_ipv4_pool.go b/apstra/resource_ipv4_pool.go index 6766fdea..0c550fdb 100644 --- a/apstra/resource_ipv4_pool.go +++ b/apstra/resource_ipv4_pool.go @@ -149,6 +149,11 @@ func (o *resourceIpv4Pool) Read(ctx context.Context, req resource.ReadRequest, r return } + newState.SetMutablesToNull(ctx, &resp.Diagnostics) + if resp.Diagnostics.HasError() { + return + } + // set state resp.Diagnostics.Append(resp.State.Set(ctx, &newState)...) } diff --git a/apstra/resource_ipv6_pool.go b/apstra/resource_ipv6_pool.go index 1667db44..4869e36c 100644 --- a/apstra/resource_ipv6_pool.go +++ b/apstra/resource_ipv6_pool.go @@ -149,6 +149,11 @@ func (o *resourceIpv6Pool) Read(ctx context.Context, req resource.ReadRequest, r return } + newState.SetMutablesToNull(ctx, &resp.Diagnostics) + if resp.Diagnostics.HasError() { + return + } + // set state resp.Diagnostics.Append(resp.State.Set(ctx, &newState)...) } diff --git a/apstra/resources/ipv4_pool_subnet.go b/apstra/resources/ipv4_pool_subnet.go index 15fd242d..eaa380ab 100644 --- a/apstra/resources/ipv4_pool_subnet.go +++ b/apstra/resources/ipv4_pool_subnet.go @@ -80,7 +80,7 @@ func (o Ipv4PoolSubnet) ResourceAttributes() map[string]resourceSchema.Attribute func (o Ipv4PoolSubnet) AttrTypes() map[string]attr.Type { return map[string]attr.Type{ "status": types.StringType, - "network": types.StringType, + "network": cidrtypes.IPv4PrefixType{}, "total": types.NumberType, "used": types.NumberType, "used_percentage": types.Float64Type, diff --git a/apstra/resources/ipv6_pool_subnet.go b/apstra/resources/ipv6_pool_subnet.go index c29a79e2..8fae0404 100644 --- a/apstra/resources/ipv6_pool_subnet.go +++ b/apstra/resources/ipv6_pool_subnet.go @@ -80,7 +80,7 @@ func (o Ipv6PoolSubnet) ResourceAttributes() map[string]resourceSchema.Attribute func (o Ipv6PoolSubnet) AttrTypes() map[string]attr.Type { return map[string]attr.Type{ "status": types.StringType, - "network": types.StringType, + "network": cidrtypes.IPv6PrefixType{}, "total": types.NumberType, "used": types.NumberType, "used_percentage": types.Float64Type, diff --git a/apstra/test_helpers_test.go b/apstra/test_helpers_test.go index 18f1d3cc..f2f4c8ee 100644 --- a/apstra/test_helpers_test.go +++ b/apstra/test_helpers_test.go @@ -4,8 +4,10 @@ package tfapstra_test import ( "context" + crand "crypto/rand" "encoding/json" "fmt" + "log" "math" "math/rand" "net" @@ -265,6 +267,59 @@ func randomJson(t testing.TB, maxInt int, strLen int, count int) json.RawMessage return result } +func TestRandomPrefix(t *testing.T) { + for _ = range 10 { + //rp := randomPrefix(t, "10.0.0.0/23", 29) + rp := randomPrefix(t, "2001:db8::/32", 127) + log.Println(rp.String()) + } +} + +func randomPrefix(t testing.TB, cidrBlock string, bits int) net.IPNet { + t.Helper() + + ip, block, err := net.ParseCIDR(cidrBlock) + if err != nil { + t.Fatalf("randomPrefix cannot parse cidrBlock - %s", err) + } + if block.IP.String() != ip.String() { + t.Fatal("invocation of randomPrefix doesn't use a base block address") + } + + mOnes, mBits := block.Mask.Size() + if mOnes >= bits { + t.Fatalf("cannot select a random /%d from within %s", bits, cidrBlock) + } + + // generate a completely random address + randomIP := make(net.IP, mBits/8) + _, err = crand.Read(randomIP) + if err != nil { + t.Fatalf("rand read failed") + } + + // mask off the "network" bits + for i, b := range randomIP { + mBitsThisByte := min(mOnes, 8) + mOnes -= mBitsThisByte + + byteMask := byte(math.MaxUint8 >> mBitsThisByte) + + randomIP[i] = b & byteMask + + block.IP[i] = block.IP[i] | (b & byteMask) + } + + block.Mask = net.CIDRMask(bits, mBits) + + _, result, err := net.ParseCIDR(block.String()) + if err != nil { + t.Fatal("failed to parse own CIDR block") + } + + return *result +} + func randomSlash31(t testing.TB, cidrBlock string) net.IPNet { t.Helper() diff --git a/go.mod b/go.mod index 2bb97d17..e3fe9087 100644 --- a/go.mod +++ b/go.mod @@ -7,7 +7,6 @@ go 1.22.5 require ( github.com/IBM/netaddr v1.5.0 github.com/Juniper/apstra-go-sdk v0.0.0-20240722155513-01f6e5e4e91a - github.com/apparentlymart/go-cidr v1.1.0 github.com/chrismarget-j/go-licenses v0.0.0-20240224210557-f22f3e06d3d4 github.com/google/go-cmp v0.6.0 github.com/hashicorp/go-version v1.7.0 diff --git a/go.sum b/go.sum index 1765f45d..350654f6 100644 --- a/go.sum +++ b/go.sum @@ -23,8 +23,6 @@ github.com/ProtonMail/go-crypto v1.1.0-alpha.3-proton h1:0RXAi0EJFs81j+MMsqvHNuA github.com/ProtonMail/go-crypto v1.1.0-alpha.3-proton/go.mod h1:rA3QumHc/FZ8pAHreoekgiAbzpNsfQAosU5td4SnOrE= github.com/agext/levenshtein v1.2.2 h1:0S/Yg6LYmFJ5stwQeRp6EeOcCbj7xiqQSdNelsXvaqE= github.com/agext/levenshtein v1.2.2/go.mod h1:JEDfjyjHDjOF/1e4FlBE/PkbqA9OfWu2ki2W0IB5558= -github.com/apparentlymart/go-cidr v1.1.0 h1:2mAhrMoF+nhXqxTzSZMUzDHkLjmIHC+Zzn4tdgBZjnU= -github.com/apparentlymart/go-cidr v1.1.0/go.mod h1:EBcsNrHc3zQeuaeCeCtQruQm+n9/YjEn/vI25Lg7Gwc= github.com/apparentlymart/go-textseg/v12 v12.0.0/go.mod h1:S/4uRK2UtaQttw1GenVJEynmyUenKwP++x/+DdGV/Ec= github.com/apparentlymart/go-textseg/v15 v15.0.0 h1:uYvfpb3DyLSCGWnctWKGj857c6ew1u1fNQOlOtuGxQY= github.com/apparentlymart/go-textseg/v15 v15.0.0/go.mod h1:K8XmNZdhEBkdlyDdvbmmsvpAG721bKi0joRfFdHIWJ4= From 4ea21dd67180e17214cfde92a9079db87bcfb854 Mon Sep 17 00:00:00 2001 From: Chris Marget Date: Tue, 30 Jul 2024 09:08:46 -0400 Subject: [PATCH 2/3] restore data source checks in ipv4/v6 pool tests --- apstra/resource_ipv4_pool_integration_test.go | 26 +++++++++++------- apstra/resource_ipv6_pool_integration_test.go | 27 ++++++++++++------- apstra/test_helpers_test.go | 16 +---------- 3 files changed, 34 insertions(+), 35 deletions(-) diff --git a/apstra/resource_ipv4_pool_integration_test.go b/apstra/resource_ipv4_pool_integration_test.go index 9b933c4c..f4221a6f 100644 --- a/apstra/resource_ipv4_pool_integration_test.go +++ b/apstra/resource_ipv4_pool_integration_test.go @@ -79,6 +79,9 @@ func (o resourceIpv4Pool) testChecks(t testing.TB, rType, rName string) testChec }) } + // ----------------------------- + // DATA SOURCE "by_id" checks below here + // ----------------------------- checks.setPath("data." + rType + "." + rName + "_by_id") var total int for _, subnet := range o.subnets { @@ -103,6 +106,9 @@ func (o resourceIpv4Pool) testChecks(t testing.TB, rType, rName string) testChec checks.append(t, "TestCheckResourceAttr", "used", "0") checks.append(t, "TestCheckResourceAttr", "used_percentage", "0") + // ----------------------------- + // DATA SOURCE "by_name" checks below here + // ----------------------------- checks.setPath("data." + rType + "." + rName + "_by_name") for _, subnet := range o.subnets { ones, _ := subnet.Mask.Size() @@ -148,9 +154,10 @@ func TestAccResourceIpv4Pool(t *testing.T) { config: resourceIpv4Pool{ name: acctest.RandString(6), subnets: []net.IPNet{ - randomSlash31(t, "10.0.0.0/24"), - randomSlash31(t, "10.0.1.0/24"), - randomSlash31(t, "10.0.2.0/24"), + randomPrefix(t, "10.0.0.0/16", 24), + randomPrefix(t, "10.1.0.0/16", 25), + randomPrefix(t, "10.2.0.0/16", 26), + randomPrefix(t, "10.3.0.0/16", 27), }, }, }, @@ -158,9 +165,8 @@ func TestAccResourceIpv4Pool(t *testing.T) { config: resourceIpv4Pool{ name: acctest.RandString(6), subnets: []net.IPNet{ - randomSlash31(t, "10.1.0.0/24"), - randomSlash31(t, "10.1.1.0/24"), - randomSlash31(t, "10.1.2.0/24"), + randomPrefix(t, "10.4.0.0/16", 24), + randomPrefix(t, "10.5.0.0/16", 25), }, }, }, @@ -168,9 +174,9 @@ func TestAccResourceIpv4Pool(t *testing.T) { config: resourceIpv4Pool{ name: acctest.RandString(6), subnets: []net.IPNet{ - randomSlash31(t, "10.2.0.0/24"), - randomSlash31(t, "10.2.1.0/24"), - randomSlash31(t, "10.2.2.0/24"), + randomPrefix(t, "10.6.0.0/16", 24), + randomPrefix(t, "10.7.0.0/16", 25), + randomPrefix(t, "10.8.0.0/16", 26), }, }, }, @@ -196,7 +202,7 @@ func TestAccResourceIpv4Pool(t *testing.T) { chkLog := checks.string() stepName := fmt.Sprintf("test case %q step %d", tName, i+1) - t.Logf("\n// ------ begin config for %s ------\n%s// -------- end config for %s ------\n\n", stepName, config, stepName) + t.Logf("\n// ------ begin config for %s ------%s// -------- end config for %s ------\n\n", stepName, config, stepName) t.Logf("\n// ------ begin checks for %s ------\n%s// -------- end checks for %s ------\n\n", stepName, chkLog, stepName) steps[i] = resource.TestStep{ diff --git a/apstra/resource_ipv6_pool_integration_test.go b/apstra/resource_ipv6_pool_integration_test.go index 7e800f36..ec2ed3ec 100644 --- a/apstra/resource_ipv6_pool_integration_test.go +++ b/apstra/resource_ipv6_pool_integration_test.go @@ -79,10 +79,14 @@ func (o resourceIpv6Pool) testChecks(t testing.TB, rType, rName string) testChec }) } + // ----------------------------- + // DATA SOURCE "by_id" checks below here + // ----------------------------- checks.setPath("data." + rType + "." + rName + "_by_id") var total int for _, subnet := range o.subnets { ones, _ := subnet.Mask.Size() + // todo: calculation of thisSubnetTotal cannot handle large values. convert to big.Int thisSubnetTotal := 1 << (128 - ones) checks.appendSetNestedCheck(t, "subnets.*", map[string]string{ @@ -103,6 +107,9 @@ func (o resourceIpv6Pool) testChecks(t testing.TB, rType, rName string) testChec checks.append(t, "TestCheckResourceAttr", "used", "0") checks.append(t, "TestCheckResourceAttr", "used_percentage", "0") + // ----------------------------- + // DATA SOURCE "by_name" checks below here + // ----------------------------- checks.setPath("data." + rType + "." + rName + "_by_name") for _, subnet := range o.subnets { ones, _ := subnet.Mask.Size() @@ -148,9 +155,9 @@ func TestAccResourceIpv6Pool(t *testing.T) { config: resourceIpv6Pool{ name: acctest.RandString(6), subnets: []net.IPNet{ - randomSlash127(t, "2001:db8:0::/64"), - randomSlash127(t, "2001:db8:0::/64"), - randomSlash127(t, "2001:db8:0::/64"), + randomPrefix(t, "2001:db8:0::/48", 112), + randomPrefix(t, "2001:db8:1::/48", 112), + randomPrefix(t, "2001:db8:2::/48", 112), }, }, }, @@ -158,9 +165,8 @@ func TestAccResourceIpv6Pool(t *testing.T) { config: resourceIpv6Pool{ name: acctest.RandString(6), subnets: []net.IPNet{ - randomSlash127(t, "2001:db8:1::/64"), - randomSlash127(t, "2001:db8:1::/64"), - randomSlash127(t, "2001:db8:1::/64"), + randomPrefix(t, "2001:db8:3::/48", 127), + randomPrefix(t, "2001:db8:4::/48", 127), }, }, }, @@ -168,9 +174,10 @@ func TestAccResourceIpv6Pool(t *testing.T) { config: resourceIpv6Pool{ name: acctest.RandString(6), subnets: []net.IPNet{ - randomSlash127(t, "2001:db8:2::/64"), - randomSlash127(t, "2001:db8:2::/64"), - randomSlash127(t, "2001:db8:2::/64"), + randomPrefix(t, "2001:db8:5::/48", 112), + randomPrefix(t, "2001:db8:6::/48", 112), + randomPrefix(t, "2001:db8:7::/48", 112), + randomPrefix(t, "2001:db8:8::/48", 112), }, }, }, @@ -196,7 +203,7 @@ func TestAccResourceIpv6Pool(t *testing.T) { chkLog := checks.string() stepName := fmt.Sprintf("test case %q step %d", tName, i+1) - t.Logf("\n// ------ begin config for %s ------\n%s// -------- end config for %s ------\n\n", stepName, config, stepName) + t.Logf("\n// ------ begin config for %s ------%s// -------- end config for %s ------\n\n", stepName, config, stepName) t.Logf("\n// ------ begin checks for %s ------\n%s// -------- end checks for %s ------\n\n", stepName, chkLog, stepName) steps[i] = resource.TestStep{ diff --git a/apstra/test_helpers_test.go b/apstra/test_helpers_test.go index f2f4c8ee..c41b57ab 100644 --- a/apstra/test_helpers_test.go +++ b/apstra/test_helpers_test.go @@ -7,7 +7,6 @@ import ( crand "crypto/rand" "encoding/json" "fmt" - "log" "math" "math/rand" "net" @@ -267,14 +266,6 @@ func randomJson(t testing.TB, maxInt int, strLen int, count int) json.RawMessage return result } -func TestRandomPrefix(t *testing.T) { - for _ = range 10 { - //rp := randomPrefix(t, "10.0.0.0/23", 29) - rp := randomPrefix(t, "2001:db8::/32", 127) - log.Println(rp.String()) - } -} - func randomPrefix(t testing.TB, cidrBlock string, bits int) net.IPNet { t.Helper() @@ -302,12 +293,7 @@ func randomPrefix(t testing.TB, cidrBlock string, bits int) net.IPNet { for i, b := range randomIP { mBitsThisByte := min(mOnes, 8) mOnes -= mBitsThisByte - - byteMask := byte(math.MaxUint8 >> mBitsThisByte) - - randomIP[i] = b & byteMask - - block.IP[i] = block.IP[i] | (b & byteMask) + block.IP[i] = block.IP[i] | (b & byte(math.MaxUint8>>mBitsThisByte)) } block.Mask = net.CIDRMask(bits, mBits) From e681907a355d70aaf341b32bfebf6feecaa3fdb2 Mon Sep 17 00:00:00 2001 From: Chris Marget Date: Tue, 30 Jul 2024 11:41:47 -0400 Subject: [PATCH 3/3] clear mutable attributes in resource `Read()` method of integer-style resource pools --- apstra/resource_asn_pool.go | 5 +++++ apstra/resource_integer_pool.go | 5 +++++ apstra/resource_vni_pool.go | 5 +++++ 3 files changed, 15 insertions(+) diff --git a/apstra/resource_asn_pool.go b/apstra/resource_asn_pool.go index 507dc867..50044abe 100644 --- a/apstra/resource_asn_pool.go +++ b/apstra/resource_asn_pool.go @@ -154,6 +154,11 @@ func (o *resourceAsnPool) Read(ctx context.Context, req resource.ReadRequest, re return } + newState.SetMutablesToNull(ctx, &resp.Diagnostics) + if resp.Diagnostics.HasError() { + return + } + // set state resp.Diagnostics.Append(resp.State.Set(ctx, &newState)...) } diff --git a/apstra/resource_integer_pool.go b/apstra/resource_integer_pool.go index 5a5c38f7..48089ca1 100644 --- a/apstra/resource_integer_pool.go +++ b/apstra/resource_integer_pool.go @@ -154,6 +154,11 @@ func (o *resourceIntegerPool) Read(ctx context.Context, req resource.ReadRequest return } + newState.SetMutablesToNull(ctx, &resp.Diagnostics) + if resp.Diagnostics.HasError() { + return + } + // set state resp.Diagnostics.Append(resp.State.Set(ctx, &newState)...) } diff --git a/apstra/resource_vni_pool.go b/apstra/resource_vni_pool.go index 2210788b..4079a2f6 100644 --- a/apstra/resource_vni_pool.go +++ b/apstra/resource_vni_pool.go @@ -154,6 +154,11 @@ func (o *resourceVniPool) Read(ctx context.Context, req resource.ReadRequest, re return } + newState.SetMutablesToNull(ctx, &resp.Diagnostics) + if resp.Diagnostics.HasError() { + return + } + // set state resp.Diagnostics.Append(resp.State.Set(ctx, &newState)...) }