1000 lines
42 KiB
Markdown
1000 lines
42 KiB
Markdown
# Google Compute Engine VM module
|
|
|
|
This module can operate in two distinct modes:
|
|
|
|
- instance creation, with optional unmanaged group
|
|
- instance template creation
|
|
|
|
In both modes, an optional service account can be created and assigned to either instances or template. If you need a managed instance group when using the module in template mode, refer to the [`compute-mig`](../compute-mig) module.
|
|
|
|
## Examples
|
|
|
|
<!-- BEGIN TOC -->
|
|
- [Examples](#examples)
|
|
- [Instance using defaults](#instance-using-defaults)
|
|
- [Service account management](#service-account-management)
|
|
- [Compute default service account](#compute-default-service-account)
|
|
- [Custom service account](#custom-service-account)
|
|
- [Custom service account, auto created](#custom-service-account-auto-created)
|
|
- [No service account](#no-service-account)
|
|
- [Disk management](#disk-management)
|
|
- [Disk sources](#disk-sources)
|
|
- [Disk types and options](#disk-types-and-options)
|
|
- [Boot disk as an independent resource](#boot-disk-as-an-independent-resource)
|
|
- [Network interfaces](#network-interfaces)
|
|
- [Internal and external IPs](#internal-and-external-ips)
|
|
- [Using Alias IPs](#using-alias-ips)
|
|
- [Using gVNIC](#using-gvnic)
|
|
- [PSC interfaces](#psc-interfaces)
|
|
- [Metadata](#metadata)
|
|
- [IAM](#iam)
|
|
- [Spot VM](#spot-vm)
|
|
- [Confidential compute](#confidential-compute)
|
|
- [Disk encryption with Cloud KMS](#disk-encryption-with-cloud-kms)
|
|
- [Advanced machine features](#advanced-machine-features)
|
|
- [Instance template](#instance-template)
|
|
- [Global template](#global-template)
|
|
- [Regional template](#regional-template)
|
|
- [Instance group](#instance-group)
|
|
- [Instance Schedule and Resource Policies](#instance-schedule-and-resource-policies)
|
|
- [Snapshot Schedules](#snapshot-schedules)
|
|
- [Resource Manager Tags](#resource-manager-tags)
|
|
- [Sole Tenancy](#sole-tenancy)
|
|
- [Variables](#variables)
|
|
- [Outputs](#outputs)
|
|
- [Fixtures](#fixtures)
|
|
<!-- END TOC -->
|
|
|
|
### Instance using defaults
|
|
|
|
The simplest example leverages defaults for the boot disk image and size, and uses a service account created by the module. Multiple instances can be managed via the `instance_count` variable.
|
|
|
|
```hcl
|
|
module "simple-vm-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
}
|
|
# tftest modules=1 resources=1 inventory=defaults.yaml e2e
|
|
```
|
|
|
|
### Service account management
|
|
|
|
VM service accounts can be managed in four different ways:
|
|
|
|
- in its default configuration, the module uses the Compute default service account with a basic set of scopes (`devstorage.read_only`, `logging.write`, `monitoring.write`)
|
|
- a custom service account can be used by passing its email in the `service_account.email` variable
|
|
- a custom service account can be created by the module and used by setting the `service_account.auto_create` variable to `true`
|
|
- the instance can be created with no service account by setting the `service_account` variable to `null`
|
|
|
|
Scopes for custom service accounts are set by default to `cloud-platform` and `userinfo.email`, and can be further customized regardless of which service account is used by directly setting the `service_account.scopes` variable.
|
|
|
|
#### Compute default service account
|
|
|
|
```hcl
|
|
module "vm-managed-sa-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test1"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
}
|
|
# tftest inventory=sa-default.yaml e2e
|
|
```
|
|
|
|
#### Custom service account
|
|
|
|
```hcl
|
|
module "vm-managed-sa-example2" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test2"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
service_account = {
|
|
email = module.iam-service-account.email
|
|
}
|
|
}
|
|
# tftest inventory=sa-custom.yaml fixtures=fixtures/iam-service-account.tf e2e
|
|
```
|
|
|
|
#### Custom service account, auto created
|
|
|
|
```hcl
|
|
module "vm-managed-sa-example2" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test2"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
service_account = {
|
|
auto_create = true
|
|
}
|
|
}
|
|
# tftest inventory=sa-managed.yaml e2e
|
|
```
|
|
|
|
#### No service account
|
|
|
|
```hcl
|
|
module "vm-managed-sa-example2" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test2"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
service_account = null
|
|
}
|
|
# tftest inventory=sa-none.yaml e2e
|
|
```
|
|
|
|
### Disk management
|
|
|
|
#### Disk sources
|
|
|
|
Attached disks can be created and optionally initialized from a pre-existing source, or attached to VMs when pre-existing. The `source` and `source_type` attributes of the `attached_disks` variable allows several modes of operation:
|
|
|
|
- `source_type = "image"` can be used with zonal disks in instances and templates, set `source` to the image name or self link
|
|
- `source_type = "snapshot"` can be used with instances only, set `source` to the snapshot name or self link
|
|
- `source_type = "attach"` can be used for both instances and templates to attach an existing disk, set source to the name (for zonal disks) or self link (for regional disks) of the existing disk to attach; no disk will be created
|
|
- `source_type = null` can be used where an empty disk is needed, `source` becomes irrelevant and can be left null
|
|
|
|
This is an example of attaching a pre-existing regional PD to a new instance:
|
|
|
|
```hcl
|
|
module "vm-disks-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
attached_disks = [{
|
|
name = "repd-1"
|
|
size = 10
|
|
source_type = "attach"
|
|
source = "regions/${var.region}/disks/repd-test-1"
|
|
options = {
|
|
replica_zone = "${var.region}-c"
|
|
}
|
|
}]
|
|
service_account = {
|
|
auto_create = true
|
|
}
|
|
}
|
|
# tftest modules=1 resources=2
|
|
```
|
|
|
|
And the same example for an instance template (where not using the full self link of the disk triggers recreation of the template)
|
|
|
|
```hcl
|
|
module "vm-disks-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
attached_disks = [{
|
|
name = "repd"
|
|
size = 10
|
|
source_type = "attach"
|
|
source = "https://www.googleapis.com/compute/v1/projects/${var.project_id}/regions/${var.region}/disks/repd-test-1"
|
|
options = {
|
|
replica_zone = "${var.region}-c"
|
|
}
|
|
}]
|
|
service_account = {
|
|
auto_create = true
|
|
}
|
|
create_template = {}
|
|
}
|
|
# tftest inventory=disks-example-template.yaml
|
|
```
|
|
|
|
#### Disk types and options
|
|
|
|
The `attached_disks` variable exposes an `option` attribute that can be used to fine tune the configuration of each disk. The following example shows a VM with multiple disks
|
|
|
|
```hcl
|
|
module "vm-disk-options-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
attached_disks = [
|
|
{
|
|
name = "data1"
|
|
size = "10"
|
|
source_type = "image"
|
|
source = "image-1"
|
|
options = {
|
|
auto_delete = false
|
|
replica_zone = "${var.region}-c"
|
|
}
|
|
},
|
|
{
|
|
name = "data2"
|
|
size = "20"
|
|
source_type = "snapshot"
|
|
source = "snapshot-2"
|
|
options = {
|
|
type = "pd-ssd"
|
|
mode = "READ_ONLY"
|
|
}
|
|
}
|
|
]
|
|
service_account = {
|
|
auto_create = true
|
|
}
|
|
}
|
|
# tftest inventory=disk-options.yaml
|
|
```
|
|
|
|
#### Boot disk as an independent resource
|
|
|
|
To create the boot disk as an independent resources instead of as part of the instance creation flow, set `boot_disk.use_independent_disk` to `true` and optionally configure `boot_disk.initialize_params`.
|
|
|
|
This will create the boot disk as its own resource and attach it to the instance, allowing to recreate the instance from Terraform while preserving the boot disk.
|
|
|
|
```hcl
|
|
module "simple-vm-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
boot_disk = {
|
|
initialize_params = {}
|
|
use_independent_disk = true
|
|
}
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
service_account = {
|
|
auto_create = true
|
|
}
|
|
}
|
|
# tftest inventory=independent-boot-disk.yaml e2e
|
|
```
|
|
|
|
### Network interfaces
|
|
|
|
#### Internal and external IPs
|
|
|
|
By default VNs are create with an automatically assigned IP addresses, but you can change it through the `addresses` and `nat` attributes of the `network_interfaces` variable:
|
|
|
|
```hcl
|
|
module "vm-internal-ip" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "vm-internal-ip"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
addresses = { internal = "10.0.0.2" }
|
|
}]
|
|
}
|
|
|
|
module "vm-external-ip" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "vm-external-ip"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
nat = true
|
|
addresses = { external = "8.8.8.8" }
|
|
}]
|
|
}
|
|
# tftest inventory=ips.yaml
|
|
```
|
|
|
|
#### Using Alias IPs
|
|
|
|
This example shows how to add additional [Alias IPs](https://cloud.google.com/vpc/docs/alias-ip) to your VM. `alias_ips` is a map of subnetwork additional range name into IP address.
|
|
|
|
```hcl
|
|
module "vm-with-alias-ips" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
alias_ips = {
|
|
services = "100.71.1.123/32"
|
|
}
|
|
}]
|
|
}
|
|
# tftest inventory=alias-ips.yaml e2e
|
|
```
|
|
|
|
#### Using gVNIC
|
|
|
|
This example shows how to enable [gVNIC](https://cloud.google.com/compute/docs/networking/using-gvnic) on your VM by customizing a `cos` image. Given that gVNIC needs to be enabled as an instance configuration and as a guest os configuration, you'll need to supply a bootable disk with `guest_os_features=GVNIC`. `SEV_CAPABLE`, `UEFI_COMPATIBLE` and `VIRTIO_SCSI_MULTIQUEUE` are enabled implicitly in the `cos`, `rhel`, `centos` and other images.
|
|
|
|
Note: most recent Google-provided images do enable `GVNIC` and no custom image is necessary.
|
|
|
|
```hcl
|
|
resource "google_compute_image" "cos-gvnic" {
|
|
project = var.project_id
|
|
name = "my-image"
|
|
source_image = "https://www.googleapis.com/compute/v1/projects/cos-cloud/global/images/cos-89-16108-534-18"
|
|
|
|
guest_os_features {
|
|
type = "GVNIC"
|
|
}
|
|
guest_os_features {
|
|
type = "SEV_CAPABLE"
|
|
}
|
|
guest_os_features {
|
|
type = "UEFI_COMPATIBLE"
|
|
}
|
|
guest_os_features {
|
|
type = "VIRTIO_SCSI_MULTIQUEUE"
|
|
}
|
|
}
|
|
|
|
module "vm-with-gvnic" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
boot_disk = {
|
|
initialize_params = {
|
|
image = google_compute_image.cos-gvnic.self_link
|
|
type = "pd-ssd"
|
|
}
|
|
}
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
nic_type = "GVNIC"
|
|
}]
|
|
service_account = {
|
|
auto_create = true
|
|
}
|
|
}
|
|
# tftest inventory=gvnic.yaml
|
|
```
|
|
|
|
#### PSC interfaces
|
|
|
|
[Private Service Connect interfaces](https://cloud.google.com/vpc/docs/about-private-service-connect-interfaces) can be configured via the `network_attached_interfaces` variable, which is a simple list of network attachment ids, one per interface. PSC interfaces will be defined after regular interfaces.
|
|
|
|
```hcl
|
|
|
|
# create the network attachment from a service project
|
|
module "net-attachment" {
|
|
source = "./fabric/modules/net-address"
|
|
project_id = var.project_id
|
|
network_attachments = {
|
|
svc-0 = {
|
|
subnet_self_link = module.vpc.subnet_self_links["${var.region}/ipv6-internal"]
|
|
producer_accept_lists = [var.project_id]
|
|
}
|
|
}
|
|
}
|
|
|
|
module "vm-psc-interface" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "vm-internal-ip"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
network_attached_interfaces = [
|
|
module.net-attachment.network_attachment_ids["svc-0"]
|
|
]
|
|
}
|
|
# tftest fixtures=fixtures/net-vpc-ipv6.tf e2e
|
|
```
|
|
|
|
### Metadata
|
|
|
|
You can define labels and custom metadata values. Metadata can be leveraged, for example, to define a custom startup script.
|
|
|
|
```hcl
|
|
module "vm-metadata-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "nginx-server"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
labels = {
|
|
env = "dev"
|
|
system = "crm"
|
|
}
|
|
metadata = {
|
|
startup-script = <<-EOF
|
|
#! /bin/bash
|
|
apt-get update
|
|
apt-get install -y nginx
|
|
EOF
|
|
}
|
|
service_account = {
|
|
auto_create = true
|
|
}
|
|
}
|
|
# tftest inventory=metadata.yaml e2e
|
|
```
|
|
|
|
### IAM
|
|
|
|
Like most modules, you can assign IAM roles to the instance using the `iam` variable.
|
|
|
|
```hcl
|
|
module "vm-iam-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "webserver"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
iam = {
|
|
"roles/compute.instanceAdmin" = [
|
|
"group:${var.group_email}",
|
|
]
|
|
}
|
|
}
|
|
# tftest inventory=iam.yaml e2e
|
|
|
|
```
|
|
|
|
### Spot VM
|
|
|
|
[Spot VMs](https://cloud.google.com/compute/docs/instances/spot) are ephemeral compute instances suitable for batch jobs and fault-tolerant workloads. Spot VMs provide new features that [preemptible instances](https://cloud.google.com/compute/docs/instances/preemptible) do not support, such as the absence of a maximum runtime.
|
|
|
|
```hcl
|
|
module "spot-vm-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
options = {
|
|
spot = true
|
|
termination_action = "STOP"
|
|
}
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
}
|
|
# tftest inventory=spot.yaml e2e
|
|
```
|
|
|
|
### Confidential compute
|
|
|
|
You can enable confidential compute with the `confidential_compute` variable, which can be used for standalone instances or for instance templates.
|
|
|
|
```hcl
|
|
module "vm-confidential-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "confidential-vm"
|
|
confidential_compute = true
|
|
instance_type = "n2d-standard-2"
|
|
boot_disk = {
|
|
initialize_params = {
|
|
image = "projects/debian-cloud/global/images/family/debian-12"
|
|
}
|
|
}
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
}
|
|
|
|
module "template-confidential-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "confidential-template"
|
|
confidential_compute = true
|
|
create_template = {}
|
|
instance_type = "n2d-standard-2"
|
|
boot_disk = {
|
|
initialize_params = {
|
|
image = "projects/debian-cloud/global/images/family/debian-12"
|
|
}
|
|
}
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
}
|
|
|
|
# tftest inventory=confidential.yaml e2e
|
|
```
|
|
|
|
### Disk encryption with Cloud KMS
|
|
|
|
This example shows how to control disk encryption via the the `encryption` variable, in this case the self link to a KMS CryptoKey that will be used to encrypt boot and attached disk. Managing the key with the `../kms` module is of course possible, but is not shown here.
|
|
|
|
```hcl
|
|
module "project" {
|
|
source = "./fabric/modules/project"
|
|
name = "gce"
|
|
billing_account = var.billing_account_id
|
|
prefix = var.prefix
|
|
parent = var.folder_id
|
|
services = [
|
|
"cloudkms.googleapis.com",
|
|
"compute.googleapis.com",
|
|
]
|
|
}
|
|
|
|
module "kms" {
|
|
source = "./fabric/modules/kms"
|
|
project_id = module.project.project_id
|
|
keyring = {
|
|
location = var.region
|
|
name = "${var.prefix}-keyring"
|
|
}
|
|
keys = {
|
|
"key-regional" = {
|
|
}
|
|
}
|
|
iam = {
|
|
"roles/cloudkms.cryptoKeyEncrypterDecrypter" = [
|
|
module.project.service_agents.compute.iam_email
|
|
]
|
|
}
|
|
}
|
|
|
|
module "vpc" {
|
|
source = "./fabric/modules/net-vpc"
|
|
project_id = module.project.project_id
|
|
name = "my-network"
|
|
subnets = [
|
|
{
|
|
ip_cidr_range = "10.0.0.0/24"
|
|
name = "production"
|
|
region = var.region
|
|
},
|
|
]
|
|
}
|
|
|
|
module "kms-vm-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = module.project.project_id
|
|
zone = "${var.region}-b"
|
|
name = "kms-test"
|
|
network_interfaces = [{
|
|
network = module.vpc.self_link
|
|
subnetwork = module.vpc.subnet_self_links["${var.region}/production"]
|
|
}]
|
|
attached_disks = [{
|
|
name = "attached-disk"
|
|
size = 10
|
|
}]
|
|
service_account = {
|
|
auto_create = true
|
|
}
|
|
encryption = {
|
|
encrypt_boot = true
|
|
kms_key_self_link = module.kms.keys.key-regional.id
|
|
}
|
|
}
|
|
# tftest inventory=cmek.yaml e2e
|
|
```
|
|
|
|
### Advanced machine features
|
|
|
|
Advanced machine features can be configured via the `options.advanced_machine_features` variable.
|
|
|
|
```hcl
|
|
module "simple-vm-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
options = {
|
|
advanced_machine_features = {
|
|
enable_nested_virtualization = true
|
|
enable_turbo_mode = true
|
|
threads_per_core = 2
|
|
}
|
|
}
|
|
}
|
|
# tftest modules=1 resources=1
|
|
```
|
|
|
|
### Instance template
|
|
|
|
#### Global template
|
|
|
|
This example shows how to use the module to manage an instance template that defines an additional attached disk for each instance, and overrides defaults for the boot disk image and service account. Instance templates are created global by default.
|
|
|
|
```hcl
|
|
module "cos-test" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
boot_disk = {
|
|
initialize_params = {
|
|
image = "projects/cos-cloud/global/images/family/cos-stable"
|
|
}
|
|
}
|
|
attached_disks = [
|
|
{ size = 10 }
|
|
]
|
|
service_account = {
|
|
email = module.iam-service-account.email
|
|
}
|
|
create_template = {}
|
|
}
|
|
# tftest inventory=template.yaml fixtures=fixtures/iam-service-account.tf e2e
|
|
```
|
|
|
|
#### Regional template
|
|
|
|
A regional template can be created by setting `var.create_template.regional`.
|
|
|
|
```hcl
|
|
module "cos-test" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
boot_disk = {
|
|
initialize_params = {
|
|
image = "projects/cos-cloud/global/images/family/cos-stable"
|
|
}
|
|
}
|
|
attached_disks = [
|
|
{ size = 10 }
|
|
]
|
|
service_account = {
|
|
email = module.iam-service-account.email
|
|
}
|
|
create_template = {
|
|
regional = true
|
|
}
|
|
}
|
|
# tftest inventory=template-regional.yaml fixtures=fixtures/iam-service-account.tf
|
|
```
|
|
|
|
### Instance group
|
|
|
|
If an instance group is needed when operating in instance mode, simply set the `group` variable to a non null map. The map can contain named port declarations, or be empty if named ports are not needed.
|
|
|
|
```hcl
|
|
locals {
|
|
cloud_config = "my cloud config"
|
|
}
|
|
|
|
module "instance-group" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "ilb-test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
boot_disk = {
|
|
initialize_params = {
|
|
image = "projects/cos-cloud/global/images/family/cos-stable"
|
|
}
|
|
}
|
|
service_account = {
|
|
email = var.service_account.email
|
|
scopes = ["https://www.googleapis.com/auth/cloud-platform"]
|
|
}
|
|
metadata = {
|
|
user-data = local.cloud_config
|
|
}
|
|
group = { named_ports = {} }
|
|
}
|
|
# tftest inventory=group.yaml e2e
|
|
```
|
|
|
|
### Instance Schedule and Resource Policies
|
|
|
|
One instance start and stop schedule can be defined via the `instance_schedule` variable. Note that this requires [additional permissions on Compute Engine Service Agent](https://cloud.google.com/compute/docs/instances/schedule-instance-start-stop#service_agent_required_roles). Already defined resource policies can be set via the `resource_policies` variable.
|
|
|
|
```hcl
|
|
module "instance" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "schedule-test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
boot_disk = {
|
|
initialize_params = {
|
|
image = "projects/cos-cloud/global/images/family/cos-stable"
|
|
}
|
|
}
|
|
resource_policies = [
|
|
"projects/${var.project_id}/regions/${var.region}/resourcePolicies/test"
|
|
]
|
|
}
|
|
# tftest inventory=instance-schedule-id.yaml
|
|
```
|
|
|
|
To create a new policy set its configuration in the `instance_schedule` variable. When removing the policy follow a two-step process by first setting `active = false` in the schedule configuration, which will unattach the policy, then removing the variable so the policy is destroyed.
|
|
|
|
```hcl
|
|
module "project" {
|
|
source = "./fabric/modules/project"
|
|
name = var.project_id
|
|
project_reuse = {
|
|
use_data_source = false
|
|
attributes = {
|
|
name = var.project_id
|
|
number = var.project_number
|
|
services_enabled = ["compute.googleapis.com"]
|
|
}
|
|
}
|
|
iam_bindings_additive = {
|
|
compute-admin-service-agent = {
|
|
member = module.project.service_agents["compute"].iam_email
|
|
role = "roles/compute.instanceAdmin.v1"
|
|
}
|
|
}
|
|
}
|
|
|
|
module "instance" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = module.project.project_id
|
|
zone = "${var.region}-b"
|
|
name = "schedule-test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
boot_disk = {
|
|
initialize_params = {
|
|
image = "projects/cos-cloud/global/images/family/cos-stable"
|
|
}
|
|
}
|
|
instance_schedule = {
|
|
vm_start = "0 8 * * *"
|
|
vm_stop = "0 17 * * *"
|
|
}
|
|
depends_on = [module.project] # ensure that grants are complete before creating schedule / instance
|
|
}
|
|
# tftest inventory=instance-schedule-create.yaml e2e
|
|
```
|
|
|
|
### Snapshot Schedules
|
|
|
|
Snapshot policies can be attached to disks with optional creation managed by the module.
|
|
|
|
```hcl
|
|
module "instance" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "schedule-test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
boot_disk = {
|
|
initialize_params = {
|
|
image = "projects/cos-cloud/global/images/family/cos-stable"
|
|
}
|
|
snapshot_schedule = ["boot"]
|
|
}
|
|
attached_disks = [
|
|
{
|
|
name = "disk-1"
|
|
size = 10
|
|
snapshot_schedule = ["boot"]
|
|
}
|
|
]
|
|
snapshot_schedules = {
|
|
boot = {
|
|
schedule = {
|
|
daily = {
|
|
days_in_cycle = 1
|
|
start_time = "03:00"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
# tftest inventory=snapshot-schedule-create.yaml e2e
|
|
```
|
|
|
|
### Resource Manager Tags
|
|
|
|
Resource manager tags bindings for use in IAM or org policy conditions are supported via three different variables:
|
|
|
|
- `network_tag_bindings` associates tags to instances after creation, and is meant for use with network firewall policies
|
|
- `tag_bindings` associates tags to instances and disks after creation, and is meant for use with IAM or organization policy conditions
|
|
- `tag_bindings_immutable` associates tags to instances and disks during the instance or template creation flow; these bindings are immutable and changes trigger resource recreation
|
|
|
|
The non-immutable variables follow our usual interface for tag bindings, and support specifying a map with arbitrary keys mapping to tag key or value ids. To prevent a provider permadiff also pass in the project number in the `project_number` variable.
|
|
|
|
The immutable variable uses a different format enforced by the Compute API, where keys need to be tag key ids, and values tag value ids.
|
|
|
|
This is an example of setting non-immutable tag bindings:
|
|
|
|
```hcl
|
|
module "simple-vm-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
project_number = 12345678
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
tag_bindings = {
|
|
dev = "tagValues/1234567890"
|
|
}
|
|
}
|
|
# tftest modules=1 resources=2
|
|
```
|
|
|
|
This example uses immutable tag bindings, and will trigger recreation if those are changed.
|
|
|
|
```hcl
|
|
module "simple-vm-example" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
tag_bindings_immutable = {
|
|
"tagKeys/1234567890" = "tagValues/7890123456"
|
|
}
|
|
}
|
|
# tftest inventory=tag-bindings.yaml
|
|
```
|
|
|
|
### Sole Tenancy
|
|
|
|
You can add node affinities (and anti-affinity) configurations to allocate the VM on sole tenant nodes.
|
|
|
|
```hcl
|
|
module "sole-tenancy" {
|
|
source = "./fabric/modules/compute-vm"
|
|
project_id = var.project_id
|
|
zone = "${var.region}-b"
|
|
instance_type = "n1-standard-1"
|
|
name = "test"
|
|
network_interfaces = [{
|
|
network = var.vpc.self_link
|
|
subnetwork = var.subnet.self_link
|
|
}]
|
|
options = {
|
|
node_affinities = {
|
|
workload = {
|
|
values = ["frontend"]
|
|
}
|
|
cpu = {
|
|
in = false
|
|
values = ["c3"]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
# tftest inventory=sole-tenancy.yaml
|
|
```
|
|
<!-- BEGIN TFDOC -->
|
|
## Variables
|
|
|
|
| name | description | type | required | default |
|
|
|---|---|:---:|:---:|:---:|
|
|
| [name](variables.tf#L283) | Instance name. | <code>string</code> | ✓ | |
|
|
| [network_interfaces](variables.tf#L295) | Network interfaces configuration. Use self links for Shared VPC, set addresses to null if not needed. | <code title="list(object({ network = string subnetwork = string alias_ips = optional(map(string), {}) nat = optional(bool, false) nic_type = optional(string) stack_type = optional(string) addresses = optional(object({ internal = optional(string) external = optional(string) }), null) network_tier = optional(string) }))">list(object({…}))</code> | ✓ | |
|
|
| [project_id](variables.tf#L380) | Project id. | <code>string</code> | ✓ | |
|
|
| [zone](variables.tf#L500) | Compute zone. | <code>string</code> | ✓ | |
|
|
| [attached_disk_defaults](variables.tf#L17) | Defaults for attached disks options. | <code title="object({ auto_delete = optional(bool, false) mode = string replica_zone = string type = string })">object({…})</code> | | <code title="{ auto_delete = true mode = "READ_WRITE" replica_zone = null type = "pd-balanced" }">{…}</code> |
|
|
| [attached_disks](variables.tf#L37) | Additional disks, if options is null defaults will be used in its place. Source type is one of 'image' (zonal disks in vms and template), 'snapshot' (vm), 'existing', and null. | <code title="list(object({ name = optional(string) device_name = optional(string) size = string snapshot_schedule = optional(list(string)) source = optional(string) source_type = optional(string) options = optional( object({ auto_delete = optional(bool, false) mode = optional(string, "READ_WRITE") replica_zone = optional(string) type = optional(string, "pd-balanced") }), { auto_delete = true mode = "READ_WRITE" replica_zone = null type = "pd-balanced" } ) }))">list(object({…}))</code> | | <code>[]</code> |
|
|
| [boot_disk](variables.tf#L82) | Boot disk properties. Initialize params are ignored when source is set. | <code title="object({ auto_delete = optional(bool, true) snapshot_schedule = optional(list(string)) source = optional(string) initialize_params = optional(object({ image = optional(string, "projects/debian-cloud/global/images/family/debian-11") size = optional(number, 10) type = optional(string, "pd-balanced") }), {}) use_independent_disk = optional(bool, false) })">object({…})</code> | | <code title="{ initialize_params = {} }">{…}</code> |
|
|
| [can_ip_forward](variables.tf#L113) | Enable IP forwarding. | <code>bool</code> | | <code>false</code> |
|
|
| [confidential_compute](variables.tf#L119) | Enable Confidential Compute for these instances. | <code>bool</code> | | <code>false</code> |
|
|
| [context](variables.tf#L125) | Context-specific interpolations. | <code title="object({ addresses = optional(map(string), {}) custom_roles = optional(map(string), {}) kms_keys = optional(map(string), {}) iam_principals = optional(map(string), {}) locations = optional(map(string), {}) networks = optional(map(string), {}) project_ids = optional(map(string), {}) subnets = optional(map(string), {}) tag_values = optional(map(string), {}) })">object({…})</code> | | <code>{}</code> |
|
|
| [create_template](variables.tf#L142) | Create instance template instead of instances. Defaults to a global template. | <code title="object({ regional = optional(bool, false) })">object({…})</code> | | <code>null</code> |
|
|
| [description](variables.tf#L151) | Description of a Compute Instance. | <code>string</code> | | <code>"Managed by the compute-vm Terraform module."</code> |
|
|
| [enable_display](variables.tf#L157) | Enable virtual display on the instances. | <code>bool</code> | | <code>false</code> |
|
|
| [encryption](variables.tf#L163) | Encryption options. Only one of kms_key_self_link and disk_encryption_key_raw may be set. If needed, you can specify to encrypt or not the boot disk. | <code title="object({ encrypt_boot = optional(bool, false) disk_encryption_key_raw = optional(string) kms_key_self_link = optional(string) })">object({…})</code> | | <code>null</code> |
|
|
| [gpu](variables.tf#L173) | GPU information. Based on https://cloud.google.com/compute/docs/gpus. | <code title="object({ count = number type = string })">object({…})</code> | | <code>null</code> |
|
|
| [group](variables.tf#L208) | Define this variable to create an instance group for instances. Disabled for template use. | <code title="object({ named_ports = map(number) })">object({…})</code> | | <code>null</code> |
|
|
| [hostname](variables.tf#L216) | Instance FQDN name. | <code>string</code> | | <code>null</code> |
|
|
| [iam](variables.tf#L222) | IAM bindings in {ROLE => [MEMBERS]} format. | <code>map(list(string))</code> | | <code>{}</code> |
|
|
| [instance_schedule](variables.tf#L228) | Assign or create and assign an instance schedule policy. Either resource policy id or create_config must be specified if not null. Set active to null to dtach a policy from vm before destroying. | <code title="object({ active = optional(bool, true) description = optional(string) expiration_time = optional(string) start_time = optional(string) timezone = optional(string, "UTC") vm_start = optional(string) vm_stop = optional(string) })">object({…})</code> | | <code>null</code> |
|
|
| [instance_type](variables.tf#L252) | Instance type. | <code>string</code> | | <code>"f1-micro"</code> |
|
|
| [labels](variables.tf#L258) | Instance labels. | <code>map(string)</code> | | <code>{}</code> |
|
|
| [metadata](variables.tf#L264) | Instance metadata. | <code>map(string)</code> | | <code>{}</code> |
|
|
| [metadata_startup_script](variables.tf#L270) | Instance startup script. Will trigger recreation on change, even after importing. | <code>string</code> | | <code>null</code> |
|
|
| [min_cpu_platform](variables.tf#L277) | Minimum CPU platform. | <code>string</code> | | <code>null</code> |
|
|
| [network_attached_interfaces](variables.tf#L288) | Network interfaces using network attachments. | <code>list(string)</code> | | <code>[]</code> |
|
|
| [network_tag_bindings](variables.tf#L316) | Resource manager tag bindings in arbitrary key => tag key or value id format. Set on both the instance only for networking purposes, and modifiable without impacting the main resource lifecycle. | <code>map(string)</code> | | <code>{}</code> |
|
|
| [options](variables.tf#L323) | Instance options. | <code title="object({ advanced_machine_features = optional(object({ enable_nested_virtualization = optional(bool) enable_turbo_mode = optional(bool) enable_uefi_networking = optional(bool) performance_monitoring_unit = optional(string) threads_per_core = optional(number) visible_core_count = optional(number) })) allow_stopping_for_update = optional(bool, true) deletion_protection = optional(bool, false) graceful_shutdown = optional(object({ enabled = optional(bool, false) max_duration_secs = optional(number) })) max_run_duration = optional(object({ nanos = optional(number) seconds = number })) node_affinities = optional(map(object({ values = list(string) in = optional(bool, true) })), {}) spot = optional(bool, false) termination_action = optional(string) })">object({…})</code> | | <code title="{ allow_stopping_for_update = true deletion_protection = false spot = false termination_action = null }">{…}</code> |
|
|
| [project_number](variables.tf#L385) | Project number. Used in tag bindings to avoid a permadiff. | <code>string</code> | | <code>null</code> |
|
|
| [resource_policies](variables.tf#L391) | Resource policies to attach to the instance or template. | <code>list(string)</code> | | <code>null</code> |
|
|
| [scratch_disks](variables.tf#L398) | Scratch disks configuration. | <code title="object({ count = number interface = string })">object({…})</code> | | <code title="{ count = 0 interface = "NVME" }">{…}</code> |
|
|
| [service_account](variables.tf#L410) | Service account email and scopes. If email is null, the default Compute service account will be used unless auto_create is true, in which case a service account will be created. Set the variable to null to avoid attaching a service account. | <code title="object({ auto_create = optional(bool, false) email = optional(string) scopes = optional(list(string)) })">object({…})</code> | | <code>{}</code> |
|
|
| [shielded_config](variables.tf#L420) | Shielded VM configuration of the instances. | <code title="object({ enable_secure_boot = bool enable_vtpm = bool enable_integrity_monitoring = bool })">object({…})</code> | | <code>null</code> |
|
|
| [snapshot_schedules](variables.tf#L430) | Snapshot schedule resource policies that can be attached to disks. | <code title="map(object({ schedule = object({ daily = optional(object({ days_in_cycle = number start_time = string })) hourly = optional(object({ hours_in_cycle = number start_time = string })) weekly = optional(list(object({ day = string start_time = string }))) }) description = optional(string) retention_policy = optional(object({ max_retention_days = number on_source_disk_delete_keep = optional(bool) })) snapshot_properties = optional(object({ chain_name = optional(string) guest_flush = optional(bool) labels = optional(map(string)) storage_locations = optional(list(string)) })) }))">map(object({…}))</code> | | <code>{}</code> |
|
|
| [tag_bindings](variables.tf#L473) | Resource manager tag bindings in arbitrary key => tag key or value id format. Set on both the instance and zonal disks, and modifiable without impacting the main resource lifecycle. | <code>map(string)</code> | | <code>{}</code> |
|
|
| [tag_bindings_immutable](variables.tf#L480) | Immutable resource manager tag bindings, in tagKeys/id => tagValues/id format. These are set on the instance or instance template at creation time, and trigger recreation if changed. | <code>map(string)</code> | | <code>null</code> |
|
|
| [tags](variables.tf#L494) | Instance network tags for firewall rule targets. | <code>list(string)</code> | | <code>[]</code> |
|
|
|
|
## Outputs
|
|
|
|
| name | description | sensitive |
|
|
|---|---|:---:|
|
|
| [external_ip](outputs.tf#L17) | Instance main interface external IP addresses. | |
|
|
| [group](outputs.tf#L26) | Instance group resource. | |
|
|
| [id](outputs.tf#L31) | Fully qualified instance id. | |
|
|
| [instance](outputs.tf#L36) | Instance resource. | ✓ |
|
|
| [internal_ip](outputs.tf#L42) | Instance main interface internal IP address. | |
|
|
| [internal_ips](outputs.tf#L50) | Instance interfaces internal IP addresses. | |
|
|
| [login_command](outputs.tf#L58) | Command to SSH into the machine. | |
|
|
| [self_link](outputs.tf#L63) | Instance self links. | |
|
|
| [service_account](outputs.tf#L68) | Service account resource. | |
|
|
| [service_account_email](outputs.tf#L73) | Service account email. | |
|
|
| [service_account_iam_email](outputs.tf#L78) | Service account email. | |
|
|
| [template](outputs.tf#L87) | Template resource. | |
|
|
| [template_name](outputs.tf#L96) | Template name. | |
|
|
|
|
## Fixtures
|
|
|
|
- [iam-service-account.tf](../../tests/fixtures/iam-service-account.tf)
|
|
- [net-vpc-ipv6.tf](../../tests/fixtures/net-vpc-ipv6.tf)
|
|
<!-- END TFDOC -->
|