// ---------------------------------------------------------------------------- // // *** 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" "strconv" "time" "github.com/hashicorp/terraform/helper/schema" "github.com/hashicorp/terraform/helper/validation" "google.golang.org/api/compute/v1" ) func resourceComputeImage() *schema.Resource { return &schema.Resource{ Create: resourceComputeImageCreate, Read: resourceComputeImageRead, Update: resourceComputeImageUpdate, Delete: resourceComputeImageDelete, Importer: &schema.ResourceImporter{ State: resourceComputeImageImport, }, 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{ "name": { Type: schema.TypeString, Required: true, ForceNew: true, }, "description": { Type: schema.TypeString, Optional: true, ForceNew: true, }, "disk_size_gb": { Type: schema.TypeInt, Computed: true, Optional: true, ForceNew: true, }, "family": { Type: schema.TypeString, Optional: true, ForceNew: true, }, "labels": { Type: schema.TypeMap, Optional: true, Elem: &schema.Schema{Type: schema.TypeString}, }, "licenses": { Type: schema.TypeList, Computed: true, Optional: true, ForceNew: true, Elem: &schema.Schema{ Type: schema.TypeString, DiffSuppressFunc: compareSelfLinkOrResourceName, }, }, "raw_disk": { Type: schema.TypeList, Optional: true, ForceNew: true, MaxItems: 1, Elem: &schema.Resource{ Schema: map[string]*schema.Schema{ "source": { Type: schema.TypeString, Required: true, ForceNew: true, }, "container_type": { Type: schema.TypeString, Optional: true, ForceNew: true, ValidateFunc: validation.StringInSlice([]string{"TAR", ""}, false), Default: "TAR", }, "sha1": { Type: schema.TypeString, Optional: true, ForceNew: true, }, }, }, }, "source_disk": { Type: schema.TypeString, Optional: true, ForceNew: true, DiffSuppressFunc: compareSelfLinkOrResourceName, }, "archive_size_bytes": { Type: schema.TypeInt, Computed: true, }, "creation_timestamp": { Type: schema.TypeString, Computed: true, }, "label_fingerprint": { Type: schema.TypeString, Computed: true, }, "project": { Type: schema.TypeString, Optional: true, Computed: true, ForceNew: true, }, "self_link": { Type: schema.TypeString, Computed: true, }, }, } } func resourceComputeImageCreate(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) obj := make(map[string]interface{}) descriptionProp, err := expandComputeImageDescription(d.Get("description"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("description"); !isEmptyValue(reflect.ValueOf(descriptionProp)) && (ok || !reflect.DeepEqual(v, descriptionProp)) { obj["description"] = descriptionProp } diskSizeGbProp, err := expandComputeImageDiskSizeGb(d.Get("disk_size_gb"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("disk_size_gb"); !isEmptyValue(reflect.ValueOf(diskSizeGbProp)) && (ok || !reflect.DeepEqual(v, diskSizeGbProp)) { obj["diskSizeGb"] = diskSizeGbProp } familyProp, err := expandComputeImageFamily(d.Get("family"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("family"); !isEmptyValue(reflect.ValueOf(familyProp)) && (ok || !reflect.DeepEqual(v, familyProp)) { obj["family"] = familyProp } labelsProp, err := expandComputeImageLabels(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 } labelFingerprintProp, err := expandComputeImageLabelFingerprint(d.Get("label_fingerprint"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("label_fingerprint"); !isEmptyValue(reflect.ValueOf(labelFingerprintProp)) && (ok || !reflect.DeepEqual(v, labelFingerprintProp)) { obj["labelFingerprint"] = labelFingerprintProp } licensesProp, err := expandComputeImageLicenses(d.Get("licenses"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("licenses"); !isEmptyValue(reflect.ValueOf(licensesProp)) && (ok || !reflect.DeepEqual(v, licensesProp)) { obj["licenses"] = licensesProp } nameProp, err := expandComputeImageName(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 } rawDiskProp, err := expandComputeImageRawDisk(d.Get("raw_disk"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("raw_disk"); !isEmptyValue(reflect.ValueOf(rawDiskProp)) && (ok || !reflect.DeepEqual(v, rawDiskProp)) { obj["rawDisk"] = rawDiskProp } sourceDiskProp, err := expandComputeImageSourceDisk(d.Get("source_disk"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("source_disk"); !isEmptyValue(reflect.ValueOf(sourceDiskProp)) && (ok || !reflect.DeepEqual(v, sourceDiskProp)) { obj["sourceDisk"] = sourceDiskProp } url, err := replaceVars(d, config, "https://www.googleapis.com/compute/v1/projects/{{project}}/global/images") if err != nil { return err } log.Printf("[DEBUG] Creating new Image: %#v", obj) res, err := sendRequestWithTimeout(config, "POST", url, obj, d.Timeout(schema.TimeoutCreate)) if err != nil { return fmt.Errorf("Error creating Image: %s", err) } // Store the ID now id, err := replaceVars(d, config, "{{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 } op := &compute.Operation{} err = Convert(res, op) if err != nil { return err } waitErr := computeOperationWaitTime( config.clientCompute, op, project, "Creating Image", int(d.Timeout(schema.TimeoutCreate).Minutes())) if waitErr != nil { // The resource didn't actually create d.SetId("") return fmt.Errorf("Error waiting to create Image: %s", waitErr) } log.Printf("[DEBUG] Finished creating Image %q: %#v", d.Id(), res) return resourceComputeImageRead(d, meta) } func resourceComputeImageRead(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) url, err := replaceVars(d, config, "https://www.googleapis.com/compute/v1/projects/{{project}}/global/images/{{name}}") if err != nil { return err } res, err := sendRequest(config, "GET", url, nil) if err != nil { return handleNotFoundError(err, d, fmt.Sprintf("ComputeImage %q", d.Id())) } project, err := getProject(d, config) if err != nil { return err } if err := d.Set("project", project); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("archive_size_bytes", flattenComputeImageArchiveSizeBytes(res["archiveSizeBytes"], d)); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("creation_timestamp", flattenComputeImageCreationTimestamp(res["creationTimestamp"], d)); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("description", flattenComputeImageDescription(res["description"], d)); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("disk_size_gb", flattenComputeImageDiskSizeGb(res["diskSizeGb"], d)); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("family", flattenComputeImageFamily(res["family"], d)); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("labels", flattenComputeImageLabels(res["labels"], d)); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("label_fingerprint", flattenComputeImageLabelFingerprint(res["labelFingerprint"], d)); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("licenses", flattenComputeImageLicenses(res["licenses"], d)); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("name", flattenComputeImageName(res["name"], d)); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("source_disk", flattenComputeImageSourceDisk(res["sourceDisk"], d)); err != nil { return fmt.Errorf("Error reading Image: %s", err) } if err := d.Set("self_link", ConvertSelfLinkToV1(res["selfLink"].(string))); err != nil { return fmt.Errorf("Error reading Image: %s", err) } return nil } func resourceComputeImageUpdate(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) d.Partial(true) if d.HasChange("labels") || d.HasChange("label_fingerprint") { obj := make(map[string]interface{}) labelsProp, err := expandComputeImageLabels(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 } labelFingerprintProp, err := expandComputeImageLabelFingerprint(d.Get("label_fingerprint"), d, config) if err != nil { return err } else if v, ok := d.GetOkExists("label_fingerprint"); !isEmptyValue(reflect.ValueOf(v)) && (ok || !reflect.DeepEqual(v, labelFingerprintProp)) { obj["labelFingerprint"] = labelFingerprintProp } url, err := replaceVars(d, config, "https://www.googleapis.com/compute/v1/projects/{{project}}/global/images/{{name}}/setLabels") if err != nil { return err } res, err := sendRequestWithTimeout(config, "POST", url, obj, d.Timeout(schema.TimeoutUpdate)) if err != nil { return fmt.Errorf("Error updating Image %q: %s", d.Id(), err) } project, err := getProject(d, config) if err != nil { return err } op := &compute.Operation{} err = Convert(res, op) if err != nil { return err } err = computeOperationWaitTime( config.clientCompute, op, project, "Updating Image", int(d.Timeout(schema.TimeoutUpdate).Minutes())) if err != nil { return err } d.SetPartial("labels") d.SetPartial("label_fingerprint") } d.Partial(false) return resourceComputeImageRead(d, meta) } func resourceComputeImageDelete(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) url, err := replaceVars(d, config, "https://www.googleapis.com/compute/v1/projects/{{project}}/global/images/{{name}}") if err != nil { return err } var obj map[string]interface{} log.Printf("[DEBUG] Deleting Image %q", d.Id()) res, err := sendRequestWithTimeout(config, "DELETE", url, obj, d.Timeout(schema.TimeoutDelete)) if err != nil { return handleNotFoundError(err, d, "Image") } project, err := getProject(d, config) if err != nil { return err } op := &compute.Operation{} err = Convert(res, op) if err != nil { return err } err = computeOperationWaitTime( config.clientCompute, op, project, "Deleting Image", int(d.Timeout(schema.TimeoutDelete).Minutes())) if err != nil { return err } log.Printf("[DEBUG] Finished deleting Image %q: %#v", d.Id(), res) return nil } func resourceComputeImageImport(d *schema.ResourceData, meta interface{}) ([]*schema.ResourceData, error) { config := meta.(*Config) if err := parseImportId([]string{"projects/(?P[^/]+)/global/images/(?P[^/]+)", "(?P[^/]+)/(?P[^/]+)", "(?P[^/]+)"}, d, config); err != nil { return nil, err } // Replace import id for the resource id id, err := replaceVars(d, config, "{{name}}") if err != nil { return nil, fmt.Errorf("Error constructing id: %s", err) } d.SetId(id) return []*schema.ResourceData{d}, nil } func flattenComputeImageArchiveSizeBytes(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 flattenComputeImageCreationTimestamp(v interface{}, d *schema.ResourceData) interface{} { return v } func flattenComputeImageDescription(v interface{}, d *schema.ResourceData) interface{} { return v } func flattenComputeImageDiskSizeGb(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 flattenComputeImageFamily(v interface{}, d *schema.ResourceData) interface{} { return v } func flattenComputeImageLabels(v interface{}, d *schema.ResourceData) interface{} { return v } func flattenComputeImageLabelFingerprint(v interface{}, d *schema.ResourceData) interface{} { return v } func flattenComputeImageLicenses(v interface{}, d *schema.ResourceData) interface{} { if v == nil { return v } return convertAndMapStringArr(v.([]interface{}), ConvertSelfLinkToV1) } func flattenComputeImageName(v interface{}, d *schema.ResourceData) interface{} { return v } func flattenComputeImageSourceDisk(v interface{}, d *schema.ResourceData) interface{} { if v == nil { return v } return ConvertSelfLinkToV1(v.(string)) } func expandComputeImageDescription(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandComputeImageDiskSizeGb(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandComputeImageFamily(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandComputeImageLabels(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 expandComputeImageLabelFingerprint(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandComputeImageLicenses(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { l := v.([]interface{}) req := make([]interface{}, 0, len(l)) for _, raw := range l { f, err := parseGlobalFieldValue("licenses", raw.(string), "project", d, config, true) if err != nil { return nil, fmt.Errorf("Invalid value for licenses: %s", err) } req = append(req, f.RelativeLink()) } return req, nil } func expandComputeImageName(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandComputeImageRawDisk(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { l := v.([]interface{}) if len(l) == 0 || l[0] == nil { return nil, nil } raw := l[0] original := raw.(map[string]interface{}) transformed := make(map[string]interface{}) transformedContainerType, err := expandComputeImageRawDiskContainerType(original["container_type"], d, config) if err != nil { return nil, err } else if val := reflect.ValueOf(transformedContainerType); val.IsValid() && !isEmptyValue(val) { transformed["containerType"] = transformedContainerType } transformedSha1, err := expandComputeImageRawDiskSha1(original["sha1"], d, config) if err != nil { return nil, err } else if val := reflect.ValueOf(transformedSha1); val.IsValid() && !isEmptyValue(val) { transformed["sha1Checksum"] = transformedSha1 } transformedSource, err := expandComputeImageRawDiskSource(original["source"], d, config) if err != nil { return nil, err } else if val := reflect.ValueOf(transformedSource); val.IsValid() && !isEmptyValue(val) { transformed["source"] = transformedSource } return transformed, nil } func expandComputeImageRawDiskContainerType(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandComputeImageRawDiskSha1(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandComputeImageRawDiskSource(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { return v, nil } func expandComputeImageSourceDisk(v interface{}, d TerraformResourceData, config *Config) (interface{}, error) { f, err := parseZonalFieldValue("disks", v.(string), "project", "zone", d, config, true) if err != nil { return nil, fmt.Errorf("Invalid value for source_disk: %s", err) } return f.RelativeLink(), nil }