github.com/meteor/terraform@v0.6.15-0.20210412225145-79ec4bc057c6/builtin/providers/google/provider.go (about) 1 package google 2 3 import ( 4 "encoding/json" 5 "fmt" 6 "log" 7 "strings" 8 9 "github.com/hashicorp/terraform/helper/schema" 10 "github.com/hashicorp/terraform/terraform" 11 "google.golang.org/api/compute/v1" 12 "google.golang.org/api/googleapi" 13 ) 14 15 // Provider returns a terraform.ResourceProvider. 16 func Provider() terraform.ResourceProvider { 17 return &schema.Provider{ 18 Schema: map[string]*schema.Schema{ 19 "credentials": &schema.Schema{ 20 Type: schema.TypeString, 21 Optional: true, 22 DefaultFunc: schema.MultiEnvDefaultFunc([]string{ 23 "GOOGLE_CREDENTIALS", 24 "GOOGLE_CLOUD_KEYFILE_JSON", 25 "GCLOUD_KEYFILE_JSON", 26 }, nil), 27 ValidateFunc: validateCredentials, 28 }, 29 30 "project": &schema.Schema{ 31 Type: schema.TypeString, 32 Optional: true, 33 DefaultFunc: schema.MultiEnvDefaultFunc([]string{ 34 "GOOGLE_PROJECT", 35 "GCLOUD_PROJECT", 36 "CLOUDSDK_CORE_PROJECT", 37 }, nil), 38 }, 39 40 "region": &schema.Schema{ 41 Type: schema.TypeString, 42 Required: true, 43 DefaultFunc: schema.MultiEnvDefaultFunc([]string{ 44 "GOOGLE_REGION", 45 "GCLOUD_REGION", 46 "CLOUDSDK_COMPUTE_REGION", 47 }, nil), 48 }, 49 }, 50 51 DataSourcesMap: map[string]*schema.Resource{ 52 "google_compute_network": dataSourceGoogleComputeNetwork(), 53 "google_compute_subnetwork": dataSourceGoogleComputeSubnetwork(), 54 "google_compute_zones": dataSourceGoogleComputeZones(), 55 "google_container_engine_versions": dataSourceGoogleContainerEngineVersions(), 56 "google_iam_policy": dataSourceGoogleIamPolicy(), 57 }, 58 59 ResourcesMap: map[string]*schema.Resource{ 60 "google_bigquery_dataset": resourceBigQueryDataset(), 61 "google_bigquery_table": resourceBigQueryTable(), 62 "google_compute_autoscaler": resourceComputeAutoscaler(), 63 "google_compute_address": resourceComputeAddress(), 64 "google_compute_backend_bucket": resourceComputeBackendBucket(), 65 "google_compute_backend_service": resourceComputeBackendService(), 66 "google_compute_disk": resourceComputeDisk(), 67 "google_compute_snapshot": resourceComputeSnapshot(), 68 "google_compute_firewall": resourceComputeFirewall(), 69 "google_compute_forwarding_rule": resourceComputeForwardingRule(), 70 "google_compute_global_address": resourceComputeGlobalAddress(), 71 "google_compute_global_forwarding_rule": resourceComputeGlobalForwardingRule(), 72 "google_compute_health_check": resourceComputeHealthCheck(), 73 "google_compute_http_health_check": resourceComputeHttpHealthCheck(), 74 "google_compute_https_health_check": resourceComputeHttpsHealthCheck(), 75 "google_compute_image": resourceComputeImage(), 76 "google_compute_instance": resourceComputeInstance(), 77 "google_compute_instance_group": resourceComputeInstanceGroup(), 78 "google_compute_instance_group_manager": resourceComputeInstanceGroupManager(), 79 "google_compute_instance_template": resourceComputeInstanceTemplate(), 80 "google_compute_network": resourceComputeNetwork(), 81 "google_compute_project_metadata": resourceComputeProjectMetadata(), 82 "google_compute_region_backend_service": resourceComputeRegionBackendService(), 83 "google_compute_route": resourceComputeRoute(), 84 "google_compute_ssl_certificate": resourceComputeSslCertificate(), 85 "google_compute_subnetwork": resourceComputeSubnetwork(), 86 "google_compute_target_http_proxy": resourceComputeTargetHttpProxy(), 87 "google_compute_target_https_proxy": resourceComputeTargetHttpsProxy(), 88 "google_compute_target_pool": resourceComputeTargetPool(), 89 "google_compute_url_map": resourceComputeUrlMap(), 90 "google_compute_vpn_gateway": resourceComputeVpnGateway(), 91 "google_compute_vpn_tunnel": resourceComputeVpnTunnel(), 92 "google_container_cluster": resourceContainerCluster(), 93 "google_container_node_pool": resourceContainerNodePool(), 94 "google_dns_managed_zone": resourceDnsManagedZone(), 95 "google_dns_record_set": resourceDnsRecordSet(), 96 "google_sql_database": resourceSqlDatabase(), 97 "google_sql_database_instance": resourceSqlDatabaseInstance(), 98 "google_sql_user": resourceSqlUser(), 99 "google_project": resourceGoogleProject(), 100 "google_project_iam_policy": resourceGoogleProjectIamPolicy(), 101 "google_project_services": resourceGoogleProjectServices(), 102 "google_pubsub_topic": resourcePubsubTopic(), 103 "google_pubsub_subscription": resourcePubsubSubscription(), 104 "google_service_account": resourceGoogleServiceAccount(), 105 "google_storage_bucket": resourceStorageBucket(), 106 "google_storage_bucket_acl": resourceStorageBucketAcl(), 107 "google_storage_bucket_object": resourceStorageBucketObject(), 108 "google_storage_object_acl": resourceStorageObjectAcl(), 109 }, 110 111 ConfigureFunc: providerConfigure, 112 } 113 } 114 115 func providerConfigure(d *schema.ResourceData) (interface{}, error) { 116 credentials := d.Get("credentials").(string) 117 config := Config{ 118 Credentials: credentials, 119 Project: d.Get("project").(string), 120 Region: d.Get("region").(string), 121 } 122 123 if err := config.loadAndValidate(); err != nil { 124 return nil, err 125 } 126 127 return &config, nil 128 } 129 130 func validateCredentials(v interface{}, k string) (warnings []string, errors []error) { 131 if v == nil || v.(string) == "" { 132 return 133 } 134 creds := v.(string) 135 var account accountFile 136 if err := json.Unmarshal([]byte(creds), &account); err != nil { 137 errors = append(errors, 138 fmt.Errorf("credentials are not valid JSON '%s': %s", creds, err)) 139 } 140 141 return 142 } 143 144 // getRegionFromZone returns the region from a zone for Google cloud. 145 func getRegionFromZone(zone string) string { 146 if zone != "" && len(zone) > 2 { 147 region := zone[:len(zone)-2] 148 return region 149 } 150 return "" 151 } 152 153 // getRegion reads the "region" field from the given resource data and falls 154 // back to the provider's value if not given. If the provider's value is not 155 // given, an error is returned. 156 func getRegion(d *schema.ResourceData, config *Config) (string, error) { 157 res, ok := d.GetOk("region") 158 if !ok { 159 if config.Region != "" { 160 return config.Region, nil 161 } 162 return "", fmt.Errorf("%q: required field is not set", "region") 163 } 164 return res.(string), nil 165 } 166 167 // getProject reads the "project" field from the given resource data and falls 168 // back to the provider's value if not given. If the provider's value is not 169 // given, an error is returned. 170 func getProject(d *schema.ResourceData, config *Config) (string, error) { 171 res, ok := d.GetOk("project") 172 if !ok { 173 if config.Project != "" { 174 return config.Project, nil 175 } 176 return "", fmt.Errorf("%q: required field is not set", "project") 177 } 178 return res.(string), nil 179 } 180 181 func getZonalResourceFromRegion(getResource func(string) (interface{}, error), region string, compute *compute.Service, project string) (interface{}, error) { 182 zoneList, err := compute.Zones.List(project).Do() 183 if err != nil { 184 return nil, err 185 } 186 var resource interface{} 187 for _, zone := range zoneList.Items { 188 if strings.Contains(zone.Name, region) { 189 resource, err = getResource(zone.Name) 190 if err != nil { 191 if gerr, ok := err.(*googleapi.Error); ok && gerr.Code == 404 { 192 // Resource was not found in this zone 193 continue 194 } 195 return nil, fmt.Errorf("Error reading Resource: %s", err) 196 } 197 // Resource was found 198 return resource, nil 199 } 200 } 201 // Resource does not exist in this region 202 return nil, nil 203 } 204 205 // getNetworkLink reads the "network" field from the given resource data and if the value: 206 // - is a resource URL, returns the string unchanged 207 // - is the network name only, then looks up the resource URL using the google client 208 func getNetworkLink(d *schema.ResourceData, config *Config, field string) (string, error) { 209 if v, ok := d.GetOk(field); ok { 210 network := v.(string) 211 212 project, err := getProject(d, config) 213 if err != nil { 214 return "", err 215 } 216 217 if !strings.HasPrefix(network, "https://www.googleapis.com/compute/") { 218 // Network value provided is just the name, lookup the network SelfLink 219 networkData, err := config.clientCompute.Networks.Get( 220 project, network).Do() 221 if err != nil { 222 return "", fmt.Errorf("Error reading network: %s", err) 223 } 224 network = networkData.SelfLink 225 } 226 227 return network, nil 228 229 } else { 230 return "", nil 231 } 232 } 233 234 // getNetworkName reads the "network" field from the given resource data and if the value: 235 // - is a resource URL, extracts the network name from the URL and returns it 236 // - is the network name only (i.e not prefixed with http://www.googleapis.com/compute/...), is returned unchanged 237 func getNetworkName(d *schema.ResourceData, field string) (string, error) { 238 if v, ok := d.GetOk(field); ok { 239 network := v.(string) 240 return getNetworkNameFromSelfLink(network) 241 } 242 return "", nil 243 } 244 245 func getNetworkNameFromSelfLink(network string) (string, error) { 246 if strings.HasPrefix(network, "https://www.googleapis.com/compute/") { 247 // extract the network name from SelfLink URL 248 networkName := network[strings.LastIndex(network, "/")+1:] 249 if networkName == "" { 250 return "", fmt.Errorf("network url not valid") 251 } 252 return networkName, nil 253 } 254 255 return network, nil 256 } 257 258 func handleNotFoundError(err error, d *schema.ResourceData, resource string) error { 259 if gerr, ok := err.(*googleapi.Error); ok && gerr.Code == 404 { 260 log.Printf("[WARN] Removing %s because it's gone", resource) 261 // The resource doesn't exist anymore 262 d.SetId("") 263 264 return nil 265 } 266 267 return fmt.Errorf("Error reading %s: %s", resource, err) 268 }