Skip to content

Commit

Permalink
helper/resource: Added error when errantly checking list, map, or set…
Browse files Browse the repository at this point in the history
… attributes in `TestCheckNoResourceAttr`, `TestCheckResourceAttr`, and `TestCheckResourceAttrSet` (#920)

Reference: #885
  • Loading branch information
bflad authored Mar 31, 2022
1 parent 40ff3ed commit f74b5a6
Show file tree
Hide file tree
Showing 3 changed files with 990 additions and 64 deletions.
7 changes: 7 additions & 0 deletions .changelog/920.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
```release-note:enhancement
helper/resource: Added error when errantly checking list, map, or set attributes in `TestCheckNoResourceAttr`, `TestCheckResourceAttr`, and `TestCheckResourceAttrSet`
```

```release-note:note
helper/resource: False positive checks of list, map, and set attributes with `TestCheckNoResourceAttr` and `TestCheckResourceAttrSet` will now return an error to explain how to accurately check those types of attributes. Some previously passing tests will now fail until the check is correctly updated.
```
100 changes: 78 additions & 22 deletions helper/resource/testing.go
Original file line number Diff line number Diff line change
Expand Up @@ -823,11 +823,33 @@ func TestCheckModuleResourceAttrSet(mp []string, name string, key string) TestCh
}

func testCheckResourceAttrSet(is *terraform.InstanceState, name string, key string) error {
if val, ok := is.Attributes[key]; !ok || val == "" {
return fmt.Errorf("%s: Attribute '%s' expected to be set", name, key)
val, ok := is.Attributes[key]

if ok && val != "" {
return nil
}

return nil
if _, ok := is.Attributes[key+".#"]; ok {
return fmt.Errorf(
"%s: list or set attribute '%s' must be checked by element count key (%s) or element value keys (e.g. %s). Set element value checks should use TestCheckTypeSet functions instead.",
name,
key,
key+".#",
key+".0",
)
}

if _, ok := is.Attributes[key+".%"]; ok {
return fmt.Errorf(
"%s: map attribute '%s' must be checked by element count key (%s) or element value keys (e.g. %s).",
name,
key,
key+".%",
key+".examplekey",
)
}

return fmt.Errorf("%s: Attribute '%s' expected to be set", name, key)
}

// TestCheckResourceAttr ensures a specific value is stored in state for the
Expand Down Expand Up @@ -892,30 +914,48 @@ func TestCheckModuleResourceAttr(mp []string, name string, key string, value str
}

func testCheckResourceAttr(is *terraform.InstanceState, name string, key string, value string) error {
// Empty containers may be elided from the state.
// If the intent here is to check for an empty container, allow the key to
// also be non-existent.
emptyCheck := false
if value == "0" && (strings.HasSuffix(key, ".#") || strings.HasSuffix(key, ".%")) {
emptyCheck = true
}
v, ok := is.Attributes[key]

if v, ok := is.Attributes[key]; !ok || v != value {
if emptyCheck && !ok {
if !ok {
// Empty containers may be elided from the state.
// If the intent here is to check for an empty container, allow the key to
// also be non-existent.
if value == "0" && (strings.HasSuffix(key, ".#") || strings.HasSuffix(key, ".%")) {
return nil
}

if !ok {
return fmt.Errorf("%s: Attribute '%s' not found", name, key)
if _, ok := is.Attributes[key+".#"]; ok {
return fmt.Errorf(
"%s: list or set attribute '%s' must be checked by element count key (%s) or element value keys (e.g. %s). Set element value checks should use TestCheckTypeSet functions instead.",
name,
key,
key+".#",
key+".0",
)
}

if _, ok := is.Attributes[key+".%"]; ok {
return fmt.Errorf(
"%s: map attribute '%s' must be checked by element count key (%s) or element value keys (e.g. %s).",
name,
key,
key+".%",
key+".examplekey",
)
}

return fmt.Errorf("%s: Attribute '%s' not found", name, key)
}

if v != value {
return fmt.Errorf(
"%s: Attribute '%s' expected %#v, got %#v",
name,
key,
value,
v)
}

return nil
}

Expand Down Expand Up @@ -976,23 +1016,39 @@ func TestCheckModuleNoResourceAttr(mp []string, name string, key string) TestChe
}

func testCheckNoResourceAttr(is *terraform.InstanceState, name string, key string) error {
v, ok := is.Attributes[key]

// Empty containers may sometimes be included in the state.
// If the intent here is to check for an empty container, allow the value to
// also be "0".
emptyCheck := false
if strings.HasSuffix(key, ".#") || strings.HasSuffix(key, ".%") {
emptyCheck = true
}

val, exists := is.Attributes[key]
if emptyCheck && val == "0" {
if v == "0" && (strings.HasSuffix(key, ".#") || strings.HasSuffix(key, ".%")) {
return nil
}

if exists {
if ok {
return fmt.Errorf("%s: Attribute '%s' found when not expected", name, key)
}

if _, ok := is.Attributes[key+".#"]; ok {
return fmt.Errorf(
"%s: list or set attribute '%s' must be checked by element count key (%s) or element value keys (e.g. %s). Set element value checks should use TestCheckTypeSet functions instead.",
name,
key,
key+".#",
key+".0",
)
}

if _, ok := is.Attributes[key+".%"]; ok {
return fmt.Errorf(
"%s: map attribute '%s' must be checked by element count key (%s) or element value keys (e.g. %s).",
name,
key,
key+".%",
key+".examplekey",
)
}

return nil
}

Expand Down
Loading

0 comments on commit f74b5a6

Please sign in to comment.