@cdktn/hcl2cdk 0.23.0-pre.7 → 0.23.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/ambient.d.ts +0 -1
- package/build/__tests__/coerceType.test.js +165 -0
- package/build/__tests__/expressionToTs.test.js +693 -0
- package/build/__tests__/expressions.test.js +415 -0
- package/build/__tests__/findExpressionType.test.js +105 -0
- package/build/__tests__/functions.test.js +605 -0
- package/build/__tests__/generation.test.js +45 -0
- package/build/__tests__/jsii-rosetta-workarounds.test.js +86 -0
- package/build/__tests__/partialCode.test.js +390 -0
- package/build/__tests__/terraformSchema.test.js +105 -0
- package/build/__tests__/testHelpers.js +16 -0
- package/build/coerceType.js +240 -0
- package/build/dynamic-blocks.js +44 -0
- package/build/expressions.js +633 -0
- package/build/function-bindings/functions.generated.js +1142 -0
- package/build/function-bindings/functions.js +73 -0
- package/{lib → build}/generation.js +4 -4
- package/{lib → build}/index.d.ts +2 -2
- package/build/index.js +364 -0
- package/build/iteration.js +87 -0
- package/build/jsii-rosetta-workarounds.js +126 -0
- package/build/partialCode.js +116 -0
- package/build/provider.js +40 -0
- package/build/references.js +141 -0
- package/build/schema.js +81 -0
- package/build/telemetryAllowList.json +24 -0
- package/build/terraformSchema.js +136 -0
- package/build/types.js +3 -0
- package/build/utils.js +25 -0
- package/build/variables.js +172 -0
- package/package.json +13 -14
- package/tsconfig.json +4 -2
- package/lib/__tests__/coerceType.test.js +0 -165
- package/lib/__tests__/expressionToTs.test.js +0 -693
- package/lib/__tests__/expressions.test.js +0 -415
- package/lib/__tests__/findExpressionType.test.js +0 -105
- package/lib/__tests__/functions.test.js +0 -605
- package/lib/__tests__/generation.test.js +0 -45
- package/lib/__tests__/jsii-rosetta-workarounds.test.js +0 -86
- package/lib/__tests__/partialCode.test.js +0 -390
- package/lib/__tests__/terraformSchema.test.js +0 -105
- package/lib/__tests__/testHelpers.js +0 -16
- package/lib/coerceType.js +0 -240
- package/lib/dynamic-blocks.js +0 -44
- package/lib/expressions.js +0 -634
- package/lib/function-bindings/functions.generated.js +0 -1142
- package/lib/function-bindings/functions.js +0 -73
- package/lib/index.js +0 -364
- package/lib/iteration.js +0 -87
- package/lib/jsii-rosetta-workarounds.js +0 -126
- package/lib/partialCode.js +0 -116
- package/lib/provider.js +0 -40
- package/lib/references.js +0 -141
- package/lib/schema.js +0 -81
- package/lib/terraformSchema.js +0 -136
- package/lib/types.js +0 -3
- package/lib/utils.js +0 -25
- package/lib/variables.js +0 -172
- /package/{lib → build}/__tests__/coerceType.test.d.ts +0 -0
- /package/{lib → build}/__tests__/expressionToTs.test.d.ts +0 -0
- /package/{lib → build}/__tests__/expressions.test.d.ts +0 -0
- /package/{lib → build}/__tests__/findExpressionType.test.d.ts +0 -0
- /package/{lib → build}/__tests__/functions.test.d.ts +0 -0
- /package/{lib → build}/__tests__/generation.test.d.ts +0 -0
- /package/{lib → build}/__tests__/jsii-rosetta-workarounds.test.d.ts +0 -0
- /package/{lib → build}/__tests__/partialCode.test.d.ts +0 -0
- /package/{lib → build}/__tests__/terraformSchema.test.d.ts +0 -0
- /package/{lib → build}/__tests__/testHelpers.d.ts +0 -0
- /package/{lib → build}/coerceType.d.ts +0 -0
- /package/{lib → build}/dynamic-blocks.d.ts +0 -0
- /package/{lib → build}/expressions.d.ts +0 -0
- /package/{lib → build}/function-bindings/functions.d.ts +0 -0
- /package/{lib → build}/function-bindings/functions.generated.d.ts +0 -0
- /package/{lib → build}/generation.d.ts +0 -0
- /package/{lib → build}/iteration.d.ts +0 -0
- /package/{lib → build}/jsii-rosetta-workarounds.d.ts +0 -0
- /package/{lib → build}/partialCode.d.ts +0 -0
- /package/{lib → build}/provider.d.ts +0 -0
- /package/{lib → build}/references.d.ts +0 -0
- /package/{lib → build}/schema.d.ts +0 -0
- /package/{lib → build}/terraformSchema.d.ts +0 -0
- /package/{lib → build}/types.d.ts +0 -0
- /package/{lib → build}/utils.d.ts +0 -0
- /package/{lib → build}/variables.d.ts +0 -0
- /package/{lib → src}/telemetryAllowList.json +0 -0
|
@@ -0,0 +1,693 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Copyright (c) HashiCorp, Inc.
|
|
4
|
+
* SPDX-License-Identifier: MPL-2.0
|
|
5
|
+
*/
|
|
6
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
7
|
+
const expressions_1 = require("../expressions");
|
|
8
|
+
const testHelpers_1 = require("./testHelpers");
|
|
9
|
+
const awsProviderSchema = {
|
|
10
|
+
format_version: "1.0",
|
|
11
|
+
provider_schemas: {
|
|
12
|
+
"registry.terraform.io/hashicorp/aws": {
|
|
13
|
+
provider: {},
|
|
14
|
+
resource_schemas: {
|
|
15
|
+
aws_s3_bucket: {
|
|
16
|
+
block: {
|
|
17
|
+
attributes: {
|
|
18
|
+
foo: {
|
|
19
|
+
type: ["map", "string"],
|
|
20
|
+
description_kind: "plain",
|
|
21
|
+
},
|
|
22
|
+
},
|
|
23
|
+
},
|
|
24
|
+
},
|
|
25
|
+
},
|
|
26
|
+
data_source_schemas: {
|
|
27
|
+
aws_availability_zones: {
|
|
28
|
+
version: 0,
|
|
29
|
+
block: {
|
|
30
|
+
attributes: {
|
|
31
|
+
all_availability_zones: {
|
|
32
|
+
type: "bool",
|
|
33
|
+
description_kind: "plain",
|
|
34
|
+
optional: true,
|
|
35
|
+
},
|
|
36
|
+
exclude_names: {
|
|
37
|
+
type: ["set", "string"],
|
|
38
|
+
description_kind: "plain",
|
|
39
|
+
optional: true,
|
|
40
|
+
},
|
|
41
|
+
exclude_zone_ids: {
|
|
42
|
+
type: ["set", "string"],
|
|
43
|
+
description_kind: "plain",
|
|
44
|
+
optional: true,
|
|
45
|
+
},
|
|
46
|
+
group_names: {
|
|
47
|
+
type: ["set", "string"],
|
|
48
|
+
description_kind: "plain",
|
|
49
|
+
computed: true,
|
|
50
|
+
},
|
|
51
|
+
id: {
|
|
52
|
+
type: "string",
|
|
53
|
+
description_kind: "plain",
|
|
54
|
+
optional: true,
|
|
55
|
+
computed: true,
|
|
56
|
+
},
|
|
57
|
+
names: {
|
|
58
|
+
type: ["list", "string"],
|
|
59
|
+
description_kind: "plain",
|
|
60
|
+
computed: true,
|
|
61
|
+
},
|
|
62
|
+
state: {
|
|
63
|
+
type: "string",
|
|
64
|
+
description_kind: "plain",
|
|
65
|
+
optional: true,
|
|
66
|
+
},
|
|
67
|
+
zone_ids: {
|
|
68
|
+
type: ["list", "string"],
|
|
69
|
+
description_kind: "plain",
|
|
70
|
+
computed: true,
|
|
71
|
+
},
|
|
72
|
+
testing_map: {
|
|
73
|
+
type: ["map", "string"],
|
|
74
|
+
description_kind: "plain",
|
|
75
|
+
computed: true,
|
|
76
|
+
},
|
|
77
|
+
},
|
|
78
|
+
block_types: {
|
|
79
|
+
filter: {
|
|
80
|
+
nesting_mode: "set",
|
|
81
|
+
block: {
|
|
82
|
+
attributes: {
|
|
83
|
+
name: {
|
|
84
|
+
type: "string",
|
|
85
|
+
description_kind: "plain",
|
|
86
|
+
required: true,
|
|
87
|
+
},
|
|
88
|
+
values: {
|
|
89
|
+
type: ["set", "string"],
|
|
90
|
+
description_kind: "plain",
|
|
91
|
+
required: true,
|
|
92
|
+
},
|
|
93
|
+
},
|
|
94
|
+
description_kind: "plain",
|
|
95
|
+
},
|
|
96
|
+
},
|
|
97
|
+
timeouts: {
|
|
98
|
+
nesting_mode: "single",
|
|
99
|
+
block: {
|
|
100
|
+
attributes: {
|
|
101
|
+
read: {
|
|
102
|
+
type: "string",
|
|
103
|
+
description_kind: "plain",
|
|
104
|
+
optional: true,
|
|
105
|
+
},
|
|
106
|
+
},
|
|
107
|
+
description_kind: "plain",
|
|
108
|
+
},
|
|
109
|
+
},
|
|
110
|
+
},
|
|
111
|
+
description_kind: "plain",
|
|
112
|
+
},
|
|
113
|
+
},
|
|
114
|
+
},
|
|
115
|
+
},
|
|
116
|
+
},
|
|
117
|
+
};
|
|
118
|
+
const externalProviderSchema = {
|
|
119
|
+
format_version: "1.0",
|
|
120
|
+
provider_schemas: {
|
|
121
|
+
"registry.terraform.io/hashicorp/external": {
|
|
122
|
+
provider: {},
|
|
123
|
+
resource_schemas: {},
|
|
124
|
+
data_source_schemas: {
|
|
125
|
+
external: {
|
|
126
|
+
version: 0,
|
|
127
|
+
block: {
|
|
128
|
+
attributes: {
|
|
129
|
+
id: {
|
|
130
|
+
type: "string",
|
|
131
|
+
description: "The id of the data source. This will always be set to `-`",
|
|
132
|
+
description_kind: "plain",
|
|
133
|
+
computed: true,
|
|
134
|
+
},
|
|
135
|
+
program: {
|
|
136
|
+
type: ["list", "string"],
|
|
137
|
+
description: "A list of strings, whose first element is the program to run and whose subsequent elements are optional command line arguments to the program. Terraform does not execute the program through a shell, so it is not necessary to escape shell metacharacters nor add quotes around arguments containing spaces.",
|
|
138
|
+
description_kind: "plain",
|
|
139
|
+
required: true,
|
|
140
|
+
},
|
|
141
|
+
query: {
|
|
142
|
+
type: ["map", "string"],
|
|
143
|
+
description: "A map of string values to pass to the external program as the query arguments. If not supplied, the program will receive an empty object as its input.",
|
|
144
|
+
description_kind: "plain",
|
|
145
|
+
optional: true,
|
|
146
|
+
},
|
|
147
|
+
result: {
|
|
148
|
+
type: ["map", "string"],
|
|
149
|
+
description: "A map of string values returned from the external program.",
|
|
150
|
+
description_kind: "plain",
|
|
151
|
+
computed: true,
|
|
152
|
+
},
|
|
153
|
+
working_dir: {
|
|
154
|
+
type: "string",
|
|
155
|
+
description: "Working directory of the program. If not supplied, the program will run in the current directory.",
|
|
156
|
+
description_kind: "plain",
|
|
157
|
+
optional: true,
|
|
158
|
+
},
|
|
159
|
+
},
|
|
160
|
+
description: 'The `external` data source allows an external program implementing a specific protocol (defined below) to act as a data source, exposing arbitrary data for use elsewhere in the Terraform configuration.\n\n**Warning** This mechanism is provided as an "escape hatch" for exceptional situations where a first-class Terraform provider is not more appropriate. Its capabilities are limited in comparison to a true data source, and implementing a data source via an external program is likely to hurt the portability of your Terraform configuration by creating dependencies on external programs and libraries that may not be available (or may need to be used differently) on different operating systems.\n\n**Warning** Terraform Enterprise does not guarantee availability of any particular language runtimes or external programs beyond standard shell utilities, so it is not recommended to use this data source within configurations that are applied within Terraform Enterprise.',
|
|
161
|
+
description_kind: "plain",
|
|
162
|
+
},
|
|
163
|
+
},
|
|
164
|
+
},
|
|
165
|
+
},
|
|
166
|
+
},
|
|
167
|
+
};
|
|
168
|
+
function getScope({ provider, resources, data, variables, locals, forEachIteratorName, scopedVariables, withinOverrideExpression = false, } = {}) {
|
|
169
|
+
const scopeVariables = {};
|
|
170
|
+
resources === null || resources === void 0 ? void 0 : resources.forEach((varName) => {
|
|
171
|
+
let [resourceType, resourceName] = varName.split(".");
|
|
172
|
+
if (!resourceName) {
|
|
173
|
+
resourceName = varName;
|
|
174
|
+
resourceType = "aws_s3_bucket";
|
|
175
|
+
}
|
|
176
|
+
scopeVariables[varName] = {
|
|
177
|
+
resource: resourceType,
|
|
178
|
+
variableName: resourceName,
|
|
179
|
+
};
|
|
180
|
+
});
|
|
181
|
+
data === null || data === void 0 ? void 0 : data.forEach((varName) => {
|
|
182
|
+
scopeVariables[varName] = {
|
|
183
|
+
resource: "data.aws_s3_bucket",
|
|
184
|
+
variableName: varName,
|
|
185
|
+
};
|
|
186
|
+
});
|
|
187
|
+
variables === null || variables === void 0 ? void 0 : variables.forEach((varName) => {
|
|
188
|
+
scopeVariables[varName] = {
|
|
189
|
+
resource: "var",
|
|
190
|
+
variableName: varName,
|
|
191
|
+
};
|
|
192
|
+
});
|
|
193
|
+
locals === null || locals === void 0 ? void 0 : locals.forEach((varName) => {
|
|
194
|
+
scopeVariables[varName] = {
|
|
195
|
+
resource: "local",
|
|
196
|
+
variableName: varName,
|
|
197
|
+
};
|
|
198
|
+
});
|
|
199
|
+
const scope = {
|
|
200
|
+
providerSchema: provider !== null && provider !== void 0 ? provider : {},
|
|
201
|
+
providerGenerator: {},
|
|
202
|
+
constructs: new Set(),
|
|
203
|
+
variables: scopeVariables,
|
|
204
|
+
hasTokenBasedTypeCoercion: true,
|
|
205
|
+
scopedVariables: scopedVariables || {},
|
|
206
|
+
forEachIteratorName,
|
|
207
|
+
withinOverrideExpression,
|
|
208
|
+
nodeIds: [],
|
|
209
|
+
importables: [],
|
|
210
|
+
topLevelConfig: {},
|
|
211
|
+
};
|
|
212
|
+
return scope;
|
|
213
|
+
}
|
|
214
|
+
const getType = () => "string";
|
|
215
|
+
describe("expressionToTs", () => {
|
|
216
|
+
test("converts a simple string to a string", async () => {
|
|
217
|
+
const expression = "hello";
|
|
218
|
+
const scope = getScope();
|
|
219
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
220
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""hello""`);
|
|
221
|
+
});
|
|
222
|
+
test("converts a simple expression with a template", async () => {
|
|
223
|
+
const expression = "${22}";
|
|
224
|
+
const scope = getScope();
|
|
225
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
226
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(22)"`);
|
|
227
|
+
});
|
|
228
|
+
test("converts a variable reference", async () => {
|
|
229
|
+
const expression = "${var.foo}";
|
|
230
|
+
const scope = getScope();
|
|
231
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
232
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"foo.stringValue"`);
|
|
233
|
+
});
|
|
234
|
+
test("convert a variable reference with snake case", async () => {
|
|
235
|
+
const expression = "${var.foo_bar}";
|
|
236
|
+
const scope = getScope({ variables: ["foo_bar"] });
|
|
237
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
238
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(fooBar.value)"`);
|
|
239
|
+
});
|
|
240
|
+
test("convert a function call", async () => {
|
|
241
|
+
const expression = `\${replace("hello", "l", "w")}`;
|
|
242
|
+
const scope = getScope();
|
|
243
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
244
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.replace("hello", "l", "w"))"`);
|
|
245
|
+
});
|
|
246
|
+
test("converts a function call with an expression", async () => {
|
|
247
|
+
const expression = `\${replace("hello-\${22+22}", "44", "world")}`;
|
|
248
|
+
const scope = getScope();
|
|
249
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
250
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.replace("hello-" + Token.asString(Op.add(22, 22)), "44", "world"))"`);
|
|
251
|
+
});
|
|
252
|
+
test("converts string concatenation of iterator key", async () => {
|
|
253
|
+
const expression = '${"dynamic-ingress-${ingress.key}"}';
|
|
254
|
+
const scope = getScope({
|
|
255
|
+
scopedVariables: {
|
|
256
|
+
ingress: "dynamic_iterator0",
|
|
257
|
+
},
|
|
258
|
+
});
|
|
259
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
260
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""dynamic-ingress-\${" + Token.asString(dynamic_iterator0.key) + "}""`);
|
|
261
|
+
});
|
|
262
|
+
test("convert a variable reference", async () => {
|
|
263
|
+
const expression = `\${var.foo_bar}`;
|
|
264
|
+
const scope = getScope({ variables: ["foo_bar"] });
|
|
265
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
266
|
+
// TODO: This seems broken
|
|
267
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(fooBar.value)"`);
|
|
268
|
+
});
|
|
269
|
+
test("convert a resource reference", async () => {
|
|
270
|
+
const expression = `"simple-\${aws_s3_bucket.foo.id}"`;
|
|
271
|
+
const scope = getScope({ resources: ["aws_s3_bucket.foo"] });
|
|
272
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
273
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""simple-\${" + foo.id + "}""`);
|
|
274
|
+
});
|
|
275
|
+
test("convert a resource reference with nested properties", async () => {
|
|
276
|
+
const expression = `"simple-\${aws_s3_bucket.foo.prop.test}"`;
|
|
277
|
+
const scope = getScope({ resources: ["aws_s3_bucket.foo"] });
|
|
278
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
279
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""simple-\${" + foo.prop.test + "}""`);
|
|
280
|
+
});
|
|
281
|
+
test("convert a local reference", async () => {
|
|
282
|
+
const expression = `"simple-\${local.foo}"`;
|
|
283
|
+
const scope = getScope({ locals: ["foo"] });
|
|
284
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
285
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""simple-\${" + foo + "}""`);
|
|
286
|
+
});
|
|
287
|
+
test("plain resource references in arithmetics", async () => {
|
|
288
|
+
const expression = "${aws_s3_bucket.examplebucket.count + aws_s3_bucket.otherbucket.count }";
|
|
289
|
+
const scope = getScope({
|
|
290
|
+
resources: ["aws_s3_bucket.examplebucket", "aws_s3_bucket.otherbucket"],
|
|
291
|
+
});
|
|
292
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
293
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Op.add(examplebucket.count, otherbucket.count))"`);
|
|
294
|
+
});
|
|
295
|
+
test("use fqn for splat reference", async () => {
|
|
296
|
+
const expression = `\${aws_s3_bucket.foo.*.id}`;
|
|
297
|
+
const scope = getScope({ resources: ["aws_s3_bucket.foo"] });
|
|
298
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
299
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(foo, ["*", "id"]))"`);
|
|
300
|
+
});
|
|
301
|
+
test("use fqn if property is present on numeric access using dot notation", async () => {
|
|
302
|
+
const expression = "${aws_s3_bucket.examplebucket.network_interface.0.access_config.0.assigned_nat_ip}";
|
|
303
|
+
const scope = getScope({ resources: ["aws_s3_bucket.examplebucket"] });
|
|
304
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
305
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(examplebucket.networkInterface, ["0", "access_config", "0", "assigned_nat_ip"]))"`);
|
|
306
|
+
});
|
|
307
|
+
test("use fqn if property is present on numeric access", async () => {
|
|
308
|
+
const expression = "${aws_s3_bucket.examplebucket.network_interface[0].access_config[0].assigned_nat_ip}";
|
|
309
|
+
const scope = getScope({ resources: ["aws_s3_bucket.examplebucket"] });
|
|
310
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
311
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(examplebucket.networkInterface, ["0", "access_config", "0", "assigned_nat_ip"]))"`);
|
|
312
|
+
});
|
|
313
|
+
test("use no fqn if property is present on numeric access using []", async () => {
|
|
314
|
+
const expression = "${aws_s3_bucket.examplebucket.network_interface[0].access_config[0].assigned_nat_ip}";
|
|
315
|
+
const scope = getScope({ resources: ["aws_s3_bucket.examplebucket"] });
|
|
316
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
317
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(examplebucket.networkInterface, ["0", "access_config", "0", "assigned_nat_ip"]))"`);
|
|
318
|
+
});
|
|
319
|
+
test("detects splat reference within function", async () => {
|
|
320
|
+
const expression = "${toset(aws_s3_bucket.examplebucket.*)}";
|
|
321
|
+
const scope = getScope({ resources: ["aws_s3_bucket.examplebucket"] });
|
|
322
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
323
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.toset(Fn.lookupNested(examplebucket, ["*"])))"`);
|
|
324
|
+
});
|
|
325
|
+
test("convert conditional", async () => {
|
|
326
|
+
const expression = "${aws_kms_key.key.deletion_window_in_days > 3 ? aws_s3_bucket.examplebucket.id : []}";
|
|
327
|
+
const scope = getScope({ resources: ["aws_kms_key.key", "examplebucket"] });
|
|
328
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
329
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(conditional(Op.gt(key.deletionWindowInDays, 3), examplebucket.id, []))"`);
|
|
330
|
+
});
|
|
331
|
+
test("converts a function with references", async () => {
|
|
332
|
+
const expression = "${element(aws_s3_bucket.examplebucket, 0).id}";
|
|
333
|
+
const scope = getScope({ resources: ["aws_s3_bucket.examplebucket"] });
|
|
334
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
335
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(Fn.element(examplebucket, 0), ["id"]))"`);
|
|
336
|
+
});
|
|
337
|
+
test("converts a function with splat reference", async () => {
|
|
338
|
+
const expression = "${element(aws_s3_bucket.examplebucket.*.id, 0)}";
|
|
339
|
+
const scope = getScope({ resources: ["aws_s3_bucket.examplebucket"] });
|
|
340
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
341
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.element(Fn.lookupNested(examplebucket, ["*", "id"]), 0))"`);
|
|
342
|
+
});
|
|
343
|
+
test("convert for loops", async () => {
|
|
344
|
+
const expression = "${{ for name, user in var.users : user.role => name...}}";
|
|
345
|
+
const scope = getScope();
|
|
346
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
347
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""\${{ for name, user in \${" + users.value + "} : user.role => name...}}""`);
|
|
348
|
+
});
|
|
349
|
+
test("for expressions", async () => {
|
|
350
|
+
const expression = "{for s in var.list : s => upper(s)}";
|
|
351
|
+
const scope = getScope();
|
|
352
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
353
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""\${{ for s in \${" + list.value + "} : s => upper(s)}}""`);
|
|
354
|
+
});
|
|
355
|
+
test("for list expression ", async () => {
|
|
356
|
+
const expression = "[for s in var.list : upper(s)]";
|
|
357
|
+
const scope = getScope();
|
|
358
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
359
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""\${[ for s in \${" + list.value + "} : upper(s)]}""`);
|
|
360
|
+
});
|
|
361
|
+
test("for list with conditional", async () => {
|
|
362
|
+
const expression = '[for s in var.list : upper(s) if s != ""]';
|
|
363
|
+
const scope = getScope();
|
|
364
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
365
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""\${[ for s in \${" + list.value + "} : upper(s) if s != \\"\\"]}""`);
|
|
366
|
+
});
|
|
367
|
+
test("convert property access resources", async () => {
|
|
368
|
+
const expression = "${aws_s3_bucket.examplebucket[0].id}";
|
|
369
|
+
const scope = getScope({ resources: ["aws_s3_bucket.examplebucket"] });
|
|
370
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
371
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(examplebucket, ["0", "id"]))"`);
|
|
372
|
+
});
|
|
373
|
+
test("convert unary operators", async () => {
|
|
374
|
+
const expression = `\${!var.enabled}`;
|
|
375
|
+
const scope = getScope({ variables: ["enabled"] });
|
|
376
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, () => "bool");
|
|
377
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asBoolean(Op.not(enabled.value))"`);
|
|
378
|
+
});
|
|
379
|
+
test("convert function with arrays and comments", async () => {
|
|
380
|
+
const expression = `\${compact([
|
|
381
|
+
# The example "bucket"
|
|
382
|
+
aws_s3_bucket.examplebucket,
|
|
383
|
+
|
|
384
|
+
# The "Learn" single page application. This is not configured in all environments.
|
|
385
|
+
var.input,
|
|
386
|
+
])}`;
|
|
387
|
+
const scope = getScope({
|
|
388
|
+
resources: ["aws_s3_bucket.examplebucket"],
|
|
389
|
+
variables: ["input"],
|
|
390
|
+
});
|
|
391
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
392
|
+
// TODO: See if we have a way to preserve comments
|
|
393
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.compact(Token.asList([examplebucket, input.value])))"`);
|
|
394
|
+
});
|
|
395
|
+
test("convert references for same reference", async () => {
|
|
396
|
+
const expression = `\${var.input == "test" ? "azure-ad-int" : "azure-ad-\${var.input}"}`;
|
|
397
|
+
const scope = getScope({ variables: ["input"] });
|
|
398
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
399
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(conditional(Op.eq(input.value, "test"), "azure-ad-int", "azure-ad-\${" + input.value + "}"))"`);
|
|
400
|
+
});
|
|
401
|
+
test("nested variable access", async () => {
|
|
402
|
+
const expression = `\${element(var.test2["val1"], 0)}`;
|
|
403
|
+
const scope = getScope({ variables: ["test2"] });
|
|
404
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
405
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.element(Fn.lookupNested(test2.value, ["\\"val1\\""]), 0))"`);
|
|
406
|
+
});
|
|
407
|
+
test("complicated nested local value", async () => {
|
|
408
|
+
const expression = "${flatten(var.vnets[*].subnets[*].name)}";
|
|
409
|
+
const scope = getScope({ variables: ["vnets"] });
|
|
410
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
411
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.flatten(Fn.lookupNested(vnets.value, ["*", "subnets", "*", "name"])))"`);
|
|
412
|
+
});
|
|
413
|
+
// TOIMPROVE: I don't think we can handle this case yet, since the variable
|
|
414
|
+
// needs to be wrapped in an iterator. I'm leaving this in, but it's broken
|
|
415
|
+
test("complicated nested variable access with map", async () => {
|
|
416
|
+
const expression = `{
|
|
417
|
+
for vnet in var.vnets[*]:
|
|
418
|
+
(vnet.vnet_name) => vnet.subnets[*].name
|
|
419
|
+
}`;
|
|
420
|
+
const scope = getScope({ variables: ["vnets"] });
|
|
421
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
422
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""\${{ for vnet in \${" + Fn.lookupNested(vnets.value, ["*"]) + "} : (vnet.vnet_name) => vnet.subnets[*].name}}""`);
|
|
423
|
+
});
|
|
424
|
+
test("complicated nested variable access with list", async () => {
|
|
425
|
+
const expression = `\${flatten([
|
|
426
|
+
for k, v in var.route : [
|
|
427
|
+
for n, s in v : [
|
|
428
|
+
{
|
|
429
|
+
key = k,
|
|
430
|
+
name = n,
|
|
431
|
+
svc_url = s
|
|
432
|
+
}
|
|
433
|
+
]
|
|
434
|
+
]
|
|
435
|
+
])}`;
|
|
436
|
+
const scope = getScope({ variables: ["route"] });
|
|
437
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
438
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.flatten("\${[ for k, v in \${" + route.value + "} : [\\n for n, s in v : [\\n {\\n key = k,\\n name = n,\\n svc_url = s\\n }\\n ]\\n ]]}"))"`);
|
|
439
|
+
});
|
|
440
|
+
test("simple local variable access", async () => {
|
|
441
|
+
const expression = "${!local.enabled}";
|
|
442
|
+
const scope = getScope({ locals: ["enabled"] });
|
|
443
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
444
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Op.not(enabled))"`);
|
|
445
|
+
});
|
|
446
|
+
test("multi local access", async () => {
|
|
447
|
+
const expression = '"${local.service_name},${local.owner},${local.is_it_great},${local.how_many}"';
|
|
448
|
+
const scope = getScope({
|
|
449
|
+
locals: ["service_name", "owner", "is_it_great", "how_many"],
|
|
450
|
+
});
|
|
451
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
452
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""\${" + serviceName + "},\${" + owner + "},\${" + isItGreat + "},\${" + howMany + "}""`);
|
|
453
|
+
});
|
|
454
|
+
test("binary operations within literals and variables", async () => {
|
|
455
|
+
const expression = '"${var.test} + 1"';
|
|
456
|
+
const scope = getScope({ variables: ["test"] });
|
|
457
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
458
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""\${" + test.value + "} + 1""`);
|
|
459
|
+
});
|
|
460
|
+
test("local variable access", async () => {
|
|
461
|
+
const expression = '"${local.enabled}"';
|
|
462
|
+
const scope = getScope({ locals: ["enabled"] });
|
|
463
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
464
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"enabled"`);
|
|
465
|
+
});
|
|
466
|
+
test("converts traversals that cannot be references", async () => {
|
|
467
|
+
const expression = "${self.path}";
|
|
468
|
+
const scope = getScope();
|
|
469
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
470
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(TerraformSelf.getAny("path"))"`);
|
|
471
|
+
});
|
|
472
|
+
test("converts join function with variables", async () => {
|
|
473
|
+
const expression = '${join("-", [var.tags.app, var.tags.env])}';
|
|
474
|
+
const scope = getScope({ variables: ["tags"] });
|
|
475
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
476
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.join("-", Token.asList([Fn.lookupNested(tags.value, ["app"]), Fn.lookupNested(tags.value, ["env"])])))"`);
|
|
477
|
+
});
|
|
478
|
+
test("converts join function with many arguments variables", async () => {
|
|
479
|
+
const expression = '${join("-", var.tags.app, var.tags.env, var.tags.other)}';
|
|
480
|
+
const scope = getScope({ variables: ["tags"] });
|
|
481
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
482
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.join("-", Token.asList(Fn.concat([Fn.lookupNested(tags.value, ["app"]), Fn.lookupNested(tags.value, ["env"]), Fn.lookupNested(tags.value, ["other"])]))))"`);
|
|
483
|
+
});
|
|
484
|
+
test("doesn't wrap any extra templates", async () => {
|
|
485
|
+
const expression = `"app-\${terraform.workspace}"`;
|
|
486
|
+
const scope = getScope();
|
|
487
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
488
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""app-\${terraform.workspace}""`);
|
|
489
|
+
});
|
|
490
|
+
test("converts a for expression", async () => {
|
|
491
|
+
const expression = `[for record in aws_route53_record.example : record.fqdn]`;
|
|
492
|
+
const scope = getScope();
|
|
493
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
494
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""\${[ for record in \${" + example.fqn + "} : record.fqdn]}""`);
|
|
495
|
+
});
|
|
496
|
+
test("converts a data source", async () => {
|
|
497
|
+
const expression = `"\${data.aws_route53_zone.example.zone_id}"`;
|
|
498
|
+
const scope = getScope();
|
|
499
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
500
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(example.zoneId)"`);
|
|
501
|
+
});
|
|
502
|
+
test("convert an each expression", async () => {
|
|
503
|
+
const expression = `"\${each.value.name}"`;
|
|
504
|
+
const scope = getScope({ forEachIteratorName: "myIterator" });
|
|
505
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
506
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(myIterator.value, ["name"]))"`);
|
|
507
|
+
});
|
|
508
|
+
test("converts a property of a function containing a resource", async () => {
|
|
509
|
+
const expression = `"\${element(aws_s3_bucket.examplebucket, 0).id}"`;
|
|
510
|
+
const scope = getScope();
|
|
511
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
512
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(Fn.element(examplebucket, 0), ["id"]))"`);
|
|
513
|
+
});
|
|
514
|
+
test("convert a function with references with splats", async () => {
|
|
515
|
+
const expression = `"\${concat(var.private_subnets.*.id, var.public_subnets.*.id)}"`;
|
|
516
|
+
const scope = getScope({
|
|
517
|
+
variables: ["private_subnets", "public_subnets"],
|
|
518
|
+
});
|
|
519
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
520
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.concat([Fn.lookupNested(privateSubnets.value, ["*", "id"]), Fn.lookupNested(publicSubnets.value, ["*", "id"])]))"`);
|
|
521
|
+
});
|
|
522
|
+
test("convert a iterator variable", async () => {
|
|
523
|
+
const expression = `"\${each.key}"`;
|
|
524
|
+
const scope = getScope({
|
|
525
|
+
variables: ["private_subnets", "public_subnets"],
|
|
526
|
+
forEachIteratorName: "myIterator",
|
|
527
|
+
});
|
|
528
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
529
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(myIterator.key)"`);
|
|
530
|
+
});
|
|
531
|
+
test("convert a self reference", async () => {
|
|
532
|
+
const expression = `"\${self.subnet.id}"`;
|
|
533
|
+
const scope = getScope({
|
|
534
|
+
variables: ["private_subnets", "public_subnets"],
|
|
535
|
+
forEachIteratorName: "myIterator",
|
|
536
|
+
});
|
|
537
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
538
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(TerraformSelf.getAny("subnet.id"))"`);
|
|
539
|
+
});
|
|
540
|
+
test("convert heredocs", async () => {
|
|
541
|
+
const expression = `<<EOF
|
|
542
|
+
[{
|
|
543
|
+
"Condition": {
|
|
544
|
+
"KeyPrefixEquals": "docs/"
|
|
545
|
+
},
|
|
546
|
+
"Redirect": {
|
|
547
|
+
"ReplaceKeyPrefixWith": "documents/"
|
|
548
|
+
}
|
|
549
|
+
}]
|
|
550
|
+
EOF
|
|
551
|
+
`;
|
|
552
|
+
const scope = getScope();
|
|
553
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
554
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""[{\\n \\"Condition\\": {\\n \\"KeyPrefixEquals\\": \\"docs/\\"\\n },\\n \\"Redirect\\": {\\n \\"ReplaceKeyPrefixWith\\": \\"documents/\\"\\n }\\n}]\\n""`);
|
|
555
|
+
});
|
|
556
|
+
test("convert heredocs without a trailing newline", async () => {
|
|
557
|
+
const expression = `<<EOF
|
|
558
|
+
hello world
|
|
559
|
+
EOF`;
|
|
560
|
+
const scope = getScope();
|
|
561
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
562
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""hello world\\n""`);
|
|
563
|
+
});
|
|
564
|
+
test("convert indented heredocs", async () => {
|
|
565
|
+
const expression = `<<-EOF
|
|
566
|
+
hello world
|
|
567
|
+
EOF`;
|
|
568
|
+
const scope = getScope();
|
|
569
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
570
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""hello world\\n""`);
|
|
571
|
+
});
|
|
572
|
+
test("convert override expressions", async () => {
|
|
573
|
+
const expression = '"${required_resource_access.value["resource_access"]}"';
|
|
574
|
+
const scope = getScope({
|
|
575
|
+
withinOverrideExpression: true,
|
|
576
|
+
scopedVariables: {
|
|
577
|
+
required_resource_access: "dynamic-block",
|
|
578
|
+
},
|
|
579
|
+
});
|
|
580
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
581
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`""\${required_resource_access.value[\\"resource_access\\"]}""`);
|
|
582
|
+
});
|
|
583
|
+
test("convert using operations while containing substring", async () => {
|
|
584
|
+
const expression =
|
|
585
|
+
// prettier-ignore
|
|
586
|
+
'"${length(var.image_id) > 4 && substr(var.image_id, 0, 4) == "ami-"}"';
|
|
587
|
+
const scope = getScope({});
|
|
588
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
589
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Op.and(Op.gt(Fn.lengthOf(imageId.value), 4), Op.eq(Fn.substr(imageId.stringValue, 0, 4), "ami-")))"`);
|
|
590
|
+
});
|
|
591
|
+
test("convert a data source with numeric access", async () => {
|
|
592
|
+
const expression = `"\${data.aws_availability_zones.changeme_az_list_ebs_snapshot.names[0]}"`;
|
|
593
|
+
const scope = getScope({
|
|
594
|
+
provider: awsProviderSchema,
|
|
595
|
+
data: ["aws_subnet_ids"],
|
|
596
|
+
});
|
|
597
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
598
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(changemeAzListEbsSnapshot.names, ["0"]))"`);
|
|
599
|
+
});
|
|
600
|
+
test("convert a reference to a map access", async () => {
|
|
601
|
+
const expression = `"\${data.aws_availability_zones.changeme_az_list_ebs_snapshot.testing_map.foo}"`;
|
|
602
|
+
const scope = getScope({
|
|
603
|
+
provider: awsProviderSchema,
|
|
604
|
+
data: ["aws_subnet_ids"],
|
|
605
|
+
});
|
|
606
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
607
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(changemeAzListEbsSnapshot, ["testing_map", "foo"]))"`);
|
|
608
|
+
});
|
|
609
|
+
test("convert attribute up to a map and not within attribute reference", async () => {
|
|
610
|
+
const expression = `"\${data.aws_availability_zones.changeme_az_list_ebs_snapshot.testing_map}"`;
|
|
611
|
+
const scope = getScope({
|
|
612
|
+
provider: awsProviderSchema,
|
|
613
|
+
});
|
|
614
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, () => ["map", "string"]);
|
|
615
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asStringMap(changemeAzListEbsSnapshot.testingMap)"`);
|
|
616
|
+
});
|
|
617
|
+
test("don't convert external unknown fields", async () => {
|
|
618
|
+
const expression = `"\${data.external.changeme_external_thumbprint_data.result.thumbprint}"`;
|
|
619
|
+
const scope = getScope({
|
|
620
|
+
provider: externalProviderSchema,
|
|
621
|
+
data: ["aws_subnet_ids"],
|
|
622
|
+
});
|
|
623
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, () => ["map", "string"]);
|
|
624
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asStringMap(Fn.lookupNested(changemeExternalThumbprintData, ["result", "thumbprint"]))"`);
|
|
625
|
+
});
|
|
626
|
+
test("convert resource reference with map attribute", async () => {
|
|
627
|
+
const expression = `"\${aws_s3_bucket.examplebucket.foo.bar}"`;
|
|
628
|
+
const scope = getScope({
|
|
629
|
+
provider: awsProviderSchema,
|
|
630
|
+
resources: ["aws_s3_bucket"],
|
|
631
|
+
});
|
|
632
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, () => "string");
|
|
633
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(examplebucket, ["foo", "bar"]))"`);
|
|
634
|
+
});
|
|
635
|
+
test("convert complex variable reference in template", async () => {
|
|
636
|
+
const expression = `"\${var.default_tags.project}-client-tg"`;
|
|
637
|
+
const scope = getScope({
|
|
638
|
+
provider: awsProviderSchema,
|
|
639
|
+
resources: ["aws_s3_bucket"],
|
|
640
|
+
variables: ["default_tags"],
|
|
641
|
+
});
|
|
642
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, () => "string");
|
|
643
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(defaultTags.value, ["project"])) + "-client-tg""`);
|
|
644
|
+
});
|
|
645
|
+
test("convert boolean values", async () => {
|
|
646
|
+
const expression = `"\${false}"`;
|
|
647
|
+
const scope = getScope({
|
|
648
|
+
provider: awsProviderSchema,
|
|
649
|
+
});
|
|
650
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, () => "bool");
|
|
651
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"false"`);
|
|
652
|
+
});
|
|
653
|
+
test("convert a data source with count", async () => {
|
|
654
|
+
const expression = `"\${data.aws_availability_zones.available.names[count.index]}"`;
|
|
655
|
+
const scope = getScope({
|
|
656
|
+
provider: awsProviderSchema,
|
|
657
|
+
data: ["aws_availability_zones"],
|
|
658
|
+
});
|
|
659
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
660
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.lookupNested(available.names, ["\${count.index}"]))"`);
|
|
661
|
+
});
|
|
662
|
+
test("accept escaped quotes within string", async () => {
|
|
663
|
+
const expression = `"\${jsonencode({
|
|
664
|
+
"Statement" = [{
|
|
665
|
+
"Action" = "s3:*",
|
|
666
|
+
"Effect" = "Allow",
|
|
667
|
+
}],
|
|
668
|
+
})}"`;
|
|
669
|
+
const scope = getScope();
|
|
670
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
671
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`
|
|
672
|
+
"Token.asString(Fn.jsonencode({
|
|
673
|
+
"Statement": [{
|
|
674
|
+
"Action": "s3:*",
|
|
675
|
+
"Effect": "Allow"
|
|
676
|
+
}]
|
|
677
|
+
}))"
|
|
678
|
+
`);
|
|
679
|
+
});
|
|
680
|
+
test("support variadic parameters in any position", async () => {
|
|
681
|
+
const expression = `"\${cidrsubnets("fd00:fd12:3456:7890::/56", 16, 16, 16, 32)}"`;
|
|
682
|
+
const scope = getScope();
|
|
683
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
684
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"Token.asString(Fn.cidrsubnets("fd00:fd12:3456:7890::/56", [16, 16, 16, 32]))"`);
|
|
685
|
+
});
|
|
686
|
+
test("converts template expression with escaped ${} expression", async () => {
|
|
687
|
+
const expression = '["$${path:name.givenName}"]'; // from aws_ssoadmin_instance_access_control_attributes example
|
|
688
|
+
const scope = getScope();
|
|
689
|
+
const result = await (0, expressions_1.convertTerraformExpressionToTs)(scope, expression, getType);
|
|
690
|
+
expect((0, testHelpers_1.astToCode)(result)).toMatchInlineSnapshot(`"["$\${path:name.givenName}"]"`);
|
|
691
|
+
});
|
|
692
|
+
});
|
|
693
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"expressionToTs.test.js","sourceRoot":"","sources":["../../src/__tests__/expressionToTs.test.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Copyright (c) HashiCorp, Inc.\n * SPDX-License-Identifier: MPL-2.0\n */\n\nimport { convertTerraformExpressionToTs } from \"../expressions\";\nimport { ResourceScope } from \"../types\";\nimport { AttributeType } from \"@cdktn/commons\";\nimport { astToCode as code } from \"./testHelpers\";\n\nconst awsProviderSchema = {\n  format_version: \"1.0\",\n  provider_schemas: {\n    \"registry.terraform.io/hashicorp/aws\": {\n      provider: {},\n      resource_schemas: {\n        aws_s3_bucket: {\n          block: {\n            attributes: {\n              foo: {\n                type: [\"map\", \"string\"],\n                description_kind: \"plain\",\n              },\n            },\n          },\n        },\n      },\n      data_source_schemas: {\n        aws_availability_zones: {\n          version: 0,\n          block: {\n            attributes: {\n              all_availability_zones: {\n                type: \"bool\",\n                description_kind: \"plain\",\n                optional: true,\n              },\n              exclude_names: {\n                type: [\"set\", \"string\"],\n                description_kind: \"plain\",\n                optional: true,\n              },\n              exclude_zone_ids: {\n                type: [\"set\", \"string\"],\n                description_kind: \"plain\",\n                optional: true,\n              },\n              group_names: {\n                type: [\"set\", \"string\"],\n                description_kind: \"plain\",\n                computed: true,\n              },\n              id: {\n                type: \"string\",\n                description_kind: \"plain\",\n                optional: true,\n                computed: true,\n              },\n              names: {\n                type: [\"list\", \"string\"],\n                description_kind: \"plain\",\n                computed: true,\n              },\n              state: {\n                type: \"string\",\n                description_kind: \"plain\",\n                optional: true,\n              },\n              zone_ids: {\n                type: [\"list\", \"string\"],\n                description_kind: \"plain\",\n                computed: true,\n              },\n              testing_map: {\n                type: [\"map\", \"string\"],\n                description_kind: \"plain\",\n                computed: true,\n              },\n            },\n            block_types: {\n              filter: {\n                nesting_mode: \"set\",\n                block: {\n                  attributes: {\n                    name: {\n                      type: \"string\",\n                      description_kind: \"plain\",\n                      required: true,\n                    },\n                    values: {\n                      type: [\"set\", \"string\"],\n                      description_kind: \"plain\",\n                      required: true,\n                    },\n                  },\n                  description_kind: \"plain\",\n                },\n              },\n              timeouts: {\n                nesting_mode: \"single\",\n                block: {\n                  attributes: {\n                    read: {\n                      type: \"string\",\n                      description_kind: \"plain\",\n                      optional: true,\n                    },\n                  },\n                  description_kind: \"plain\",\n                },\n              },\n            },\n            description_kind: \"plain\",\n          },\n        },\n      },\n    },\n  },\n};\n\nconst externalProviderSchema = {\n  format_version: \"1.0\",\n  provider_schemas: {\n    \"registry.terraform.io/hashicorp/external\": {\n      provider: {},\n      resource_schemas: {},\n      data_source_schemas: {\n        external: {\n          version: 0,\n          block: {\n            attributes: {\n              id: {\n                type: \"string\",\n                description:\n                  \"The id of the data source. This will always be set to `-`\",\n                description_kind: \"plain\",\n                computed: true,\n              },\n              program: {\n                type: [\"list\", \"string\"],\n                description:\n                  \"A list of strings, whose first element is the program to run and whose subsequent elements are optional command line arguments to the program. Terraform does not execute the program through a shell, so it is not necessary to escape shell metacharacters nor add quotes around arguments containing spaces.\",\n                description_kind: \"plain\",\n                required: true,\n              },\n              query: {\n                type: [\"map\", \"string\"],\n                description:\n                  \"A map of string values to pass to the external program as the query arguments. If not supplied, the program will receive an empty object as its input.\",\n                description_kind: \"plain\",\n                optional: true,\n              },\n              result: {\n                type: [\"map\", \"string\"],\n                description:\n                  \"A map of string values returned from the external program.\",\n                description_kind: \"plain\",\n                computed: true,\n              },\n              working_dir: {\n                type: \"string\",\n                description:\n                  \"Working directory of the program. If not supplied, the program will run in the current directory.\",\n                description_kind: \"plain\",\n                optional: true,\n              },\n            },\n            description:\n              'The `external` data source allows an external program implementing a specific protocol (defined below) to act as a data source, exposing arbitrary data for use elsewhere in the Terraform configuration.\\n\\n**Warning** This mechanism is provided as an \"escape hatch\" for exceptional situations where a first-class Terraform provider is not more appropriate. Its capabilities are limited in comparison to a true data source, and implementing a data source via an external program is likely to hurt the portability of your Terraform configuration by creating dependencies on external programs and libraries that may not be available (or may need to be used differently) on different operating systems.\\n\\n**Warning** Terraform Enterprise does not guarantee availability of any particular language runtimes or external programs beyond standard shell utilities, so it is not recommended to use this data source within configurations that are applied within Terraform Enterprise.',\n            description_kind: \"plain\",\n          },\n        },\n      },\n    },\n  },\n};\n\ntype GetScopeParams = {\n  provider?: Record<string, any>;\n  resources?: string[];\n  data?: string[];\n  variables?: string[];\n  locals?: string[];\n  forEachIteratorName?: string;\n  withinOverrideExpression?: boolean;\n  scopedVariables?: Record<string, string>;\n};\n\nfunction getScope({\n  provider,\n  resources,\n  data,\n  variables,\n  locals,\n  forEachIteratorName,\n  scopedVariables,\n  withinOverrideExpression = false,\n}: GetScopeParams = {}): ResourceScope {\n  const scopeVariables: ResourceScope[\"variables\"] = {};\n\n  resources?.forEach((varName) => {\n    let [resourceType, resourceName] = varName.split(\".\");\n    if (!resourceName) {\n      resourceName = varName;\n      resourceType = \"aws_s3_bucket\";\n    }\n\n    scopeVariables[varName] = {\n      resource: resourceType,\n      variableName: resourceName,\n    };\n  });\n\n  data?.forEach((varName) => {\n    scopeVariables[varName] = {\n      resource: \"data.aws_s3_bucket\",\n      variableName: varName,\n    };\n  });\n\n  variables?.forEach((varName) => {\n    scopeVariables[varName] = {\n      resource: \"var\",\n      variableName: varName,\n    };\n  });\n\n  locals?.forEach((varName) => {\n    scopeVariables[varName] = {\n      resource: \"local\",\n      variableName: varName,\n    };\n  });\n\n  const scope = {\n    providerSchema: provider ?? {},\n    providerGenerator: {},\n    constructs: new Set<string>(),\n    variables: scopeVariables,\n    hasTokenBasedTypeCoercion: true,\n    scopedVariables: scopedVariables || {},\n    forEachIteratorName,\n    withinOverrideExpression,\n    nodeIds: [],\n    importables: [],\n    topLevelConfig: {},\n  };\n\n  return scope;\n}\n\nconst getType = (): AttributeType => \"string\";\n\ndescribe(\"expressionToTs\", () => {\n  test(\"converts a simple string to a string\", async () => {\n    const expression = \"hello\";\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(`\"\"hello\"\"`);\n  });\n\n  test(\"converts a simple expression with a template\", async () => {\n    const expression = \"${22}\";\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(`\"Token.asString(22)\"`);\n  });\n\n  test(\"converts a variable reference\", async () => {\n    const expression = \"${var.foo}\";\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(`\"foo.stringValue\"`);\n  });\n\n  test(\"convert a variable reference with snake case\", async () => {\n    const expression = \"${var.foo_bar}\";\n    const scope = getScope({ variables: [\"foo_bar\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(fooBar.value)\"`,\n    );\n  });\n\n  test(\"convert a function call\", async () => {\n    const expression = `\\${replace(\"hello\", \"l\", \"w\")}`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.replace(\"hello\", \"l\", \"w\"))\"`,\n    );\n  });\n\n  test(\"converts a function call with an expression\", async () => {\n    const expression = `\\${replace(\"hello-\\${22+22}\", \"44\", \"world\")}`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.replace(\"hello-\" + Token.asString(Op.add(22, 22)), \"44\", \"world\"))\"`,\n    );\n  });\n\n  test(\"converts string concatenation of iterator key\", async () => {\n    const expression = '${\"dynamic-ingress-${ingress.key}\"}';\n    const scope = getScope({\n      scopedVariables: {\n        ingress: \"dynamic_iterator0\",\n      },\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"dynamic-ingress-\\${\" + Token.asString(dynamic_iterator0.key) + \"}\"\"`,\n    );\n  });\n\n  test(\"convert a variable reference\", async () => {\n    const expression = `\\${var.foo_bar}`;\n    const scope = getScope({ variables: [\"foo_bar\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    // TODO: This seems broken\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(fooBar.value)\"`,\n    );\n  });\n\n  test(\"convert a resource reference\", async () => {\n    const expression = `\"simple-\\${aws_s3_bucket.foo.id}\"`;\n    const scope = getScope({ resources: [\"aws_s3_bucket.foo\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(`\"\"simple-\\${\" + foo.id + \"}\"\"`);\n  });\n\n  test(\"convert a resource reference with nested properties\", async () => {\n    const expression = `\"simple-\\${aws_s3_bucket.foo.prop.test}\"`;\n    const scope = getScope({ resources: [\"aws_s3_bucket.foo\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"simple-\\${\" + foo.prop.test + \"}\"\"`,\n    );\n  });\n\n  test(\"convert a local reference\", async () => {\n    const expression = `\"simple-\\${local.foo}\"`;\n    const scope = getScope({ locals: [\"foo\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(`\"\"simple-\\${\" + foo + \"}\"\"`);\n  });\n\n  test(\"plain resource references in arithmetics\", async () => {\n    const expression =\n      \"${aws_s3_bucket.examplebucket.count + aws_s3_bucket.otherbucket.count }\";\n    const scope = getScope({\n      resources: [\"aws_s3_bucket.examplebucket\", \"aws_s3_bucket.otherbucket\"],\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Op.add(examplebucket.count, otherbucket.count))\"`,\n    );\n  });\n\n  test(\"use fqn for splat reference\", async () => {\n    const expression = `\\${aws_s3_bucket.foo.*.id}`;\n    const scope = getScope({ resources: [\"aws_s3_bucket.foo\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(foo, [\"*\", \"id\"]))\"`,\n    );\n  });\n\n  test(\"use fqn if property is present on numeric access using dot notation\", async () => {\n    const expression =\n      \"${aws_s3_bucket.examplebucket.network_interface.0.access_config.0.assigned_nat_ip}\";\n    const scope = getScope({ resources: [\"aws_s3_bucket.examplebucket\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(examplebucket.networkInterface, [\"0\", \"access_config\", \"0\", \"assigned_nat_ip\"]))\"`,\n    );\n  });\n\n  test(\"use fqn if property is present on numeric access\", async () => {\n    const expression =\n      \"${aws_s3_bucket.examplebucket.network_interface[0].access_config[0].assigned_nat_ip}\";\n    const scope = getScope({ resources: [\"aws_s3_bucket.examplebucket\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(examplebucket.networkInterface, [\"0\", \"access_config\", \"0\", \"assigned_nat_ip\"]))\"`,\n    );\n  });\n\n  test(\"use no fqn if property is present on numeric access using []\", async () => {\n    const expression =\n      \"${aws_s3_bucket.examplebucket.network_interface[0].access_config[0].assigned_nat_ip}\";\n    const scope = getScope({ resources: [\"aws_s3_bucket.examplebucket\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(examplebucket.networkInterface, [\"0\", \"access_config\", \"0\", \"assigned_nat_ip\"]))\"`,\n    );\n  });\n\n  test(\"detects splat reference within function\", async () => {\n    const expression = \"${toset(aws_s3_bucket.examplebucket.*)}\";\n    const scope = getScope({ resources: [\"aws_s3_bucket.examplebucket\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.toset(Fn.lookupNested(examplebucket, [\"*\"])))\"`,\n    );\n  });\n\n  test(\"convert conditional\", async () => {\n    const expression =\n      \"${aws_kms_key.key.deletion_window_in_days > 3 ? aws_s3_bucket.examplebucket.id : []}\";\n    const scope = getScope({ resources: [\"aws_kms_key.key\", \"examplebucket\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(conditional(Op.gt(key.deletionWindowInDays, 3), examplebucket.id, []))\"`,\n    );\n  });\n\n  test(\"converts a function with references\", async () => {\n    const expression = \"${element(aws_s3_bucket.examplebucket, 0).id}\";\n    const scope = getScope({ resources: [\"aws_s3_bucket.examplebucket\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(Fn.element(examplebucket, 0), [\"id\"]))\"`,\n    );\n  });\n\n  test(\"converts a function with splat reference\", async () => {\n    const expression = \"${element(aws_s3_bucket.examplebucket.*.id, 0)}\";\n    const scope = getScope({ resources: [\"aws_s3_bucket.examplebucket\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.element(Fn.lookupNested(examplebucket, [\"*\", \"id\"]), 0))\"`,\n    );\n  });\n\n  test(\"convert for loops\", async () => {\n    const expression =\n      \"${{ for name, user in var.users : user.role => name...}}\";\n\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"\\${{ for name, user in \\${\" + users.value + \"} : user.role => name...}}\"\"`,\n    );\n  });\n\n  test(\"for expressions\", async () => {\n    const expression = \"{for s in var.list : s => upper(s)}\";\n\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"\\${{ for s in \\${\" + list.value + \"} : s => upper(s)}}\"\"`,\n    );\n  });\n\n  test(\"for list expression \", async () => {\n    const expression = \"[for s in var.list : upper(s)]\";\n\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"\\${[ for s in \\${\" + list.value + \"} : upper(s)]}\"\"`,\n    );\n  });\n\n  test(\"for list with conditional\", async () => {\n    const expression = '[for s in var.list : upper(s) if s != \"\"]';\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"\\${[ for s in \\${\" + list.value + \"} : upper(s) if s != \\\\\"\\\\\"]}\"\"`,\n    );\n  });\n\n  test(\"convert property access resources\", async () => {\n    const expression = \"${aws_s3_bucket.examplebucket[0].id}\";\n    const scope = getScope({ resources: [\"aws_s3_bucket.examplebucket\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(examplebucket, [\"0\", \"id\"]))\"`,\n    );\n  });\n\n  test(\"convert unary operators\", async () => {\n    const expression = `\\${!var.enabled}`;\n    const scope = getScope({ variables: [\"enabled\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      () => \"bool\",\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asBoolean(Op.not(enabled.value))\"`,\n    );\n  });\n\n  test(\"convert function with arrays and comments\", async () => {\n    const expression = `\\${compact([\n            # The example \"bucket\"\n            aws_s3_bucket.examplebucket,\n        \n            # The \"Learn\" single page application. This is not configured in all environments.\n            var.input,\n          ])}`;\n    const scope = getScope({\n      resources: [\"aws_s3_bucket.examplebucket\"],\n      variables: [\"input\"],\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    // TODO: See if we have a way to preserve comments\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.compact(Token.asList([examplebucket, input.value])))\"`,\n    );\n  });\n\n  test(\"convert references for same reference\", async () => {\n    const expression = `\\${var.input == \"test\" ? \"azure-ad-int\" : \"azure-ad-\\${var.input}\"}`;\n    const scope = getScope({ variables: [\"input\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(conditional(Op.eq(input.value, \"test\"), \"azure-ad-int\", \"azure-ad-\\${\" + input.value + \"}\"))\"`,\n    );\n  });\n\n  test(\"nested variable access\", async () => {\n    const expression = `\\${element(var.test2[\"val1\"], 0)}`;\n    const scope = getScope({ variables: [\"test2\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.element(Fn.lookupNested(test2.value, [\"\\\\\"val1\\\\\"\"]), 0))\"`,\n    );\n  });\n\n  test(\"complicated nested local value\", async () => {\n    const expression = \"${flatten(var.vnets[*].subnets[*].name)}\";\n    const scope = getScope({ variables: [\"vnets\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.flatten(Fn.lookupNested(vnets.value, [\"*\", \"subnets\", \"*\", \"name\"])))\"`,\n    );\n  });\n\n  // TOIMPROVE: I don't think we can handle this case yet, since the variable\n  // needs to be wrapped in an iterator. I'm leaving this in, but it's broken\n  test(\"complicated nested variable access with map\", async () => {\n    const expression = `{ \n    for vnet in var.vnets[*]:\n    (vnet.vnet_name) => vnet.subnets[*].name\n  }`;\n    const scope = getScope({ variables: [\"vnets\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"\\${{ for vnet in \\${\" + Fn.lookupNested(vnets.value, [\"*\"]) + \"} : (vnet.vnet_name) => vnet.subnets[*].name}}\"\"`,\n    );\n  });\n\n  test(\"complicated nested variable access with list\", async () => {\n    const expression = `\\${flatten([\n    for k, v in var.route : [\n      for n, s in v : [\n        {\n          key = k,\n          name = n,\n          svc_url = s\n        }\n      ]\n    ]\n  ])}`;\n    const scope = getScope({ variables: [\"route\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.flatten(\"\\${[ for k, v in \\${\" + route.value + \"} : [\\\\n      for n, s in v : [\\\\n        {\\\\n          key = k,\\\\n          name = n,\\\\n          svc_url = s\\\\n        }\\\\n      ]\\\\n    ]]}\"))\"`,\n    );\n  });\n\n  test(\"simple local variable access\", async () => {\n    const expression = \"${!local.enabled}\";\n    const scope = getScope({ locals: [\"enabled\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Op.not(enabled))\"`,\n    );\n  });\n\n  test(\"multi local access\", async () => {\n    const expression =\n      '\"${local.service_name},${local.owner},${local.is_it_great},${local.how_many}\"';\n    const scope = getScope({\n      locals: [\"service_name\", \"owner\", \"is_it_great\", \"how_many\"],\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"\\${\" + serviceName + \"},\\${\" + owner + \"},\\${\" + isItGreat + \"},\\${\" + howMany + \"}\"\"`,\n    );\n  });\n\n  test(\"binary operations within literals and variables\", async () => {\n    const expression = '\"${var.test} + 1\"';\n    const scope = getScope({ variables: [\"test\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"\\${\" + test.value + \"} + 1\"\"`,\n    );\n  });\n\n  test(\"local variable access\", async () => {\n    const expression = '\"${local.enabled}\"';\n    const scope = getScope({ locals: [\"enabled\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(`\"enabled\"`);\n  });\n\n  test(\"converts traversals that cannot be references\", async () => {\n    const expression = \"${self.path}\";\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(TerraformSelf.getAny(\"path\"))\"`,\n    );\n  });\n\n  test(\"converts join function with variables\", async () => {\n    const expression = '${join(\"-\", [var.tags.app, var.tags.env])}';\n    const scope = getScope({ variables: [\"tags\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.join(\"-\", Token.asList([Fn.lookupNested(tags.value, [\"app\"]), Fn.lookupNested(tags.value, [\"env\"])])))\"`,\n    );\n  });\n\n  test(\"converts join function with many arguments variables\", async () => {\n    const expression =\n      '${join(\"-\", var.tags.app, var.tags.env, var.tags.other)}';\n    const scope = getScope({ variables: [\"tags\"] });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.join(\"-\", Token.asList(Fn.concat([Fn.lookupNested(tags.value, [\"app\"]), Fn.lookupNested(tags.value, [\"env\"]), Fn.lookupNested(tags.value, [\"other\"])]))))\"`,\n    );\n  });\n\n  test(\"doesn't wrap any extra templates\", async () => {\n    const expression = `\"app-\\${terraform.workspace}\"`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"app-\\${terraform.workspace}\"\"`,\n    );\n  });\n\n  test(\"converts a for expression\", async () => {\n    const expression = `[for record in aws_route53_record.example : record.fqdn]`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"\\${[ for record in \\${\" + example.fqn + \"} : record.fqdn]}\"\"`,\n    );\n  });\n\n  test(\"converts a data source\", async () => {\n    const expression = `\"\\${data.aws_route53_zone.example.zone_id}\"`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(example.zoneId)\"`,\n    );\n  });\n\n  test(\"convert an each expression\", async () => {\n    const expression = `\"\\${each.value.name}\"`;\n    const scope = getScope({ forEachIteratorName: \"myIterator\" });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(myIterator.value, [\"name\"]))\"`,\n    );\n  });\n\n  test(\"converts a property of a function containing a resource\", async () => {\n    const expression = `\"\\${element(aws_s3_bucket.examplebucket, 0).id}\"`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(Fn.element(examplebucket, 0), [\"id\"]))\"`,\n    );\n  });\n\n  test(\"convert a function with references with splats\", async () => {\n    const expression = `\"\\${concat(var.private_subnets.*.id, var.public_subnets.*.id)}\"`;\n    const scope = getScope({\n      variables: [\"private_subnets\", \"public_subnets\"],\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.concat([Fn.lookupNested(privateSubnets.value, [\"*\", \"id\"]), Fn.lookupNested(publicSubnets.value, [\"*\", \"id\"])]))\"`,\n    );\n  });\n\n  test(\"convert a iterator variable\", async () => {\n    const expression = `\"\\${each.key}\"`;\n    const scope = getScope({\n      variables: [\"private_subnets\", \"public_subnets\"],\n      forEachIteratorName: \"myIterator\",\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(myIterator.key)\"`,\n    );\n  });\n\n  test(\"convert a self reference\", async () => {\n    const expression = `\"\\${self.subnet.id}\"`;\n    const scope = getScope({\n      variables: [\"private_subnets\", \"public_subnets\"],\n      forEachIteratorName: \"myIterator\",\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(TerraformSelf.getAny(\"subnet.id\"))\"`,\n    );\n  });\n\n  test(\"convert heredocs\", async () => {\n    const expression = `<<EOF\n[{\n    \"Condition\": {\n        \"KeyPrefixEquals\": \"docs/\"\n    },\n    \"Redirect\": {\n        \"ReplaceKeyPrefixWith\": \"documents/\"\n    }\n}]\nEOF\n`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"[{\\\\n    \\\\\"Condition\\\\\": {\\\\n        \\\\\"KeyPrefixEquals\\\\\": \\\\\"docs/\\\\\"\\\\n    },\\\\n    \\\\\"Redirect\\\\\": {\\\\n        \\\\\"ReplaceKeyPrefixWith\\\\\": \\\\\"documents/\\\\\"\\\\n    }\\\\n}]\\\\n\"\"`,\n    );\n  });\n\n  test(\"convert heredocs without a trailing newline\", async () => {\n    const expression = `<<EOF\nhello world\nEOF`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(`\"\"hello world\\\\n\"\"`);\n  });\n\n  test(\"convert indented heredocs\", async () => {\n    const expression = `<<-EOF\n              hello world\n          EOF`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(`\"\"hello world\\\\n\"\"`);\n  });\n\n  test(\"convert override expressions\", async () => {\n    const expression = '\"${required_resource_access.value[\"resource_access\"]}\"';\n    const scope = getScope({\n      withinOverrideExpression: true,\n      scopedVariables: {\n        required_resource_access: \"dynamic-block\",\n      },\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"\"\\${required_resource_access.value[\\\\\"resource_access\\\\\"]}\"\"`,\n    );\n  });\n\n  test(\"convert using operations while containing substring\", async () => {\n    const expression =\n      // prettier-ignore\n      '\"${length(var.image_id) > 4 && substr(var.image_id, 0, 4) == \"ami-\"}\"';\n    const scope = getScope({});\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Op.and(Op.gt(Fn.lengthOf(imageId.value), 4), Op.eq(Fn.substr(imageId.stringValue, 0, 4), \"ami-\")))\"`,\n    );\n  });\n\n  test(\"convert a data source with numeric access\", async () => {\n    const expression = `\"\\${data.aws_availability_zones.changeme_az_list_ebs_snapshot.names[0]}\"`;\n    const scope = getScope({\n      provider: awsProviderSchema,\n      data: [\"aws_subnet_ids\"],\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(changemeAzListEbsSnapshot.names, [\"0\"]))\"`,\n    );\n  });\n\n  test(\"convert a reference to a map access\", async () => {\n    const expression = `\"\\${data.aws_availability_zones.changeme_az_list_ebs_snapshot.testing_map.foo}\"`;\n    const scope = getScope({\n      provider: awsProviderSchema,\n      data: [\"aws_subnet_ids\"],\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(changemeAzListEbsSnapshot, [\"testing_map\", \"foo\"]))\"`,\n    );\n  });\n\n  test(\"convert attribute up to a map and not within attribute reference\", async () => {\n    const expression = `\"\\${data.aws_availability_zones.changeme_az_list_ebs_snapshot.testing_map}\"`;\n    const scope = getScope({\n      provider: awsProviderSchema,\n    });\n\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      () => [\"map\", \"string\"],\n    );\n\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asStringMap(changemeAzListEbsSnapshot.testingMap)\"`,\n    );\n  });\n\n  test(\"don't convert external unknown fields\", async () => {\n    const expression = `\"\\${data.external.changeme_external_thumbprint_data.result.thumbprint}\"`;\n    const scope = getScope({\n      provider: externalProviderSchema,\n      data: [\"aws_subnet_ids\"],\n    });\n\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      () => [\"map\", \"string\"],\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asStringMap(Fn.lookupNested(changemeExternalThumbprintData, [\"result\", \"thumbprint\"]))\"`,\n    );\n  });\n\n  test(\"convert resource reference with map attribute\", async () => {\n    const expression = `\"\\${aws_s3_bucket.examplebucket.foo.bar}\"`;\n    const scope = getScope({\n      provider: awsProviderSchema,\n      resources: [\"aws_s3_bucket\"],\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      () => \"string\",\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(examplebucket, [\"foo\", \"bar\"]))\"`,\n    );\n  });\n\n  test(\"convert complex variable reference in template\", async () => {\n    const expression = `\"\\${var.default_tags.project}-client-tg\"`;\n    const scope = getScope({\n      provider: awsProviderSchema,\n      resources: [\"aws_s3_bucket\"],\n      variables: [\"default_tags\"],\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      () => \"string\",\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(defaultTags.value, [\"project\"])) + \"-client-tg\"\"`,\n    );\n  });\n\n  test(\"convert boolean values\", async () => {\n    const expression = `\"\\${false}\"`;\n    const scope = getScope({\n      provider: awsProviderSchema,\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      () => \"bool\",\n    );\n    expect(code(result)).toMatchInlineSnapshot(`\"false\"`);\n  });\n\n  test(\"convert a data source with count\", async () => {\n    const expression = `\"\\${data.aws_availability_zones.available.names[count.index]}\"`;\n    const scope = getScope({\n      provider: awsProviderSchema,\n      data: [\"aws_availability_zones\"],\n    });\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.lookupNested(available.names, [\"\\${count.index}\"]))\"`,\n    );\n  });\n\n  test(\"accept escaped quotes within string\", async () => {\n    const expression = `\"\\${jsonencode({\n          \"Statement\" = [{\n            \"Action\" = \"s3:*\",\n            \"Effect\" = \"Allow\",\n          }],\n        })}\"`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n    expect(code(result)).toMatchInlineSnapshot(`\n      \"Token.asString(Fn.jsonencode({\n        \"Statement\": [{\n          \"Action\": \"s3:*\",\n          \"Effect\": \"Allow\"\n        }]\n      }))\"\n    `);\n  });\n\n  test(\"support variadic parameters in any position\", async () => {\n    const expression = `\"\\${cidrsubnets(\"fd00:fd12:3456:7890::/56\", 16, 16, 16, 32)}\"`;\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"Token.asString(Fn.cidrsubnets(\"fd00:fd12:3456:7890::/56\", [16, 16, 16, 32]))\"`,\n    );\n  });\n\n  test(\"converts template expression with escaped ${} expression\", async () => {\n    const expression = '[\"$${path:name.givenName}\"]'; // from aws_ssoadmin_instance_access_control_attributes example\n    const scope = getScope();\n    const result = await convertTerraformExpressionToTs(\n      scope,\n      expression,\n      getType,\n    );\n\n    expect(code(result)).toMatchInlineSnapshot(\n      `\"[\"$\\${path:name.givenName}\"]\"`,\n    );\n  });\n});\n"]}
|