Skip to content

Commit

Permalink
have metadata item fail on create if the key is already present
Browse files Browse the repository at this point in the history
  • Loading branch information
danawillow committed Jan 31, 2020
1 parent 62f0a99 commit 7e038dd
Show file tree
Hide file tree
Showing 2 changed files with 71 additions and 61 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,7 @@ func resourceComputeProjectMetadataItemCreate(d *schema.ResourceData, meta inter
key := d.Get("key").(string)
val := d.Get("value").(string)

err = updateComputeCommonInstanceMetadata(config, projectID, key, &val, int(d.Timeout(schema.TimeoutCreate).Minutes()))
err = updateComputeCommonInstanceMetadata(config, projectID, key, &val, int(d.Timeout(schema.TimeoutCreate).Minutes()), true)
if err != nil {
return err
}
Expand Down Expand Up @@ -108,7 +108,7 @@ func resourceComputeProjectMetadataItemUpdate(d *schema.ResourceData, meta inter
_, n := d.GetChange("value")
new := n.(string)

err = updateComputeCommonInstanceMetadata(config, projectID, key, &new, int(d.Timeout(schema.TimeoutUpdate).Minutes()))
err = updateComputeCommonInstanceMetadata(config, projectID, key, &new, int(d.Timeout(schema.TimeoutUpdate).Minutes()), false)
if err != nil {
return err
}
Expand All @@ -126,7 +126,7 @@ func resourceComputeProjectMetadataItemDelete(d *schema.ResourceData, meta inter

key := d.Get("key").(string)

err = updateComputeCommonInstanceMetadata(config, projectID, key, nil, int(d.Timeout(schema.TimeoutDelete).Minutes()))
err = updateComputeCommonInstanceMetadata(config, projectID, key, nil, int(d.Timeout(schema.TimeoutDelete).Minutes()), false)
if err != nil {
return err
}
Expand All @@ -135,7 +135,7 @@ func resourceComputeProjectMetadataItemDelete(d *schema.ResourceData, meta inter
return nil
}

func updateComputeCommonInstanceMetadata(config *Config, projectID string, key string, afterVal *string, timeout int) error {
func updateComputeCommonInstanceMetadata(config *Config, projectID string, key string, afterVal *string, timeout int, failIfPresent bool) error {
updateMD := func() error {
log.Printf("[DEBUG] Loading project metadata: %s", projectID)
project, err := config.clientCompute.Projects.Get(projectID).Do()
Expand All @@ -153,6 +153,9 @@ func updateComputeCommonInstanceMetadata(config *Config, projectID string, key s
return nil
}
} else {
if failIfPresent {
return fmt.Errorf("key %q already present in metadata for project %q. Use `terraform import` to manage it with Terraform", a)
}
if afterVal != nil && *afterVal == val {
// Asked to set a value and it's already set - we're done.
return nil
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ package google

import (
"fmt"
"regexp"
"testing"

"github.com/hashicorp/terraform-plugin-sdk/helper/acctest"
Expand All @@ -10,10 +11,9 @@ import (
)

func TestAccComputeProjectMetadataItem_basic(t *testing.T) {
t.Parallel(
// Key must be unique to avoid concurrent tests interfering with each other
)
t.Parallel()

// Key must be unique to avoid concurrent tests interfering with each other
key := "myKey" + acctest.RandString(10)

resource.Test(t, resource.TestCase{
Expand All @@ -22,10 +22,7 @@ func TestAccComputeProjectMetadataItem_basic(t *testing.T) {
CheckDestroy: testAccCheckProjectMetadataItemDestroy,
Steps: []resource.TestStep{
{
Config: testAccProjectMetadataItem_basicWithResourceName("foobar", key, "myValue"),
Check: resource.ComposeTestCheckFunc(
testAccCheckProjectMetadataItem_hasMetadata(key, "myValue"),
),
Config: testAccProjectMetadataItem_basic("foobar", key, "myValue"),
},
{
ResourceName: "google_compute_project_metadata_item.foobar",
Expand All @@ -37,33 +34,40 @@ func TestAccComputeProjectMetadataItem_basic(t *testing.T) {
}

func TestAccComputeProjectMetadataItem_basicMultiple(t *testing.T) {
t.Parallel(
t.Parallel()

// Generate a config of two config keys
)
key1 := "myKey" + acctest.RandString(10)
key2 := "myKey" + acctest.RandString(10)
config := testAccProjectMetadataItem_basic("foobar", key1, "myValue") +
testAccProjectMetadataItem_basic("foobar2", key2, "myOtherValue")

config := testAccProjectMetadataItem_basic("myKey", "myValue") +
testAccProjectMetadataItem_basic("myOtherKey", "myOtherValue")
resource.Test(t, resource.TestCase{
PreCheck: func() { testAccPreCheck(t) },
Providers: testAccProviders,
CheckDestroy: testAccCheckProjectMetadataItemDestroy,
Steps: []resource.TestStep{
{
Config: config,
Check: resource.ComposeTestCheckFunc(
testAccCheckProjectMetadataItem_hasMetadata("myKey", "myValue"),
testAccCheckProjectMetadataItem_hasMetadata("myOtherKey", "myOtherValue"),
),
},
{
ResourceName: "google_compute_project_metadata_item.foobar",
ImportState: true,
ImportStateVerify: true,
},
{
ResourceName: "google_compute_project_metadata_item.foobar2",
ImportState: true,
ImportStateVerify: true,
},
},
})
}

func TestAccComputeProjectMetadataItem_basicWithEmptyVal(t *testing.T) {
t.Parallel(
// Key must be unique to avoid concurrent tests interfering with each other
)
t.Parallel()

// Key must be unique to avoid concurrent tests interfering with each other
key := "myKey" + acctest.RandString(10)

resource.Test(t, resource.TestCase{
Expand All @@ -72,10 +76,7 @@ func TestAccComputeProjectMetadataItem_basicWithEmptyVal(t *testing.T) {
CheckDestroy: testAccCheckProjectMetadataItemDestroy,
Steps: []resource.TestStep{
{
Config: testAccProjectMetadataItem_basicWithResourceName("foobar", key, ""),
Check: resource.ComposeTestCheckFunc(
testAccCheckProjectMetadataItem_hasMetadata(key, ""),
),
Config: testAccProjectMetadataItem_basic("foobar", key, ""),
},
{
ResourceName: "google_compute_project_metadata_item.foobar",
Expand All @@ -87,10 +88,9 @@ func TestAccComputeProjectMetadataItem_basicWithEmptyVal(t *testing.T) {
}

func TestAccComputeProjectMetadataItem_basicUpdate(t *testing.T) {
t.Parallel(
// Key must be unique to avoid concurrent tests interfering with each other
)
t.Parallel()

// Key must be unique to avoid concurrent tests interfering with each other
key := "myKey" + acctest.RandString(10)

resource.Test(t, resource.TestCase{
Expand All @@ -99,41 +99,52 @@ func TestAccComputeProjectMetadataItem_basicUpdate(t *testing.T) {
CheckDestroy: testAccCheckProjectMetadataItemDestroy,
Steps: []resource.TestStep{
{
Config: testAccProjectMetadataItem_basicWithResourceName("foobar", key, "myValue"),
Check: resource.ComposeTestCheckFunc(
testAccCheckProjectMetadataItem_hasMetadata(key, "myValue"),
),
Config: testAccProjectMetadataItem_basic("foobar", key, "myValue"),
},
{
Config: testAccProjectMetadataItem_basicWithResourceName("foobar", key, "myUpdatedValue"),
Check: resource.ComposeTestCheckFunc(
testAccCheckProjectMetadataItem_hasMetadata(key, "myUpdatedValue"),
),
ResourceName: "google_compute_project_metadata_item.foobar",
ImportState: true,
ImportStateVerify: true,
},
{
Config: testAccProjectMetadataItem_basic("foobar", key, "myUpdatedValue"),
},
{
ResourceName: "google_compute_project_metadata_item.foobar",
ImportState: true,
ImportStateVerify: true,
},
},
})
}

func testAccCheckProjectMetadataItem_hasMetadata(key, value string) resource.TestCheckFunc {
return func(s *terraform.State) error {
config := testAccProvider.Meta().(*Config)

project, err := config.clientCompute.Projects.Get(config.Project).Do()
if err != nil {
return err
}

metadata := flattenMetadata(project.CommonInstanceMetadata)
func TestAccComputeProjectMetadataItem_exists(t *testing.T) {
t.Parallel()

val, ok := metadata[key]
if !ok {
return fmt.Errorf("Unable to find a value for key '%s'", key)
}
if val != value {
return fmt.Errorf("Value for key '%s' does not match. Expected '%s' but found '%s'", key, value, val)
}
return nil
}
// Key must be unique to avoid concurrent tests interfering with each other
key := "myKey" + acctest.RandString(10)
originalConfig := testAccProjectMetadataItem_basic("foobar", key, "myValue"),

resource.Test(t, resource.TestCase{
PreCheck: func() { testAccPreCheck(t) },
Providers: testAccProviders,
CheckDestroy: testAccCheckProjectMetadataItemDestroy,
Steps: []resource.TestStep{
{
Config: originalConfig,
},
{
ResourceName: "google_compute_project_metadata_item.foobar",
ImportState: true,
ImportStateVerify: true,
},
// Add a second resource with the same key
{
Config: originalConfig + testAccProjectMetadataItem_basic("foobar2", key, "myValue"),
ExpectError: regexp.MustCompile("already present in metadata for project"),
},
},
})
}

func testAccCheckProjectMetadataItemDestroy(s *terraform.State) error {
Expand All @@ -160,11 +171,7 @@ func testAccCheckProjectMetadataItemDestroy(s *terraform.State) error {
return nil
}

func testAccProjectMetadataItem_basic(key, val string) string {
return testAccProjectMetadataItem_basicWithResourceName(fmt.Sprintf("test_%s", acctest.RandString(10)), key, val)
}

func testAccProjectMetadataItem_basicWithResourceName(resourceName, key, val string) string {
func testAccProjectMetadataItem_basic(resourceName, key, val string) string {
return fmt.Sprintf(`
resource "google_compute_project_metadata_item" "%s" {
key = "%s"
Expand Down

0 comments on commit 7e038dd

Please sign in to comment.