Terraform升级到0.12时出现错误”Reference to “count” in non-counted context”

44次阅读
没有评论

问题描述

在尝试将Terraform从0.11升级到0.12时,遇到了以下错误:

Error: Reference to "count" in non-counted context  on main.tf line 33, in data "aws_subnet" "public":  33:   id = data.aws_subnet_ids.public.ids[count.index]
The "count" object can be used only in "resource" and "data" blocks, and only when the "count" argument is set.

以下是用户的部分代码:

data "aws_subnet_ids" "public" {
  vpc_id = data.aws_vpc.vpc.id
  tags = {
    Tier = "public"
  }
}

data "aws_subnet" "public" {
  id = data.aws_subnet_ids.public.ids[count.index]
}

用户想知道如何修复这个错误。

解决方案

请注意以下操作注意版本差异及修改前做好备份。

方案1

根据你的代码,你的目标似乎是获取所有标记为Tier = public的子网的完整属性集。以下是使用Terraform 0.12.6的不同方法来实现这个目标:

data "aws_subnet_ids" "public" {
  vpc_id = data.aws_vpc.vpc.id
  tags = {
    Tier = "public"
  }
}

data "aws_subnet" "public" {
  for_each = data.aws_subnet_ids.public.ids
  id = each.value
}

通过上述代码,你会发现data.aws_subnet.public是一个从子网ID到该特定子网属性的映射。
在你的代码中出现错误的原因是你使用了count.index,但没有设置count参数,因此没有索引可返回。
以下是基于count的等效代码,与你的原始示例更相似:

data "aws_subnet_ids" "public" {
  vpc_id = data.aws_vpc.vpc.id
  tags = {
    Tier = "public"
  }
}

data "aws_subnet" "public" {
  count = length(data.aws_subnet_ids.public.ids)
  id = sort(data.aws_subnet_ids.public.ids)[count.index]
}

这样做的结果与我上面给出的基于for_each的示例非常相似,但有一个重要的区别:在这种情况下,data.aws_subnet.public将是一个按子网ID的词法顺序排序的子网对象列表。这通常会使结果在配置中的其他位置更难使用,因此我建议使用for_each方法,除非你有特定的原因需要一个列表。
(如果你使用了一个映射,然后后来发现在特定上下文中需要一个列表,你可以始终使用values(data.aws_subnet.public)从映射中获取值,按照词法顺序排列。)

方案2

使用脚本或工具来管理容器的启动顺序可能会增加复杂性,并且需要确保容器A和容器B之间的依赖关系正确设置。
另一种方法是编写脚本或使用工具来控制容器的运行顺序。你可以使用docker run命令来手动控制容器的启动顺序,或者使用一些第三方工具来管理容器的依赖关系。

示例:

以下是一个简单的bash脚本示例,可以在容器A启动后启动容器B:

#!/bin/bash
# 启动容器A
docker run -d --name container_a your_image_a
# 等待容器A完全启动
while ! docker exec container_a echo "Container A is ready"; do
  sleep 1
done
# 启动容器B
docker run -d --name container_b your_image_b

在这个示例中,我们首先使用docker run命令启动容器A,并将其命名为container_a。然后,使用一个循环来等待容器A完全启动(这里是通过在容器内运行echo命令来测试)。一旦容器A就绪,我们再使用docker run命令启动容器B,并将其命名为container_b

正文完