package google import ( "bytes" "fmt" "log" "os" "regexp" "github.com/hashicorp/terraform/helper/hashcode" "github.com/hashicorp/terraform/helper/schema" "google.golang.org/api/compute/v0.beta" "google.golang.org/api/googleapi" ) func resourceComputeRegionBackendService() *schema.Resource { return &schema.Resource{ Create: resourceComputeRegionBackendServiceCreate, Read: resourceComputeRegionBackendServiceRead, Update: resourceComputeRegionBackendServiceUpdate, Delete: resourceComputeRegionBackendServiceDelete, Schema: map[string]*schema.Schema{ "name": &schema.Schema{ Type: schema.TypeString, Required: true, ForceNew: true, ValidateFunc: func(v interface{}, k string) (ws []string, errors []error) { value := v.(string) re := `^(?:[a-z](?:[-a-z0-9]{0,61}[a-z0-9])?)$` if !regexp.MustCompile(re).MatchString(value) { errors = append(errors, fmt.Errorf( "%q (%q) doesn't match regexp %q", k, value, re)) } return }, }, "health_checks": &schema.Schema{ Type: schema.TypeSet, Elem: &schema.Schema{Type: schema.TypeString}, Required: true, Set: schema.HashString, }, "backend": &schema.Schema{ Type: schema.TypeSet, Elem: &schema.Resource{ Schema: map[string]*schema.Schema{ "group": &schema.Schema{ Type: schema.TypeString, Optional: true, }, "balancing_mode": &schema.Schema{ Type: schema.TypeString, Optional: true, Default: "UTILIZATION", }, "capacity_scaler": &schema.Schema{ Type: schema.TypeFloat, Optional: true, Default: 1, }, "description": &schema.Schema{ Type: schema.TypeString, Optional: true, }, "max_rate": &schema.Schema{ Type: schema.TypeInt, Optional: true, }, "max_rate_per_instance": &schema.Schema{ Type: schema.TypeFloat, Optional: true, }, "max_utilization": &schema.Schema{ Type: schema.TypeFloat, Optional: true, Default: 0.8, }, }, }, Optional: true, Set: resourceGoogleComputeRegionBackendServiceBackendHash, }, "description": &schema.Schema{ Type: schema.TypeString, Optional: true, }, "enable_cdn": &schema.Schema{ Type: schema.TypeBool, Optional: true, Default: false, }, "fingerprint": &schema.Schema{ Type: schema.TypeString, Computed: true, }, "load_balancing_scheme": &schema.Schema{ Type: schema.TypeString, Optional: true, ForceNew: true, }, "port_name": &schema.Schema{ Type: schema.TypeString, Optional: true, Computed: true, }, "project": &schema.Schema{ Type: schema.TypeString, Optional: true, ForceNew: true, }, "protocol": &schema.Schema{ Type: schema.TypeString, Optional: true, Computed: true, }, "region": &schema.Schema{ Type: schema.TypeString, Optional: true, ForceNew: true, }, "self_link": &schema.Schema{ Type: schema.TypeString, Computed: true, }, "timeout_sec": &schema.Schema{ Type: schema.TypeInt, Optional: true, Computed: true, }, }, } } func resourceComputeRegionBackendServiceCreate(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) hc := d.Get("health_checks").(*schema.Set).List() healthChecks := make([]string, 0, len(hc)) for _, v := range hc { healthChecks = append(healthChecks, v.(string)) } service := compute.BackendService{ Name: d.Get("name").(string), HealthChecks: healthChecks, } if v, ok := d.GetOk("backend"); ok { service.Backends = expandBackends(v.(*schema.Set).List()) } if v, ok := d.GetOk("description"); ok { service.Description = v.(string) } if v, ok := d.GetOk("port_name"); ok { service.PortName = v.(string) } if v, ok := d.GetOk("protocol"); ok { service.Protocol = v.(string) } if v, ok := d.GetOk("timeout_sec"); ok { service.TimeoutSec = int64(v.(int)) } if v, ok := d.GetOk("enable_cdn"); ok { service.EnableCDN = v.(bool) } if v, ok := d.GetOk("load_balancing_scheme"); ok { service.LoadBalancingScheme = v.(string) } project, err := getProject(d, config) if err != nil { return err } region, err := getRegion(d, config) if err != nil { return err } fmt.Fprintf(os.Stderr, "[DEBUG] Creating new Region Backend Service: %#v", service) // DO NOT SUBMIT log.Printf("[DEBUG] Creating new Region Backend Service: %#v", service) op, err := config.clientComputeBeta.RegionBackendServices.Insert( project, region, &service).Do() if err != nil { return fmt.Errorf("Error creating backend service: %s", err) } log.Printf("[DEBUG] Waiting for new backend service, operation: %#v", op) d.SetId(service.Name) err = computeOperationWaitGlobalBeta(config, op, project, "Creating Backend Service") if err != nil { return err } return resourceComputeRegionBackendServiceRead(d, meta) } func resourceComputeRegionBackendServiceRead(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) project, err := getProject(d, config) if err != nil { return err } region, err := getRegion(d, config) if err != nil { return err } service, err := config.clientComputeBeta.RegionBackendServices.Get( project, region, d.Id()).Do() if err != nil { if gerr, ok := err.(*googleapi.Error); ok && gerr.Code == 404 { // The resource doesn't exist anymore log.Printf("[WARN] Removing Backend Service %q because it's gone", d.Get("name").(string)) d.SetId("") return nil } return fmt.Errorf("Error reading service: %s", err) } d.Set("description", service.Description) d.Set("enable_cdn", service.EnableCDN) d.Set("port_name", service.PortName) d.Set("protocol", service.Protocol) d.Set("timeout_sec", service.TimeoutSec) d.Set("fingerprint", service.Fingerprint) d.Set("load_balancing_scheme", service.LoadBalancingScheme) d.Set("self_link", service.SelfLink) d.Set("backend", flattenBackends(service.Backends)) d.Set("health_checks", service.HealthChecks) return nil } func resourceComputeRegionBackendServiceUpdate(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) project, err := getProject(d, config) if err != nil { return err } region, err := getRegion(d, config) if err != nil { return err } hc := d.Get("health_checks").(*schema.Set).List() healthChecks := make([]string, 0, len(hc)) for _, v := range hc { healthChecks = append(healthChecks, v.(string)) } service := compute.BackendService{ Name: d.Get("name").(string), Fingerprint: d.Get("fingerprint").(string), HealthChecks: healthChecks, } // Optional things if v, ok := d.GetOk("backend"); ok { service.Backends = expandBackends(v.(*schema.Set).List()) } if v, ok := d.GetOk("description"); ok { service.Description = v.(string) } if v, ok := d.GetOk("port_name"); ok { service.PortName = v.(string) } if v, ok := d.GetOk("protocol"); ok { service.Protocol = v.(string) } if v, ok := d.GetOk("timeout_sec"); ok { service.TimeoutSec = int64(v.(int)) } if v, ok := d.GetOk("load_balancing_scheme"); ok { service.LoadBalancingScheme = v.(string) } if d.HasChange("enable_cdn") { service.EnableCDN = d.Get("enable_cdn").(bool) } log.Printf("[DEBUG] Updating existing Backend Service %q: %#v", d.Id(), service) op, err := config.clientComputeBeta.RegionBackendServices.Update( project, region, d.Id(), &service).Do() if err != nil { return fmt.Errorf("Error updating backend service: %s", err) } d.SetId(service.Name) err = computeOperationWaitGlobalBeta(config, op, project, "Updating Backend Service") if err != nil { return err } return resourceComputeRegionBackendServiceRead(d, meta) } func resourceComputeRegionBackendServiceDelete(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) project, err := getProject(d, config) if err != nil { return err } region, err := getRegion(d, config) if err != nil { return err } log.Printf("[DEBUG] Deleting backend service %s", d.Id()) op, err := config.clientComputeBeta.RegionBackendServices.Delete( project, region, d.Id()).Do() if err != nil { return fmt.Errorf("Error deleting backend service: %s", err) } err = computeOperationWaitGlobalBeta(config, op, project, "Deleting Backend Service") if err != nil { return err } d.SetId("") return nil } // func expandBackends(configured []interface{}) []*compute.Backend { // backends := make([]*compute.Backend, 0, len(configured)) // for _, raw := range configured { // data := raw.(map[string]interface{}) // b := compute.Backend{ // Group: data["group"].(string), // } // if v, ok := data["balancing_mode"]; ok { // b.BalancingMode = v.(string) // } // if v, ok := data["capacity_scaler"]; ok { // b.CapacityScaler = v.(float64) // } // if v, ok := data["description"]; ok { // b.Description = v.(string) // } // if v, ok := data["max_rate"]; ok { // b.MaxRate = int64(v.(int)) // } // if v, ok := data["max_rate_per_instance"]; ok { // b.MaxRatePerInstance = v.(float64) // } // if v, ok := data["max_utilization"]; ok { // b.MaxUtilization = v.(float64) // } // backends = append(backends, &b) // } // return backends // } // func flattenBackends(backends []*compute.Backend) []map[string]interface{} { // result := make([]map[string]interface{}, 0, len(backends)) // for _, b := range backends { // data := make(map[string]interface{}) // data["balancing_mode"] = b.BalancingMode // data["capacity_scaler"] = b.CapacityScaler // data["description"] = b.Description // data["group"] = b.Group // data["max_rate"] = b.MaxRate // data["max_rate_per_instance"] = b.MaxRatePerInstance // data["max_utilization"] = b.MaxUtilization // result = append(result, data) // } // return result // } func resourceGoogleComputeRegionBackendServiceBackendHash(v interface{}) int { if v == nil { return 0 } var buf bytes.Buffer m := v.(map[string]interface{}) buf.WriteString(fmt.Sprintf("%s-", m["group"].(string))) if v, ok := m["balancing_mode"]; ok { buf.WriteString(fmt.Sprintf("%s-", v.(string))) } if v, ok := m["capacity_scaler"]; ok { buf.WriteString(fmt.Sprintf("%f-", v.(float64))) } if v, ok := m["description"]; ok { buf.WriteString(fmt.Sprintf("%s-", v.(string))) } if v, ok := m["max_rate"]; ok { buf.WriteString(fmt.Sprintf("%d-", int64(v.(int)))) } if v, ok := m["max_rate_per_instance"]; ok { buf.WriteString(fmt.Sprintf("%f-", v.(float64))) } if v, ok := m["max_rate_per_instance"]; ok { buf.WriteString(fmt.Sprintf("%f-", v.(float64))) } return hashcode.String(buf.String()) }