2017-07-28 16:51:29 +00:00
|
|
|
package google
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"sort"
|
|
|
|
|
|
|
|
"github.com/hashicorp/terraform/helper/schema"
|
|
|
|
"google.golang.org/api/compute/v1"
|
|
|
|
"google.golang.org/api/googleapi"
|
|
|
|
)
|
|
|
|
|
|
|
|
const peerNetworkLinkRegex = "projects/(" + ProjectRegex + ")/global/networks/((?:[a-z](?:[-a-z0-9]*[a-z0-9])?))$"
|
|
|
|
|
|
|
|
func resourceComputeNetworkPeering() *schema.Resource {
|
|
|
|
return &schema.Resource{
|
|
|
|
Create: resourceComputeNetworkPeeringCreate,
|
|
|
|
Read: resourceComputeNetworkPeeringRead,
|
|
|
|
Delete: resourceComputeNetworkPeeringDelete,
|
|
|
|
|
|
|
|
Schema: map[string]*schema.Schema{
|
|
|
|
"name": &schema.Schema{
|
|
|
|
Type: schema.TypeString,
|
|
|
|
Required: true,
|
|
|
|
ForceNew: true,
|
|
|
|
ValidateFunc: validateGCPName,
|
|
|
|
},
|
|
|
|
"network": &schema.Schema{
|
|
|
|
Type: schema.TypeString,
|
|
|
|
Required: true,
|
|
|
|
ForceNew: true,
|
|
|
|
ValidateFunc: validateRegexp(peerNetworkLinkRegex),
|
|
|
|
DiffSuppressFunc: compareSelfLinkRelativePaths,
|
|
|
|
},
|
|
|
|
"peer_network": &schema.Schema{
|
|
|
|
Type: schema.TypeString,
|
|
|
|
Required: true,
|
|
|
|
ForceNew: true,
|
|
|
|
ValidateFunc: validateRegexp(peerNetworkLinkRegex),
|
|
|
|
DiffSuppressFunc: compareSelfLinkRelativePaths,
|
|
|
|
},
|
|
|
|
"auto_create_routes": &schema.Schema{
|
|
|
|
Type: schema.TypeBool,
|
|
|
|
ForceNew: true,
|
|
|
|
Optional: true,
|
|
|
|
Default: true,
|
|
|
|
},
|
|
|
|
"state": &schema.Schema{
|
|
|
|
Type: schema.TypeString,
|
|
|
|
Computed: true,
|
|
|
|
},
|
|
|
|
"state_details": &schema.Schema{
|
|
|
|
Type: schema.TypeString,
|
|
|
|
Computed: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func resourceComputeNetworkPeeringCreate(d *schema.ResourceData, meta interface{}) error {
|
|
|
|
config := meta.(*Config)
|
2017-10-10 16:53:57 +00:00
|
|
|
networkFieldValue, err := ParseNetworkFieldValue(d.Get("network").(string), d, config)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-07-28 16:51:29 +00:00
|
|
|
|
|
|
|
request := &compute.NetworksAddPeeringRequest{
|
|
|
|
Name: d.Get("name").(string),
|
|
|
|
PeerNetwork: d.Get("peer_network").(string),
|
|
|
|
AutoCreateRoutes: d.Get("auto_create_routes").(bool),
|
|
|
|
}
|
|
|
|
|
2017-10-03 20:30:31 +00:00
|
|
|
addOp, err := config.clientCompute.Networks.AddPeering(networkFieldValue.Project, networkFieldValue.Name, request).Do()
|
2017-07-28 16:51:29 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Error adding network peering: %s", err)
|
|
|
|
}
|
|
|
|
|
2017-10-13 22:36:03 +00:00
|
|
|
err = computeOperationWait(config.clientCompute, addOp, networkFieldValue.Project, "Adding Network Peering")
|
2017-07-28 16:51:29 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2017-10-03 20:30:31 +00:00
|
|
|
d.SetId(fmt.Sprintf("%s/%s", networkFieldValue.Name, d.Get("name").(string)))
|
2017-07-28 16:51:29 +00:00
|
|
|
|
|
|
|
return resourceComputeNetworkPeeringRead(d, meta)
|
|
|
|
}
|
|
|
|
|
|
|
|
func resourceComputeNetworkPeeringRead(d *schema.ResourceData, meta interface{}) error {
|
|
|
|
config := meta.(*Config)
|
|
|
|
|
|
|
|
peeringName := d.Get("name").(string)
|
2017-10-10 16:53:57 +00:00
|
|
|
networkFieldValue, err := ParseNetworkFieldValue(d.Get("network").(string), d, config)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-07-28 16:51:29 +00:00
|
|
|
|
2017-10-03 20:30:31 +00:00
|
|
|
network, err := config.clientCompute.Networks.Get(networkFieldValue.Project, networkFieldValue.Name).Do()
|
2017-07-28 16:51:29 +00:00
|
|
|
if err != nil {
|
2018-01-04 21:17:03 +00:00
|
|
|
return handleNotFoundError(err, d, fmt.Sprintf("Network %q", networkFieldValue.Name))
|
2017-07-28 16:51:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
peering := findPeeringFromNetwork(network, peeringName)
|
|
|
|
if peering == nil {
|
2017-10-03 20:30:31 +00:00
|
|
|
log.Printf("[WARN] Removing network peering %s from network %s because it's gone", peeringName, network.Name)
|
2017-07-28 16:51:29 +00:00
|
|
|
d.SetId("")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
d.Set("peer_network", peering.Network)
|
|
|
|
d.Set("auto_create_routes", peering.AutoCreateRoutes)
|
|
|
|
d.Set("state", peering.State)
|
|
|
|
d.Set("state_details", peering.StateDetails)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func resourceComputeNetworkPeeringDelete(d *schema.ResourceData, meta interface{}) error {
|
|
|
|
config := meta.(*Config)
|
|
|
|
|
|
|
|
// Remove the `network` to `peer_network` peering
|
|
|
|
name := d.Get("name").(string)
|
2017-10-10 16:53:57 +00:00
|
|
|
networkFieldValue, err := ParseNetworkFieldValue(d.Get("network").(string), d, config)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
peerNetworkFieldValue, err := ParseNetworkFieldValue(d.Get("peer_network").(string), d, config)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-07-28 16:51:29 +00:00
|
|
|
|
|
|
|
request := &compute.NetworksRemovePeeringRequest{
|
|
|
|
Name: name,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only one delete peering operation at a time can be performed inside any peered VPCs.
|
2017-10-03 20:30:31 +00:00
|
|
|
peeringLockName := getNetworkPeeringLockName(networkFieldValue.Name, peerNetworkFieldValue.Name)
|
2017-07-28 16:51:29 +00:00
|
|
|
mutexKV.Lock(peeringLockName)
|
|
|
|
defer mutexKV.Unlock(peeringLockName)
|
|
|
|
|
2017-10-03 20:30:31 +00:00
|
|
|
removeOp, err := config.clientCompute.Networks.RemovePeering(networkFieldValue.Project, networkFieldValue.Name, request).Do()
|
2017-07-28 16:51:29 +00:00
|
|
|
if err != nil {
|
|
|
|
if gerr, ok := err.(*googleapi.Error); ok && gerr.Code == 404 {
|
2017-10-03 20:30:31 +00:00
|
|
|
log.Printf("[WARN] Peering `%s` already removed from network `%s`", name, networkFieldValue.Name)
|
2017-07-28 16:51:29 +00:00
|
|
|
} else {
|
2017-10-03 20:30:31 +00:00
|
|
|
return fmt.Errorf("Error removing peering `%s` from network `%s`: %s", name, networkFieldValue.Name, err)
|
2017-07-28 16:51:29 +00:00
|
|
|
}
|
|
|
|
} else {
|
2017-10-13 22:36:03 +00:00
|
|
|
err = computeOperationWait(config.clientCompute, removeOp, networkFieldValue.Project, "Removing Network Peering")
|
2017-07-28 16:51:29 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func findPeeringFromNetwork(network *compute.Network, peeringName string) *compute.NetworkPeering {
|
|
|
|
for _, p := range network.Peerings {
|
|
|
|
if p.Name == peeringName {
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getNetworkPeeringLockName(networkName, peerNetworkName string) string {
|
|
|
|
// Whether you delete the peering from network A to B or the one from B to A, they
|
|
|
|
// cannot happen at the same time.
|
|
|
|
networks := []string{networkName, peerNetworkName}
|
|
|
|
sort.Strings(networks)
|
|
|
|
|
|
|
|
return fmt.Sprintf("network_peering/%s/%s", networks[0], networks[1])
|
|
|
|
}
|