@cdktn/hcl2cdk 0.21.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.
Files changed (61) hide show
  1. package/README.md +95 -0
  2. package/ambient.d.ts +14 -0
  3. package/jest.config.js +16 -0
  4. package/lib/__tests__/coerceType.test.d.ts +2 -0
  5. package/lib/__tests__/coerceType.test.js +165 -0
  6. package/lib/__tests__/expressionToTs.test.d.ts +6 -0
  7. package/lib/__tests__/expressionToTs.test.js +693 -0
  8. package/lib/__tests__/expressions.test.d.ts +4 -0
  9. package/lib/__tests__/expressions.test.js +415 -0
  10. package/lib/__tests__/findExpressionType.test.d.ts +6 -0
  11. package/lib/__tests__/findExpressionType.test.js +105 -0
  12. package/lib/__tests__/functions.test.d.ts +6 -0
  13. package/lib/__tests__/functions.test.js +605 -0
  14. package/lib/__tests__/generation.test.d.ts +6 -0
  15. package/lib/__tests__/generation.test.js +45 -0
  16. package/lib/__tests__/jsii-rosetta-workarounds.test.d.ts +2 -0
  17. package/lib/__tests__/jsii-rosetta-workarounds.test.js +86 -0
  18. package/lib/__tests__/partialCode.test.d.ts +6 -0
  19. package/lib/__tests__/partialCode.test.js +390 -0
  20. package/lib/__tests__/terraformSchema.test.d.ts +6 -0
  21. package/lib/__tests__/terraformSchema.test.js +105 -0
  22. package/lib/__tests__/testHelpers.d.ts +7 -0
  23. package/lib/__tests__/testHelpers.js +16 -0
  24. package/lib/coerceType.d.ts +7 -0
  25. package/lib/coerceType.js +240 -0
  26. package/lib/dynamic-blocks.d.ts +8 -0
  27. package/lib/dynamic-blocks.js +44 -0
  28. package/lib/expressions.d.ts +9 -0
  29. package/lib/expressions.js +634 -0
  30. package/lib/function-bindings/functions.d.ts +31 -0
  31. package/lib/function-bindings/functions.generated.d.ts +806 -0
  32. package/lib/function-bindings/functions.generated.js +1142 -0
  33. package/lib/function-bindings/functions.js +73 -0
  34. package/lib/generation.d.ts +26 -0
  35. package/lib/generation.js +676 -0
  36. package/lib/index.d.ts +136 -0
  37. package/lib/index.js +364 -0
  38. package/lib/iteration.d.ts +23 -0
  39. package/lib/iteration.js +87 -0
  40. package/lib/jsii-rosetta-workarounds.d.ts +5 -0
  41. package/lib/jsii-rosetta-workarounds.js +126 -0
  42. package/lib/partialCode.d.ts +11 -0
  43. package/lib/partialCode.js +116 -0
  44. package/lib/provider.d.ts +8 -0
  45. package/lib/provider.js +40 -0
  46. package/lib/references.d.ts +11 -0
  47. package/lib/references.js +141 -0
  48. package/lib/schema.d.ts +297 -0
  49. package/lib/schema.js +81 -0
  50. package/lib/telemetryAllowList.json +24 -0
  51. package/lib/terraformSchema.d.ts +6 -0
  52. package/lib/terraformSchema.js +136 -0
  53. package/lib/types.d.ts +50 -0
  54. package/lib/types.js +3 -0
  55. package/lib/utils.d.ts +6 -0
  56. package/lib/utils.js +25 -0
  57. package/lib/variables.d.ts +10 -0
  58. package/lib/variables.js +172 -0
  59. package/package.json +71 -0
  60. package/package.sh +9 -0
  61. package/tsconfig.json +42 -0
