Skip to content

Commit

Permalink
resource/aws_dynamodb_table: Update and validate attributes
Browse files Browse the repository at this point in the history
  • Loading branch information
radeksimko committed Mar 20, 2018
1 parent d6eb19e commit 05cdd35
Show file tree
Hide file tree
Showing 3 changed files with 188 additions and 6 deletions.
23 changes: 17 additions & 6 deletions aws/resource_aws_dynamodb_table.go
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,9 @@ func resourceAwsDynamoDbTable() *schema.Resource {
func(diff *schema.ResourceDiff, v interface{}) error {
return validateDynamoDbStreamSpec(diff)
},
func(diff *schema.ResourceDiff, v interface{}) error {
return validateDynamoDbTableAttributes(diff)
},
func(diff *schema.ResourceDiff, v interface{}) error {
if diff.Id() != "" && diff.HasChange("server_side_encryption") {
o, n := diff.GetChange("server_side_encryption")
Expand Down Expand Up @@ -374,11 +377,10 @@ func resourceAwsDynamoDbTableUpdate(d *schema.ResourceData, meta interface{}) er
}
}

if d.HasChange("global_secondary_index") && !d.IsNewResource() {
var attributes []*dynamodb.AttributeDefinition
if v, ok := d.GetOk("attribute"); ok {
attributes = expandDynamoDbAttributes(v.(*schema.Set).List())
}
// Indexes and attributes are tightly coupled (DynamoDB requires attribute definitions
// for all indexed attributes) so it's necessary to update these together.
if (d.HasChange("global_secondary_index") || d.HasChange("attribute")) && !d.IsNewResource() {
attributes := d.Get("attribute").(*schema.Set).List()

o, n := d.GetChange("global_secondary_index")
ops, err := diffDynamoDbGSI(o.(*schema.Set).List(), n.(*schema.Set).List())
Expand All @@ -389,12 +391,13 @@ func resourceAwsDynamoDbTableUpdate(d *schema.ResourceData, meta interface{}) er

input := &dynamodb.UpdateTableInput{
TableName: aws.String(d.Id()),
AttributeDefinitions: attributes,
AttributeDefinitions: expandDynamoDbAttributes(attributes),
}

// Only 1 online index can be created or deleted simultaneously per table
for _, op := range ops {
input.GlobalSecondaryIndexUpdates = []*dynamodb.GlobalSecondaryIndexUpdate{op}
log.Printf("[DEBUG] Updating DynamoDB Table: %s", input)
_, err := conn.UpdateTable(input)
if err != nil {
return err
Expand All @@ -419,6 +422,14 @@ func resourceAwsDynamoDbTableUpdate(d *schema.ResourceData, meta interface{}) er
}
}

// We may only be changing the attribute type
if len(ops) == 0 {
_, err := conn.UpdateTable(input)
if err != nil {
return err
}
}

if err := waitForDynamoDbTableToBeActive(d.Id(), d.Timeout(schema.TimeoutUpdate), conn); err != nil {
return fmt.Errorf("Error waiting for DynamoDB Table op: %s", err)
}
Expand Down
121 changes: 121 additions & 0 deletions aws/resource_aws_dynamodb_table_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -656,6 +656,65 @@ func TestAccAWSDynamoDbTable_ttl(t *testing.T) {
},
})
}

func TestAccAWSDynamoDbTable_attributeUpdate(t *testing.T) {
var conf dynamodb.DescribeTableOutput

rName := acctest.RandomWithPrefix("TerraformTestTable-")

resource.Test(t, resource.TestCase{
PreCheck: func() { testAccPreCheck(t) },
Providers: testAccProviders,
CheckDestroy: testAccCheckAWSDynamoDbTableDestroy,
Steps: []resource.TestStep{
{
Config: testAccAWSDynamoDbConfigOneAttribute(rName, "firstKey", "firstKey", "S"),
Check: resource.ComposeTestCheckFunc(
testAccCheckInitialAWSDynamoDbTableExists("aws_dynamodb_table.basic-dynamodb-table", &conf),
),
},
{ // Attribute type change
Config: testAccAWSDynamoDbConfigOneAttribute(rName, "firstKey", "firstKey", "N"),
Check: resource.ComposeTestCheckFunc(
testAccCheckInitialAWSDynamoDbTableExists("aws_dynamodb_table.basic-dynamodb-table", &conf),
),
},
{ // New attribute addition (index update)
Config: testAccAWSDynamoDbConfigTwoAttributes(rName, "firstKey", "secondKey", "firstKey", "N", "secondKey", "S"),
Check: resource.ComposeTestCheckFunc(
testAccCheckInitialAWSDynamoDbTableExists("aws_dynamodb_table.basic-dynamodb-table", &conf),
),
},
{ // Attribute removal (index update)
Config: testAccAWSDynamoDbConfigOneAttribute(rName, "firstKey", "firstKey", "S"),
Check: resource.ComposeTestCheckFunc(
testAccCheckInitialAWSDynamoDbTableExists("aws_dynamodb_table.basic-dynamodb-table", &conf),
),
},
},
})
}

