Terraform - how to use for_each loop on a list of objects to create resources

后端 未结 1 1932
刺人心
刺人心 2021-01-12 00:11

I have an object containing the list of subnets I want to create.

variable \"subnet-map\" {
  default = {
    ec2 = [
      {
        cidr_block        = \"1         


        
1条回答
  •  孤城傲影
    2021-01-12 00:34

    I'm not sure I fully follow all of what you tried here because your initial snippet of var.subnet-map shows it being a map of maps of lists of objects, but later on when you used for_each = var.subnet-map it seems to have treated it as a map of lists instead. Did you remove that extra level of maps (the "default" key) before trying for_each here?

    Working with your original definition of variable "subnet-map", your first step with for_each will be similar to what you did with count: you need to flatten the structure, this time into a map of objects rather than a list of objects. The easiest way to get there is to derive a map from your existing flattened list:

    locals {
      subnets = flatten([
        for resource in keys(var.subnet-map) : [
          for subnet in var.subnet-map[resource] : {
            resource          = resource
            cidr_block        = subnet.cidr_block
            availability_zone = subnet.availability_zone
          }
        ]
      ])
    
      subnets_map = {
        for s in local.subnets: "${s.resource}:${s.availability_zone}" => s
      }
    }
    

    Here I assumed that your "resource" string and your availability zone together are a suitable unique identifier for a subnet. If not, you can adjust the "${s.resource}:${s.availability_zone}" part to whatever unique key you want to use for these.

    Now you can use the flattened map as the for_each map:

    resource "aws_subnet" "subnets-dev" {
      for_each          = local.subnets_map
      vpc_id            = aws_vpc.vpc-dev.id
      cidr_block        = each.value.cidr_block
      availability_zone = each.value.availability_zone
    
      tags = {
        Name        = "subnet-dev-${each.value.resource}-${each.value.availability_zone}"
        environment = "dev"
      }
    }
    

    This will give you instances with addresses like aws_subnet.subnets-dev["ec2:eu-west-1a"].


    Note that if you are migrating from count with existing subnets that you wish to retain, you'll need to also do a one-time migration step to tell Terraform which indexes from the existing state correspond to which keys in the new configuration.

    For example, if (and only if) index 0 was previously the one for ec2 in eu-west-1a, the migration command for that one would be:

    terraform state mv 'aws_subnet.subnets-dev[0]' 'aws_subnet.subnets-dev["ec2:eu-west-1a"]'
    

    If you're not sure how they correlate, you can run terraform plan after adding for_each and look at the instances that Terraform is planning to destroy. If you work through each one of those in turn, taking the address Terraform currently knows along with the resource and availability zone names shown in the Name tag, you can migrate each of them to its new address so that Terraform will no longer think you're asking for it to destroy the numbered instances and replace them with named ones.

    0 讨论(0)
提交回复
热议问题