@cdktn/hcl2cdk 0.23.0-pre.8 → 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
package/build/index.js
ADDED
|
@@ -0,0 +1,364 @@
|
|
|
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.attributeNameToCdktfName = exports.isRegistryModule = exports.convertProject = exports.getTerraformConfigFromDir = exports.convert = exports.convertToTypescript = exports.parseProviderRequirements = exports.getParsedHcl = exports.CODE_MARKER = void 0;
|
|
30
|
+
// Copyright (c) HashiCorp, Inc
|
|
31
|
+
// SPDX-License-Identifier: MPL-2.0
|
|
32
|
+
const hcl2json_1 = require("@cdktn/hcl2json");
|
|
33
|
+
const provider_generator_1 = require("@cdktn/provider-generator");
|
|
34
|
+
Object.defineProperty(exports, "isRegistryModule", { enumerable: true, get: function () { return provider_generator_1.isRegistryModule; } });
|
|
35
|
+
const t = __importStar(require("@babel/types"));
|
|
36
|
+
const prettier_1 = __importDefault(require("prettier"));
|
|
37
|
+
const path = __importStar(require("path"));
|
|
38
|
+
const glob = __importStar(require("glob"));
|
|
39
|
+
const fs = __importStar(require("fs"));
|
|
40
|
+
const graphology_1 = require("graphology");
|
|
41
|
+
const rosetta = __importStar(require("jsii-rosetta"));
|
|
42
|
+
const schema_1 = require("./schema");
|
|
43
|
+
const references_1 = require("./references");
|
|
44
|
+
const generation_1 = require("./generation");
|
|
45
|
+
const iteration_1 = require("./iteration");
|
|
46
|
+
const provider_1 = require("./provider");
|
|
47
|
+
const utils_1 = require("./utils");
|
|
48
|
+
const generation_2 = require("./generation");
|
|
49
|
+
Object.defineProperty(exports, "attributeNameToCdktfName", { enumerable: true, get: function () { return generation_2.attributeNameToCdktfName; } });
|
|
50
|
+
const jsii_rosetta_workarounds_1 = require("./jsii-rosetta-workarounds");
|
|
51
|
+
const iteration_2 = require("./iteration");
|
|
52
|
+
exports.CODE_MARKER = "// define resources here";
|
|
53
|
+
async function getParsedHcl(hcl) {
|
|
54
|
+
utils_1.logger.debug(`Parsing HCL: ${hcl}`);
|
|
55
|
+
// Get the JSON representation of the HCL
|
|
56
|
+
let json;
|
|
57
|
+
try {
|
|
58
|
+
json = await (0, hcl2json_1.parse)("terraform.tf", hcl);
|
|
59
|
+
}
|
|
60
|
+
catch (err) {
|
|
61
|
+
utils_1.logger.error(`Failed to parse HCL: ${err}`);
|
|
62
|
+
throw new Error(`Error: Could not parse HCL, this means either that the HCL passed is invalid or that you found a bug. If the HCL seems valid, please file a bug under https://cdk.tf/bugs/new/convert`);
|
|
63
|
+
}
|
|
64
|
+
// Ensure the JSON representation matches the expected structure
|
|
65
|
+
let plan;
|
|
66
|
+
try {
|
|
67
|
+
plan = schema_1.schema.parse(json);
|
|
68
|
+
}
|
|
69
|
+
catch (err) {
|
|
70
|
+
throw new Error(`Error: HCL-JSON does not conform to schema. This is not expected, please file a bug under https://cdk.tf/bugs/new/convert
|
|
71
|
+
Please include this information:
|
|
72
|
+
${JSON.stringify(err.errors)}`);
|
|
73
|
+
}
|
|
74
|
+
return plan;
|
|
75
|
+
}
|
|
76
|
+
exports.getParsedHcl = getParsedHcl;
|
|
77
|
+
async function parseProviderRequirements(hcl) {
|
|
78
|
+
utils_1.logger.debug("Parsing provider requirements");
|
|
79
|
+
const plan = await getParsedHcl(hcl);
|
|
80
|
+
return (0, provider_1.getProviderRequirements)(plan);
|
|
81
|
+
}
|
|
82
|
+
exports.parseProviderRequirements = parseProviderRequirements;
|
|
83
|
+
async function convertToTypescript(hcl, providerSchema, codeContainer) {
|
|
84
|
+
var _a;
|
|
85
|
+
utils_1.logger.debug("Converting to typescript");
|
|
86
|
+
const plan = await getParsedHcl(hcl);
|
|
87
|
+
// Each key in the scope needs to be unique, therefore we save them in a set
|
|
88
|
+
// Each variable needs to be unique as well, we save them in a record so we can identify if two variables are the same
|
|
89
|
+
const scope = {
|
|
90
|
+
providerSchema,
|
|
91
|
+
providerGenerator: Object.keys(providerSchema.provider_schemas || {}).reduce((carry, fqpn) => {
|
|
92
|
+
const providerGenerator = new provider_generator_1.TerraformProviderGenerator(new provider_generator_1.CodeMaker(), providerSchema);
|
|
93
|
+
providerGenerator.buildResourceModels(fqpn); // can't use that type on the keys yet, since we are not on TS >=4.4 yet :sadcat:
|
|
94
|
+
return { ...carry, [fqpn]: providerGenerator };
|
|
95
|
+
}, {}),
|
|
96
|
+
constructs: new Set(),
|
|
97
|
+
variables: {},
|
|
98
|
+
hasTokenBasedTypeCoercion: false,
|
|
99
|
+
nodeIds: [],
|
|
100
|
+
importables: [],
|
|
101
|
+
topLevelConfig: {},
|
|
102
|
+
};
|
|
103
|
+
const graph = new graphology_1.DirectedGraph();
|
|
104
|
+
// Get all items in the JSON as a map of id to function that generates the AST
|
|
105
|
+
// We will use this to construct the nodes for a dependency graph
|
|
106
|
+
// We need to use a function here because the same node has different representation based on if it's referenced by another one
|
|
107
|
+
const nodeMap = {
|
|
108
|
+
...(0, iteration_1.forEachProvider)(scope, plan.provider, generation_1.provider),
|
|
109
|
+
...(0, iteration_1.forEachGlobal)(scope, "var", plan.variable, generation_1.variable),
|
|
110
|
+
// locals are a special case
|
|
111
|
+
...(0, iteration_1.forEachGlobal)(scope, "local", Array.isArray(plan.locals)
|
|
112
|
+
? plan.locals.reduce((carry, locals) => ({ ...carry, ...locals }), {})
|
|
113
|
+
: {}, generation_1.local),
|
|
114
|
+
...(0, iteration_1.forEachGlobal)(scope, "out", plan.output, generation_1.output),
|
|
115
|
+
...(0, iteration_1.forEachGlobal)(scope, "module", plan.module, generation_1.modules),
|
|
116
|
+
...(0, iteration_2.forEachImport)(scope, "import", plan.import, generation_1.imports),
|
|
117
|
+
...(0, iteration_1.forEachNamespaced)(scope, plan.resource, generation_1.resource),
|
|
118
|
+
...(0, iteration_1.forEachNamespaced)(scope, plan.data, generation_1.resource, "data"),
|
|
119
|
+
};
|
|
120
|
+
// Add all nodes to the dependency graph so we can detect if an edge is added for an unknown link
|
|
121
|
+
Object.entries(nodeMap).forEach(([key, value]) => {
|
|
122
|
+
utils_1.logger.debug(`Adding node '${key}' to graph`);
|
|
123
|
+
graph.addNode(key, value);
|
|
124
|
+
});
|
|
125
|
+
// Finding references becomes easier of the to be referenced ids are already known
|
|
126
|
+
const nodeIds = Object.keys(nodeMap);
|
|
127
|
+
scope.nodeIds = nodeIds;
|
|
128
|
+
async function addEdges(id, value) {
|
|
129
|
+
(await (0, references_1.findUsedReferences)(nodeIds, value)).forEach((ref) => {
|
|
130
|
+
if (!graph.hasDirectedEdge(ref.referencee.id, id) &&
|
|
131
|
+
graph.hasNode(ref.referencee.id) // in case the referencee is a dynamic variable
|
|
132
|
+
) {
|
|
133
|
+
if (!graph.hasNode(id)) {
|
|
134
|
+
throw new Error(`The dependency graph is expected to link from ${ref.referencee.id} to ${id} but ${id} does not exist.
|
|
135
|
+
These nodes exist: ${graph.nodes().join("\n")}`);
|
|
136
|
+
}
|
|
137
|
+
// The graph should have no self-references
|
|
138
|
+
if (id === ref.referencee.id) {
|
|
139
|
+
utils_1.logger.debug(`Skipping self-reference for ${id}`);
|
|
140
|
+
return;
|
|
141
|
+
}
|
|
142
|
+
utils_1.logger.debug(`Adding edge from ${ref.referencee.id} to ${id}`);
|
|
143
|
+
graph.addDirectedEdge(ref.referencee.id, id, { ref });
|
|
144
|
+
}
|
|
145
|
+
});
|
|
146
|
+
}
|
|
147
|
+
// We recursively inspect each resource value to find references to other values
|
|
148
|
+
// We add these to a dependency graph so that the programming code has the right order
|
|
149
|
+
async function addGlobalEdges(_scope, _key, id, value) {
|
|
150
|
+
await addEdges(id, value);
|
|
151
|
+
}
|
|
152
|
+
async function addProviderEdges(_scope, _key, id, value) {
|
|
153
|
+
await addEdges(id, value);
|
|
154
|
+
}
|
|
155
|
+
async function addNamespacedEdges(_scope, _type, _key, id, value) {
|
|
156
|
+
await addEdges(id, value);
|
|
157
|
+
}
|
|
158
|
+
await Promise.all(Object.values({
|
|
159
|
+
...(0, iteration_1.forEachProvider)(scope, plan.provider, addProviderEdges),
|
|
160
|
+
...(0, iteration_1.forEachGlobal)(scope, "var", plan.variable, addGlobalEdges),
|
|
161
|
+
// locals are a special case
|
|
162
|
+
...(0, iteration_1.forEachGlobal)(scope, "local", Array.isArray(plan.locals)
|
|
163
|
+
? plan.locals.reduce((carry, locals) => ({ ...carry, ...locals }), {})
|
|
164
|
+
: {}, addGlobalEdges),
|
|
165
|
+
...(0, iteration_1.forEachGlobal)(scope, "out", plan.output, addGlobalEdges),
|
|
166
|
+
...(0, iteration_1.forEachGlobal)(scope, "module", plan.module, addGlobalEdges),
|
|
167
|
+
...(0, iteration_1.forEachNamespaced)(scope, plan.resource, addNamespacedEdges),
|
|
168
|
+
...(0, iteration_1.forEachNamespaced)(scope, plan.data, addNamespacedEdges, "data"),
|
|
169
|
+
}).map(({ code: addEdgesToGraph }) => addEdgesToGraph(graph)));
|
|
170
|
+
utils_1.logger.debug(`Graph: ${JSON.stringify(graph, null, 2)}`);
|
|
171
|
+
utils_1.logger.debug(`Starting to assemble the typescript code`);
|
|
172
|
+
// We traverse the dependency graph to get the unordered JSON nodes into an ordered array
|
|
173
|
+
// where no node is referenced before it's defined
|
|
174
|
+
// As we check that the nodes on both ends of an edge exist we can be sure
|
|
175
|
+
// that no infinite loop exists, there can be no stray dependency on a node
|
|
176
|
+
const expressions = [];
|
|
177
|
+
let nodesToVisit = [...nodeIds];
|
|
178
|
+
// This ensures we detect cycles and don't end up in an endless loop
|
|
179
|
+
let nodesVisitedThisIteration = 0;
|
|
180
|
+
do {
|
|
181
|
+
nodesVisitedThisIteration = 0;
|
|
182
|
+
// Find next nodes to visit
|
|
183
|
+
const nodeExpressionGenerators = graph.mapNodes((nodeId, { code }) => {
|
|
184
|
+
if (!nodesToVisit.includes(nodeId)) {
|
|
185
|
+
return undefined;
|
|
186
|
+
}
|
|
187
|
+
const unresolvedDependencies = graph
|
|
188
|
+
.inNeighbors(nodeId)
|
|
189
|
+
.filter((item) => nodesToVisit.includes(item));
|
|
190
|
+
if (unresolvedDependencies.length === 0) {
|
|
191
|
+
nodesToVisit = nodesToVisit.filter((id) => nodeId !== id);
|
|
192
|
+
nodesVisitedThisIteration = nodesVisitedThisIteration + 1;
|
|
193
|
+
utils_1.logger.debug(`Visiting node ${nodeId}`);
|
|
194
|
+
return code;
|
|
195
|
+
}
|
|
196
|
+
return undefined;
|
|
197
|
+
});
|
|
198
|
+
// Generate the code for the nodes
|
|
199
|
+
for (const code of nodeExpressionGenerators) {
|
|
200
|
+
if (code) {
|
|
201
|
+
expressions.push(...(await code(graph)));
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
utils_1.logger.debug(`${nodesToVisit.length} unvisited nodes: ${nodesToVisit.join(", ")}`);
|
|
205
|
+
} while (nodesToVisit.length > 0 && nodesVisitedThisIteration != 0);
|
|
206
|
+
if (nodesToVisit.length > 0) {
|
|
207
|
+
throw new Error(`There are ${nodesToVisit.length} terraform elements that could not be visited.
|
|
208
|
+
This is likely due to a cycle in the dependency graph.
|
|
209
|
+
These nodes are: ${nodesToVisit.join(", ")}`);
|
|
210
|
+
}
|
|
211
|
+
utils_1.logger.debug(`${nodesToVisit.length} unvisited nodes: ${nodesToVisit.join(", ")}`);
|
|
212
|
+
const backendExpressions = (await Promise.all(((_a = plan.terraform) === null || _a === void 0 ? void 0 : _a.map((terraform) => (0, generation_1.backendToExpression)(scope, terraform.backend))) || [Promise.resolve([])])).reduce((carry, item) => [...carry, ...item], []);
|
|
213
|
+
utils_1.logger.debug(`Using these backend expressions: ${JSON.stringify(backendExpressions, null, 2)}`);
|
|
214
|
+
// We collect all module sources
|
|
215
|
+
const moduleRequirements = [
|
|
216
|
+
...new Set(Object.values(plan.module || {}).reduce((carry, moduleBlock) => [
|
|
217
|
+
...carry,
|
|
218
|
+
...moduleBlock.reduce((arr, { source, version }) => [
|
|
219
|
+
...arr,
|
|
220
|
+
version ? `${source}@${version}` : source,
|
|
221
|
+
], []),
|
|
222
|
+
], []) || []),
|
|
223
|
+
];
|
|
224
|
+
utils_1.logger.debug(`Found these modules: ${JSON.stringify(moduleRequirements, null, 2)}`);
|
|
225
|
+
if (Object.keys(plan.variable || {}).length > 0 && expressions.length > 0) {
|
|
226
|
+
expressions[0] = t.addComment(expressions[0], "leading", `Terraform Variables are not always the best fit for getting inputs in the context of Terraform CDK.
|
|
227
|
+
You can read more about this at https://cdk.tf/variables`);
|
|
228
|
+
}
|
|
229
|
+
const providerRequirements = (0, provider_1.getProviderRequirements)(plan);
|
|
230
|
+
utils_1.logger.debug(`Found these provider requirements: ${JSON.stringify(providerRequirements, null, 2)}`);
|
|
231
|
+
// We add a comment if there are providers with missing schema information
|
|
232
|
+
const providersLackingSchema = Object.keys(providerRequirements).filter((providerName) => providerName !== "terraform" &&
|
|
233
|
+
!Object.keys(providerSchema.provider_schemas || {}).some((schemaName) => schemaName.endsWith(providerName)));
|
|
234
|
+
utils_1.logger.debug(`${providersLackingSchema.length} providers lack schema information: ${providersLackingSchema.join(", ")}`);
|
|
235
|
+
if (providersLackingSchema.length > 0) {
|
|
236
|
+
expressions[0] = t.addComment(expressions[0], "leading", `The following providers are missing schema information and might need manual adjustments to synthesize correctly: ${providersLackingSchema.join(", ")}.
|
|
237
|
+
For a more precise conversion please use the --provider flag in convert.`);
|
|
238
|
+
}
|
|
239
|
+
// Always add constructs
|
|
240
|
+
scope.importables.push({
|
|
241
|
+
constructName: "Construct",
|
|
242
|
+
provider: "constructs",
|
|
243
|
+
});
|
|
244
|
+
if (scope.hasTokenBasedTypeCoercion) {
|
|
245
|
+
scope.importables.push({
|
|
246
|
+
constructName: "Token",
|
|
247
|
+
provider: "cdktn",
|
|
248
|
+
});
|
|
249
|
+
}
|
|
250
|
+
// Add specific import for codeContainer
|
|
251
|
+
(0, generation_1.addImportForCodeContainer)(scope, codeContainer);
|
|
252
|
+
const constructImports = (0, generation_1.buildImports)(scope.importables);
|
|
253
|
+
const code = [...(backendExpressions || []), ...expressions];
|
|
254
|
+
const configTypeName = Object.keys(scope.topLevelConfig).length > 0 ? "MyConfig" : undefined;
|
|
255
|
+
const classConfig = configTypeName
|
|
256
|
+
? [(0, generation_1.generateConfigType)(configTypeName, scope.topLevelConfig)]
|
|
257
|
+
: [];
|
|
258
|
+
// We split up the generated code so that users can have more control over what to insert where
|
|
259
|
+
return {
|
|
260
|
+
// TODO: Remove imports and code because rosetta won't be able to translate them
|
|
261
|
+
all: await (0, generation_1.gen)([
|
|
262
|
+
...constructImports,
|
|
263
|
+
...(0, generation_1.moduleImports)(plan.module),
|
|
264
|
+
...classConfig,
|
|
265
|
+
(0, generation_1.wrapCodeInConstructor)(codeContainer, code, "MyConvertedCode", configTypeName),
|
|
266
|
+
]),
|
|
267
|
+
imports: await (0, generation_1.gen)([...constructImports, ...(0, generation_1.moduleImports)(plan.module)]),
|
|
268
|
+
code: await (0, generation_1.gen)(code),
|
|
269
|
+
providers: Object.entries(providerRequirements).map(([source, version]) => version === "*" ? source : `${source}@${version}`),
|
|
270
|
+
modules: moduleRequirements,
|
|
271
|
+
// We track some usage data to make it easier to understand what is used
|
|
272
|
+
stats: {
|
|
273
|
+
numberOfModules: moduleRequirements.length,
|
|
274
|
+
numberOfProviders: Object.keys(providerRequirements).length,
|
|
275
|
+
resources: (0, iteration_1.resourceStats)(plan.resource || {}),
|
|
276
|
+
data: (0, iteration_1.resourceStats)(plan.data || {}),
|
|
277
|
+
convertedLines: hcl.split("\n").length,
|
|
278
|
+
},
|
|
279
|
+
};
|
|
280
|
+
}
|
|
281
|
+
exports.convertToTypescript = convertToTypescript;
|
|
282
|
+
const translators = {
|
|
283
|
+
python: {
|
|
284
|
+
visitor: () => new rosetta.PythonVisitor(),
|
|
285
|
+
postTranslationMutation: jsii_rosetta_workarounds_1.replacePythonImports,
|
|
286
|
+
},
|
|
287
|
+
java: {
|
|
288
|
+
visitor: () => new rosetta.JavaVisitor(),
|
|
289
|
+
postTranslationMutation: jsii_rosetta_workarounds_1.replaceJavaImports,
|
|
290
|
+
},
|
|
291
|
+
csharp: {
|
|
292
|
+
visitor: () => new rosetta.CSharpVisitor(),
|
|
293
|
+
postTranslationMutation: jsii_rosetta_workarounds_1.replaceCsharpImports,
|
|
294
|
+
},
|
|
295
|
+
go: {
|
|
296
|
+
visitor: () => new rosetta.GoVisitor(),
|
|
297
|
+
postTranslationMutation: jsii_rosetta_workarounds_1.replaceGoImports,
|
|
298
|
+
},
|
|
299
|
+
};
|
|
300
|
+
function translatorForLanguage(language) {
|
|
301
|
+
return (file, throwOnTranslationError) => {
|
|
302
|
+
const { visitor, postTranslationMutation } = translators[language];
|
|
303
|
+
const { translation, diagnostics } = rosetta.translateTypeScript(file, visitor(), throwOnTranslationError ? { includeCompilerDiagnostics: true } : {});
|
|
304
|
+
if (throwOnTranslationError &&
|
|
305
|
+
diagnostics.filter((diag) => diag.isError).length > 0) {
|
|
306
|
+
utils_1.logger.debug(`Could not translate TS to ${language}:\n${file.contents}`);
|
|
307
|
+
throw new Error(`Could not translate TS to ${language}: ${diagnostics
|
|
308
|
+
.map((diag) => diag.formattedMessage)
|
|
309
|
+
.join("\n")}`);
|
|
310
|
+
}
|
|
311
|
+
return postTranslationMutation(translation);
|
|
312
|
+
};
|
|
313
|
+
}
|
|
314
|
+
async function convert(hcl, { language, providerSchema, throwOnTranslationError = false, codeContainer = "cdktn.TerraformStack", }) {
|
|
315
|
+
const fileName = "terraform.tf";
|
|
316
|
+
const translater = language === "typescript"
|
|
317
|
+
? (file, _throwOnTranslationError) => file.contents
|
|
318
|
+
: translatorForLanguage(language);
|
|
319
|
+
if (!translater) {
|
|
320
|
+
throw new Error("Unsupported language used: " + language);
|
|
321
|
+
}
|
|
322
|
+
const tsCode = await convertToTypescript(hcl, providerSchema, codeContainer);
|
|
323
|
+
return {
|
|
324
|
+
...tsCode,
|
|
325
|
+
all: translater({ fileName, contents: tsCode.all }, throwOnTranslationError),
|
|
326
|
+
imports: translater({ fileName, contents: tsCode.imports }, false),
|
|
327
|
+
code: translater({ fileName, contents: tsCode.code }, false),
|
|
328
|
+
stats: { ...tsCode.stats, language },
|
|
329
|
+
};
|
|
330
|
+
}
|
|
331
|
+
exports.convert = convert;
|
|
332
|
+
function getTerraformConfigFromDir(importPath) {
|
|
333
|
+
const absPath = path.resolve(importPath);
|
|
334
|
+
const fileContents = glob
|
|
335
|
+
.sync("./*.tf", { cwd: absPath })
|
|
336
|
+
.map((p) => fs.readFileSync(path.resolve(absPath, p), "utf8"));
|
|
337
|
+
return fileContents.join("\n");
|
|
338
|
+
}
|
|
339
|
+
exports.getTerraformConfigFromDir = getTerraformConfigFromDir;
|
|
340
|
+
async function convertProject(combinedHcl, { language, providerSchema }) {
|
|
341
|
+
if (language !== "typescript") {
|
|
342
|
+
throw new Error("Unsupported language used: " + language);
|
|
343
|
+
}
|
|
344
|
+
const { imports, code, providers, modules: tfModules, stats, } = await convert(combinedHcl, {
|
|
345
|
+
language,
|
|
346
|
+
providerSchema,
|
|
347
|
+
});
|
|
348
|
+
return {
|
|
349
|
+
code: (inputMainFile) => {
|
|
350
|
+
const importMainFile = [imports, inputMainFile].join("\n");
|
|
351
|
+
const outputMainFile = importMainFile.replace(exports.CODE_MARKER, code);
|
|
352
|
+
return prettier_1.default.format(outputMainFile, { parser: "babel" });
|
|
353
|
+
},
|
|
354
|
+
cdktfJson: (inputCdktfJson) => {
|
|
355
|
+
const cdktfJson = { ...inputCdktfJson };
|
|
356
|
+
cdktfJson.terraformProviders = providers;
|
|
357
|
+
cdktfJson.terraformModules = tfModules;
|
|
358
|
+
return cdktfJson;
|
|
359
|
+
},
|
|
360
|
+
stats,
|
|
361
|
+
};
|
|
362
|
+
}
|
|
363
|
+
exports.convertProject = convertProject;
|
|
364
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright (c) HashiCorp, Inc\n// SPDX-License-Identifier: MPL-2.0\nimport { parse } from \"@cdktn/hcl2json\";\nimport {\n  isRegistryModule,\n  TerraformProviderGenerator,\n  CodeMaker,\n} from \"@cdktn/provider-generator\";\n\nimport * as t from \"@babel/types\";\nimport prettier from \"prettier\";\nimport * as path from \"path\";\nimport * as glob from \"glob\";\nimport * as fs from \"fs\";\nimport { DirectedGraph } from \"graphology\";\nimport * as rosetta from \"jsii-rosetta\";\nimport * as z from \"zod\";\n\nimport { schema } from \"./schema\";\nimport { findUsedReferences } from \"./references\";\nimport {\n  backendToExpression,\n  gen,\n  local,\n  moduleImports,\n  modules,\n  output,\n  provider,\n  resource,\n  variable,\n  wrapCodeInConstructor,\n  addImportForCodeContainer,\n  buildImports,\n  generateConfigType,\n  imports,\n} from \"./generation\";\nimport { TerraformResourceBlock, ProgramScope } from \"./types\";\nimport {\n  forEachProvider,\n  forEachGlobal,\n  forEachNamespaced,\n  resourceStats,\n} from \"./iteration\";\nimport { getProviderRequirements } from \"./provider\";\nimport { logger } from \"./utils\";\nimport { FQPN } from \"@cdktn/provider-schema\";\nimport { attributeNameToCdktfName } from \"./generation\";\nimport {\n  replaceCsharpImports,\n  replaceGoImports,\n  replaceJavaImports,\n  replacePythonImports,\n} from \"./jsii-rosetta-workarounds\";\nimport { ProviderSchema } from \"@cdktn/commons\";\nimport { forEachImport } from \"./iteration\";\n\nexport const CODE_MARKER = \"// define resources here\";\n\nexport async function getParsedHcl(hcl: string) {\n  logger.debug(`Parsing HCL: ${hcl}`);\n  // Get the JSON representation of the HCL\n  let json: Record<string, unknown>;\n  try {\n    json = await parse(\"terraform.tf\", hcl);\n  } catch (err) {\n    logger.error(`Failed to parse HCL: ${err}`);\n    throw new Error(\n      `Error: Could not parse HCL, this means either that the HCL passed is invalid or that you found a bug. If the HCL seems valid, please file a bug under https://cdk.tf/bugs/new/convert`,\n    );\n  }\n\n  // Ensure the JSON representation matches the expected structure\n  let plan: z.infer<typeof schema>;\n  try {\n    plan = schema.parse(json);\n  } catch (err) {\n    throw new Error(`Error: HCL-JSON does not conform to schema. This is not expected, please file a bug under https://cdk.tf/bugs/new/convert\nPlease include this information:\n${JSON.stringify((err as z.ZodError).errors)}`);\n  }\n\n  return plan;\n}\n\nexport async function parseProviderRequirements(hcl: string) {\n  logger.debug(\"Parsing provider requirements\");\n  const plan = await getParsedHcl(hcl);\n  return getProviderRequirements(plan);\n}\n\nexport async function convertToTypescript(\n  hcl: string,\n  providerSchema: ProviderSchema,\n  codeContainer: string,\n) {\n  logger.debug(\"Converting to typescript\");\n  const plan = await getParsedHcl(hcl);\n\n  // Each key in the scope needs to be unique, therefore we save them in a set\n  // Each variable needs to be unique as well, we save them in a record so we can identify if two variables are the same\n  const scope: ProgramScope = {\n    providerSchema,\n    providerGenerator: Object.keys(\n      providerSchema.provider_schemas || {},\n    ).reduce((carry, fqpn) => {\n      const providerGenerator = new TerraformProviderGenerator(\n        new CodeMaker(),\n        providerSchema,\n      );\n      providerGenerator.buildResourceModels(fqpn as FQPN); // can't use that type on the keys yet, since we are not on TS >=4.4 yet :sadcat:\n      return { ...carry, [fqpn]: providerGenerator };\n    }, {}),\n    constructs: new Set<string>(),\n    variables: {},\n    hasTokenBasedTypeCoercion: false,\n    nodeIds: [],\n    importables: [],\n    topLevelConfig: {},\n  };\n\n  const graph = new DirectedGraph<{\n    code: (\n      g: DirectedGraph<any>,\n    ) => Promise<Array<t.Statement | t.VariableDeclaration>>;\n  }>();\n\n  // Get all items in the JSON as a map of id to function that generates the AST\n  // We will use this to construct the nodes for a dependency graph\n  // We need to use a function here because the same node has different representation based on if it's referenced by another one\n  const nodeMap: Record<\n    string,\n    {\n      code: (\n        g: typeof graph,\n      ) => Promise<Array<t.Statement | t.VariableDeclaration>>;\n      value: unknown;\n    }\n  > = {\n    ...forEachProvider(scope, plan.provider, provider),\n    ...forEachGlobal(scope, \"var\", plan.variable, variable),\n    // locals are a special case\n    ...forEachGlobal(\n      scope,\n      \"local\",\n      Array.isArray(plan.locals)\n        ? plan.locals.reduce((carry, locals) => ({ ...carry, ...locals }), {})\n        : {},\n      local,\n    ),\n    ...forEachGlobal(scope, \"out\", plan.output, output),\n    ...forEachGlobal(scope, \"module\", plan.module, modules),\n    ...forEachImport(scope, \"import\", plan.import, imports),\n    ...forEachNamespaced(scope, plan.resource, resource),\n    ...forEachNamespaced(scope, plan.data, resource, \"data\"),\n  };\n\n  // Add all nodes to the dependency graph so we can detect if an edge is added for an unknown link\n  Object.entries(nodeMap).forEach(([key, value]) => {\n    logger.debug(`Adding node '${key}' to graph`);\n    graph.addNode(key, value);\n  });\n\n  // Finding references becomes easier of the to be referenced ids are already known\n  const nodeIds = Object.keys(nodeMap);\n  scope.nodeIds = nodeIds;\n  async function addEdges(id: string, value: TerraformResourceBlock) {\n    (await findUsedReferences(nodeIds, value)).forEach((ref) => {\n      if (\n        !graph.hasDirectedEdge(ref.referencee.id, id) &&\n        graph.hasNode(ref.referencee.id) // in case the referencee is a dynamic variable\n      ) {\n        if (!graph.hasNode(id)) {\n          throw new Error(\n            `The dependency graph is expected to link from ${\n              ref.referencee.id\n            } to ${id} but ${id} does not exist.\n            These nodes exist: ${graph.nodes().join(\"\\n\")}`,\n          );\n        }\n\n        // The graph should have no self-references\n        if (id === ref.referencee.id) {\n          logger.debug(`Skipping self-reference for ${id}`);\n          return;\n        }\n\n        logger.debug(`Adding edge from ${ref.referencee.id} to ${id}`);\n        graph.addDirectedEdge(ref.referencee.id, id, { ref });\n      }\n    });\n  }\n\n  // We recursively inspect each resource value to find references to other values\n  // We add these to a dependency graph so that the programming code has the right order\n  async function addGlobalEdges(\n    _scope: ProgramScope,\n    _key: string,\n    id: string,\n    value: TerraformResourceBlock,\n  ) {\n    await addEdges(id, value);\n  }\n  async function addProviderEdges(\n    _scope: ProgramScope,\n    _key: string,\n    id: string,\n    value: TerraformResourceBlock,\n  ) {\n    await addEdges(id, value);\n  }\n  async function addNamespacedEdges(\n    _scope: ProgramScope,\n    _type: string,\n    _key: string,\n    id: string,\n    value: TerraformResourceBlock,\n  ) {\n    await addEdges(id, value);\n  }\n\n  await Promise.all(\n    Object.values({\n      ...forEachProvider(scope, plan.provider, addProviderEdges),\n      ...forEachGlobal(scope, \"var\", plan.variable, addGlobalEdges),\n      // locals are a special case\n      ...forEachGlobal(\n        scope,\n        \"local\",\n        Array.isArray(plan.locals)\n          ? plan.locals.reduce((carry, locals) => ({ ...carry, ...locals }), {})\n          : {},\n        addGlobalEdges,\n      ),\n      ...forEachGlobal(scope, \"out\", plan.output, addGlobalEdges),\n      ...forEachGlobal(scope, \"module\", plan.module, addGlobalEdges),\n      ...forEachNamespaced(scope, plan.resource, addNamespacedEdges),\n      ...forEachNamespaced(scope, plan.data, addNamespacedEdges, \"data\"),\n    }).map(({ code: addEdgesToGraph }) => addEdgesToGraph(graph)),\n  );\n\n  logger.debug(`Graph: ${JSON.stringify(graph, null, 2)}`);\n  logger.debug(`Starting to assemble the typescript code`);\n  // We traverse the dependency graph to get the unordered JSON nodes into an ordered array\n  // where no node is referenced before it's defined\n  // As we check that the nodes on both ends of an edge exist we can be sure\n  // that no infinite loop exists, there can be no stray dependency on a node\n  const expressions: t.Statement[] = [];\n  let nodesToVisit = [...nodeIds];\n  // This ensures we detect cycles and don't end up in an endless loop\n  let nodesVisitedThisIteration = 0;\n  do {\n    nodesVisitedThisIteration = 0;\n\n    // Find next nodes to visit\n    const nodeExpressionGenerators = graph.mapNodes((nodeId, { code }) => {\n      if (!nodesToVisit.includes(nodeId)) {\n        return undefined;\n      }\n\n      const unresolvedDependencies = graph\n        .inNeighbors(nodeId)\n        .filter((item) => nodesToVisit.includes(item));\n\n      if (unresolvedDependencies.length === 0) {\n        nodesToVisit = nodesToVisit.filter((id) => nodeId !== id);\n        nodesVisitedThisIteration = nodesVisitedThisIteration + 1;\n\n        logger.debug(`Visiting node ${nodeId}`);\n        return code;\n      }\n      return undefined;\n    });\n\n    // Generate the code for the nodes\n    for (const code of nodeExpressionGenerators) {\n      if (code) {\n        expressions.push(...(await code(graph)));\n      }\n    }\n\n    logger.debug(\n      `${nodesToVisit.length} unvisited nodes: ${nodesToVisit.join(\", \")}`,\n    );\n  } while (nodesToVisit.length > 0 && nodesVisitedThisIteration != 0);\n\n  if (nodesToVisit.length > 0) {\n    throw new Error(\n      `There are ${\n        nodesToVisit.length\n      } terraform elements that could not be visited.\n      This is likely due to a cycle in the dependency graph.\n      These nodes are: ${nodesToVisit.join(\", \")}`,\n    );\n  }\n\n  logger.debug(\n    `${nodesToVisit.length} unvisited nodes: ${nodesToVisit.join(\", \")}`,\n  );\n\n  const backendExpressions = (\n    await Promise.all(\n      plan.terraform?.map((terraform) =>\n        backendToExpression(scope, terraform.backend),\n      ) || [Promise.resolve([])],\n    )\n  ).reduce((carry, item) => [...carry, ...item], []);\n\n  logger.debug(\n    `Using these backend expressions: ${JSON.stringify(\n      backendExpressions,\n      null,\n      2,\n    )}`,\n  );\n\n  // We collect all module sources\n  const moduleRequirements = [\n    ...new Set(\n      Object.values(plan.module || {}).reduce(\n        (carry, moduleBlock) => [\n          ...carry,\n          ...moduleBlock.reduce(\n            (arr, { source, version }) => [\n              ...arr,\n              version ? `${source}@${version}` : source,\n            ],\n            [] as string[],\n          ),\n        ],\n        [] as string[],\n      ) || [],\n    ),\n  ];\n\n  logger.debug(\n    `Found these modules: ${JSON.stringify(moduleRequirements, null, 2)}`,\n  );\n\n  if (Object.keys(plan.variable || {}).length > 0 && expressions.length > 0) {\n    expressions[0] = t.addComment(\n      expressions[0],\n      \"leading\",\n      `Terraform Variables are not always the best fit for getting inputs in the context of Terraform CDK.\nYou can read more about this at https://cdk.tf/variables`,\n    );\n  }\n\n  const providerRequirements = getProviderRequirements(plan);\n  logger.debug(\n    `Found these provider requirements: ${JSON.stringify(\n      providerRequirements,\n      null,\n      2,\n    )}`,\n  );\n\n  // We add a comment if there are providers with missing schema information\n  const providersLackingSchema = Object.keys(providerRequirements).filter(\n    (providerName) =>\n      providerName !== \"terraform\" &&\n      !Object.keys(providerSchema.provider_schemas || {}).some((schemaName) =>\n        schemaName.endsWith(providerName),\n      ),\n  );\n  logger.debug(\n    `${\n      providersLackingSchema.length\n    } providers lack schema information: ${providersLackingSchema.join(\", \")}`,\n  );\n\n  if (providersLackingSchema.length > 0) {\n    expressions[0] = t.addComment(\n      expressions[0],\n      \"leading\",\n      `The following providers are missing schema information and might need manual adjustments to synthesize correctly: ${providersLackingSchema.join(\n        \", \",\n      )}.\nFor a more precise conversion please use the --provider flag in convert.`,\n    );\n  }\n\n  // Always add constructs\n  scope.importables.push({\n    constructName: \"Construct\",\n    provider: \"constructs\",\n  });\n\n  if (scope.hasTokenBasedTypeCoercion) {\n    scope.importables.push({\n      constructName: \"Token\",\n      provider: \"cdktn\",\n    });\n  }\n\n  // Add specific import for codeContainer\n  addImportForCodeContainer(scope, codeContainer);\n  const constructImports = buildImports(scope.importables);\n\n  const code = [...(backendExpressions || []), ...expressions];\n  const configTypeName =\n    Object.keys(scope.topLevelConfig).length > 0 ? \"MyConfig\" : undefined;\n\n  const classConfig = configTypeName\n    ? [generateConfigType(configTypeName, scope.topLevelConfig)]\n    : [];\n\n  // We split up the generated code so that users can have more control over what to insert where\n  return {\n    // TODO: Remove imports and code because rosetta won't be able to translate them\n    all: await gen([\n      ...constructImports,\n      ...moduleImports(plan.module),\n      ...classConfig,\n      wrapCodeInConstructor(\n        codeContainer,\n        code,\n        \"MyConvertedCode\",\n        configTypeName,\n      ),\n    ]),\n    imports: await gen([...constructImports, ...moduleImports(plan.module)]),\n    code: await gen(code),\n    providers: Object.entries(providerRequirements).map(([source, version]) =>\n      version === \"*\" ? source : `${source}@${version}`,\n    ),\n    modules: moduleRequirements,\n    // We track some usage data to make it easier to understand what is used\n    stats: {\n      numberOfModules: moduleRequirements.length,\n      numberOfProviders: Object.keys(providerRequirements).length,\n      resources: resourceStats(plan.resource || {}),\n      data: resourceStats(plan.data || {}),\n      convertedLines: hcl.split(\"\\n\").length,\n    },\n  };\n}\n\ntype File = { contents: string; fileName: string };\nconst translators = {\n  python: {\n    visitor: () => new rosetta.PythonVisitor(),\n    postTranslationMutation: replacePythonImports,\n  },\n  java: {\n    visitor: () => new rosetta.JavaVisitor(),\n    postTranslationMutation: replaceJavaImports,\n  },\n  csharp: {\n    visitor: () => new rosetta.CSharpVisitor(),\n    postTranslationMutation: replaceCsharpImports,\n  },\n  go: {\n    visitor: () => new rosetta.GoVisitor(),\n    postTranslationMutation: replaceGoImports,\n  },\n};\n\nfunction translatorForLanguage(language: keyof typeof translators) {\n  return (file: File, throwOnTranslationError: boolean) => {\n    const { visitor, postTranslationMutation } = translators[language];\n    const { translation, diagnostics } = rosetta.translateTypeScript(\n      file,\n      visitor(),\n      throwOnTranslationError ? { includeCompilerDiagnostics: true } : {},\n    );\n\n    if (\n      throwOnTranslationError &&\n      diagnostics.filter((diag) => diag.isError).length > 0\n    ) {\n      logger.debug(`Could not translate TS to ${language}:\\n${file.contents}`);\n      throw new Error(\n        `Could not translate TS to ${language}: ${diagnostics\n          .map((diag) => diag.formattedMessage)\n          .join(\"\\n\")}`,\n      );\n    }\n\n    return postTranslationMutation(translation);\n  };\n}\n\ntype ConvertOptions = {\n  /**\n   * The language to convert to\n   */\n  language: keyof typeof translators | \"typescript\";\n  /**\n   * The provider schema to use for conversion\n   */\n  providerSchema: ProviderSchema;\n  /**\n   * The base class to extend from. Defaults to `constructs.Construct`\n   */\n  codeContainer?: string;\n  /**\n   * Whether to throw an error if the translation fails\n   * Defaults to false\n   */\n  throwOnTranslationError?: boolean;\n};\n\nexport async function convert(\n  hcl: string,\n  {\n    language,\n    providerSchema,\n    throwOnTranslationError = false,\n    codeContainer = \"cdktn.TerraformStack\",\n  }: ConvertOptions,\n) {\n  const fileName = \"terraform.tf\";\n  const translater =\n    language === \"typescript\"\n      ? (file: File, _throwOnTranslationError: boolean) => file.contents\n      : translatorForLanguage(language);\n\n  if (!translater) {\n    throw new Error(\"Unsupported language used: \" + language);\n  }\n\n  const tsCode = await convertToTypescript(hcl, providerSchema, codeContainer);\n\n  return {\n    ...tsCode,\n    all: translater(\n      { fileName, contents: tsCode.all },\n      throwOnTranslationError,\n    ),\n    imports: translater({ fileName, contents: tsCode.imports }, false),\n    code: translater({ fileName, contents: tsCode.code }, false),\n    stats: { ...tsCode.stats, language },\n  };\n}\n\nexport function getTerraformConfigFromDir(importPath: string) {\n  const absPath = path.resolve(importPath);\n  const fileContents = glob\n    .sync(\"./*.tf\", { cwd: absPath })\n    .map((p) => fs.readFileSync(path.resolve(absPath, p), \"utf8\"));\n\n  return fileContents.join(\"\\n\");\n}\n\ntype CdktfJson = Record<string, unknown> & {\n  terraformProviders: any[];\n  terraformModules: any[];\n};\nexport async function convertProject(\n  combinedHcl: string,\n  { language, providerSchema }: ConvertOptions,\n) {\n  if (language !== \"typescript\") {\n    throw new Error(\"Unsupported language used: \" + language);\n  }\n\n  const {\n    imports,\n    code,\n    providers,\n    modules: tfModules,\n    stats,\n  } = await convert(combinedHcl, {\n    language,\n    providerSchema,\n  });\n\n  return {\n    code: (inputMainFile: string) => {\n      const importMainFile = [imports, inputMainFile].join(\"\\n\");\n      const outputMainFile = importMainFile.replace(CODE_MARKER, code);\n      return prettier.format(outputMainFile, { parser: \"babel\" });\n    },\n    cdktfJson: (inputCdktfJson: CdktfJson) => {\n      const cdktfJson = { ...inputCdktfJson };\n      cdktfJson.terraformProviders = providers;\n      cdktfJson.terraformModules = tfModules;\n      return cdktfJson;\n    },\n    stats,\n  };\n}\n\nexport { isRegistryModule, attributeNameToCdktfName };\n"]}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.resourceStats = exports.forEachNamespaced = exports.forEachProvider = exports.forEachImport = exports.forEachGlobal = void 0;
|
|
4
|
+
const telemetryAllowList_json_1 = require("./telemetryAllowList.json");
|
|
5
|
+
// locals, variables, and outputs are global key value maps
|
|
6
|
+
function forEachGlobal(scope, prefix, record, iterator) {
|
|
7
|
+
return Object.entries(record || {}).reduce((carry, [key, item]) => {
|
|
8
|
+
const id = `${prefix}.${key}`;
|
|
9
|
+
return {
|
|
10
|
+
...carry,
|
|
11
|
+
[id]: {
|
|
12
|
+
code: async (graph) => await iterator(scope, key, id, item, graph),
|
|
13
|
+
value: item,
|
|
14
|
+
},
|
|
15
|
+
};
|
|
16
|
+
}, {});
|
|
17
|
+
}
|
|
18
|
+
exports.forEachGlobal = forEachGlobal;
|
|
19
|
+
function forEachImport(scope, prefix, record, iterator) {
|
|
20
|
+
return (record || []).reduce((carry, item) => {
|
|
21
|
+
const target = item.to.startsWith("${") && item.to.endsWith("}")
|
|
22
|
+
? item.to.substring(2, item.to.length - 1)
|
|
23
|
+
: item.to;
|
|
24
|
+
const id = `${prefix}.${target}`;
|
|
25
|
+
return {
|
|
26
|
+
...carry,
|
|
27
|
+
[id]: {
|
|
28
|
+
code: async (graph) => await iterator(scope, id, item, graph),
|
|
29
|
+
value: item,
|
|
30
|
+
},
|
|
31
|
+
};
|
|
32
|
+
}, {});
|
|
33
|
+
}
|
|
34
|
+
exports.forEachImport = forEachImport;
|
|
35
|
+
function forEachProvider(scope, record, iterator) {
|
|
36
|
+
return Object.entries(record || {}).reduce((carry, [key, items]) => {
|
|
37
|
+
return {
|
|
38
|
+
...carry,
|
|
39
|
+
...items.reduce((innerCarry, item) => {
|
|
40
|
+
const id = item.alias ? `${key}.${item.alias}` : `${key}`;
|
|
41
|
+
return {
|
|
42
|
+
...innerCarry,
|
|
43
|
+
[id]: {
|
|
44
|
+
code: async (graph) => await iterator(scope, key, id, item, graph),
|
|
45
|
+
value: item,
|
|
46
|
+
},
|
|
47
|
+
};
|
|
48
|
+
}, {}),
|
|
49
|
+
};
|
|
50
|
+
}, {});
|
|
51
|
+
}
|
|
52
|
+
exports.forEachProvider = forEachProvider;
|
|
53
|
+
// data and resource are namespaced key value maps
|
|
54
|
+
function forEachNamespaced(scope, record, iterator, prefix) {
|
|
55
|
+
return Object.entries(record || {}).reduce((outerCarry, [type, items]) => ({
|
|
56
|
+
...outerCarry,
|
|
57
|
+
...Object.entries(items).reduce((innerCarry, [key, item]) => {
|
|
58
|
+
const prefixedType = prefix ? `${prefix}.${type}` : type;
|
|
59
|
+
const id = prefix ? `${prefix}.${type}.${key}` : `${type}.${key}`;
|
|
60
|
+
return {
|
|
61
|
+
...innerCarry,
|
|
62
|
+
[id]: {
|
|
63
|
+
code: async (graph) => await iterator(scope, prefixedType, key, id, item, graph),
|
|
64
|
+
value: item,
|
|
65
|
+
},
|
|
66
|
+
};
|
|
67
|
+
}, {}),
|
|
68
|
+
}), {});
|
|
69
|
+
}
|
|
70
|
+
exports.forEachNamespaced = forEachNamespaced;
|
|
71
|
+
function resourceStats(obj) {
|
|
72
|
+
return Object.entries(obj).reduce((carry, [key, value]) => {
|
|
73
|
+
const [provider, ...resourceParts] = key.split("_");
|
|
74
|
+
const shouldBeTracked = telemetryAllowList_json_1.providers.includes(provider);
|
|
75
|
+
const providerKey = shouldBeTracked ? provider : "other";
|
|
76
|
+
const resourceName = shouldBeTracked ? resourceParts.join("_") : "other";
|
|
77
|
+
return {
|
|
78
|
+
...carry,
|
|
79
|
+
[providerKey]: {
|
|
80
|
+
...(carry[providerKey] || {}),
|
|
81
|
+
[resourceName]: Object.keys(value).length,
|
|
82
|
+
},
|
|
83
|
+
};
|
|
84
|
+
}, {});
|
|
85
|
+
}
|
|
86
|
+
exports.resourceStats = resourceStats;
|
|
87
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"iteration.js","sourceRoot":"","sources":["../src/iteration.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright (c) HashiCorp, Inc\n// SPDX-License-Identifier: MPL-2.0\nimport { DirectedGraph } from \"graphology\";\nimport { providers as telemetryAllowedProviders } from \"./telemetryAllowList.json\";\nimport { ProgramScope } from \"./types\";\nimport { Import } from \"./schema\";\n\n// locals, variables, and outputs are global key value maps\nexport function forEachGlobal<T, R>(\n  scope: ProgramScope,\n  prefix: string,\n  record: Record<string, T> | undefined,\n  iterator: (\n    scope: ProgramScope,\n    key: string,\n    id: string,\n    value: T,\n    graph: DirectedGraph,\n  ) => Promise<R>,\n): Record<string, { code: (graph: DirectedGraph) => Promise<R>; value: T }> {\n  return Object.entries(record || {}).reduce((carry, [key, item]) => {\n    const id = `${prefix}.${key}`;\n    return {\n      ...carry,\n      [id]: {\n        code: async (graph: DirectedGraph) =>\n          await iterator(scope, key, id, item, graph),\n        value: item,\n      },\n    };\n  }, {});\n}\n\nexport function forEachImport<R>(\n  scope: ProgramScope,\n  prefix: string,\n  record: Import[] | undefined,\n  iterator: (\n    scope: ProgramScope,\n    id: string,\n    value: Import,\n    graph: DirectedGraph,\n  ) => Promise<R>,\n): Record<\n  string,\n  { code: (graph: DirectedGraph) => Promise<R>; value: Import }\n> {\n  return (record || []).reduce((carry, item) => {\n    const target =\n      item.to.startsWith(\"${\") && item.to.endsWith(\"}\")\n        ? item.to.substring(2, item.to.length - 1)\n        : item.to;\n\n    const id = `${prefix}.${target}`;\n    return {\n      ...carry,\n      [id]: {\n        code: async (graph: DirectedGraph) =>\n          await iterator(scope, id, item, graph),\n        value: item,\n      },\n    };\n  }, {});\n}\n\nexport function forEachProvider<T extends { alias?: string }, R>(\n  scope: ProgramScope,\n  record: Record<string, T[]> | undefined,\n  iterator: (\n    scope: ProgramScope,\n    key: string,\n    id: string,\n    value: T,\n    graph: DirectedGraph,\n  ) => Promise<R>,\n): Record<string, { code: (graph: DirectedGraph) => Promise<R>; value: T }> {\n  return Object.entries(record || {}).reduce((carry, [key, items]) => {\n    return {\n      ...carry,\n      ...items.reduce((innerCarry, item: T) => {\n        const id = item.alias ? `${key}.${item.alias}` : `${key}`;\n        return {\n          ...innerCarry,\n          [id]: {\n            code: async (graph: DirectedGraph) =>\n              await iterator(scope, key, id, item, graph),\n            value: item,\n          },\n        };\n      }, {}),\n    };\n  }, {});\n}\n\n// data and resource are namespaced key value maps\nexport function forEachNamespaced<T, R>(\n  scope: ProgramScope,\n  record: Record<string, Record<string, T>> | undefined,\n  iterator: (\n    scope: ProgramScope,\n    type: string,\n    key: string,\n    id: string,\n    value: T,\n    graph: DirectedGraph,\n  ) => Promise<R>,\n  prefix?: string,\n): Record<string, { code: (graph: DirectedGraph) => Promise<R>; value: T }> {\n  return Object.entries(record || {}).reduce(\n    (outerCarry, [type, items]) => ({\n      ...outerCarry,\n      ...Object.entries(items).reduce(\n        (innerCarry, [key, item]) => {\n          const prefixedType = prefix ? `${prefix}.${type}` : type;\n          const id = prefix ? `${prefix}.${type}.${key}` : `${type}.${key}`;\n          return {\n            ...innerCarry,\n            [id]: {\n              code: async (graph: DirectedGraph) =>\n                await iterator(scope, prefixedType, key, id, item, graph),\n              value: item,\n            },\n          };\n        },\n        {} as Record<\n          string,\n          { code: (graph: DirectedGraph) => Promise<R>; value: T }\n        >,\n      ),\n    }),\n    {} as Record<\n      string,\n      { code: (graph: DirectedGraph) => Promise<R>; value: T }\n    >,\n  );\n}\n\nexport function resourceStats(obj: Record<string, Record<string, unknown>>) {\n  return Object.entries(obj).reduce(\n    (carry, [key, value]) => {\n      const [provider, ...resourceParts] = key.split(\"_\");\n      const shouldBeTracked = telemetryAllowedProviders.includes(provider);\n      const providerKey = shouldBeTracked ? provider : \"other\";\n      const resourceName = shouldBeTracked ? resourceParts.join(\"_\") : \"other\";\n\n      return {\n        ...carry,\n        [providerKey]: {\n          ...(carry[providerKey] || {}),\n          [resourceName]: Object.keys(value).length,\n        },\n      };\n    },\n    {} as Record<string, Record<string, number>>,\n  );\n}\n"]}
|