func TestAccAWSDynamoDbTable_attributeUpdateValidation(t *testing.T) {
rName := acctest.RandomWithPrefix("TerraformTestTable-")

resource.Test(t, resource.TestCase{
PreCheck: func() { testAccPreCheck(t) },
Providers: testAccProviders,
CheckDestroy: testAccCheckAWSDynamoDbTableDestroy,
Steps: []resource.TestStep{
{
Config: testAccAWSDynamoDbConfigOneAttribute(rName, "firstKey", "unusedKey", "S"),
ExpectError: regexp.MustCompile(`All attributes must be indexed. Unused attributes: \["unusedKey"\]$`),
},
{
Config: testAccAWSDynamoDbConfigTwoAttributes(rName, "firstKey", "secondKey", "firstUnused", "N", "secondUnused", "S"),
ExpectError: regexp.MustCompile(`All attributes must be indexed. Unused attributes: \["firstUnused"\ \"secondUnused\"]$`),
},
},
})
}

func testAccCheckDynamoDbTableTimeToLiveWasUpdated(n string) resource.TestCheckFunc {
return func(s *terraform.State) error {
log.Printf("[DEBUG] Trying to create initial table state!")
Expand Down Expand Up @@ -1468,3 +1527,65 @@ resource "aws_dynamodb_table" "basic-dynamodb-table" {
}
`, rName)
}

func testAccAWSDynamoDbConfigOneAttribute(rName, hashKey, attrName, attrType string) string {
return fmt.Sprintf(`
resource "aws_dynamodb_table" "basic-dynamodb-table" {
name = "%s"
read_capacity = 10
write_capacity = 10
hash_key = "staticHashKey"
attribute {
name = "staticHashKey"
type = "S"
}
attribute {
name = "%s"
type = "%s"
}
global_secondary_index {
name = "gsiName"
hash_key = "%s"
write_capacity = 10
read_capacity = 10
projection_type = "KEYS_ONLY"
}
}
`, rName, attrName, attrType, hashKey)
}

func testAccAWSDynamoDbConfigTwoAttributes(rName, hashKey, rangeKey, attrName1, attrType1, attrName2, attrType2 string) string {
return fmt.Sprintf(`
resource "aws_dynamodb_table" "basic-dynamodb-table" {
name = "%s"
read_capacity = 10
write_capacity = 10
hash_key = "staticHashKey"
attribute {
name = "staticHashKey"
type = "S"
}
attribute {
name = "%s"
type = "%s"
}
attribute {
name = "%s"
type = "%s"
}
global_secondary_index {
name = "gsiName"
hash_key = "%s"
range_key = "%s"
write_capacity = 10
read_capacity = 10
projection_type = "KEYS_ONLY"
}
}
`, rName, attrName1, attrType1, attrName2, attrType2, hashKey, rangeKey)
}
50 changes: 50 additions & 0 deletions aws/validators.go
Original file line number Diff line number Diff line change
Expand Up @@ -1690,3 +1690,53 @@ func validateIotThingTypeSearchableAttribute(v interface{}, k string) (ws []stri
}
return
}

func validateDynamoDbTableAttributes(d *schema.ResourceDiff) error {
// Collect all indexed attributes
primaryHashKey := d.Get("hash_key").(string)
indexedAttributes := map[string]bool{
primaryHashKey: true,
}
if v, ok := d.GetOk("range_key"); ok {
indexedAttributes[v.(string)] = true
}
if v, ok := d.GetOk("local_secondary_index"); ok {
indexes := v.(*schema.Set).List()
for _, idx := range indexes {
index := idx.(map[string]interface{})
rangeKey := index["range_key"].(string)
indexedAttributes[rangeKey] = true
}
}
if v, ok := d.GetOk("global_secondary_index"); ok {
indexes := v.(*schema.Set).List()
for _, idx := range indexes {
index := idx.(map[string]interface{})

hashKey := index["hash_key"].(string)
indexedAttributes[hashKey] = true

if rk, ok := index["range_key"]; ok {
indexedAttributes[rk.(string)] = true
}
}
}

// Check if all indexed attributes have an attribute definition
attributes := d.Get("attribute").(*schema.Set).List()
missingAttrDefs := []string{}
for _, attr := range attributes {
attribute := attr.(map[string]interface{})
attrName := attribute["name"].(string)

if _, ok := indexedAttributes[attrName]; !ok {
missingAttrDefs = append(missingAttrDefs, attrName)
}
}

if len(missingAttrDefs) > 0 {
return fmt.Errorf("All attributes must be indexed. Unused attributes: %q", missingAttrDefs)
}

return nil
}

0 comments on commit 05cdd35

Please sign in to comment.