@@ -0,0 +1,634 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || function (mod) {
19
+ if (mod && mod.__esModule) return mod;
20
+ var result = {};
21
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
+ __setModuleDefault(result, mod);
23
+ return result;
24
+ };
25
+ var __importDefault = (this && this.__importDefault) || function (mod) {
26
+ return (mod && mod.__esModule) ? mod : { "default": mod };
27
+ };
28
+ Object.defineProperty(exports, "__esModule", { value: true });
29
+ exports.dynamicVariableToAst = exports.extractIteratorVariablesFromExpression = exports.convertTerraformExpressionToTs = exports.expressionAst = void 0;
30
+ // Copyright (c) HashiCorp, Inc
31
+ // SPDX-License-Identifier: MPL-2.0
32
+ const t = __importStar(require("@babel/types"));
33
+ const template_1 = __importDefault(require("@babel/template"));
34
+ const utils_1 = require("./utils");
35
+ const hcl2json_1 = require("@cdktn/hcl2json");
36
+ const hcl2json_2 = require("@cdktn/hcl2json");
37
+ const functions_1 = require("./function-bindings/functions");
38
+ const coerceType_1 = require("./coerceType");
39
+ const terraformSchema_1 = require("./terraformSchema");
40
+ const references_1 = require("./references");
41
+ const variables_1 = require("./variables");
42
+ const syntax_tree_1 = require("@cdktn/hcl2json/lib/syntax-tree");
43
+ const tfBinaryOperatorsToCdktf = {
44
+ logicalOr: "or",
45
+ logicalAnd: "and",
46
+ greaterThan: "gt",
47
+ greaterThanOrEqual: "gte",
48
+ lessThan: "lt",
49
+ lessThanOrEqual: "lte",
50
+ equal: "eq",
51
+ notEqual: "neq",
52
+ add: "add",
53
+ subtract: "sub",
54
+ multiply: "mul",
55
+ divide: "div",
56
+ modulo: "mod",
57
+ };
58
+ const tfUnaryOperatorsToCdktf = {
59
+ logicalNot: "not",
60
+ negate: "negate",
61
+ };
62
+ function traversalPartsToString(traversals, asSuffix = false) {
63
+ let seed = "";
64
+ if (asSuffix && hcl2json_2.TFExpressionSyntaxTree.isNameTraversalPart(traversals[0])) {
65
+ seed = ".";
66
+ }
67
+ return traversals.reduce((acc, part) => {
68
+ if (part.type === "nameTraversal") {
69
+ if (acc === seed) {
70
+ return `${acc}${part.segment}`;
71
+ }
72
+ return `${acc}.${part.segment}`;
73
+ }
74
+ return `${acc}[${part.segment}]`;
75
+ }, seed);
76
+ }
77
+ function canUseFqn(expression) {
78
+ if (!hcl2json_2.TFExpressionSyntaxTree.isScopeTraversalExpression(expression)) {
79
+ return false;
80
+ }
81
+ const rootSegment = expression.meta.traversal[0].segment;
82
+ return !["var", "local"].includes(rootSegment);
83
+ }
84
+ function traversalToVariableName(scope, node) {
85
+ if (!hcl2json_2.TFExpressionSyntaxTree.isScopeTraversalExpression(node)) {
86
+ utils_1.logger.error(`Unexpected expression type ${node.type} with value ${node.meta.value} passed to convert to a variable.
87
+ ${utils_1.leaveCommentText}`);
88
+ return "";
89
+ }
90
+ const segments = node.meta.traversal;
91
+ if (segments.length === 1) {
92
+ return node.meta.fullAccessor;
93
+ }
94
+ const rootSegment = segments[0].segment;
95
+ const resource = rootSegment === "data"
96
+ ? `${segments[0].segment}.${segments[1].segment}`
97
+ : rootSegment;
98
+ const name = rootSegment === "data" ? segments[2].segment : segments[1].segment;
99
+ return (0, variables_1.variableName)(scope, resource, name);
100
+ }
101
+ function expressionForSerialStringConcatenation(scope, nodes) {
102
+ const reducedNodes = nodes.reduce((acc, node) => {
103
+ const prev = acc[acc.length - 1];
104
+ if (!prev)
105
+ return [node];
106
+ if (t.isStringLiteral(prev) && t.isStringLiteral(node)) {
107
+ acc.pop();
108
+ acc.push(t.stringLiteral(prev.value + node.value));
109
+ return acc;
110
+ }
111
+ acc.push(node);
112
+ return acc;
113
+ }, []);
114
+ return reducedNodes.reduce((acc, node) => {
115
+ if (!acc) {
116
+ return node;
117
+ }
118
+ // wrap access to dynamic blocks in Token.asString() as they return a Lazy
119
+ // for .key and .value which can't be concatenated in languages like Python
120
+ // because JSII currently has no support for the toString() method in
121
+ // languages other than TypeScript: https://github.com/aws/jsii/issues/380
122
+ // example: dynamic_iterator0.key / dynamic_iterator0.value
123
+ if (t.isMemberExpression(node) &&
124
+ t.isIdentifier(node.object) &&
125
+ Object.values(scope.scopedVariables || {}).includes(node.object.name) &&
126
+ t.isIdentifier(node.property) &&
127
+ ["key", "value"].includes(node.property.name)) {
128
+ node = (0, coerceType_1.coerceType)(scope, node, "dynamic", "string");
129
+ }
130
+ return t.binaryExpression("+", acc, node);
131
+ });
132
+ }
133
+ function getTfResourcePathFromNode(node) {
134
+ const segments = node.meta.traversal;
135
+ let resource = segments[0].segment;
136
+ let result = [];
137
+ let attributes = [];
138
+ if (segments[0].segment === "data") {
139
+ result.push(segments[0].segment);
140
+ resource = segments[1].segment;
141
+ attributes = segments.slice(3); // we want to skip the variable name
142
+ }
143
+ else {
144
+ attributes = segments.slice(2); // we want to skip the variable name
145
+ }
146
+ const [provider, ...resourceNameFragments] = resource.split("_");
147
+ // Hack: This happens in the case of `external` provider
148
+ // where the data source does not have a provider name prefix
149
+ if (resourceNameFragments.length === 0) {
150
+ resourceNameFragments.push(provider);
151
+ }
152
+ result.push(provider);
153
+ result.push(resourceNameFragments.join("_"));
154
+ result = [
155
+ ...result,
156
+ ...attributes.map((seg) => {
157
+ if (hcl2json_2.TFExpressionSyntaxTree.isIndexTraversalPart(seg)) {
158
+ return `[${seg.segment}]`;
159
+ }
160
+ return seg.segment;
161
+ }),
162
+ ];
163
+ return result.join(".");
164
+ }
165
+ function convertLiteralValueExpressionToTs(_scope, node) {
166
+ const literalType = node.meta.type;
167
+ if (literalType === "number") {
168
+ return t.numericLiteral(Number(node.meta.value));
169
+ }
170
+ if (literalType === "bool") {
171
+ return t.booleanLiteral(node.meta.value === "true" ? true : false);
172
+ }
173
+ return t.stringLiteral(node.meta.value);
174
+ }
175
+ function convertScopeTraversalExpressionToTs(scope, node) {
176
+ var _a;
177
+ const hasReference = (0, references_1.containsReference)(node);
178
+ const segments = node.meta.traversal;
179
+ if (segments[0].segment === "each" && scope.forEachIteratorName) {
180
+ return dynamicVariableToAst(scope, node, scope.forEachIteratorName);
181
+ }
182
+ if (segments[0].segment === "count" && scope.countIteratorName) {
183
+ return dynamicVariableToAst(scope, node, scope.countIteratorName, "count");
184
+ }
185
+ if (segments[0].segment === "self") {
186
+ scope.importables.push({
187
+ constructName: "TerraformSelf",
188
+ provider: "cdktn",
189
+ });
190
+ return t.callExpression(t.memberExpression(t.identifier("TerraformSelf"), t.identifier("getAny")), [t.stringLiteral(traversalPartsToString(segments.slice(1)))]);
191
+ }
192
+ // setting.value, setting.value[1].id
193
+ const dynamicBlock = (_a = scope.scopedVariables) === null || _a === void 0 ? void 0 : _a[segments[0].segment];
194
+ if (dynamicBlock) {
195
+ if (dynamicBlock === "dynamic-block") {
196
+ return dynamicVariableToAst(scope, node, dynamicBlock, traversalPartsToString(segments));
197
+ }
198
+ return dynamicVariableToAst(scope, node, dynamicBlock, segments[0].segment);
199
+ }
200
+ // This may be a variable reference that we don't understand yet, so we wrap it in a template string
201
+ // for Terraform to handle
202
+ let varIdentifier = t.stringLiteral(`\${${node.meta.fullAccessor}}`);
203
+ if (hasReference) {
204
+ varIdentifier = t.identifier((0, utils_1.camelCase)(traversalToVariableName(scope, node)));
205
+ }
206
+ if (["var", "local"].includes(segments[0].segment)) {
207
+ const variableAccessor = segments[0].segment === "var"
208
+ ? t.memberExpression(varIdentifier, t.identifier("value"))
209
+ : varIdentifier;
210
+ if (segments.length > 2) {
211
+ scope.importables.push({
212
+ constructName: "Fn",
213
+ provider: "cdktn",
214
+ });
215
+ const callee = t.memberExpression(t.identifier("Fn"), t.identifier("lookupNested"));
216
+ return t.callExpression(callee, [
217
+ variableAccessor,
218
+ t.arrayExpression(segments.slice(2).map((s) => t.stringLiteral(s.segment))),
219
+ ]);
220
+ }
221
+ return variableAccessor;
222
+ }
223
+ if (!hasReference || scope.withinOverrideExpression) {
224
+ return varIdentifier;
225
+ }
226
+ const rootSegment = segments[0].segment;
227
+ const attributeIndex = rootSegment === "data" ? 3 : 2;
228
+ const attributeSegments = segments.slice(attributeIndex);
229
+ const numericAccessorIndex = attributeSegments.findIndex((seg) => hcl2json_2.TFExpressionSyntaxTree.isIndexTraversalPart(seg));
230
+ let minAccessorIndex = numericAccessorIndex;
231
+ let mapAccessorIndex = -1;
232
+ if (numericAccessorIndex === -1) {
233
+ // only do this if we have to, if we already have a
234
+ // numeric accessor, we don't have to do this additional work
235
+ const resourcePath = getTfResourcePathFromNode(node);
236
+ let usingSubPathType = false;
237
+ let parts = resourcePath.split(".").filter((p) => p !== "");
238
+ const minParts = attributeIndex; // we need to stop before data.aws.resource_name or aws.resource_name
239
+ const originalParts = parts.length;
240
+ let hasMapAccessor = false;
241
+ while (parts.length >= minParts) {
242
+ const type = (0, terraformSchema_1.getTypeAtPath)(scope.providerSchema, parts.join("."));
243
+ if (type !== null) {
244
+ if (Array.isArray(type) && type[0] === "map" && usingSubPathType) {
245
+ hasMapAccessor = true;
246
+ break;
247
+ }
248
+ }
249
+ parts.pop();
250
+ usingSubPathType = true;
251
+ }
252
+ if (hasMapAccessor) {
253
+ mapAccessorIndex = originalParts - parts.length - 1;
254
+ minAccessorIndex = mapAccessorIndex;
255
+ }
256
+ }
257
+ const needsPropertyAccess = minAccessorIndex >= 0;
258
+ const refSegments = needsPropertyAccess
259
+ ? attributeSegments.slice(0, minAccessorIndex)
260
+ : attributeSegments;
261
+ const nonRefSegments = needsPropertyAccess
262
+ ? attributeSegments.slice(minAccessorIndex)
263
+ : [];
264
+ const ref = refSegments.reduce((acc, seg, index) => t.memberExpression(acc, t.identifier(index === 0 && rootSegment === "module"
265
+ ? (0, utils_1.camelCase)(seg.segment + "Output")
266
+ : (0, utils_1.camelCase)(seg.segment))), varIdentifier);
267
+ if (nonRefSegments.length === 0) {
268
+ return ref;
269
+ }
270
+ scope.importables.push({
271
+ constructName: "Fn",
272
+ provider: "cdktn",
273
+ });
274
+ const callee = t.memberExpression(t.identifier("Fn"), t.identifier("lookupNested"));
275
+ return t.callExpression(callee, [
276
+ ref,
277
+ t.arrayExpression(nonRefSegments.map((s) => t.stringLiteral(s.segment))),
278
+ ]);
279
+ }
280
+ function convertUnaryOpExpressionToTs(scope, node) {
281
+ const operand = convertTFExpressionAstToTs(scope, hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.valueExpression));
282
+ let fnName = node.meta.operator;
283
+ if (tfUnaryOperatorsToCdktf[fnName]) {
284
+ fnName = tfUnaryOperatorsToCdktf[fnName];
285
+ }
286
+ else {
287
+ throw new Error(`Cannot convert unknown operator ${node.meta.operator}`);
288
+ }
289
+ scope.importables.push({
290
+ constructName: "Op",
291
+ provider: "cdktn",
292
+ });
293
+ const fn = t.memberExpression(t.identifier("Op"), t.identifier(fnName));
294
+ return t.callExpression(fn, [operand]);
295
+ }
296
+ function convertBinaryOpExpressionToTs(scope, node) {
297
+ const left = convertTFExpressionAstToTs(scope, hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.lhsExpression));
298
+ const right = convertTFExpressionAstToTs(scope, hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.rhsExpression));
299
+ let fnName = node.meta.operator;
300
+ if (tfBinaryOperatorsToCdktf[fnName]) {
301
+ fnName = tfBinaryOperatorsToCdktf[fnName];
302
+ }
303
+ else {
304
+ throw new Error(`Cannot convert unknown operator ${node.meta.operator}`);
305
+ }
306
+ scope.importables.push({
307
+ constructName: "Op",
308
+ provider: "cdktn",
309
+ });
310
+ const fn = t.memberExpression(t.identifier("Op"), t.identifier(fnName));
311
+ return t.callExpression(fn, [left, right]);
312
+ }
313
+ function convertTemplateExpressionToTs(scope, node) {
314
+ const parts = node.children.map((child) => ({
315
+ node: child,
316
+ expr: convertTFExpressionAstToTs(scope, child),
317
+ }));
318
+ const lastPart = parts[parts.length - 1];
319
+ if (t.isStringLiteral(lastPart.expr) && lastPart.expr.value === "\n") {
320
+ // This is a bit of a hack, but the trailing newline we add due to
321
+ // heredocs looks ugly and unnecessary in the generated code, so we
322
+ // try to remove it
323
+ parts.pop();
324
+ }
325
+ if (parts.length === 0) {
326
+ return t.stringLiteral(node.meta.value);
327
+ }
328
+ if (parts.length === 1) {
329
+ return parts[0].expr;
330
+ }
331
+ let isScopedTraversal = false;
332
+ let expressions = [];
333
+ for (const { node, expr } of parts) {
334
+ if (hcl2json_2.TFExpressionSyntaxTree.isScopeTraversalExpression(node) &&
335
+ !t.isStringLiteral(expr) &&
336
+ !t.isCallExpression(expr)) {
337
+ expressions.push(t.stringLiteral("${"));
338
+ isScopedTraversal = true;
339
+ }
340
+ else if (
341
+ // we should ideally be doing type coercion more
342
+ // carefully here, because it may not always be needed
343
+ t.isCallExpression(expr)) {
344
+ scope.importables.push({
345
+ constructName: "Token",
346
+ provider: "cdktn",
347
+ });
348
+ expressions.push(template_1.default.expression(`Token.asString(%%expr%%)`)({
349
+ expr,
350
+ }));
351
+ continue;
352
+ }
353
+ else {
354
+ if (isScopedTraversal) {
355
+ expressions.push(t.stringLiteral("}"));
356
+ isScopedTraversal = false;
357
+ }
358
+ }
359
+ expressions.push(expr);
360
+ }
361
+ if (isScopedTraversal) {
362
+ expressions.push(t.stringLiteral("}"));
363
+ }
364
+ return expressionForSerialStringConcatenation(scope, expressions);
365
+ }
366
+ function convertObjectExpressionToTs(scope, node) {
367
+ return t.objectExpression(Object.entries(node.meta.items)
368
+ .map(([key, value]) => {
369
+ const valueChild = (0, syntax_tree_1.getChildWithValue)(node, value);
370
+ if (!valueChild) {
371
+ utils_1.logger.error(`Unable to value for object key '${key}': ${value}`);
372
+ return null;
373
+ }
374
+ return t.objectProperty(t.identifier(key), convertTFExpressionAstToTs(scope, valueChild));
375
+ })
376
+ .filter((s) => s !== null));
377
+ }
378
+ function convertFunctionCallExpressionToTs(scope, node) {
379
+ const functionName = node.meta.name;
380
+ const mapping = functions_1.functionsMap[functionName];
381
+ if (!mapping) {
382
+ utils_1.logger.error(`Unknown function ${functionName} encountered. ${utils_1.leaveCommentText}`);
383
+ const argumentExpressions = node.children.map((child) => convertTFExpressionAstToTs(scope, child));
384
+ return t.callExpression(t.identifier(functionName), argumentExpressions);
385
+ }
386
+ let transformedNode = mapping.transformer
387
+ ? mapping.transformer(node)
388
+ : node;
389
+ const argumentExpressions = transformedNode.children.map((child) => convertTFExpressionAstToTs(scope, child));
390
+ scope.importables.push({
391
+ constructName: "Fn",
392
+ provider: "cdktn",
393
+ });
394
+ const callee = t.memberExpression(t.identifier("Fn"), t.identifier(mapping.name));
395
+ if (mapping.parameters.length > 0 &&
396
+ mapping.parameters[mapping.parameters.length - 1].variadic) {
397
+ const lastParameterType = mapping.parameters[mapping.parameters.length - 1].type;
398
+ const nonVariadicArguments = argumentExpressions.slice(0, mapping.parameters.length - 1);
399
+ const fnCallArguments = [
400
+ ...nonVariadicArguments.map((argExpr, index) => (0, coerceType_1.coerceType)(scope, argExpr, (0, coerceType_1.findExpressionType)(scope, argExpr), mapping.parameters[index].type)),
401
+ t.arrayExpression(argumentExpressions
402
+ .slice(mapping.parameters.length - 1)
403
+ .map((argExpr) => (0, coerceType_1.coerceType)(scope, argExpr, (0, coerceType_1.findExpressionType)(scope, argExpr), lastParameterType))),
404
+ ];
405
+ return t.callExpression(callee, fnCallArguments);
406
+ }
407
+ return t.callExpression(callee, argumentExpressions.map((argExpr, index) => (0, coerceType_1.coerceType)(scope, argExpr, (0, coerceType_1.findExpressionType)(scope, argExpr), mapping.parameters[index].type)));
408
+ }
409
+ function convertIndexExpressionToTs(scope, node) {
410
+ const collectionExpressionChild = hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.collectionExpression);
411
+ const keyExpressionChild = hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.keyExpression);
412
+ const collectionExpression = convertTFExpressionAstToTs(scope, collectionExpressionChild);
413
+ const keyExpression = convertTFExpressionAstToTs(scope, keyExpressionChild);
414
+ scope.importables.push({
415
+ constructName: "Fn",
416
+ provider: "cdktn",
417
+ });
418
+ const callee = t.memberExpression(t.identifier("Fn"), t.identifier("lookupNested"));
419
+ return t.callExpression(callee, [
420
+ collectionExpression,
421
+ t.arrayExpression([keyExpression]),
422
+ ]);
423
+ }
424
+ function convertSplatExpressionToTs(scope, node) {
425
+ const sourceExpressionChild = hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.sourceExpression);
426
+ let sourceExpression = convertTFExpressionAstToTs(scope, sourceExpressionChild);
427
+ // We don't convert the relative expression because everything after the splat is going to be
428
+ // a string
429
+ let relativeExpression = node.meta.eachExpression.startsWith(node.meta.anonSymbolExpression)
430
+ ? node.meta.eachExpression.slice(node.meta.anonSymbolExpression.length)
431
+ : node.meta.eachExpression;
432
+ const segments = relativeExpression.split(/\.|\[|\]/).filter((s) => s);
433
+ scope.importables.push({
434
+ constructName: "Fn",
435
+ provider: "cdktn",
436
+ });
437
+ const callee = t.memberExpression(t.identifier("Fn"), t.identifier("lookupNested"));
438
+ return t.callExpression(callee, [
439
+ sourceExpression,
440
+ t.arrayExpression([
441
+ // we don't need to use the anonSymbolExpression here because
442
+ // it only changes between .* and [*] which we don't care about
443
+ t.stringLiteral("*"),
444
+ ...segments.map(t.stringLiteral),
445
+ ]),
446
+ ]);
447
+ }
448
+ function convertConditionalExpressionToTs(scope, node) {
449
+ const conditionChild = hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.conditionExpression);
450
+ let condition = convertTFExpressionAstToTs(scope, conditionChild);
451
+ if (t.isIdentifier(condition) && canUseFqn(conditionChild)) {
452
+ // We have a resource or data source here, which we would need to
453
+ // reference using fqn
454
+ condition = t.memberExpression(condition, t.identifier("fqn"));
455
+ }
456
+ const trueExpression = convertTFExpressionAstToTs(scope, hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.trueExpression));
457
+ const falseExpression = convertTFExpressionAstToTs(scope, hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.falseExpression));
458
+ scope.importables.push({
459
+ constructName: "conditional",
460
+ provider: "cdktn",
461
+ });
462
+ return t.callExpression(t.identifier("conditional"), [
463
+ condition,
464
+ trueExpression,
465
+ falseExpression,
466
+ ]);
467
+ }
468
+ function convertTupleExpressionToTs(scope, node) {
469
+ const expressions = node.children.map((child) => convertTFExpressionAstToTs(scope, child));
470
+ return t.arrayExpression(expressions);
471
+ }
472
+ function convertRelativeTraversalExpressionToTs(scope, node) {
473
+ const segments = node.meta.traversal;
474
+ // The left hand side / source of a relative traversal is not a proper
475
+ // object / resource / data thing that is being referenced
476
+ const source = convertTFExpressionAstToTs(scope, hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.sourceExpression));
477
+ scope.importables.push({
478
+ constructName: "Fn",
479
+ provider: "cdktn",
480
+ });
481
+ const callee = t.memberExpression(t.identifier("Fn"), t.identifier("lookupNested"));
482
+ return t.callExpression(callee, [
483
+ source,
484
+ t.arrayExpression(segments.map((s) => t.stringLiteral(s.segment))),
485
+ ]);
486
+ }
487
+ function convertForExpressionToTs(scope, node) {
488
+ const collectionChild = hcl2json_2.TFExpressionSyntaxTree.getChildWithValue(node, node.meta.collectionExpression);
489
+ let collectionExpression = convertTFExpressionAstToTs(scope, collectionChild);
490
+ if (t.isIdentifier(collectionExpression) && canUseFqn(collectionChild)) {
491
+ // We have a resource or data source here, which we would need to
492
+ // reference using fqn
493
+ collectionExpression = t.memberExpression(collectionExpression, t.identifier("fqn"));
494
+ }
495
+ const collectionRequiresWrapping = !t.isStringLiteral(collectionExpression);
496
+ const expressions = [];
497
+ const conditionBody = node.meta.keyVar
498
+ ? `${node.meta.keyVar}, ${node.meta.valVar}`
499
+ : node.meta.valVar;
500
+ const openBrace = node.meta.openRangeValue;
501
+ const closeBrace = node.meta.closeRangeValue;
502
+ const grouped = node.meta.groupedValue ? "..." : "";
503
+ const valueExpression = `${node.meta.valueExpression}${grouped}`;
504
+ const prefix = `\${${openBrace} for ${conditionBody} in `;
505
+ const keyValue = node.meta.keyExpression
506
+ ? ` : ${node.meta.keyExpression} => ${valueExpression}`
507
+ : ` : ${valueExpression}`;
508
+ const conditional = node.meta.conditionalExpression;
509
+ const suffix = `${keyValue}${conditional ? ` if ${conditional}` : ""}${closeBrace}}`;
510
+ expressions.push(t.stringLiteral(prefix));
511
+ if (collectionRequiresWrapping) {
512
+ expressions.push(t.stringLiteral("${"));
513
+ }
514
+ expressions.push(collectionExpression);
515
+ if (collectionRequiresWrapping) {
516
+ expressions.push(t.stringLiteral("}"));
517
+ }
518
+ expressions.push(t.stringLiteral(suffix));
519
+ return expressionForSerialStringConcatenation(scope, expressions);
520
+ }
521
+ function convertTFExpressionAstToTs(scope, node) {
522
+ if (hcl2json_2.TFExpressionSyntaxTree.isLiteralValueExpression(node)) {
523
+ return convertLiteralValueExpressionToTs(scope, node);
524
+ }
525
+ if (hcl2json_2.TFExpressionSyntaxTree.isScopeTraversalExpression(node)) {
526
+ return convertScopeTraversalExpressionToTs(scope, node);
527
+ }
528
+ if (hcl2json_2.TFExpressionSyntaxTree.isUnaryOpExpression(node)) {
529
+ return convertUnaryOpExpressionToTs(scope, node);
530
+ }
531
+ if (hcl2json_2.TFExpressionSyntaxTree.isBinaryOpExpression(node)) {
532
+ return convertBinaryOpExpressionToTs(scope, node);
533
+ }
534
+ if (hcl2json_2.TFExpressionSyntaxTree.isTemplateExpression(node) || hcl2json_2.TFExpressionSyntaxTree.isTemplateWrapExpression(node)) {
535
+ return convertTemplateExpressionToTs(scope, node);
536
+ }
537
+ if (hcl2json_2.TFExpressionSyntaxTree.isObjectExpression(node)) {
538
+ return convertObjectExpressionToTs(scope, node);
539
+ }
540
+ if (hcl2json_2.TFExpressionSyntaxTree.isFunctionCallExpression(node)) {
541
+ return convertFunctionCallExpressionToTs(scope, node);
542
+ }
543
+ if (hcl2json_2.TFExpressionSyntaxTree.isIndexExpression(node)) {
544
+ return convertIndexExpressionToTs(scope, node);
545
+ }
546
+ if (hcl2json_2.TFExpressionSyntaxTree.isSplatExpression(node)) {
547
+ return convertSplatExpressionToTs(scope, node);
548
+ }
549
+ if (hcl2json_2.TFExpressionSyntaxTree.isConditionalExpression(node)) {
550
+ return convertConditionalExpressionToTs(scope, node);
551
+ }
552
+ if (hcl2json_2.TFExpressionSyntaxTree.isTupleExpression(node)) {
553
+ return convertTupleExpressionToTs(scope, node);
554
+ }
555
+ if (hcl2json_2.TFExpressionSyntaxTree.isRelativeTraversalExpression(node)) {
556
+ return convertRelativeTraversalExpressionToTs(scope, node);
557
+ }
558
+ if (hcl2json_2.TFExpressionSyntaxTree.isForExpression(node)) {
559
+ return convertForExpressionToTs(scope, node);
560
+ }
561
+ return t.stringLiteral("");
562
+ }
563
+ async function expressionAst(input) {
564
+ const { wrap, wrapOffset } = (0, hcl2json_2.wrapTerraformExpression)(input);
565
+ const ast = await (0, hcl2json_1.getExpressionAst)("main.tf", wrap);
566
+ if (!ast) {
567
+ throw new Error(`Unable to parse terraform expression: ${input}`);
568
+ }
569
+ if (wrapOffset != 0 && hcl2json_2.TFExpressionSyntaxTree.isTemplateWrapExpression(ast)) {
570
+ return ast.children[0];
571
+ }
572
+ return ast;
573
+ }
574
+ exports.expressionAst = expressionAst;
575
+ async function convertTerraformExpressionToTs(scope, input, targetType) {
576
+ utils_1.logger.debug(`convertTerraformExpressionToTs(${input})`);
577
+ const tsExpression = convertTFExpressionAstToTs(scope, await expressionAst(input));
578
+ return (0, coerceType_1.coerceType)(scope, tsExpression, (0, coerceType_1.findExpressionType)(scope, tsExpression), targetType());
579
+ }
580
+ exports.convertTerraformExpressionToTs = convertTerraformExpressionToTs;
581
+ async function extractIteratorVariablesFromExpression(input) {
582
+ const possibleVariableSpots = await (0, hcl2json_1.getReferencesInExpression)("main.tf", input);
583
+ return possibleVariableSpots
584
+ .filter((spot) => spot.value.startsWith("each."))
585
+ .map((spot) => ({
586
+ start: spot.startPosition,
587
+ end: spot.endPosition,
588
+ value: spot.value,
589
+ }));
590
+ }
591
+ exports.extractIteratorVariablesFromExpression = extractIteratorVariablesFromExpression;
592
+ function dynamicVariableToAst(scope, node, iteratorName, block = "each") {
593
+ if (iteratorName === "dynamic-block") {
594
+ return expressionForSerialStringConcatenation(scope, [
595
+ t.stringLiteral("${"),
596
+ t.stringLiteral(block),
597
+ t.stringLiteral("}"),
598
+ ]);
599
+ }
600
+ if (node.meta.value === `${block}.key`) {
601
+ return t.memberExpression(t.identifier(iteratorName), t.identifier("key"));
602
+ }
603
+ if (node.meta.value === `${block}.value`) {
604
+ return t.memberExpression(t.identifier(iteratorName), t.identifier("value"));
605
+ }
606
+ if (block === "count" && node.meta.value === `${block}.index`) {
607
+ return t.memberExpression(t.identifier(iteratorName), t.identifier("index"));
608
+ }
609
+ const segments = node.meta.traversal;
610
+ if (segments.length > 2 &&
611
+ segments[0].segment === block &&
612
+ segments[1].segment === "value") {
613
+ const segmentsAfterEachValue = segments.slice(2);
614
+ scope.importables.push({
615
+ constructName: "Fn",
616
+ provider: "cdktn",
617
+ });
618
+ const callee = t.memberExpression(t.identifier("Fn"), t.identifier("lookupNested"));
619
+ return t.callExpression(callee, [
620
+ t.memberExpression(t.identifier(iteratorName), t.identifier("value")),
621
+ t.arrayExpression(segmentsAfterEachValue.map((part) => {
622
+ if (part.type === "nameTraversal") {
623
+ return t.stringLiteral(part.segment);
624
+ }
625
+ else {
626
+ return t.stringLiteral(`[${part.segment}]`);
627
+ }
628
+ })),
629
+ ]);
630
+ }
631
+ throw new Error(`Can not create AST for iterator variable of '${node.meta.value}'`);
632
+ }
633
+ exports.dynamicVariableToAst = dynamicVariableToAst;
634
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"expressions.js","sourceRoot":"","sources":["expressions.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright (c) HashiCorp, Inc\n// SPDX-License-Identifier: MPL-2.0\nimport * as t from \"@babel/types\";\nimport template from \"@babel/template\";\nimport { camelCase, leaveCommentText, logger } from \"./utils\";\nimport {\n  IteratorVariableReference,\n  ProgramScope,\n  ResourceScope,\n} from \"./types\";\nimport { getReferencesInExpression, getExpressionAst } from \"@cdktn/hcl2json\";\nimport {\n  TFExpressionSyntaxTree as tex,\n  wrapTerraformExpression,\n} from \"@cdktn/hcl2json\";\nimport { functionsMap } from \"./function-bindings/functions\";\nimport { coerceType, findExpressionType } from \"./coerceType\";\nimport { AttributeType } from \"@cdktn/commons\";\nimport { getTypeAtPath } from \"./terraformSchema\";\nimport { containsReference } from \"./references\";\nimport { variableName } from \"./variables\";\nimport { getChildWithValue } from \"@cdktn/hcl2json/lib/syntax-tree\";\n\nconst tfBinaryOperatorsToCdktf = {\n  logicalOr: \"or\",\n  logicalAnd: \"and\",\n  greaterThan: \"gt\",\n  greaterThanOrEqual: \"gte\",\n  lessThan: \"lt\",\n  lessThanOrEqual: \"lte\",\n  equal: \"eq\",\n  notEqual: \"neq\",\n  add: \"add\",\n  subtract: \"sub\",\n  multiply: \"mul\",\n  divide: \"div\",\n  modulo: \"mod\",\n};\n\nconst tfUnaryOperatorsToCdktf = {\n  logicalNot: \"not\",\n  negate: \"negate\",\n};\n\ntype supportedBinaryOperators = keyof typeof tfBinaryOperatorsToCdktf;\ntype supportedUnaryOperators = keyof typeof tfUnaryOperatorsToCdktf;\n\nfunction traversalPartsToString(\n  traversals: tex.TerraformTraversalPart[],\n  asSuffix = false,\n) {\n  let seed = \"\";\n  if (asSuffix && tex.isNameTraversalPart(traversals[0])) {\n    seed = \".\";\n  }\n  return traversals.reduce((acc, part) => {\n    if (part.type === \"nameTraversal\") {\n      if (acc === seed) {\n        return `${acc}${part.segment}`;\n      }\n      return `${acc}.${part.segment}`;\n    }\n    return `${acc}[${part.segment}]`;\n  }, seed);\n}\n\nfunction canUseFqn(expression: tex.ExpressionType) {\n  if (!tex.isScopeTraversalExpression(expression)) {\n    return false;\n  }\n\n  const rootSegment = expression.meta.traversal[0].segment;\n\n  return ![\"var\", \"local\"].includes(rootSegment);\n}\n\nfunction traversalToVariableName(\n  scope: ProgramScope,\n  node: tex.ExpressionType,\n) {\n  if (!tex.isScopeTraversalExpression(node)) {\n    logger.error(\n      `Unexpected expression type ${node.type} with value ${node.meta.value} passed to convert to a variable. \n        ${leaveCommentText}`,\n    );\n    return \"\";\n  }\n\n  const segments = node.meta.traversal;\n  if (segments.length === 1) {\n    return node.meta.fullAccessor;\n  }\n  const rootSegment = segments[0].segment;\n  const resource =\n    rootSegment === \"data\"\n      ? `${segments[0].segment}.${segments[1].segment}`\n      : rootSegment;\n  const name =\n    rootSegment === \"data\" ? segments[2].segment : segments[1].segment;\n\n  return variableName(scope, resource, name);\n}\n\nfunction expressionForSerialStringConcatenation(\n  scope: ResourceScope,\n  nodes: t.Expression[],\n) {\n  const reducedNodes = nodes.reduce((acc, node) => {\n    const prev = acc[acc.length - 1];\n    if (!prev) return [node];\n\n    if (t.isStringLiteral(prev) && t.isStringLiteral(node)) {\n      acc.pop();\n      acc.push(t.stringLiteral(prev.value + node.value));\n      return acc;\n    }\n\n    acc.push(node);\n    return acc;\n  }, [] as t.Expression[]);\n\n  return reducedNodes.reduce(\n    (acc: t.Expression | undefined, node: t.Expression) => {\n      if (!acc) {\n        return node;\n      }\n\n      // wrap access to dynamic blocks in Token.asString() as they return a Lazy\n      // for .key and .value which can't be concatenated in languages like Python\n      // because JSII currently has no support for the toString() method in\n      // languages other than TypeScript: https://github.com/aws/jsii/issues/380\n      // example: dynamic_iterator0.key / dynamic_iterator0.value\n      if (\n        t.isMemberExpression(node) &&\n        t.isIdentifier(node.object) &&\n        Object.values(scope.scopedVariables || {}).includes(node.object.name) &&\n        t.isIdentifier(node.property) &&\n        [\"key\", \"value\"].includes(node.property.name)\n      ) {\n        node = coerceType(scope, node, \"dynamic\", \"string\");\n      }\n\n      return t.binaryExpression(\"+\", acc as t.Expression, node);\n    },\n  );\n}\n\nfunction getTfResourcePathFromNode(node: tex.ScopeTraversalExpression) {\n  const segments = node.meta.traversal;\n  let resource = segments[0].segment;\n  let result = [];\n  let attributes = [];\n\n  if (segments[0].segment === \"data\") {\n    result.push(segments[0].segment);\n    resource = segments[1].segment;\n    attributes = segments.slice(3); // we want to skip the variable name\n  } else {\n    attributes = segments.slice(2); // we want to skip the variable name\n  }\n\n  const [provider, ...resourceNameFragments] = resource.split(\"_\");\n\n  // Hack: This happens in the case of `external` provider\n  // where the data source does not have a provider name prefix\n  if (resourceNameFragments.length === 0) {\n    resourceNameFragments.push(provider);\n  }\n\n  result.push(provider);\n  result.push(resourceNameFragments.join(\"_\"));\n  result = [\n    ...result,\n    ...attributes.map((seg) => {\n      if (tex.isIndexTraversalPart(seg)) {\n        return `[${seg.segment}]`;\n      }\n      return seg.segment;\n    }),\n  ];\n\n  return result.join(\".\");\n}\n\nfunction convertLiteralValueExpressionToTs(\n  _scope: ResourceScope,\n  node: tex.LiteralValueExpression,\n) {\n  const literalType = node.meta.type;\n  if (literalType === \"number\") {\n    return t.numericLiteral(Number(node.meta.value));\n  }\n  if (literalType === \"bool\") {\n    return t.booleanLiteral(node.meta.value === \"true\" ? true : false);\n  }\n\n  return t.stringLiteral(node.meta.value);\n}\n\nfunction convertScopeTraversalExpressionToTs(\n  scope: ResourceScope,\n  node: tex.ScopeTraversalExpression,\n) {\n  const hasReference = containsReference(node);\n\n  const segments = node.meta.traversal;\n\n  if (segments[0].segment === \"each\" && scope.forEachIteratorName) {\n    return dynamicVariableToAst(scope, node, scope.forEachIteratorName);\n  }\n\n  if (segments[0].segment === \"count\" && scope.countIteratorName) {\n    return dynamicVariableToAst(scope, node, scope.countIteratorName, \"count\");\n  }\n\n  if (segments[0].segment === \"self\") {\n    scope.importables.push({\n      constructName: \"TerraformSelf\",\n      provider: \"cdktn\",\n    });\n\n    return t.callExpression(\n      t.memberExpression(t.identifier(\"TerraformSelf\"), t.identifier(\"getAny\")),\n\n      [t.stringLiteral(traversalPartsToString(segments.slice(1)))],\n    );\n  }\n\n  // setting.value, setting.value[1].id\n  const dynamicBlock = scope.scopedVariables?.[segments[0].segment];\n  if (dynamicBlock) {\n    if (dynamicBlock === \"dynamic-block\") {\n      return dynamicVariableToAst(\n        scope,\n        node,\n        dynamicBlock,\n        traversalPartsToString(segments),\n      );\n    }\n    return dynamicVariableToAst(scope, node, dynamicBlock, segments[0].segment);\n  }\n\n  // This may be a variable reference that we don't understand yet, so we wrap it in a template string\n  // for Terraform to handle\n  let varIdentifier: t.Expression = t.stringLiteral(\n    `\\${${node.meta.fullAccessor}}`,\n  );\n\n  if (hasReference) {\n    varIdentifier = t.identifier(\n      camelCase(traversalToVariableName(scope, node)),\n    );\n  }\n\n  if ([\"var\", \"local\"].includes(segments[0].segment)) {\n    const variableAccessor =\n      segments[0].segment === \"var\"\n        ? t.memberExpression(varIdentifier, t.identifier(\"value\"))\n        : varIdentifier;\n\n    if (segments.length > 2) {\n      scope.importables.push({\n        constructName: \"Fn\",\n        provider: \"cdktn\",\n      });\n      const callee = t.memberExpression(\n        t.identifier(\"Fn\"),\n        t.identifier(\"lookupNested\"),\n      );\n      return t.callExpression(callee, [\n        variableAccessor,\n        t.arrayExpression(\n          segments.slice(2).map((s) => t.stringLiteral(s.segment)),\n        ),\n      ]);\n    }\n\n    return variableAccessor;\n  }\n\n  if (!hasReference || scope.withinOverrideExpression) {\n    return varIdentifier;\n  }\n\n  const rootSegment = segments[0].segment;\n  const attributeIndex = rootSegment === \"data\" ? 3 : 2;\n  const attributeSegments = segments.slice(attributeIndex);\n  const numericAccessorIndex = attributeSegments.findIndex((seg) =>\n    tex.isIndexTraversalPart(seg),\n  );\n  let minAccessorIndex = numericAccessorIndex;\n  let mapAccessorIndex = -1;\n  if (numericAccessorIndex === -1) {\n    // only do this if we have to, if we already have a\n    // numeric accessor, we don't have to do this additional work\n    const resourcePath = getTfResourcePathFromNode(node);\n    let usingSubPathType = false;\n    let parts = resourcePath.split(\".\").filter((p) => p !== \"\");\n    const minParts = attributeIndex; // we need to stop before data.aws.resource_name or aws.resource_name\n    const originalParts = parts.length;\n    let hasMapAccessor = false;\n    while (parts.length >= minParts) {\n      const type = getTypeAtPath(scope.providerSchema, parts.join(\".\"));\n      if (type !== null) {\n        if (Array.isArray(type) && type[0] === \"map\" && usingSubPathType) {\n          hasMapAccessor = true;\n          break;\n        }\n      }\n      parts.pop();\n      usingSubPathType = true;\n    }\n\n    if (hasMapAccessor) {\n      mapAccessorIndex = originalParts - parts.length - 1;\n      minAccessorIndex = mapAccessorIndex;\n    }\n  }\n\n  const needsPropertyAccess = minAccessorIndex >= 0;\n\n  const refSegments = needsPropertyAccess\n    ? attributeSegments.slice(0, minAccessorIndex)\n    : attributeSegments;\n  const nonRefSegments = needsPropertyAccess\n    ? attributeSegments.slice(minAccessorIndex)\n    : [];\n\n  const ref = refSegments.reduce(\n    (acc: t.Expression, seg, index) =>\n      t.memberExpression(\n        acc,\n        t.identifier(\n          index === 0 && rootSegment === \"module\"\n            ? camelCase(seg.segment + \"Output\")\n            : camelCase(seg.segment),\n        ),\n      ),\n    varIdentifier,\n  );\n\n  if (nonRefSegments.length === 0) {\n    return ref;\n  }\n\n  scope.importables.push({\n    constructName: \"Fn\",\n    provider: \"cdktn\",\n  });\n  const callee = t.memberExpression(\n    t.identifier(\"Fn\"),\n    t.identifier(\"lookupNested\"),\n  );\n  return t.callExpression(callee, [\n    ref,\n    t.arrayExpression(nonRefSegments.map((s) => t.stringLiteral(s.segment))),\n  ]);\n}\n\nfunction convertUnaryOpExpressionToTs(\n  scope: ResourceScope,\n  node: tex.UnaryOpExpression,\n) {\n  const operand = convertTFExpressionAstToTs(\n    scope,\n    tex.getChildWithValue(node, node.meta.valueExpression)!,\n  );\n\n  let fnName = node.meta.operator;\n  if (tfUnaryOperatorsToCdktf[fnName as supportedUnaryOperators]) {\n    fnName = tfUnaryOperatorsToCdktf[fnName as supportedUnaryOperators];\n  } else {\n    throw new Error(`Cannot convert unknown operator ${node.meta.operator}`);\n  }\n\n  scope.importables.push({\n    constructName: \"Op\",\n    provider: \"cdktn\",\n  });\n\n  const fn = t.memberExpression(t.identifier(\"Op\"), t.identifier(fnName));\n\n  return t.callExpression(fn, [operand]);\n}\n\nfunction convertBinaryOpExpressionToTs(\n  scope: ResourceScope,\n  node: tex.BinaryOpExpression,\n) {\n  const left = convertTFExpressionAstToTs(\n    scope,\n    tex.getChildWithValue(node, node.meta.lhsExpression)!,\n  );\n  const right = convertTFExpressionAstToTs(\n    scope,\n    tex.getChildWithValue(node, node.meta.rhsExpression)!,\n  );\n\n  let fnName = node.meta.operator;\n  if (tfBinaryOperatorsToCdktf[fnName as supportedBinaryOperators]) {\n    fnName = tfBinaryOperatorsToCdktf[fnName as supportedBinaryOperators];\n  } else {\n    throw new Error(`Cannot convert unknown operator ${node.meta.operator}`);\n  }\n\n  scope.importables.push({\n    constructName: \"Op\",\n    provider: \"cdktn\",\n  });\n\n  const fn = t.memberExpression(t.identifier(\"Op\"), t.identifier(fnName));\n  return t.callExpression(fn, [left, right]);\n}\n\nfunction convertTemplateExpressionToTs(\n  scope: ResourceScope,\n  node: tex.TemplateExpression | tex.TemplateWrapExpression,\n) {\n  const parts = node.children.map((child) => ({\n    node: child,\n    expr: convertTFExpressionAstToTs(scope, child),\n  }));\n\n  const lastPart = parts[parts.length - 1];\n  if (t.isStringLiteral(lastPart.expr) && lastPart.expr.value === \"\\n\") {\n    // This is a bit of a hack, but the trailing newline we add due to\n    // heredocs looks ugly and unnecessary in the generated code, so we\n    // try to remove it\n    parts.pop();\n  }\n\n  if (parts.length === 0) {\n    return t.stringLiteral(node.meta.value);\n  }\n\n  if (parts.length === 1) {\n    return parts[0].expr;\n  }\n\n  let isScopedTraversal = false;\n  let expressions: t.Expression[] = [];\n  for (const { node, expr } of parts) {\n    if (\n      tex.isScopeTraversalExpression(node) &&\n      !t.isStringLiteral(expr) &&\n      !t.isCallExpression(expr)\n    ) {\n      expressions.push(t.stringLiteral(\"${\"));\n      isScopedTraversal = true;\n    } else if (\n      // we should ideally be doing type coercion more\n      // carefully here, because it may not always be needed\n      t.isCallExpression(expr)\n    ) {\n      scope.importables.push({\n        constructName: \"Token\",\n        provider: \"cdktn\",\n      });\n\n      expressions.push(\n        template.expression(`Token.asString(%%expr%%)`)({\n          expr,\n        }) as t.Expression,\n      );\n      continue;\n    } else {\n      if (isScopedTraversal) {\n        expressions.push(t.stringLiteral(\"}\"));\n        isScopedTraversal = false;\n      }\n    }\n    expressions.push(expr);\n  }\n\n  if (isScopedTraversal) {\n    expressions.push(t.stringLiteral(\"}\"));\n  }\n\n  return expressionForSerialStringConcatenation(scope, expressions);\n}\n\nfunction convertObjectExpressionToTs(\n  scope: ResourceScope,\n  node: tex.ObjectExpression,\n) {\n  return t.objectExpression(\n    Object.entries(node.meta.items)\n      .map(([key, value]) => {\n        const valueChild = getChildWithValue(node, value);\n        if (!valueChild) {\n          logger.error(`Unable to value for object key '${key}': ${value}`);\n          return null;\n        }\n\n        return t.objectProperty(\n          t.identifier(key),\n          convertTFExpressionAstToTs(scope, valueChild),\n        );\n      })\n      .filter((s) => s !== null) as t.ObjectProperty[],\n  );\n}\n\nfunction convertFunctionCallExpressionToTs(\n  scope: ResourceScope,\n  node: tex.FunctionCallExpression,\n) {\n  const functionName = node.meta.name;\n  const mapping = functionsMap[functionName];\n\n  if (!mapping) {\n    logger.error(\n      `Unknown function ${functionName} encountered. ${leaveCommentText}`,\n    );\n    const argumentExpressions = node.children.map((child) =>\n      convertTFExpressionAstToTs(scope, child),\n    );\n\n    return t.callExpression(t.identifier(functionName), argumentExpressions);\n  }\n\n  let transformedNode: tex.FunctionCallExpression = mapping.transformer\n    ? mapping.transformer(node)\n    : node;\n\n  const argumentExpressions = transformedNode.children.map((child) =>\n    convertTFExpressionAstToTs(scope, child),\n  );\n\n  scope.importables.push({\n    constructName: \"Fn\",\n    provider: \"cdktn\",\n  });\n\n  const callee = t.memberExpression(\n    t.identifier(\"Fn\"),\n    t.identifier(mapping.name),\n  );\n\n  if (\n    mapping.parameters.length > 0 &&\n    mapping.parameters[mapping.parameters.length - 1].variadic\n  ) {\n    const lastParameterType =\n      mapping.parameters[mapping.parameters.length - 1].type;\n    const nonVariadicArguments = argumentExpressions.slice(\n      0,\n      mapping.parameters.length - 1,\n    );\n\n    const fnCallArguments = [\n      ...nonVariadicArguments.map((argExpr, index) =>\n        coerceType(\n          scope,\n          argExpr,\n          findExpressionType(scope, argExpr),\n          mapping.parameters[index].type,\n        ),\n      ),\n\n      t.arrayExpression(\n        argumentExpressions\n          .slice(mapping.parameters.length - 1)\n          .map((argExpr) =>\n            coerceType(\n              scope,\n              argExpr,\n              findExpressionType(scope, argExpr),\n              lastParameterType,\n            ),\n          ),\n      ),\n    ];\n\n    return t.callExpression(callee, fnCallArguments);\n  }\n\n  return t.callExpression(\n    callee,\n    argumentExpressions.map((argExpr, index) =>\n      coerceType(\n        scope,\n        argExpr,\n        findExpressionType(scope, argExpr),\n        mapping.parameters[index].type,\n      ),\n    ),\n  );\n}\n\nfunction convertIndexExpressionToTs(\n  scope: ResourceScope,\n  node: tex.IndexExpression,\n) {\n  const collectionExpressionChild = tex.getChildWithValue(\n    node,\n    node.meta.collectionExpression,\n  );\n  const keyExpressionChild = tex.getChildWithValue(\n    node,\n    node.meta.keyExpression,\n  );\n\n  const collectionExpression = convertTFExpressionAstToTs(\n    scope,\n    collectionExpressionChild!,\n  );\n  const keyExpression = convertTFExpressionAstToTs(scope, keyExpressionChild!);\n\n  scope.importables.push({\n    constructName: \"Fn\",\n    provider: \"cdktn\",\n  });\n  const callee = t.memberExpression(\n    t.identifier(\"Fn\"),\n    t.identifier(\"lookupNested\"),\n  );\n  return t.callExpression(callee, [\n    collectionExpression,\n    t.arrayExpression([keyExpression]),\n  ]);\n}\n\nfunction convertSplatExpressionToTs(\n  scope: ResourceScope,\n  node: tex.SplatExpression,\n) {\n  const sourceExpressionChild = tex.getChildWithValue(\n    node,\n    node.meta.sourceExpression,\n  )!;\n  let sourceExpression = convertTFExpressionAstToTs(\n    scope,\n    sourceExpressionChild,\n  );\n\n  // We don't convert the relative expression because everything after the splat is going to be\n  // a string\n  let relativeExpression = node.meta.eachExpression.startsWith(\n    node.meta.anonSymbolExpression,\n  )\n    ? node.meta.eachExpression.slice(node.meta.anonSymbolExpression.length)\n    : node.meta.eachExpression;\n\n  const segments = relativeExpression.split(/\\.|\\[|\\]/).filter((s) => s);\n  scope.importables.push({\n    constructName: \"Fn\",\n    provider: \"cdktn\",\n  });\n  const callee = t.memberExpression(\n    t.identifier(\"Fn\"),\n    t.identifier(\"lookupNested\"),\n  );\n\n  return t.callExpression(callee, [\n    sourceExpression,\n    t.arrayExpression([\n      // we don't need to use the anonSymbolExpression here because\n      // it only changes between .* and [*] which we don't care about\n      t.stringLiteral(\"*\"),\n      ...segments.map(t.stringLiteral),\n    ]),\n  ]);\n}\n\nfunction convertConditionalExpressionToTs(\n  scope: ResourceScope,\n  node: tex.ConditionalExpression,\n) {\n  const conditionChild = tex.getChildWithValue(\n    node,\n    node.meta.conditionExpression,\n  )!;\n  let condition = convertTFExpressionAstToTs(scope, conditionChild);\n  if (t.isIdentifier(condition) && canUseFqn(conditionChild)) {\n    // We have a resource or data source here, which we would need to\n    // reference using fqn\n    condition = t.memberExpression(condition, t.identifier(\"fqn\"));\n  }\n\n  const trueExpression = convertTFExpressionAstToTs(\n    scope,\n    tex.getChildWithValue(node, node.meta.trueExpression)!,\n  );\n\n  const falseExpression = convertTFExpressionAstToTs(\n    scope,\n    tex.getChildWithValue(node, node.meta.falseExpression)!,\n  );\n\n  scope.importables.push({\n    constructName: \"conditional\",\n    provider: \"cdktn\",\n  });\n\n  return t.callExpression(t.identifier(\"conditional\"), [\n    condition,\n    trueExpression,\n    falseExpression,\n  ]);\n}\n\nfunction convertTupleExpressionToTs(\n  scope: ResourceScope,\n  node: tex.TupleExpression,\n) {\n  const expressions = node.children.map((child) =>\n    convertTFExpressionAstToTs(scope, child),\n  );\n\n  return t.arrayExpression(expressions);\n}\n\nfunction convertRelativeTraversalExpressionToTs(\n  scope: ResourceScope,\n  node: tex.RelativeTraversalExpression,\n) {\n  const segments = node.meta.traversal;\n\n  // The left hand side / source of a relative traversal is not a proper\n  // object / resource / data thing that is being referenced\n  const source = convertTFExpressionAstToTs(\n    scope,\n    tex.getChildWithValue(node, node.meta.sourceExpression)!,\n  );\n\n  scope.importables.push({\n    constructName: \"Fn\",\n    provider: \"cdktn\",\n  });\n  const callee = t.memberExpression(\n    t.identifier(\"Fn\"),\n    t.identifier(\"lookupNested\"),\n  );\n\n  return t.callExpression(callee, [\n    source,\n    t.arrayExpression(segments.map((s) => t.stringLiteral(s.segment))),\n  ]);\n}\n\nfunction convertForExpressionToTs(\n  scope: ResourceScope,\n  node: tex.ForExpression,\n) {\n  const collectionChild = tex.getChildWithValue(\n    node,\n    node.meta.collectionExpression,\n  )!;\n\n  let collectionExpression = convertTFExpressionAstToTs(scope, collectionChild);\n\n  if (t.isIdentifier(collectionExpression) && canUseFqn(collectionChild)) {\n    // We have a resource or data source here, which we would need to\n    // reference using fqn\n    collectionExpression = t.memberExpression(\n      collectionExpression,\n      t.identifier(\"fqn\"),\n    );\n  }\n\n  const collectionRequiresWrapping = !t.isStringLiteral(collectionExpression);\n  const expressions = [];\n  const conditionBody = node.meta.keyVar\n    ? `${node.meta.keyVar}, ${node.meta.valVar}`\n    : node.meta.valVar;\n\n  const openBrace = node.meta.openRangeValue;\n  const closeBrace = node.meta.closeRangeValue;\n  const grouped = node.meta.groupedValue ? \"...\" : \"\";\n  const valueExpression = `${node.meta.valueExpression}${grouped}`;\n\n  const prefix = `\\${${openBrace} for ${conditionBody} in `;\n  const keyValue = node.meta.keyExpression\n    ? ` : ${node.meta.keyExpression} => ${valueExpression}`\n    : ` : ${valueExpression}`;\n  const conditional = node.meta.conditionalExpression;\n  const suffix = `${keyValue}${\n    conditional ? ` if ${conditional}` : \"\"\n  }${closeBrace}}`;\n\n  expressions.push(t.stringLiteral(prefix));\n  if (collectionRequiresWrapping) {\n    expressions.push(t.stringLiteral(\"${\"));\n  }\n  expressions.push(collectionExpression);\n  if (collectionRequiresWrapping) {\n    expressions.push(t.stringLiteral(\"}\"));\n  }\n  expressions.push(t.stringLiteral(suffix));\n\n  return expressionForSerialStringConcatenation(scope, expressions);\n}\n\nfunction convertTFExpressionAstToTs(\n  scope: ResourceScope,\n  node: tex.ExpressionType,\n): t.Expression {\n  if (tex.isLiteralValueExpression(node)) {\n    return convertLiteralValueExpressionToTs(scope, node);\n  }\n\n  if (tex.isScopeTraversalExpression(node)) {\n    return convertScopeTraversalExpressionToTs(scope, node);\n  }\n\n  if (tex.isUnaryOpExpression(node)) {\n    return convertUnaryOpExpressionToTs(scope, node);\n  }\n\n  if (tex.isBinaryOpExpression(node)) {\n    return convertBinaryOpExpressionToTs(scope, node);\n  }\n\n  if (tex.isTemplateExpression(node) || tex.isTemplateWrapExpression(node)) {\n    return convertTemplateExpressionToTs(scope, node);\n  }\n\n  if (tex.isObjectExpression(node)) {\n    return convertObjectExpressionToTs(scope, node);\n  }\n\n  if (tex.isFunctionCallExpression(node)) {\n    return convertFunctionCallExpressionToTs(scope, node);\n  }\n\n  if (tex.isIndexExpression(node)) {\n    return convertIndexExpressionToTs(scope, node);\n  }\n\n  if (tex.isSplatExpression(node)) {\n    return convertSplatExpressionToTs(scope, node);\n  }\n\n  if (tex.isConditionalExpression(node)) {\n    return convertConditionalExpressionToTs(scope, node);\n  }\n\n  if (tex.isTupleExpression(node)) {\n    return convertTupleExpressionToTs(scope, node);\n  }\n\n  if (tex.isRelativeTraversalExpression(node)) {\n    return convertRelativeTraversalExpressionToTs(scope, node);\n  }\n\n  if (tex.isForExpression(node)) {\n    return convertForExpressionToTs(scope, node);\n  }\n\n  return t.stringLiteral(\"\");\n}\n\nexport async function expressionAst(\n  input: string,\n): Promise<tex.ExpressionType> {\n  const { wrap, wrapOffset } = wrapTerraformExpression(input);\n  const ast = await getExpressionAst(\"main.tf\", wrap);\n\n  if (!ast) {\n    throw new Error(`Unable to parse terraform expression: ${input}`);\n  }\n\n  if (wrapOffset != 0 && tex.isTemplateWrapExpression(ast)) {\n    return ast.children[0];\n  }\n\n  return ast;\n}\n\nexport async function convertTerraformExpressionToTs(\n  scope: ResourceScope,\n  input: string,\n  targetType: () => AttributeType,\n): Promise<t.Expression> {\n  logger.debug(`convertTerraformExpressionToTs(${input})`);\n  const tsExpression = convertTFExpressionAstToTs(\n    scope,\n    await expressionAst(input),\n  );\n\n  return coerceType(\n    scope,\n    tsExpression,\n    findExpressionType(scope, tsExpression),\n    targetType(),\n  );\n}\n\nexport async function extractIteratorVariablesFromExpression(\n  input: string,\n): Promise<IteratorVariableReference[]> {\n  const possibleVariableSpots = await getReferencesInExpression(\n    \"main.tf\",\n    input,\n  );\n\n  return possibleVariableSpots\n    .filter((spot) => spot.value.startsWith(\"each.\"))\n    .map((spot) => ({\n      start: spot.startPosition,\n      end: spot.endPosition,\n      value: spot.value,\n    }));\n}\n\nexport function dynamicVariableToAst(\n  scope: ProgramScope,\n  node: tex.ScopeTraversalExpression,\n  iteratorName: string,\n  block: string = \"each\",\n): t.Expression {\n  if (iteratorName === \"dynamic-block\") {\n    return expressionForSerialStringConcatenation(scope, [\n      t.stringLiteral(\"${\"),\n      t.stringLiteral(block),\n      t.stringLiteral(\"}\"),\n    ]);\n  }\n  if (node.meta.value === `${block}.key`) {\n    return t.memberExpression(t.identifier(iteratorName), t.identifier(\"key\"));\n  }\n  if (node.meta.value === `${block}.value`) {\n    return t.memberExpression(\n      t.identifier(iteratorName),\n      t.identifier(\"value\"),\n    );\n  }\n\n  if (block === \"count\" && node.meta.value === `${block}.index`) {\n    return t.memberExpression(\n      t.identifier(iteratorName),\n      t.identifier(\"index\"),\n    );\n  }\n\n  const segments = node.meta.traversal;\n\n  if (\n    segments.length > 2 &&\n    segments[0].segment === block &&\n    segments[1].segment === \"value\"\n  ) {\n    const segmentsAfterEachValue = segments.slice(2);\n    scope.importables.push({\n      constructName: \"Fn\",\n      provider: \"cdktn\",\n    });\n    const callee = t.memberExpression(\n      t.identifier(\"Fn\"),\n      t.identifier(\"lookupNested\"),\n    );\n    return t.callExpression(callee, [\n      t.memberExpression(t.identifier(iteratorName), t.identifier(\"value\")),\n      t.arrayExpression(\n        segmentsAfterEachValue.map((part) => {\n          if (part.type === \"nameTraversal\") {\n            return t.stringLiteral(part.segment);\n          } else {\n            return t.stringLiteral(`[${part.segment}]`);\n          }\n        }),\n      ),\n    ]);\n  }\n\n  throw new Error(\n    `Can not create AST for iterator variable of '${node.meta.value}'`,\n  );\n}\n"]}