package google import ( "fmt" "log" "strings" "github.com/hashicorp/terraform/helper/schema" "google.golang.org/api/compute/v1" ) func resourceComputeSubnetwork() *schema.Resource { return &schema.Resource{ Create: resourceComputeSubnetworkCreate, Read: resourceComputeSubnetworkRead, Update: resourceComputeSubnetworkUpdate, Delete: resourceComputeSubnetworkDelete, Importer: &schema.ResourceImporter{ State: resourceComputeSubnetworkImportState, }, Schema: map[string]*schema.Schema{ "ip_cidr_range": &schema.Schema{ Type: schema.TypeString, Required: true, ForceNew: true, }, "name": &schema.Schema{ Type: schema.TypeString, Required: true, ForceNew: true, }, "network": &schema.Schema{ Type: schema.TypeString, Required: true, ForceNew: true, DiffSuppressFunc: compareSelfLinkOrResourceName, }, "description": &schema.Schema{ Type: schema.TypeString, Optional: true, ForceNew: true, }, "gateway_address": &schema.Schema{ Type: schema.TypeString, Computed: true, }, "project": &schema.Schema{ Type: schema.TypeString, Optional: true, ForceNew: true, }, "region": &schema.Schema{ Type: schema.TypeString, Optional: true, ForceNew: true, }, "private_ip_google_access": &schema.Schema{ Type: schema.TypeBool, Optional: true, }, "secondary_ip_range": &schema.Schema{ Type: schema.TypeList, Optional: true, ForceNew: true, Elem: &schema.Resource{ Schema: map[string]*schema.Schema{ "range_name": &schema.Schema{ Type: schema.TypeString, Required: true, ForceNew: true, ValidateFunc: validateGCPName, }, "ip_cidr_range": &schema.Schema{ Type: schema.TypeString, Required: true, ForceNew: true, }, }, }, }, "self_link": &schema.Schema{ Type: schema.TypeString, Computed: true, }, }, } } func resourceComputeSubnetworkCreate(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) region, err := getRegion(d, config) if err != nil { return err } project, err := getProject(d, config) if err != nil { return err } // Build the subnetwork parameters subnetwork := &compute.Subnetwork{ Name: d.Get("name").(string), Description: d.Get("description").(string), IpCidrRange: d.Get("ip_cidr_range").(string), PrivateIpGoogleAccess: d.Get("private_ip_google_access").(bool), SecondaryIpRanges: expandSecondaryRanges(d.Get("secondary_ip_range").([]interface{})), Network: ParseNetworkFieldValue(d.Get("network").(string), config).RelativeLink(), } log.Printf("[DEBUG] Subnetwork insert request: %#v", subnetwork) op, err := config.clientCompute.Subnetworks.Insert(project, region, subnetwork).Do() if err != nil { return fmt.Errorf("Error creating subnetwork: %s", err) } // It probably maybe worked, so store the ID now. ID is a combination of region + subnetwork // name because subnetwork names are not unique in a project, per the Google docs: // "When creating a new subnetwork, its name has to be unique in that project for that region, even across networks. // The same name can appear twice in a project, as long as each one is in a different region." // https://cloud.google.com/compute/docs/subnetworks subnetwork.Region = region d.SetId(createSubnetID(subnetwork)) err = computeSharedOperationWait(config, op, project, "Creating Subnetwork") if err != nil { return err } return resourceComputeSubnetworkRead(d, meta) } func resourceComputeSubnetworkRead(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) region, err := getRegion(d, config) if err != nil { return err } project, err := getProject(d, config) if err != nil { return err } name := d.Get("name").(string) subnetwork, err := config.clientCompute.Subnetworks.Get(project, region, name).Do() if err != nil { return handleNotFoundError(err, d, fmt.Sprintf("Subnetwork %q", name)) } d.Set("name", subnetwork.Name) d.Set("ip_cidr_range", subnetwork.IpCidrRange) d.Set("network", subnetwork.Network) d.Set("description", subnetwork.Description) d.Set("private_ip_google_access", subnetwork.PrivateIpGoogleAccess) d.Set("gateway_address", subnetwork.GatewayAddress) d.Set("secondary_ip_range", flattenSecondaryRanges(subnetwork.SecondaryIpRanges)) d.Set("self_link", ConvertSelfLinkToV1(subnetwork.SelfLink)) return nil } func resourceComputeSubnetworkUpdate(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) region, err := getRegion(d, config) if err != nil { return err } project, err := getProject(d, config) if err != nil { return err } d.Partial(true) if d.HasChange("private_ip_google_access") { subnetworksSetPrivateIpGoogleAccessRequest := &compute.SubnetworksSetPrivateIpGoogleAccessRequest{ PrivateIpGoogleAccess: d.Get("private_ip_google_access").(bool), } log.Printf("[DEBUG] Updating Subnetwork PrivateIpGoogleAccess %q: %#v", d.Id(), subnetworksSetPrivateIpGoogleAccessRequest) op, err := config.clientCompute.Subnetworks.SetPrivateIpGoogleAccess( project, region, d.Get("name").(string), subnetworksSetPrivateIpGoogleAccessRequest).Do() if err != nil { return fmt.Errorf("Error updating subnetwork PrivateIpGoogleAccess: %s", err) } err = computeSharedOperationWait(config, op, project, "Updating Subnetwork PrivateIpGoogleAccess") if err != nil { return err } d.SetPartial("private_ip_google_access") } d.Partial(false) return resourceComputeSubnetworkRead(d, meta) } func resourceComputeSubnetworkDelete(d *schema.ResourceData, meta interface{}) error { config := meta.(*Config) region, err := getRegion(d, config) if err != nil { return err } project, err := getProject(d, config) if err != nil { return err } // Delete the subnetwork op, err := config.clientCompute.Subnetworks.Delete( project, region, d.Get("name").(string)).Do() if err != nil { return fmt.Errorf("Error deleting subnetwork: %s", err) } err = computeSharedOperationWait(config, op, project, "Deleting Subnetwork") if err != nil { return err } d.SetId("") return nil } func resourceComputeSubnetworkImportState(d *schema.ResourceData, meta interface{}) ([]*schema.ResourceData, error) { parts := strings.Split(d.Id(), "/") if len(parts) != 2 { return nil, fmt.Errorf("Invalid compute subnetwork specifier. Expecting {region}/{name}") } region, name := parts[0], parts[1] d.Set("region", region) d.Set("name", name) d.SetId(createSubnetID(&compute.Subnetwork{ Region: region, Name: name, })) return []*schema.ResourceData{d}, nil } func createSubnetID(s *compute.Subnetwork) string { return fmt.Sprintf("%s/%s", s.Region, s.Name) } func splitSubnetID(id string) (region string, name string) { parts := strings.Split(id, "/") region = parts[0] name = parts[1] return } func expandSecondaryRanges(configured []interface{}) []*compute.SubnetworkSecondaryRange { secondaryRanges := make([]*compute.SubnetworkSecondaryRange, 0, len(configured)) for _, raw := range configured { data := raw.(map[string]interface{}) secondaryRange := compute.SubnetworkSecondaryRange{ RangeName: data["range_name"].(string), IpCidrRange: data["ip_cidr_range"].(string), } secondaryRanges = append(secondaryRanges, &secondaryRange) } return secondaryRanges } func flattenSecondaryRanges(secondaryRanges []*compute.SubnetworkSecondaryRange) []map[string]interface{} { secondaryRangesSchema := make([]map[string]interface{}, 0, len(secondaryRanges)) for _, secondaryRange := range secondaryRanges { data := map[string]interface{}{ "range_name": secondaryRange.RangeName, "ip_cidr_range": secondaryRange.IpCidrRange, } secondaryRangesSchema = append(secondaryRangesSchema, data) } return secondaryRangesSchema }