From 4c33028edd858d064b2928a5e1caa5ae546dd55a Mon Sep 17 00:00:00 2001 From: The Magician Date: Wed, 12 Feb 2020 14:37:47 -0800 Subject: [PATCH] Add state waiting to google_container_node_pool (#3114) (#1758) Signed-off-by: Modular Magician --- .changelog/3114.txt | 6 ++ google-beta/resource_container_node_pool.go | 66 ++++++++++++++++++++- 2 files changed, 71 insertions(+), 1 deletion(-) create mode 100644 .changelog/3114.txt diff --git a/.changelog/3114.txt b/.changelog/3114.txt new file mode 100644 index 0000000000..30fba691ac --- /dev/null +++ b/.changelog/3114.txt @@ -0,0 +1,6 @@ +```release-note:enhancement +container: `google_container_node_pool` will allow importing / updating / deleting node pools in error states and will wait for a stable state after any changes. +``` +```release-note:enhancement +container: `google_container_node_pool` resources created in an error state will be marked as tainted on creation. +``` diff --git a/google-beta/resource_container_node_pool.go b/google-beta/resource_container_node_pool.go index a759502cf2..173f769fdf 100644 --- a/google-beta/resource_container_node_pool.go +++ b/google-beta/resource_container_node_pool.go @@ -300,7 +300,20 @@ func resourceContainerNodePoolCreate(d *schema.ResourceData, meta interface{}) e log.Printf("[INFO] GKE NodePool %s has been created", nodePool.Name) - return resourceContainerNodePoolRead(d, meta) + if err = resourceContainerNodePoolRead(d, meta); err != nil { + return err + } + + state, err := containerNodePoolAwaitRestingState(config, d.Id(), d.Timeout(schema.TimeoutCreate)) + if err != nil { + return err + } + + if containerNodePoolRestingStates[state] == ErrorState { + return fmt.Errorf("NodePool %s was created in the error state %q", nodePool.Name, state) + } + + return nil } func resourceContainerNodePoolRead(d *schema.ResourceData, meta interface{}) error { @@ -355,12 +368,22 @@ func resourceContainerNodePoolUpdate(d *schema.ResourceData, meta interface{}) e return err } + _, err = containerNodePoolAwaitRestingState(config, d.Id(), d.Timeout(schema.TimeoutUpdate)) + if err != nil { + return err + } + d.Partial(true) if err := nodePoolUpdate(d, meta, nodePoolInfo, "", timeoutInMinutes); err != nil { return err } d.Partial(false) + _, err = containerNodePoolAwaitRestingState(config, d.Id(), d.Timeout(schema.TimeoutUpdate)) + if err != nil { + return err + } + return resourceContainerNodePoolRead(d, meta) } @@ -374,6 +397,11 @@ func resourceContainerNodePoolDelete(d *schema.ResourceData, meta interface{}) e name := getNodePoolName(d.Id()) + _, err = containerNodePoolAwaitRestingState(config, d.Id(), d.Timeout(schema.TimeoutDelete)) + if err != nil { + return err + } + timeoutInMinutes := int(d.Timeout(schema.TimeoutDelete).Minutes()) mutexKV.Lock(nodePoolInfo.lockKey()) @@ -444,8 +472,13 @@ func resourceContainerNodePoolStateImporter(d *schema.ResourceData, meta interfa if err != nil { return nil, err } + d.SetId(id) + if _, err := containerNodePoolAwaitRestingState(config, d.Id(), d.Timeout(schema.TimeoutCreate)); err != nil { + return nil, err + } + return []*schema.ResourceData{d}, nil } @@ -854,3 +887,34 @@ func getNodePoolName(id string) string { splits := strings.Split(id, "/") return splits[len(splits)-1] } + +var containerNodePoolRestingStates = RestingStates{ + "RUNNING": ReadyState, + "RUNNING_WITH_ERROR": ErrorState, + "ERROR": ErrorState, +} + +// takes in a config object, full node pool name, and the current CRUD action timeout +// returns a state with no error if the state is a resting state, and the last state with an error otherwise +func containerNodePoolAwaitRestingState(config *Config, name string, timeout time.Duration) (state string, err error) { + err = resource.Retry(timeout, func() *resource.RetryError { + nodePool, gErr := config.clientContainerBeta.Projects.Locations.Clusters.NodePools.Get(name).Do() + if gErr != nil { + return resource.NonRetryableError(gErr) + } + + state = nodePool.Status + switch stateType := containerNodePoolRestingStates[state]; stateType { + case ReadyState: + log.Printf("[DEBUG] NodePool %q has status %q with message %q.", name, state, nodePool.StatusMessage) + return nil + case ErrorState: + log.Printf("[DEBUG] NodePool %q has error state %q with message %q.", name, state, nodePool.StatusMessage) + return nil + default: + return resource.RetryableError(fmt.Errorf("NodePool %q has state %q with message %q", name, state, nodePool.StatusMessage)) + } + }) + + return state, err +}