// ---------------------------------------------------------------------------- // // *** AUTO GENERATED CODE *** AUTO GENERATED CODE *** // // ---------------------------------------------------------------------------- // // This file is automatically generated by Magic Modules and manual // changes will be clobbered when the file is regenerated. // // Please read more about how to change this file in // .github/CONTRIBUTING.md. // // ---------------------------------------------------------------------------- package google import ( "fmt" "log" "reflect" "regexp" "strconv" "strings" "time" "github.com/hashicorp/terraform/helper/resource" "github.com/hashicorp/terraform/helper/schema" ) func resourceSpannerInstance() *schema.Resource { return &schema.Resource{ Create: resourceSpannerInstanceCreate, Read: resourceSpannerInstanceRead, Update: resourceSpannerInstanceUpdate, Delete: resourceSpannerInstanceDelete, Importer: &schema.ResourceImporter{ State: resourceSpannerInstanceImport, }, Timeouts: &schema.ResourceTimeout{ Create: schema.DefaultTimeout(240 * time.Second), Update: schema.DefaultTimeout(240 * time.Second), Delete: schema.DefaultTimeout(240 * time.Second), }, Schema: map[string]*schema.Schema{ "config": { Type: schema.TypeString, Required: true, DiffSuppressFunc: compareSelfLinkOrResourceName, }, "display_name": { Type: schema.TypeString, Required: true, }, "name": { Type: schema.TypeString, Computed: true, Optional: true, ForceNew: true, ValidateFunc: validateRegexp(`^[a-z][-a-z0-9]*[a-z0-9]$`), }, "labels": { Type: schema.TypeMap, Optional: true, Elem: &schema.Schema{Type: schema.TypeString}, }, "num_nodes": { Type: schema.TypeInt, Optional: true, Default: 1, }, "state": { Type: schema.TypeString, Computed: true, }, "project": { Type: schema.TypeString, Optional: true, Computed: true, ForceNew: true, }, }, } } func resourceSpannerInstanceCreate(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) obj := make(map[string]interface{}) nameProp, err := expandSpannerInstanceName(d.Get("name"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("name"); !isEmptyValue(reflect.ValueOf(nameProp)) && (ok || !reflect.DeepEqual(v, nameProp)) { obj["name"] = nameProp } configProp, err := expandSpannerInstanceConfig(d.Get("config"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("config"); !isEmptyValue(reflect.ValueOf(configProp)) && (ok || !reflect.DeepEqual(v, configProp)) { obj["config"] = configProp } displayNameProp, err := expandSpannerInstanceDisplayName(d.Get("display_name"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("display_name"); !isEmptyValue(reflect.ValueOf(displayNameProp)) && (ok || !reflect.DeepEqual(v, displayNameProp)) { obj["displayName"] = displayNameProp } nodeCountProp, err := expandSpannerInstanceNum_nodes(d.Get("num_nodes"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("num_nodes"); !isEmptyValue(reflect.ValueOf(nodeCountProp)) && (ok || !reflect.DeepEqual(v, nodeCountProp)) { obj["nodeCount"] = nodeCountProp } labelsProp, err := expandSpannerInstanceLabels(d.Get("labels"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("labels"); !isEmptyValue(reflect.ValueOf(labelsProp)) && (ok || !reflect.DeepEqual(v, labelsProp)) { obj["labels"] = labelsProp } obj, err = resourceSpannerInstanceEncoder(d, meta, obj) if err != nil { return err } url, err := replaceVars(d, config, "https://spanner.googleapis.com/v1/projects/{{project}}/instances") if err != nil { return err } log.Printf("[DEBUG] Creating new Instance: %#v", obj) res, err := sendRequestWithTimeout(config, "POST", url, obj, d.Timeout(schema.TimeoutCreate)) if err != nil { return fmt.Errorf("Error creating Instance: %s", err) } // Store the ID now id, err := replaceVars(d, config, "{{project}}/{{name}}") if err != nil { return fmt.Errorf("Error constructing id: %s", err) } d.SetId(id) project, err := getProject(d, config) if err != nil { return err } waitErr := spannerOperationWaitTime( config, res, project, "Creating Instance", int(d.Timeout(schema.TimeoutCreate).Minutes())) if waitErr != nil { // The resource didn't actually create d.SetId("") return fmt.Errorf("Error waiting to create Instance: %s", waitErr) } log.Printf("[DEBUG] Finished creating Instance %q: %#v", d.Id(), res) // This is useful if the resource in question doesn't have a perfectly consistent API // That is, the Operation for Create might return before the Get operation shows the // completed state of the resource. time.Sleep(5 * time.Second) return resourceSpannerInstanceRead(d, meta) } func resourceSpannerInstanceRead(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) url, err := replaceVars(d, config, "https://spanner.googleapis.com/v1/projects/{{project}}/instances/{{name}}") if err != nil { return err } res, err := sendRequest(config, "GET", url, nil) if err != nil { return handleNotFoundError(err, d, fmt.Sprintf("SpannerInstance %q", d.Id())) } res, err = resourceSpannerInstanceDecoder(d, meta, res) if err != nil { return err } project, err := getProject(d, config) if err != nil { return err } if err := d.Set("project", project); err != nil { return fmt.Errorf("Error reading Instance: %s", err) } if err := d.Set("name", flattenSpannerInstanceName(res["name"], d)); err != nil { return fmt.Errorf("Error reading Instance: %s", err) } if err := d.Set("config", flattenSpannerInstanceConfig(res["config"], d)); err != nil { return fmt.Errorf("Error reading Instance: %s", err) } if err := d.Set("display_name", flattenSpannerInstanceDisplayName(res["displayName"], d)); err != nil { return fmt.Errorf("Error reading Instance: %s", err) } if err := d.Set("num_nodes", flattenSpannerInstanceNum_nodes(res["nodeCount"], d)); err != nil { return fmt.Errorf("Error reading Instance: %s", err) } if err := d.Set("labels", flattenSpannerInstanceLabels(res["labels"], d)); err != nil { return fmt.Errorf("Error reading Instance: %s", err) } if err := d.Set("state", flattenSpannerInstanceState(res["state"], d)); err != nil { return fmt.Errorf("Error reading Instance: %s", err) } return nil } func resourceSpannerInstanceUpdate(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) obj := make(map[string]interface{}) configProp, err := expandSpannerInstanceConfig(d.Get("config"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("config"); !isEmptyValue(reflect.ValueOf(v)) && (ok || !reflect.DeepEqual(v, configProp)) { obj["config"] = configProp } displayNameProp, err := expandSpannerInstanceDisplayName(d.Get("display_name"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("display_name"); !isEmptyValue(reflect.ValueOf(v)) && (ok || !reflect.DeepEqual(v, displayNameProp)) { obj["displayName"] = displayNameProp } nodeCountProp, err := expandSpannerInstanceNum_nodes(d.Get("num_nodes"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("num_nodes"); !isEmptyValue(reflect.ValueOf(v)) && (ok || !reflect.DeepEqual(v, nodeCountProp)) { obj["nodeCount"] = nodeCountProp } labelsProp, err := expandSpannerInstanceLabels(d.Get("labels"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("labels"); !isEmptyValue(reflect.ValueOf(v)) && (ok || !reflect.DeepEqual(v, labelsProp)) { obj["labels"] = labelsProp } obj, err = resourceSpannerInstanceUpdateEncoder(d, meta, obj) if err != nil { return err } url, err := replaceVars(d, config, "https://spanner.googleapis.com/v1/projects/{{project}}/instances/{{name}}") if err != nil { return err } log.Printf("[DEBUG] Updating Instance %q: %#v", d.Id(), obj) res, err := sendRequestWithTimeout(config, "PATCH", url, obj, d.Timeout(schema.TimeoutUpdate)) if err != nil { return fmt.Errorf("Error updating Instance %q: %s", d.Id(), err) } project, err := getProject(d, config) if err != nil { return err } err = spannerOperationWaitTime( config, res, project, "Updating Instance", int(d.Timeout(schema.TimeoutUpdate).Minutes())) if err != nil { return err } return resourceSpannerInstanceRead(d, meta) } func resourceSpannerInstanceDelete(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) url, err := replaceVars(d, config, "https://spanner.googleapis.com/v1/projects/{{project}}/instances/{{name}}") if err != nil { return err } var obj map[string]interface{} log.Printf("[DEBUG] Deleting Instance %q", d.Id()) res, err := sendRequestWithTimeout(config, "DELETE", url, obj, d.Timeout(schema.TimeoutDelete)) if err != nil { return handleNotFoundError(err, d, "Instance") } project, err := getProject(d, config) if err != nil { return err } err = spannerOperationWaitTime( config, res, project, "Deleting Instance", int(d.Timeout(schema.TimeoutDelete).Minutes())) if err != nil { return err } log.Printf("[DEBUG] Finished deleting Instance %q: %#v", d.Id(), res) return nil } func resourceSpannerInstanceImport(d *schema.ResourceData, meta interface{}) ([]*schema.ResourceData, error) { config := meta.(*Config) if err := parseImportId([]string{"projects/(?P[^/]+)/instances/(?P[^/]+)", "(?P[^/]+)/(?P[^/]+)", "(?P[^/]+)"}, d, config); err != nil { return nil, err } // Replace import id for the resource id id, err := replaceVars(d, config, "{{project}}/{{name}}") if err != nil { return nil, fmt.Errorf("Error constructing id: %s", err) } d.SetId(id) return []*schema.ResourceData{d}, nil } func flattenSpannerInstanceName(v interface{}, d *schema.ResourceData) interface{} { return v } func flattenSpannerInstanceConfig(v interface{}, d *schema.ResourceData) interface{} { if v == nil { return v } return ConvertSelfLinkToV1(v.(string)) } func flattenSpannerInstanceDisplayName(v interface{}, d *schema.ResourceData) interface{} { return v } func flattenSpannerInstanceNum_nodes(v interface{}, d *schema.ResourceData) interface{} { // Handles the string fixed64 format if strVal, ok := v.(string); ok { if intVal, err := strconv.ParseInt(strVal, 10, 64); err == nil { return intVal } // let terraform core handle it if we can't convert the string to an int. } return v } func flattenSpannerInstanceLabels(v interface{}, d *schema.ResourceData) interface{} { return v } func flattenSpannerInstanceState(v interface{}, d *schema.ResourceData) interface{} { return v } func expandSpannerInstanceName(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandSpannerInstanceConfig(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { r := regexp.MustCompile("projects/(.+)/instanceConfigs/(.+)") if r.MatchString(v.(string)) { return v.(string), nil } project, err := getProject(d, config) if err != nil { return nil, err } return fmt.Sprintf("projects/%s/instanceConfigs/%s", project, v.(string)), nil } func expandSpannerInstanceDisplayName(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandSpannerInstanceNum_nodes(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandSpannerInstanceLabels(v interface{}, d TerraformResourceData, config *Config) (map[string]string, error) { if v == nil { return map[string]string{}, nil } m := make(map[string]string) for k, val := range v.(map[string]interface{}) { m[k] = val.(string) } return m, nil } func resourceSpannerInstanceEncoder(d *schema.ResourceData, meta interface{}, obj map[string]interface{}) (map[string]interface{}, error) { newObj := make(map[string]interface{}) newObj["instance"] = obj if obj["name"] == nil { d.Set("name", resource.PrefixedUniqueId("tfgen-spanid-")[:30]) newObj["instanceId"] = d.Get("name").(string) } else { newObj["instanceId"] = obj["name"] } delete(obj, "name") return newObj, nil } func resourceSpannerInstanceUpdateEncoder(d *schema.ResourceData, meta interface{}, obj map[string]interface{}) (map[string]interface{}, error) { project, err := getProject(d, meta.(*Config)) if err != nil { return nil, err } obj["name"] = fmt.Sprintf("projects/%s/instances/%s", project, obj["name"]) newObj := make(map[string]interface{}) newObj["instance"] = obj updateMask := make([]string, 0) if d.HasChange("num_nodes") { updateMask = append(updateMask, "nodeCount") } if d.HasChange("display_name") { updateMask = append(updateMask, "displayName") } if d.HasChange("labels") { updateMask = append(updateMask, "labels") } newObj["fieldMask"] = strings.Join(updateMask, ",") return newObj, nil } func resourceSpannerInstanceDecoder(d *schema.ResourceData, meta interface{}, res map[string]interface{}) (map[string]interface{}, error) { config := meta.(*Config) d.SetId(res["name"].(string)) if err := parseImportId([]string{"projects/(?P[^/]+)/instances/(?P[^/]+)"}, d, config); err != nil { return nil, err } res["project"] = d.Get("project").(string) res["name"] = d.Get("name").(string) id, err := replaceVars(d, config, "{{project}}/{{name}}") if err != nil { return nil, err } d.SetId(id) return res, nil }