# Cloud HA VPN Module This module makes it easy to deploy either GCP-to-GCP or GCP-to-On-prem [Cloud HA VPN](https://cloud.google.com/network-connectivity/docs/vpn/concepts/overview#ha-vpn). - [Examples](#examples) - [GCP to GCP](#gcp-to-gcp) - [GCP to on-prem](#gcp-to-on-prem) - [GCP to on-prem with custom ciphers](#gcp-to-on-prem-with-custom-ciphers) - [IPv6 (dual-stack)](#ipv6-dual-stack) - [Recipes](#recipes) - [Variables](#variables) - [Outputs](#outputs) ## Examples ### GCP to GCP ```hcl module "vpn-1" { source = "./fabric/modules/net-vpn-ha" project_id = var.project_id region = "europe-west4" network = var.vpc1.self_link name = "net1-to-net-2" peer_gateways = { default = { gcp = module.vpn-2.self_link } } router_config = { asn = 64514 custom_advertise = { all_subnets = true ip_ranges = { "10.0.0.0/8" = "default" } } } tunnels = { remote-0 = { bgp_peer = { address = "169.254.1.1" asn = 64513 } bgp_session_range = "169.254.1.2/30" vpn_gateway_interface = 0 } remote-1 = { bgp_peer = { address = "169.254.2.1" asn = 64513 } bgp_session_range = "169.254.2.2/30" vpn_gateway_interface = 1 } } } module "vpn-2" { source = "./fabric/modules/net-vpn-ha" project_id = var.project_id region = "europe-west4" network = var.vpc2.self_link name = "net2-to-net1" router_config = { asn = 64513 } peer_gateways = { default = { gcp = module.vpn-1.self_link } } tunnels = { remote-0 = { bgp_peer = { address = "169.254.1.2" asn = 64514 } bgp_session_range = "169.254.1.1/30" shared_secret = module.vpn-1.shared_secrets["remote-0"] vpn_gateway_interface = 0 } remote-1 = { bgp_peer = { address = "169.254.2.2" asn = 64514 } bgp_session_range = "169.254.2.1/30" shared_secret = module.vpn-1.shared_secrets["remote-1"] vpn_gateway_interface = 1 } } } # tftest modules=2 resources=22 inventory=gcp-to-gcp.yaml ``` Note: When using the `for_each` meta-argument you might experience a Cycle Error due to the multiple `net-vpn-ha` modules referencing each other. To fix this you can create the [google_compute_ha_vpn_gateway](https://registry.terraform.io/providers/hashicorp/google/latest/docs/resources/compute_ha_vpn_gateway) resources separately and reference them in the `net-vpn-ha` module via the `vpn_gateway` and `peer_gcp_gateway` variables. ### GCP to on-prem ```hcl module "vpn_ha" { source = "./fabric/modules/net-vpn-ha" project_id = var.project_id region = var.region network = var.vpc.self_link name = "mynet-to-onprem" peer_gateways = { default = { external = { redundancy_type = "SINGLE_IP_INTERNALLY_REDUNDANT" interfaces = ["8.8.8.8"] # on-prem router ip address } } } router_config = { asn = 64514 } tunnels = { remote-0 = { bgp_peer = { address = "169.254.1.1" asn = 64513 # Custom learned routes are optional custom_learned_ip_ranges = { ip_ranges = { "onprem-range" = "10.128.0.0/16" } } # MD5 Authentication is optional md5_authentication_key = { name = "foo" key = "bar" } } bgp_session_range = "169.254.1.2/30" peer_external_gateway_interface = 0 shared_secret = "mySecret" vpn_gateway_interface = 0 } remote-1 = { bgp_peer = { address = "169.254.2.1" asn = 64513 # Custom learned routes are optional custom_learned_ip_ranges = { ip_ranges = { "onprem-range" = "10.128.0.0/16" } } # MD5 Authentication is optional md5_authentication_key = { name = "foo" key = "bar" } } bgp_session_range = "169.254.2.2/30" peer_external_gateway_interface = 0 shared_secret = "mySecret" vpn_gateway_interface = 1 } } } # tftest modules=1 resources=12 inventory=gcp-to-onprem.yaml ``` ### GCP to on-prem with custom ciphers ```hcl module "vpn_ha" { source = "./fabric/modules/net-vpn-ha" project_id = var.project_id region = var.region network = var.vpc.self_link name = "mynet-to-onprem" peer_gateways = { default = { external = { redundancy_type = "SINGLE_IP_INTERNALLY_REDUNDANT" interfaces = ["8.8.8.8"] # on-prem router ip address } } } router_config = { asn = 64514 } tunnels = { remote-0 = { bgp_peer = { address = "169.254.1.1" asn = 64513 # Custom learned routes are optional custom_learned_ip_ranges = { ip_ranges = { "onprem-range" = "10.128.0.0/16" } } # MD5 Authentication is optional md5_authentication_key = { name = "foo" key = "bar" } } bgp_session_range = "169.254.1.2/30" cipher_suite = { phase1 = { dh = ["Group-14"] encryption = ["AES-CBC-256"] integrity = ["HMAC-SHA2-256-128"] prf = ["PRF-HMAC-SHA2-256"] } phase2 = { encryption = ["AES-CBC-128"] integrity = ["HMAC-SHA2-256-128"] pfs = ["Group-14"] } } peer_external_gateway_interface = 0 shared_secret = "mySecret" vpn_gateway_interface = 0 } remote-1 = { bgp_peer = { address = "169.254.2.1" asn = 64513 # Custom learned routes are optional custom_learned_ip_ranges = { ip_ranges = { "onprem-range" = "10.128.0.0/16" } } # MD5 Authentication is optional md5_authentication_key = { name = "foo" key = "bar" } } bgp_session_range = "169.254.2.2/30" cipher_suite = { phase1 = { dh = ["Group-14"] encryption = ["AES-CBC-256"] integrity = ["HMAC-SHA2-256-128"] prf = ["PRF-HMAC-SHA2-256"] } phase2 = { encryption = ["AES-CBC-128"] integrity = ["HMAC-SHA2-256-128"] pfs = ["Group-14"] } } peer_external_gateway_interface = 0 shared_secret = "mySecret" vpn_gateway_interface = 1 } } } # tftest modules=1 resources=12 inventory=gcp-to-onprem-custom-ciphers.yaml ``` ### IPv6 (dual-stack) You can optionally set your HA VPN gateway (and BGP sessions) to carry both IPv4 and IPv6 traffic. IPv6 only is not supported. ```hcl module "vpn_ha" { source = "./fabric/modules/net-vpn-ha" project_id = var.project_id region = var.region name = "mynet-to-onprem" network = var.vpc.self_link peer_gateways = { default = { external = { redundancy_type = "SINGLE_IP_INTERNALLY_REDUNDANT" interfaces = ["8.8.8.8"] # on-prem router ip address } } } router_config = { asn = 64514 } tunnels = { remote-0 = { bgp_peer = { address = "169.254.1.1" asn = 64513 ipv6 = {} } bgp_session_range = "169.254.1.2/30" peer_external_gateway_interface = 0 shared_secret = "mySecret" vpn_gateway_interface = 0 } remote-1 = { bgp_peer = { address = "169.254.2.1" asn = 64513 ipv6 = { nexthop_address = "2600:2d00:0:2::1" peer_nexthop_address = "2600:2d00:0:3::1" } } bgp_session_range = "169.254.2.2/30" peer_external_gateway_interface = 0 shared_secret = "mySecret" vpn_gateway_interface = 1 } } vpn_gateway_create = { ipv6 = true } } # tftest modules=1 resources=12 inventory=ipv6.yaml ``` You can optionally avoid to specify MD5 keys and the module will automatically generate them for you. ## Recipes - [HA VPN connections between Google Cloud and AWS](https://github.com/GoogleCloudPlatform/cloud-foundation-fabric/blob/master/modules/net-vpn-ha/recipe-vpn-aws-gcp) ## Variables | name | description | type | required | default | |---|---|:---:|:---:|:---:| | [name](variables.tf#L31) | VPN Gateway name (if an existing VPN Gateway is not used), and prefix used for dependent resources. | string | ✓ | | | [network](variables.tf#L36) | VPC used for the gateway and routes. | string | ✓ | | | [project_id](variables.tf#L62) | Project where resources will be created. | string | ✓ | | | [region](variables.tf#L67) | Region used for resources. | string | ✓ | | | [router_config](variables.tf#L72) | Cloud Router configuration for the VPN. If you want to reuse an existing router, set create to false and use name to specify the desired router. | object({…}) | ✓ | | | [context](variables.tf#L17) | Context-specific interpolations. | object({…}) | | {} | | [peer_gateways](variables.tf#L41) | Configuration of the (external or GCP) peer gateway. | map(object({…})) | | {} | | [tunnels](variables.tf#L88) | VPN tunnel configurations. | map(object({…})) | | {} | | [vpn_gateway](variables.tf#L142) | HA VPN Gateway Self Link for using an existing HA VPN Gateway. Ignored if `vpn_gateway_create` is set to `true`. | string | | null | | [vpn_gateway_create](variables.tf#L148) | Create HA VPN Gateway. Set to null to avoid creation. | object({…}) | | {} | ## Outputs | name | description | sensitive | |---|---|:---:| | [bgp_peers](outputs.tf#L18) | BGP peer resources. | | | [external_gateway](outputs.tf#L25) | External VPN gateway resource. | | | [gateway](outputs.tf#L30) | VPN gateway resource (only if auto-created). | | | [id](outputs.tf#L35) | Fully qualified VPN gateway id. | | | [md5_keys](outputs.tf#L42) | BGP tunnels MD5 keys. | ✓ | | [name](outputs.tf#L54) | VPN gateway name (only if auto-created). | | | [random_secret](outputs.tf#L59) | Generated secret. | ✓ | | [router](outputs.tf#L65) | Router resource (only if auto-created). | | | [router_name](outputs.tf#L70) | Router name. | | | [self_link](outputs.tf#L75) | HA VPN gateway self link. | | | [shared_secrets](outputs.tf#L80) | IPSEC tunnels shared secrets. | ✓ | | [tunnel_names](outputs.tf#L89) | VPN tunnel names. | | | [tunnel_self_links](outputs.tf#L97) | VPN tunnel self links. | | | [tunnels](outputs.tf#L105) | VPN tunnel resources. | |