我有一个包含很多行的第三方维护文件,它看起来像:
# a comment lines
aaa.bbb.cc 423
ddd.ee.fff 452
...
tons.like.them 111
有没有办法将文件加载到本地地图并使用以下语法在 Terraform 中访问其键?:
"${lookup(var.locals.mylist, 'ddd.ee.fff', 0)}"
使用用通用编程语言编写的单独程序来预处理这样的文件通常会更好,因为 Terraform 的 DSL 不适合这种临时解析逻辑。
Terraform v0.11 及更早版本缺乏在语言本身内实现此类解析所需的原语。v0.12.0 版本(在撰写本文时处于测试阶段)引入了一些允许实现这种解析的原语,但如果可能的话,我通常仍然更愿意在 Terraform 之外将其作为预处理步骤进行.
话虽如此,这是一个使用 Terraform v0.12 功能将像您的示例这样的文件解析为地图的示例:
locals {
raw_lines = [
for line in split("\n", file("${path.module}/input.txt")) :
split(" ", trimspace(line))
]
lines = [
for line in local.raw_lines :
line if length(line[0]) > 0 && substr(line[0], 0, 1) != "#"
]
records = { for line in local.lines : line[0] => line[1] }
}
output "result" {
value = local.records
}
给定以下输入文件:
# a comment line
aaa.bbb.cc 423
ddd.ee.fff 452
tons.like.them 111
# another comment
another.record abc
这使用 Terraform v0.12.0-beta1 产生以下结果:
$ terraform apply
Apply complete! Resources: 0 added, 0 changed, 0 destroyed.
Outputs:
result = {
"aaa.bbb.cc" = "423"
"another.record" = "abc"
"ddd.ee.fff" = "452"
"tons.like.them" = "111"
}
文件名是myfile.txt
,内容是:
# a comment line
aaa.bbb.cc 423
ddd.ee.fff 452
...
tons.like.them 111
数据查询应如下所示:
data "external" "myfile" {
program = [
"bash",
"${path.module}/template/parser.sh",
"${path.module}/template/myfile.txt",
"ddd.ee.fff"
]
}
并且template/parser.sh
包含:
#!/usr/bin/env bash
set -eo pipefail
CONFIG=$(cat $1|grep -v '#' | sed 's/ /=/g' | jq -R -n -c '.data = ([inputs|split("=")|{(.[0]):.[1]}] |add)')
OUTPUT=$(echo $CONFIG | jq -r '.data["'$2'"]')
jq -n --arg output "$OUTPUT" '{"output":$output}'
这个棘手的脚本将返回:
{"output": 452}
并且可以使用插值来获得结果值:
${data.external.myfile.result.output}
不幸的是,我没有找到如何将完整的行列表加载到 Terraform 并使用“变量的变量”或 lookup() 访问它们。如果 bash 脚本返回第一次jq
调用的结果,Terraform 会抛出异常:
command "bash" produced invalid JSON: json: cannot unmarshal object into Go value of type string
Update2:要访问结果值,您需要在脚本的实际“.output”键之前使用附加键“.result”。