@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.
- package/README.md +95 -0
- package/ambient.d.ts +14 -0
- package/jest.config.js +16 -0
- package/lib/__tests__/coerceType.test.d.ts +2 -0
- package/lib/__tests__/coerceType.test.js +165 -0
- package/lib/__tests__/expressionToTs.test.d.ts +6 -0
- package/lib/__tests__/expressionToTs.test.js +693 -0
- package/lib/__tests__/expressions.test.d.ts +4 -0
- package/lib/__tests__/expressions.test.js +415 -0
- package/lib/__tests__/findExpressionType.test.d.ts +6 -0
- package/lib/__tests__/findExpressionType.test.js +105 -0
- package/lib/__tests__/functions.test.d.ts +6 -0
- package/lib/__tests__/functions.test.js +605 -0
- package/lib/__tests__/generation.test.d.ts +6 -0
- package/lib/__tests__/generation.test.js +45 -0
- package/lib/__tests__/jsii-rosetta-workarounds.test.d.ts +2 -0
- package/lib/__tests__/jsii-rosetta-workarounds.test.js +86 -0
- package/lib/__tests__/partialCode.test.d.ts +6 -0
- package/lib/__tests__/partialCode.test.js +390 -0
- package/lib/__tests__/terraformSchema.test.d.ts +6 -0
- package/lib/__tests__/terraformSchema.test.js +105 -0
- package/lib/__tests__/testHelpers.d.ts +7 -0
- package/lib/__tests__/testHelpers.js +16 -0
- package/lib/coerceType.d.ts +7 -0
- package/lib/coerceType.js +240 -0
- package/lib/dynamic-blocks.d.ts +8 -0
- package/lib/dynamic-blocks.js +44 -0
- package/lib/expressions.d.ts +9 -0
- package/lib/expressions.js +634 -0
- package/lib/function-bindings/functions.d.ts +31 -0
- package/lib/function-bindings/functions.generated.d.ts +806 -0
- package/lib/function-bindings/functions.generated.js +1142 -0
- package/lib/function-bindings/functions.js +73 -0
- package/lib/generation.d.ts +26 -0
- package/lib/generation.js +676 -0
- package/lib/index.d.ts +136 -0
- package/lib/index.js +364 -0
- package/lib/iteration.d.ts +23 -0
- package/lib/iteration.js +87 -0
- package/lib/jsii-rosetta-workarounds.d.ts +5 -0
- package/lib/jsii-rosetta-workarounds.js +126 -0
- package/lib/partialCode.d.ts +11 -0
- package/lib/partialCode.js +116 -0
- package/lib/provider.d.ts +8 -0
- package/lib/provider.js +40 -0
- package/lib/references.d.ts +11 -0
- package/lib/references.js +141 -0
- package/lib/schema.d.ts +297 -0
- package/lib/schema.js +81 -0
- package/lib/telemetryAllowList.json +24 -0
- package/lib/terraformSchema.d.ts +6 -0
- package/lib/terraformSchema.js +136 -0
- package/lib/types.d.ts +50 -0
- package/lib/types.js +3 -0
- package/lib/utils.d.ts +6 -0
- package/lib/utils.js +25 -0
- package/lib/variables.d.ts +10 -0
- package/lib/variables.js +172 -0
- package/package.json +71 -0
- package/package.sh +9 -0
- package/tsconfig.json +42 -0
package/lib/index.d.ts
ADDED
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
import { isRegistryModule } from "@cdktn/provider-generator";
|
|
2
|
+
import * as rosetta from "jsii-rosetta";
|
|
3
|
+
import * as z from "zod";
|
|
4
|
+
import { attributeNameToCdktfName } from "./generation";
|
|
5
|
+
import { replaceCsharpImports, replaceGoImports, replaceJavaImports, replacePythonImports } from "./jsii-rosetta-workarounds";
|
|
6
|
+
import { ProviderSchema } from "@cdktn/commons";
|
|
7
|
+
export declare const CODE_MARKER = "// define resources here";
|
|
8
|
+
export declare function getParsedHcl(hcl: string): Promise<{
|
|
9
|
+
terraform?: {
|
|
10
|
+
backend?: Record<string, Record<string, any>[]> | undefined;
|
|
11
|
+
required_providers?: Record<string, string | {
|
|
12
|
+
source?: string | undefined;
|
|
13
|
+
version?: string | undefined;
|
|
14
|
+
}>[] | undefined;
|
|
15
|
+
required_version?: string | undefined;
|
|
16
|
+
}[] | undefined;
|
|
17
|
+
provider?: Record<string, Record<string, any>[]> | undefined;
|
|
18
|
+
module?: Record<string, z.objectOutputType<{
|
|
19
|
+
source: z.ZodString;
|
|
20
|
+
}, z.ZodTypeAny, "passthrough">[]> | undefined;
|
|
21
|
+
resource?: Record<string, Record<string, Record<string, any>[]>> | undefined;
|
|
22
|
+
data?: Record<string, Record<string, Record<string, any>[]>> | undefined;
|
|
23
|
+
output?: Record<string, {
|
|
24
|
+
value?: any;
|
|
25
|
+
description?: string | undefined;
|
|
26
|
+
sensitive?: boolean | undefined;
|
|
27
|
+
depends_on?: string | undefined;
|
|
28
|
+
}[]> | undefined;
|
|
29
|
+
variable?: Record<string, {
|
|
30
|
+
type?: string | undefined;
|
|
31
|
+
default?: any;
|
|
32
|
+
description?: string | undefined;
|
|
33
|
+
sensitive?: boolean | undefined;
|
|
34
|
+
nullable?: boolean | undefined;
|
|
35
|
+
validation?: {
|
|
36
|
+
error_message: string;
|
|
37
|
+
condition?: any;
|
|
38
|
+
}[] | undefined;
|
|
39
|
+
}[]> | undefined;
|
|
40
|
+
import?: {
|
|
41
|
+
to: string;
|
|
42
|
+
id: string;
|
|
43
|
+
provider?: any;
|
|
44
|
+
}[] | undefined;
|
|
45
|
+
locals?: Record<string, any>[] | undefined;
|
|
46
|
+
}>;
|
|
47
|
+
export declare function parseProviderRequirements(hcl: string): Promise<Record<string, string>>;
|
|
48
|
+
export declare function convertToTypescript(hcl: string, providerSchema: ProviderSchema, codeContainer: string): Promise<{
|
|
49
|
+
all: string;
|
|
50
|
+
imports: string;
|
|
51
|
+
code: string;
|
|
52
|
+
providers: string[];
|
|
53
|
+
modules: string[];
|
|
54
|
+
stats: {
|
|
55
|
+
numberOfModules: number;
|
|
56
|
+
numberOfProviders: number;
|
|
57
|
+
resources: Record<string, Record<string, number>>;
|
|
58
|
+
data: Record<string, Record<string, number>>;
|
|
59
|
+
convertedLines: number;
|
|
60
|
+
};
|
|
61
|
+
}>;
|
|
62
|
+
declare const translators: {
|
|
63
|
+
python: {
|
|
64
|
+
visitor: () => rosetta.PythonVisitor;
|
|
65
|
+
postTranslationMutation: typeof replacePythonImports;
|
|
66
|
+
};
|
|
67
|
+
java: {
|
|
68
|
+
visitor: () => rosetta.JavaVisitor;
|
|
69
|
+
postTranslationMutation: typeof replaceJavaImports;
|
|
70
|
+
};
|
|
71
|
+
csharp: {
|
|
72
|
+
visitor: () => rosetta.CSharpVisitor;
|
|
73
|
+
postTranslationMutation: typeof replaceCsharpImports;
|
|
74
|
+
};
|
|
75
|
+
go: {
|
|
76
|
+
visitor: () => rosetta.GoVisitor;
|
|
77
|
+
postTranslationMutation: typeof replaceGoImports;
|
|
78
|
+
};
|
|
79
|
+
};
|
|
80
|
+
type ConvertOptions = {
|
|
81
|
+
/**
|
|
82
|
+
* The language to convert to
|
|
83
|
+
*/
|
|
84
|
+
language: keyof typeof translators | "typescript";
|
|
85
|
+
/**
|
|
86
|
+
* The provider schema to use for conversion
|
|
87
|
+
*/
|
|
88
|
+
providerSchema: ProviderSchema;
|
|
89
|
+
/**
|
|
90
|
+
* The base class to extend from. Defaults to `constructs.Construct`
|
|
91
|
+
*/
|
|
92
|
+
codeContainer?: string;
|
|
93
|
+
/**
|
|
94
|
+
* Whether to throw an error if the translation fails
|
|
95
|
+
* Defaults to false
|
|
96
|
+
*/
|
|
97
|
+
throwOnTranslationError?: boolean;
|
|
98
|
+
};
|
|
99
|
+
export declare function convert(hcl: string, { language, providerSchema, throwOnTranslationError, codeContainer, }: ConvertOptions): Promise<{
|
|
100
|
+
all: string;
|
|
101
|
+
imports: string;
|
|
102
|
+
code: string;
|
|
103
|
+
stats: {
|
|
104
|
+
language: "typescript" | "python" | "csharp" | "java" | "go";
|
|
105
|
+
numberOfModules: number;
|
|
106
|
+
numberOfProviders: number;
|
|
107
|
+
resources: Record<string, Record<string, number>>;
|
|
108
|
+
data: Record<string, Record<string, number>>;
|
|
109
|
+
convertedLines: number;
|
|
110
|
+
};
|
|
111
|
+
providers: string[];
|
|
112
|
+
modules: string[];
|
|
113
|
+
}>;
|
|
114
|
+
export declare function getTerraformConfigFromDir(importPath: string): string;
|
|
115
|
+
type CdktfJson = Record<string, unknown> & {
|
|
116
|
+
terraformProviders: any[];
|
|
117
|
+
terraformModules: any[];
|
|
118
|
+
};
|
|
119
|
+
export declare function convertProject(combinedHcl: string, { language, providerSchema }: ConvertOptions): Promise<{
|
|
120
|
+
code: (inputMainFile: string) => string;
|
|
121
|
+
cdktfJson: (inputCdktfJson: CdktfJson) => {
|
|
122
|
+
[x: string]: unknown;
|
|
123
|
+
terraformProviders: any[];
|
|
124
|
+
terraformModules: any[];
|
|
125
|
+
};
|
|
126
|
+
stats: {
|
|
127
|
+
language: "typescript" | "python" | "csharp" | "java" | "go";
|
|
128
|
+
numberOfModules: number;
|
|
129
|
+
numberOfProviders: number;
|
|
130
|
+
resources: Record<string, Record<string, number>>;
|
|
131
|
+
data: Record<string, Record<string, number>>;
|
|
132
|
+
convertedLines: number;
|
|
133
|
+
};
|
|
134
|
+
}>;
|
|
135
|
+
export { isRegistryModule, attributeNameToCdktfName };
|
|
136
|
+
//# sourceMappingURL=index.d.ts.map
|
package/lib/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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLCtCQUErQjtBQUMvQixtQ0FBbUM7QUFDbkMsOENBQXdDO0FBQ3hDLGtFQUltQztBQWdrQjFCLGlHQW5rQlAscUNBQWdCLE9BbWtCTztBQTlqQnpCLGdEQUFrQztBQUNsQyx3REFBZ0M7QUFDaEMsMkNBQTZCO0FBQzdCLDJDQUE2QjtBQUM3Qix1Q0FBeUI7QUFDekIsMkNBQTJDO0FBQzNDLHNEQUF3QztBQUd4QyxxQ0FBa0M7QUFDbEMsNkNBQWtEO0FBQ2xELDZDQWVzQjtBQUV0QiwyQ0FLcUI7QUFDckIseUNBQXFEO0FBQ3JELG1DQUFpQztBQUVqQyw2Q0FBd0Q7QUF5aEI3Qix5R0F6aEJsQixxQ0FBd0IsT0F5aEJrQjtBQXhoQm5ELHlFQUtvQztBQUVwQywyQ0FBNEM7QUFFL0IsUUFBQSxXQUFXLEdBQUcsMEJBQTBCLENBQUM7QUFFL0MsS0FBSyxVQUFVLFlBQVksQ0FBQyxHQUFXO0lBQzVDLGNBQU0sQ0FBQyxLQUFLLENBQUMsZ0JBQWdCLEdBQUcsRUFBRSxDQUFDLENBQUM7SUFDcEMseUNBQXlDO0lBQ3pDLElBQUksSUFBNkIsQ0FBQztJQUNsQyxJQUFJLENBQUM7UUFDSCxJQUFJLEdBQUcsTUFBTSxJQUFBLGdCQUFLLEVBQUMsY0FBYyxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQzFDLENBQUM7SUFBQyxPQUFPLEdBQUcsRUFBRSxDQUFDO1FBQ2IsY0FBTSxDQUFDLEtBQUssQ0FBQyx3QkFBd0IsR0FBRyxFQUFFLENBQUMsQ0FBQztRQUM1QyxNQUFNLElBQUksS0FBSyxDQUNiLHVMQUF1TCxDQUN4TCxDQUFDO0lBQ0osQ0FBQztJQUVELGdFQUFnRTtJQUNoRSxJQUFJLElBQTRCLENBQUM7SUFDakMsSUFBSSxDQUFDO1FBQ0gsSUFBSSxHQUFHLGVBQU0sQ0FBQyxLQUFLLENB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QUN2QyxNQUFNLFNBQVMsR0FBRyxFQUFFLEdBQUcsY0FBYyxFQUFFLENBQUM7WUFDeEMsU0FBUyxDQUFDLGtCQUFrQixHQUFHLFNBQVMsQ0FBQztZQUN6QyxTQUFTLENBQUMsZ0JBQWdCLEdBQUcsU0FBUyxDQUFDO1lBQ3ZDLE9BQU8sU0FBUyxDQUFDO1FBQ25CLENBQUM7UUFDRCxLQUFLO0tBQ04sQ0FBQztBQUNKLENBQUM7QUFqQ0Qsd0NBaUNDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQ29weXJpZ2h0IChjKSBIYXNoaUNvcnAsIEluY1xuLy8gU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IE1QTC0yLjBcbmltcG9ydCB7IHBhcnNlIH0gZnJvbSBcIkBjZGt0bi9oY2wyanNvblwiO1xuaW1wb3J0IHtcbiAgaXNSZWdpc3RyeU1vZHVsZSxcbiAgVGVycmFmb3JtUHJvdmlkZXJHZW5lcmF0b3IsXG4gIENvZGVNYWtlcixcbn0gZnJvbSBcIkBjZGt0bi9wcm92aWRlci1nZW5lcmF0b3JcIjtcblxuaW1wb3J0ICogYXMgdCBmcm9tIFwiQGJhYmVsL3R5cGVzXCI7XG5pbXBvcnQgcHJldHRpZXIgZnJvbSBcInByZXR0aWVyXCI7XG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gXCJwYXRoXCI7XG5pbXBvcnQgKiBhcyBnbG9iIGZyb20gXCJnbG9iXCI7XG5pbXBvcnQgKiBhcyBmcyBmcm9tIFwiZnNcIjtcbmltcG9ydCB7IERpcmVjdGVkR3JhcGggfSBmcm9tIFwiZ3JhcGhvbG9neVwiO1xuaW1wb3J0ICogYXMgcm9zZXR0YSBmcm9tIFwianNpaS1yb3NldHRhXCI7XG5pbXBvcnQgKiBhcyB6IGZyb20gXCJ6b2RcIjtcblxuaW1wb3J0IHsgc2NoZW1hIH0gZnJvbSBcIi4vc2NoZW1hXCI7XG5pbXBvcnQgeyBmaW5kVXNlZFJlZmVyZW5jZXMgfSBmcm9tIFwiLi9yZWZlcmVuY2VzXCI7XG5pbXBvcnQge1xuICBiYWNrZW5kVG9FeHByZXNzaW9uLFxuICBnZW4sXG4gIGxvY2FsLFxuICBtb2R1bGVJbXBvcnRzLFxuICBtb2R1bGVzLFxuICBvdXRwdXQsXG4gIHByb3ZpZGVyLFxuICByZXNvdXJjZSxcbiAgdmFyaWFibGUsXG4gIHdyYXBDb2RlSW5Db25zdHJ1Y3RvcixcbiAgYWRkSW1wb3J0Rm9yQ29kZUNvbnRhaW5lcixcbiAgYnVpbGRJbXBvcnRzLFxuICBnZW5lcmF0ZUNvbmZpZ1R5cGUsXG4gIGltcG9ydHMsXG59IGZyb20gXCIuL2dlbmVyYXRpb25cIjtcbmltcG9ydCB7IFRlcnJhZm9ybVJlc291cmNlQmxvY2ssIFByb2dyYW1TY29wZSB9IGZyb20gXCIuL3R5cGVzXCI7XG5pbXBvcnQge1xuICBmb3JFYWNoUHJvdmlkZXIsXG4gIGZvckVhY2hHbG9iYWwsXG4gIGZvckVhY2hOYW1lc3BhY2VkLFxuICByZXNvdXJjZVN0YXRzLFxufSBmcm9tIFwiLi9pdGVyYXRpb25cIjtcbmltcG9ydCB7IGdldFByb3ZpZGVyUmVxdWlyZW1lbnRzIH0gZnJvbSBcIi4vcHJvdmlkZXJcIjtcbmltcG9ydCB7IGxvZ2dlciB9IGZyb20gXCIuL3V0aWxzXCI7XG5pbXBvcnQgeyBGUVBOIH0gZnJvbSBcIkBjZGt0bi9wcm92aWRlci1zY2hlbWFcIjtcbmltcG9ydCB7IGF0dHJpYnV0ZU5hbWVUb0Nka3RmTmFtZSB9IGZyb20gXCIuL2dlbmVyYXRpb25cIjtcbmltcG9ydCB7XG4gIHJlcGxhY2VDc2hhcnBJbXBvcnRzLFxuICByZXBsYWNlR29JbXBvcnRzLFxuICByZXBsYWNlSmF2YUltcG9ydHMsXG4gIHJlcGxhY2VQeXRob25JbXBvcnRzLFxufSBmcm9tIFwiLi9qc2lpLXJvc2V0dGEtd29ya2Fyb3VuZHNcIjtcbmltcG9ydCB7IFByb3ZpZGVyU2NoZW1hIH0gZnJvbSBcIkBjZGt0bi9jb21tb25zXCI7XG5pbXBvcnQgeyBmb3JFYWNoSW1wb3J0IH0gZnJvbSBcIi4vaXRlcmF0aW9uXCI7XG5cbmV4cG9ydCBjb25zdCBDT0RFX01BUktFUiA9IFwiLy8gZGVmaW5lIHJlc291cmNlcyBoZXJlXCI7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBnZXRQYXJzZWRIY2woaGNsOiBzdHJpbmcpIHtcbiAgbG9nZ2VyLmRlYnVnKGBQYXJzaW5nIEhDTDogJHtoY2x9YCk7XG4gIC8vIEdldCB0aGUgSlNPTiByZXByZXNlbnRhdGlvbiBvZiB0aGUgSENMXG4gIGxldCBqc29uOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgdHJ5IHtcbiAgICBqc29uID0gYXdhaXQgcGFyc2UoXCJ0ZXJyYWZvcm0udGZcIiwgaGNsKTtcbiAgfSBjYXRjaCAoZXJyKSB7XG4gICAgbG9nZ2VyLmVycm9yKGBGYWlsZWQgdG8gcGFyc2UgSENMOiAke2Vycn1gKTtcbiAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICBgRXJyb3I6IENvdWxkIG5vdCBwYXJzZSBIQ0wsIHRoaXMgbWVhbnMgZWl0aGVyIHRoYXQgdGhlIEhDTCBwYXNzZWQgaXMgaW52YWxpZCBvciB0aGF0IHlvdSBmb3VuZCBhIGJ1Zy4gSWYgdGhlIEhDTCBzZWVtcyB2YWxpZCwgcGxlYXNlIGZpbGUgYSBidWcgdW5kZXIgaHR0cHM6Ly9jZGsudGYvYnVncy9uZXcvY29udmVydGAsXG4gICAgKTtcbiAgfVxuXG4gIC8vIEVuc3VyZSB0aGUgSlNPTiByZXByZXNlbnRhdGlvbiBtYXRjaGVzIHRoZSBleHBlY3RlZCBzdHJ1Y3R1cmVcbiAgbGV0IHBsYW46IHouaW5mZXI8dHlwZW9mIHNjaGVtYT47XG4gIHRyeSB7XG4gICAgcGxhbiA9IHNjaGVtYS5wYXJzZShqc29uKTtcbiAgfSBjYXRjaCAoZXJyKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBFcnJvcjogSENMLUpTT04gZG9lcyBub3QgY29uZm9ybSB0byBzY2hlbWEuIFRoaXMgaXMgbm90IGV4cGVjdGVkLCBwbGVhc2UgZmlsZSBhIGJ1ZyB1bmRlciBodHRwczovL2Nkay50Zi9idWdzL25ldy9jb252ZXJ0XG5QbGVhc2UgaW5jbHVkZSB0aGlzIGluZm9ybWF0aW9uOlxuJHtKU09OLnN0cmluZ2lmeSgoZXJyIGFzIHouWm9kRXJyb3IpLmVycm9ycyl9YCk7XG4gIH1cblxuICByZXR1cm4gcGxhbjtcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHBhcnNlUHJvdmlkZXJSZXF1aXJlbWVudHMoaGNsOiBzdHJpbmcpIHtcbiAgbG9nZ2VyLmRlYnVnKFwiUGFyc2luZyBwcm92aWRlciByZXF1aXJlbWVudHNcIik7XG4gIGNvbnN0IHBsYW4gPSBhd2FpdCBnZXRQYXJzZWRIY2woaGNsKTtcbiAgcmV0dXJuIGdldFByb3ZpZGVyUmVxdWlyZW1lbnRzKHBsYW4pO1xufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gY29udmVydFRvVHlwZXNjcmlwdChcbiAgaGNsOiBzdHJpbmcsXG4gIHByb3ZpZGVyU2NoZW1hOiBQcm92aWRlclNjaGVtYSxcbiAgY29kZUNvbnRhaW5lcjogc3RyaW5nLFxuKSB7XG4gIGxvZ2dlci5kZWJ1ZyhcIkNvbnZlcnRpbmcgdG8gdHlwZXNjcmlwdFwiKTtcbiAgY29uc3QgcGxhbiA9IGF3YWl0IGdldFBhcnNlZEhjbChoY2wpO1xuXG4gIC8vIEVhY2gga2V5IGluIHRoZSBzY29wZSBuZWVkcyB0byBiZSB1bmlxdWUsIHRoZXJlZm9yZSB3ZSBzYXZlIHRoZW0gaW4gYSBzZXRcbiAgLy8gRWFjaCB2YXJpYWJsZSBuZWVkcyB0byBiZSB1bmlxdWUgYXMgd2VsbCwgd2Ugc2F2ZSB0aGVtIGluIGEgcmVjb3JkIHNvIHdlIGNhbiBpZGVudGlmeSBpZiB0d28gdmFyaWFibGVzIGFyZSB0aGUgc2FtZVxuICBjb25zdCBzY29wZTogUHJvZ3JhbVNjb3BlID0ge1xuICAgIHByb3ZpZGVyU2NoZW1hLFxuICAgIHByb3ZpZGVyR2VuZXJhdG9yOiBPYmplY3Qua2V5cyhcbiAgICAgIHByb3ZpZGVyU2NoZW1hLnByb3ZpZGVyX3NjaGVtYXMgfHwge30sXG4gICAgKS5yZWR1Y2UoKGNhcnJ5LCBmcXBuKSA9PiB7XG4gICAgICBjb25zdCBwcm92aWRlckdlbmVyYXRvciA9IG5ldyBUZXJyYWZvcm1Qcm92aWRlckdlbmVyYXRvcihcbiAgICAgICAgbmV3IENvZGVNYWtlcigpLFxuICAgICAgICBwcm92aWRlclNjaGVtYSxcbiAgICAgICk7XG4gICAgICBwcm92aWRlckdlbmVyYXRvci5idWlsZFJlc291cmNlTW9kZWxzKGZxcG4gYXMgRlFQTik7IC8vIGNhbid0IHVzZSB0aGF0IHR5cGUgb24gdGhlIGtleXMgeWV0LCBzaW5jZSB3ZSBhcmUgbm90IG9uIFRTID49NC40IHlldCA6c2FkY2F0OlxuICAgICAgcmV0dXJuIHsgLi4uY2FycnksIFtmcXBuXTogcHJvdmlkZXJHZW5lcmF0b3IgfTtcbiAgICB9LCB7fSksXG4gICAgY29uc3RydWN0czogbmV3IFNldDxzdHJpbmc+KCksXG4gICAgdmFyaWFibGVzOiB7fSxcbiAgICBoYXNUb2tlbkJhc2VkVHlwZUNvZXJjaW9uOiBmYWxzZSxcbiAgICBub2RlSWRzOiBbXSxcbiAgICBpbXBvcnRhYmxlczogW10sXG4gICAgdG9wTGV2ZWxDb25maWc6IHt9LFxuICB9O1xuXG4gIGNvbnN0IGdyYXBoID0gbmV3IERpcmVjdGVkR3JhcGg8e1xuICAgIGNvZGU6IChcbiAgICAgIGc6IERpcmVjdGVkR3JhcGg8YW55PixcbiAgICApID0+IFByb21pc2U8QXJyYXk8dC5TdGF0ZW1lbnQgfCB0LlZhcmlhYmxlRGVjbGFyYXRpb24+PjtcbiAgfT4oKTtcblxuICAvLyBHZXQgYWxsIGl0ZW1zIGluIHRoZSBKU09OIGFzIGEgbWFwIG9mIGlkIHRvIGZ1bmN0aW9uIHRoYXQgZ2VuZXJhdGVzIHRoZSBBU1RcbiAgLy8gV2Ugd2lsbCB1c2UgdGhpcyB0byBjb25zdHJ1Y3QgdGhlIG5vZGVzIGZvciBhIGRlcGVuZGVuY3kgZ3JhcGhcbiAgLy8gV2UgbmVlZCB0byB1c2UgYSBmdW5jdGlvbiBoZXJlIGJlY2F1c2UgdGhlIHNhbWUgbm9kZSBoYXMgZGlmZmVyZW50IHJlcHJlc2VudGF0aW9uIGJhc2VkIG9uIGlmIGl0J3MgcmVmZXJlbmNlZCBieSBhbm90aGVyIG9uZVxuICBjb25zdCBub2RlTWFwOiBSZWNvcmQ8XG4gICAgc3RyaW5nLFxuICAgIHtcbiAgICAgIGNvZGU6IChcbiAgICAgICAgZzogdHlwZW9mIGdyYXBoLFxuICAgICAgKSA9PiBQcm9taXNlPEFycmF5PHQuU3RhdGVtZW50IHwgdC5WYXJpYWJsZURlY2xhcmF0aW9uPj47XG4gICAgICB2YWx1ZTogdW5rbm93bjtcbiAgICB9XG4gID4gPSB7XG4gICAgLi4uZm9yRWFjaFByb3ZpZGVyKHNjb3BlLCBwbGFuLnByb3ZpZGVyLCBwcm92aWRlciksXG4gICAgLi4uZm9yRWFjaEdsb2JhbChzY29wZSwgXCJ2YXJcIiwgcGxhbi52YXJpYWJsZSwgdmFyaWFibGUpLFxuICAgIC8vIGxvY2FscyBhcmUgYSBzcGVjaWFsIGNhc2VcbiAgICAuLi5mb3JFYWNoR2xvYmFsKFxuICAgICAgc2NvcGUsXG4gICAgICBcImxvY2FsXCIsXG4gICAgICBBcnJheS5pc0FycmF5KHBsYW4ubG9jYWxzKVxuICAgICAgICA/IHBsYW4ubG9jYWxzLnJlZHVjZSgoY2FycnksIGxvY2FscykgPT4gKHsgLi4uY2FycnksIC4uLmxvY2FscyB9KSwge30pXG4gICAgICAgIDoge30sXG4gICAgICBsb2NhbCxcbiAgICApLFxuICAgIC4uLmZvckVhY2hHbG9iYWwoc2NvcGUsIFwib3V0XCIsIHBsYW4ub3V0cHV0LCBvdXRwdXQpLFxuICAgIC4uLmZvckVhY2hHbG9iYWwoc2NvcGUsIFwibW9kdWxlXCIsIHBsYW4ubW9kdWxlLCBtb2R1bGVzKSxcbiAgICAuLi5mb3JFYWNoSW1wb3J0KHNjb3BlLCBcImltcG9ydFwiLCBwbGFuLmltcG9ydCwgaW1wb3J0cyksXG4gICAgLi4uZm9yRWFjaE5hbWVzcGFjZWQoc2NvcGUsIHBsYW4ucmVzb3VyY2UsIHJlc291cmNlKSxcbiAgICAuLi5mb3JFYWNoTmFtZXNwYWNlZChzY29wZSwgcGxhbi5kYXRhLCByZXNvdXJjZSwgXCJkYXRhXCIpLFxuICB9O1xuXG4gIC8vIEFkZCBhbGwgbm9kZXMgdG8gdGhlIGRlcGVuZGVuY3kgZ3JhcGggc28gd2UgY2FuIGRldGVjdCBpZiBhbiBlZGdlIGlzIGFkZGVkIGZvciBhbiB1bmtub3duIGxpbmtcbiAgT2JqZWN0LmVudHJpZXMobm9kZU1hcCkuZm9yRWFjaCgoW2tleSwgdmFsdWVdKSA9PiB7XG4gICAgbG9nZ2VyLmRlYnVnKGBBZGRpbmcgbm9kZSAnJHtrZXl9JyB0byBncmFwaGApO1xuICAgIGdyYXBoLmFkZE5vZGUoa2V5LCB2YWx1ZSk7XG4gIH0pO1xuXG4gIC8vIEZpbmRpbmcgcmVmZXJlbmNlcyBiZWNvbWVzIGVhc2llciBvZiB0aGUgdG8gYmUgcmVmZXJlbmNlZCBpZHMgYXJlIGFscmVhZHkga25vd25cbiAgY29uc3Qgbm9kZUlkcyA9IE9iamVjdC5rZXlzKG5vZGVNYXApO1xuICBzY29wZS5ub2RlSWRzID0gbm9kZUlkcztcbiAgYXN5bmMgZnVuY3Rpb24gYWRkRWRnZXMoaWQ6IHN0cmluZywgdmFsdWU6IFRlcnJhZm9ybVJlc291cmNlQmxvY2spIHtcbiAgICAoYXdhaXQgZmluZFVzZWRSZWZlcmVuY2VzKG5vZGVJZHMsIHZhbHVlKSkuZm9yRWFjaCgocmVmKSA9PiB7XG4gICAgICBpZiAoXG4gICAgICAgICFncmFwaC5oYXNEaXJlY3RlZEVkZ2UocmVmLnJlZmVyZW5jZWUuaWQsIGlkKSAmJlxuICAgICAgICBncmFwaC5oYXNOb2RlKHJlZi5yZWZlcmVuY2VlLmlkKSAvLyBpbiBjYXNlIHRoZSByZWZlcmVuY2VlIGlzIGEgZHluYW1pYyB2YXJpYWJsZVxuICAgICAgKSB7XG4gICAgICAgIGlmICghZ3JhcGguaGFzTm9kZShpZCkpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgICBgVGhlIGRlcGVuZGVuY3kgZ3JhcGggaXMgZXhwZWN0ZWQgdG8gbGluayBmcm9tICR7XG4gICAgICAgICAgICAgIHJlZi5yZWZlcmVuY2VlLmlkXG4gICAgICAgICAgICB9IHRvICR7aWR9IGJ1dCAke2lkfSBkb2VzIG5vdCBleGlzdC5cbiAgICAgICAgICAgIFRoZXNlIG5vZGVzIGV4aXN0OiAke2dyYXBoLm5vZGVzKCkuam9pbihcIlxcblwiKX1gLFxuICAgICAgICAgICk7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBUaGUgZ3JhcGggc2hvdWxkIGhhdmUgbm8gc2VsZi1yZWZlcmVuY2VzXG4gICAgICAgIGlmIChpZCA9PT0gcmVmLnJlZmVyZW5jZWUuaWQpIHtcbiAgICAgICAgICBsb2dnZXIuZGVidWcoYFNraXBwaW5nIHNlbGYtcmVmZXJlbmNlIGZvciAke2lkfWApO1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgQWRkaW5nIGVkZ2UgZnJvbSAke3JlZi5yZWZlcmVuY2VlLmlkfSB0byAke2lkfWApO1xuICAgICAgICBncmFwaC5hZGREaXJlY3RlZEVkZ2UocmVmLnJlZmVyZW5jZWUuaWQsIGlkLCB7IHJlZiB9KTtcbiAgICAgIH1cbiAgICB9KTtcbiAgfVxuXG4gIC8vIFdlIHJlY3Vyc2l2ZWx5IGluc3BlY3QgZWFjaCByZXNvdXJjZSB2YWx1ZSB0byBmaW5kIHJlZmVyZW5jZXMgdG8gb3RoZXIgdmFsdWVzXG4gIC8vIFdlIGFkZCB0aGVzZSB0byBhIGRlcGVuZGVuY3kgZ3JhcGggc28gdGhhdCB0aGUgcHJvZ3JhbW1pbmcgY29kZSBoYXMgdGhlIHJpZ2h0IG9yZGVyXG4gIGFzeW5jIGZ1bmN0aW9uIGFkZEdsb2JhbEVkZ2VzKFxuICAgIF9zY29wZTogUHJvZ3JhbVNjb3BlLFxuICAgIF9rZXk6IHN0cmluZyxcbiAgICBpZDogc3RyaW5nLFxuICAgIHZhbHVlOiBUZXJyYWZvcm1SZXNvdXJjZUJsb2NrLFxuICApIHtcbiAgICBhd2FpdCBhZGRFZGdlcyhpZCwgdmFsdWUpO1xuICB9XG4gIGFzeW5jIGZ1bmN0aW9uIGFkZFByb3ZpZGVyRWRnZXMoXG4gICAgX3Njb3BlOiBQcm9ncmFtU2NvcGUsXG4gICAgX2tleTogc3RyaW5nLFxuICAgIGlkOiBzdHJpbmcsXG4gICAgdmFsdWU6IFRlcnJhZm9ybVJlc291cmNlQmxvY2ssXG4gICkge1xuICAgIGF3YWl0IGFkZEVkZ2VzKGlkLCB2YWx1ZSk7XG4gIH1cbiAgYXN5bmMgZnVuY3Rpb24gYWRkTmFtZXNwYWNlZEVkZ2VzKFxuICAgIF9zY29wZTogUHJvZ3JhbVNjb3BlLFxuICAgIF90eXBlOiBzdHJpbmcsXG4gICAgX2tleTogc3RyaW5nLFxuICAgIGlkOiBzdHJpbmcsXG4gICAgdmFsdWU6IFRlcnJhZm9ybVJlc291cmNlQmxvY2ssXG4gICkge1xuICAgIGF3YWl0IGFkZEVkZ2VzKGlkLCB2YWx1ZSk7XG4gIH1cblxuICBhd2FpdCBQcm9taXNlLmFsbChcbiAgICBPYmplY3QudmFsdWVzKHtcbiAgICAgIC4uLmZvckVhY2hQcm92aWRlcihzY29wZSwgcGxhbi5wcm92aWRlciwgYWRkUHJvdmlkZXJFZGdlcyksXG4gICAgICAuLi5mb3JFYWNoR2xvYmFsKHNjb3BlLCBcInZhclwiLCBwbGFuLnZhcmlhYmxlLCBhZGRHbG9iYWxFZGdlcyksXG4gICAgICAvLyBsb2NhbHMgYXJlIGEgc3BlY2lhbCBjYXNlXG4gICAgICAuLi5mb3JFYWNoR2xvYmFsKFxuICAgICAgICBzY29wZSxcbiAgICAgICAgXCJsb2NhbFwiLFxuICAgICAgICBBcnJheS5pc0FycmF5KHBsYW4ubG9jYWxzKVxuICAgICAgICAgID8gcGxhbi5sb2NhbHMucmVkdWNlKChjYXJyeSwgbG9jYWxzKSA9PiAoeyAuLi5jYXJyeSwgLi4ubG9jYWxzIH0pLCB7fSlcbiAgICAgICAgICA6IHt9LFxuICAgICAgICBhZGRHbG9iYWxFZGdlcyxcbiAgICAgICksXG4gICAgICAuLi5mb3JFYWNoR2xvYmFsKHNjb3BlLCBcIm91dFwiLCBwbGFuLm91dHB1dCwgYWRkR2xvYmFsRWRnZXMpLFxuICAgICAgLi4uZm9yRWFjaEdsb2JhbChzY29wZSwgXCJtb2R1bGVcIiwgcGxhbi5tb2R1bGUsIGFkZEdsb2JhbEVkZ2VzKSxcbiAgICAgIC4uLmZvckVhY2hOYW1lc3BhY2VkKHNjb3BlLCBwbGFuLnJlc291cmNlLCBhZGROYW1lc3BhY2VkRWRnZXMpLFxuICAgICAgLi4uZm9yRWFjaE5hbWVzcGFjZWQoc2NvcGUsIHBsYW4uZGF0YSwgYWRkTmFtZXNwYWNlZEVkZ2VzLCBcImRhdGFcIiksXG4gICAgfSkubWFwKCh7IGNvZGU6IGFkZEVkZ2VzVG9HcmFwaCB9KSA9PiBhZGRFZGdlc1RvR3JhcGgoZ3JhcGgpKSxcbiAgKTtcblxuICBsb2dnZXIuZGVidWcoYEdyYXBoOiAke0pTT04uc3RyaW5naWZ5KGdyYXBoLCBudWxsLCAyKX1gKTtcbiAgbG9nZ2VyLmRlYnVnKGBTdGFydGluZyB0byBhc3NlbWJsZSB0aGUgdHlwZXNjcmlwdCBjb2RlYCk7XG4gIC8vIFdlIHRyYXZlcnNlIHRoZSBkZXBlbmRlbmN5IGdyYXBoIHRvIGdldCB0aGUgdW5vcmRlcmVkIEpTT04gbm9kZXMgaW50byBhbiBvcmRlcmVkIGFycmF5XG4gIC8vIHdoZXJlIG5vIG5vZGUgaXMgcmVmZXJlbmNlZCBiZWZvcmUgaXQncyBkZWZpbmVkXG4gIC8vIEFzIHdlIGNoZWNrIHRoYXQgdGhlIG5vZGVzIG9uIGJvdGggZW5kcyBvZiBhbiBlZGdlIGV4aXN0IHdlIGNhbiBiZSBzdXJlXG4gIC8vIHRoYXQgbm8gaW5maW5pdGUgbG9vcCBleGlzdHMsIHRoZXJlIGNhbiBiZSBubyBzdHJheSBkZXBlbmRlbmN5IG9uIGEgbm9kZVxuICBjb25zdCBleHByZXNzaW9uczogdC5TdGF0ZW1lbnRbXSA9IFtdO1xuICBsZXQgbm9kZXNUb1Zpc2l0ID0gWy4uLm5vZGVJZHNdO1xuICAvLyBUaGlzIGVuc3VyZXMgd2UgZGV0ZWN0IGN5Y2xlcyBhbmQgZG9uJ3QgZW5kIHVwIGluIGFuIGVuZGxlc3MgbG9vcFxuICBsZXQgbm9kZXNWaXNpdGVkVGhpc0l0ZXJhdGlvbiA9IDA7XG4gIGRvIHtcbiAgICBub2Rlc1Zpc2l0ZWRUaGlzSXRlcmF0aW9uID0gMDtcblxuICAgIC8vIEZpbmQgbmV4dCBub2RlcyB0byB2aXNpdFxuICAgIGNvbnN0IG5vZGVFeHByZXNzaW9uR2VuZXJhdG9ycyA9IGdyYXBoLm1hcE5vZGVzKChub2RlSWQsIHsgY29kZSB9KSA9PiB7XG4gICAgICBpZiAoIW5vZGVzVG9WaXNpdC5pbmNsdWRlcyhub2RlSWQpKSB7XG4gICAgICAgIHJldHVybiB1bmRlZmluZWQ7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHVucmVzb2x2ZWREZXBlbmRlbmNpZXMgPSBncmFwaFxuICAgICAgICAuaW5OZWlnaGJvcnMobm9kZUlkKVxuICAgICAgICAuZmlsdGVyKChpdGVtKSA9PiBub2Rlc1RvVmlzaXQuaW5jbHVkZXMoaXRlbSkpO1xuXG4gICAgICBpZiAodW5yZXNvbHZlZERlcGVuZGVuY2llcy5sZW5ndGggPT09IDApIHtcbiAgICAgICAgbm9kZXNUb1Zpc2l0ID0gbm9kZXNUb1Zpc2l0LmZpbHRlcigoaWQpID0+IG5vZGVJZCAhPT0gaWQpO1xuICAgICAgICBub2Rlc1Zpc2l0ZWRUaGlzSXRlcmF0aW9uID0gbm9kZXNWaXNpdGVkVGhpc0l0ZXJhdGlvbiArIDE7XG5cbiAgICAgICAgbG9nZ2VyLmRlYnVnKGBWaXNpdGluZyBub2RlICR7bm9kZUlkfWApO1xuICAgICAgICByZXR1cm4gY29kZTtcbiAgICAgIH1cbiAgICAgIHJldHVybiB1bmRlZmluZWQ7XG4gICAgfSk7XG5cbiAgICAvLyBHZW5lcmF0ZSB0aGUgY29kZSBmb3IgdGhlIG5vZGVzXG4gICAgZm9yIChjb25zdCBjb2RlIG9mIG5vZGVFeHByZXNzaW9uR2VuZXJhdG9ycykge1xuICAgICAgaWYgKGNvZGUpIHtcbiAgICAgICAgZXhwcmVzc2lvbnMucHVzaCguLi4oYXdhaXQgY29kZShncmFwaCkpKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBsb2dnZXIuZGVidWcoXG4gICAgICBgJHtub2Rlc1RvVmlzaXQubGVuZ3RofSB1bnZpc2l0ZWQgbm9kZXM6ICR7bm9kZXNUb1Zpc2l0LmpvaW4oXCIsIFwiKX1gLFxuICAgICk7XG4gIH0gd2hpbGUgKG5vZGVzVG9WaXNpdC5sZW5ndGggPiAwICYmIG5vZGVzVmlzaXRlZFRoaXNJdGVyYXRpb24gIT0gMCk7XG5cbiAgaWYgKG5vZGVzVG9WaXNpdC5sZW5ndGggPiAwKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgYFRoZXJlIGFyZSAke1xuICAgICAgICBub2Rlc1RvVmlzaXQubGVuZ3RoXG4gICAgICB9IHRlcnJhZm9ybSBlbGVtZW50cyB0aGF0IGNvdWxkIG5vdCBiZSB2aXNpdGVkLlxuICAgICAgVGhpcyBpcyBsaWtlbHkgZHVlIHRvIGEgY3ljbGUgaW4gdGhlIGRlcGVuZGVuY3kgZ3JhcGguXG4gICAgICBUaGVzZSBub2RlcyBhcmU6ICR7bm9kZXNUb1Zpc2l0LmpvaW4oXCIsIFwiKX1gLFxuICAgICk7XG4gIH1cblxuICBsb2dnZXIuZGVidWcoXG4gICAgYCR7bm9kZXNUb1Zpc2l0Lmxlbmd0aH0gdW52aXNpdGVkIG5vZGVzOiAke25vZGVzVG9WaXNpdC5qb2luKFwiLCBcIil9YCxcbiAgKTtcblxuICBjb25zdCBiYWNrZW5kRXhwcmVzc2lvbnMgPSAoXG4gICAgYXdhaXQgUHJvbWlzZS5hbGwoXG4gICAgICBwbGFuLnRlcnJhZm9ybT8ubWFwKCh0ZXJyYWZvcm0pID0+XG4gICAgICAgIGJhY2tlbmRUb0V4cHJlc3Npb24oc2NvcGUsIHRlcnJhZm9ybS5iYWNrZW5kKSxcbiAgICAgICkgfHwgW1Byb21pc2UucmVzb2x2ZShbXSldLFxuICAgIClcbiAgKS5yZWR1Y2UoKGNhcnJ5LCBpdGVtKSA9PiBbLi4uY2FycnksIC4uLml0ZW1dLCBbXSk7XG5cbiAgbG9nZ2VyLmRlYnVnKFxuICAgIGBVc2luZyB0aGVzZSBiYWNrZW5kIGV4cHJlc3Npb25zOiAke0pTT04uc3RyaW5naWZ5KFxuICAgICAgYmFja2VuZEV4cHJlc3Npb25zLFxuICAgICAgbnVsbCxcbiAgICAgIDIsXG4gICAgKX1gLFxuICApO1xuXG4gIC8vIFdlIGNvbGxlY3QgYWxsIG1vZHVsZSBzb3VyY2VzXG4gIGNvbnN0IG1vZHVsZVJlcXVpcmVtZW50cyA9IFtcbiAgICAuLi5uZXcgU2V0KFxuICAgICAgT2JqZWN0LnZhbHVlcyhwbGFuLm1vZHVsZSB8fCB7fSkucmVkdWNlKFxuICAgICAgICAoY2FycnksIG1vZHVsZUJsb2NrKSA9PiBbXG4gICAgICAgICAgLi4uY2FycnksXG4gICAgICAgICAgLi4ubW9kdWxlQmxvY2sucmVkdWNlKFxuICAgICAgICAgICAgKGFyciwgeyBzb3VyY2UsIHZlcnNpb24gfSkgPT4gW1xuICAgICAgICAgICAgICAuLi5hcnIsXG4gICAgICAgICAgICAgIHZlcnNpb24gPyBgJHtzb3VyY2V9QCR7dmVyc2lvbn1gIDogc291cmNlLFxuICAgICAgICAgICAgXSxcbiAgICAgICAgICAgIFtdIGFzIHN0cmluZ1tdLFxuICAgICAgICAgICksXG4gICAgICAgIF0sXG4gICAgICAgIFtdIGFzIHN0cmluZ1tdLFxuICAgICAgKSB8fCBbXSxcbiAgICApLFxuICBdO1xuXG4gIGxvZ2dlci5kZWJ1ZyhcbiAgICBgRm91bmQgdGhlc2UgbW9kdWxlczogJHtKU09OLnN0cmluZ2lmeShtb2R1bGVSZXF1aXJlbWVudHMsIG51bGwsIDIpfWAsXG4gICk7XG5cbiAgaWYgKE9iamVjdC5rZXlzKHBsYW4udmFyaWFibGUgfHwge30pLmxlbmd0aCA+IDAgJiYgZXhwcmVzc2lvbnMubGVuZ3RoID4gMCkge1xuICAgIGV4cHJlc3Npb25zWzBdID0gdC5hZGRDb21tZW50KFxuICAgICAgZXhwcmVzc2lvbnNbMF0sXG4gICAgICBcImxlYWRpbmdcIixcbiAgICAgIGBUZXJyYWZvcm0gVmFyaWFibGVzIGFyZSBub3QgYWx3YXlzIHRoZSBiZXN0IGZpdCBmb3IgZ2V0dGluZyBpbnB1dHMgaW4gdGhlIGNvbnRleHQgb2YgVGVycmFmb3JtIENESy5cbllvdSBjYW4gcmVhZCBtb3JlIGFib3V0IHRoaXMgYXQgaHR0cHM6Ly9jZGsudGYvdmFyaWFibGVzYCxcbiAgICApO1xuICB9XG5cbiAgY29uc3QgcHJvdmlkZXJSZXF1aXJlbWVudHMgPSBnZXRQcm92aWRlclJlcXVpcmVtZW50cyhwbGFuKTtcbiAgbG9nZ2VyLmRlYnVnKFxuICAgIGBGb3VuZCB0aGVzZSBwcm92aWRlciByZXF1aXJlbWVudHM6ICR7SlNPTi5zdHJpbmdpZnkoXG4gICAgICBwcm92aWRlclJlcXVpcmVtZW50cyxcbiAgICAgIG51bGwsXG4gICAgICAyLFxuICAgICl9YCxcbiAgKTtcblxuICAvLyBXZSBhZGQgYSBjb21tZW50IGlmIHRoZXJlIGFyZSBwcm92aWRlcnMgd2l0aCBtaXNzaW5nIHNjaGVtYSBpbmZvcm1hdGlvblxuICBjb25zdCBwcm92aWRlcnNMYWNraW5nU2NoZW1hID0gT2JqZWN0LmtleXMocHJvdmlkZXJSZXF1aXJlbWVudHMpLmZpbHRlcihcbiAgICAocHJvdmlkZXJOYW1lKSA9PlxuICAgICAgcHJvdmlkZXJOYW1lICE9PSBcInRlcnJhZm9ybVwiICYmXG4gICAgICAhT2JqZWN0LmtleXMocHJvdmlkZXJTY2hlbWEucHJvdmlkZXJfc2NoZW1hcyB8fCB7fSkuc29tZSgoc2NoZW1hTmFtZSkgPT5cbiAgICAgICAgc2NoZW1hTmFtZS5lbmRzV2l0aChwcm92aWRlck5hbWUpLFxuICAgICAgKSxcbiAgKTtcbiAgbG9nZ2VyLmRlYnVnKFxuICAgIGAke1xuICAgICAgcHJvdmlkZXJzTGFja2luZ1NjaGVtYS5sZW5ndGhcbiAgICB9IHByb3ZpZGVycyBsYWNrIHNjaGVtYSBpbmZvcm1hdGlvbjogJHtwcm92aWRlcnNMYWNraW5nU2NoZW1hLmpvaW4oXCIsIFwiKX1gLFxuICApO1xuXG4gIGlmIChwcm92aWRlcnNMYWNraW5nU2NoZW1hLmxlbmd0aCA+IDApIHtcbiAgICBleHByZXNzaW9uc1swXSA9IHQuYWRkQ29tbWVudChcbiAgICAgIGV4cHJlc3Npb25zWzBdLFxuICAgICAgXCJsZWFkaW5nXCIsXG4gICAgICBgVGhlIGZvbGxvd2luZyBwcm92aWRlcnMgYXJlIG1pc3Npbmcgc2NoZW1hIGluZm9ybWF0aW9uIGFuZCBtaWdodCBuZWVkIG1hbnVhbCBhZGp1c3RtZW50cyB0byBzeW50aGVzaXplIGNvcnJlY3RseTogJHtwcm92aWRlcnNMYWNraW5nU2NoZW1hLmpvaW4oXG4gICAgICAgIFwiLCBcIixcbiAgICAgICl9LlxuRm9yIGEgbW9yZSBwcmVjaXNlIGNvbnZlcnNpb24gcGxlYXNlIHVzZSB0aGUgLS1wcm92aWRlciBmbGFnIGluIGNvbnZlcnQuYCxcbiAgICApO1xuICB9XG5cbiAgLy8gQWx3YXlzIGFkZCBjb25zdHJ1Y3RzXG4gIHNjb3BlLmltcG9ydGFibGVzLnB1c2goe1xuICAgIGNvbnN0cnVjdE5hbWU6IFwiQ29uc3RydWN0XCIsXG4gICAgcHJvdmlkZXI6IFwiY29uc3RydWN0c1wiLFxuICB9KTtcblxuICBpZiAoc2NvcGUuaGFzVG9rZW5CYXNlZFR5cGVDb2VyY2lvbikge1xuICAgIHNjb3BlLmltcG9ydGFibGVzLnB1c2goe1xuICAgICAgY29uc3RydWN0TmFtZTogXCJUb2tlblwiLFxuICAgICAgcHJvdmlkZXI6IFwiY2RrdG5cIixcbiAgICB9KTtcbiAgfVxuXG4gIC8vIEFkZCBzcGVjaWZpYyBpbXBvcnQgZm9yIGNvZGVDb250YWluZXJcbiAgYWRkSW1wb3J0Rm9yQ29kZUNvbnRhaW5lcihzY29wZSwgY29kZUNvbnRhaW5lcik7XG4gIGNvbnN0IGNvbnN0cnVjdEltcG9ydHMgPSBidWlsZEltcG9ydHMoc2NvcGUuaW1wb3J0YWJsZXMpO1xuXG4gIGNvbnN0IGNvZGUgPSBbLi4uKGJhY2tlbmRFeHByZXNzaW9ucyB8fCBbXSksIC4uLmV4cHJlc3Npb25zXTtcbiAgY29uc3QgY29uZmlnVHlwZU5hbWUgPVxuICAgIE9iamVjdC5rZXlzKHNjb3BlLnRvcExldmVsQ29uZmlnKS5sZW5ndGggPiAwID8gXCJNeUNvbmZpZ1wiIDogdW5kZWZpbmVkO1xuXG4gIGNvbnN0IGNsYXNzQ29uZmlnID0gY29uZmlnVHlwZU5hbWVcbiAgICA/IFtnZW5lcmF0ZUNvbmZpZ1R5cGUoY29uZmlnVHlwZU5hbWUsIHNjb3BlLnRvcExldmVsQ29uZmlnKV1cbiAgICA6IFtdO1xuXG4gIC8vIFdlIHNwbGl0IHVwIHRoZSBnZW5lcmF0ZWQgY29kZSBzbyB0aGF0IHVzZXJzIGNhbiBoYXZlIG1vcmUgY29udHJvbCBvdmVyIHdoYXQgdG8gaW5zZXJ0IHdoZXJlXG4gIHJldHVybiB7XG4gICAgLy8gVE9ETzogUmVtb3ZlIGltcG9ydHMgYW5kIGNvZGUgYmVjYXVzZSByb3NldHRhIHdvbid0IGJlIGFibGUgdG8gdHJhbnNsYXRlIHRoZW1cbiAgICBhbGw6IGF3YWl0IGdlbihbXG4gICAgICAuLi5jb25zdHJ1Y3RJbXBvcnRzLFxuICAgICAgLi4ubW9kdWxlSW1wb3J0cyhwbGFuLm1vZHVsZSksXG4gICAgICAuLi5jbGFzc0NvbmZpZyxcbiAgICAgIHdyYXBDb2RlSW5Db25zdHJ1Y3RvcihcbiAgICAgICAgY29kZUNvbnRhaW5lcixcbiAgICAgICAgY29kZSxcbiAgICAgICAgXCJNeUNvbnZlcnRlZENvZGVcIixcbiAgICAgICAgY29uZmlnVHlwZU5hbWUsXG4gICAgICApLFxuICAgIF0pLFxuICAgIGltcG9ydHM6IGF3YWl0IGdlbihbLi4uY29uc3RydWN0SW1wb3J0cywgLi4ubW9kdWxlSW1wb3J0cyhwbGFuLm1vZHVsZSldKSxcbiAgICBjb2RlOiBhd2FpdCBnZW4oY29kZSksXG4gICAgcHJvdmlkZXJzOiBPYmplY3QuZW50cmllcyhwcm92aWRlclJlcXVpcmVtZW50cykubWFwKChbc291cmNlLCB2ZXJzaW9uXSkgPT5cbiAgICAgIHZlcnNpb24gPT09IFwiKlwiID8gc291cmNlIDogYCR7c291cmNlfUAke3ZlcnNpb259YCxcbiAgICApLFxuICAgIG1vZHVsZXM6IG1vZHVsZVJlcXVpcmVtZW50cyxcbiAgICAvLyBXZSB0cmFjayBzb21lIHVzYWdlIGRhdGEgdG8gbWFrZSBpdCBlYXNpZXIgdG8gdW5kZXJzdGFuZCB3aGF0IGlzIHVzZWRcbiAgICBzdGF0czoge1xuICAgICAgbnVtYmVyT2ZNb2R1bGVzOiBtb2R1bGVSZXF1aXJlbWVudHMubGVuZ3RoLFxuICAgICAgbnVtYmVyT2ZQcm92aWRlcnM6IE9iamVjdC5rZXlzKHByb3ZpZGVyUmVxdWlyZW1lbnRzKS5sZW5ndGgsXG4gICAgICByZXNvdXJjZXM6IHJlc291cmNlU3RhdHMocGxhbi5yZXNvdXJjZSB8fCB7fSksXG4gICAgICBkYXRhOiByZXNvdXJjZVN0YXRzKHBsYW4uZGF0YSB8fCB7fSksXG4gICAgICBjb252ZXJ0ZWRMaW5lczogaGNsLnNwbGl0KFwiXFxuXCIpLmxlbmd0aCxcbiAgICB9LFxuICB9O1xufVxuXG50eXBlIEZpbGUgPSB7IGNvbnRlbnRzOiBzdHJpbmc7IGZpbGVOYW1lOiBzdHJpbmcgfTtcbmNvbnN0IHRyYW5zbGF0b3JzID0ge1xuICBweXRob246IHtcbiAgICB2aXNpdG9yOiAoKSA9PiBuZXcgcm9zZXR0YS5QeXRob25WaXNpdG9yKCksXG4gICAgcG9zdFRyYW5zbGF0aW9uTXV0YXRpb246IHJlcGxhY2VQeXRob25JbXBvcnRzLFxuICB9LFxuICBqYXZhOiB7XG4gICAgdmlzaXRvcjogKCkgPT4gbmV3IHJvc2V0dGEuSmF2YVZpc2l0b3IoKSxcbiAgICBwb3N0VHJhbnNsYXRpb25NdXRhdGlvbjogcmVwbGFjZUphdmFJbXBvcnRzLFxuICB9LFxuICBjc2hhcnA6IHtcbiAgICB2aXNpdG9yOiAoKSA9PiBuZXcgcm9zZXR0YS5DU2hhcnBWaXNpdG9yKCksXG4gICAgcG9zdFRyYW5zbGF0aW9uTXV0YXRpb246IHJlcGxhY2VDc2hhcnBJbXBvcnRzLFxuICB9LFxuICBnbzoge1xuICAgIHZpc2l0b3I6ICgpID0+IG5ldyByb3NldHRhLkdvVmlzaXRvcigpLFxuICAgIHBvc3RUcmFuc2xhdGlvbk11dGF0aW9uOiByZXBsYWNlR29JbXBvcnRzLFxuICB9LFxufTtcblxuZnVuY3Rpb24gdHJhbnNsYXRvckZvckxhbmd1YWdlKGxhbmd1YWdlOiBrZXlvZiB0eXBlb2YgdHJhbnNsYXRvcnMpIHtcbiAgcmV0dXJuIChmaWxlOiBGaWxlLCB0aHJvd09uVHJhbnNsYXRpb25FcnJvcjogYm9vbGVhbikgPT4ge1xuICAgIGNvbnN0IHsgdmlzaXRvciwgcG9zdFRyYW5zbGF0aW9uTXV0YXRpb24gfSA9IHRyYW5zbGF0b3JzW2xhbmd1YWdlXTtcbiAgICBjb25zdCB7IHRyYW5zbGF0aW9uLCBkaWFnbm9zdGljcyB9ID0gcm9zZXR0YS50cmFuc2xhdGVUeXBlU2NyaXB0KFxuICAgICAgZmlsZSxcbiAgICAgIHZpc2l0b3IoKSxcbiAgICAgIHRocm93T25UcmFuc2xhdGlvbkVycm9yID8geyBpbmNsdWRlQ29tcGlsZXJEaWFnbm9zdGljczogdHJ1ZSB9IDoge30sXG4gICAgKTtcblxuICAgIGlmIChcbiAgICAgIHRocm93T25UcmFuc2xhdGlvbkVycm9yICYmXG4gICAgICBkaWFnbm9zdGljcy5maWx0ZXIoKGRpYWcpID0+IGRpYWcuaXNFcnJvcikubGVuZ3RoID4gMFxuICAgICkge1xuICAgICAgbG9nZ2VyLmRlYnVnKGBDb3VsZCBub3QgdHJhbnNsYXRlIFRTIHRvICR7bGFuZ3VhZ2V9OlxcbiR7ZmlsZS5jb250ZW50c31gKTtcbiAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgYENvdWxkIG5vdCB0cmFuc2xhdGUgVFMgdG8gJHtsYW5ndWFnZX06ICR7ZGlhZ25vc3RpY3NcbiAgICAgICAgICAubWFwKChkaWFnKSA9PiBkaWFnLmZvcm1hdHRlZE1lc3NhZ2UpXG4gICAgICAgICAgLmpvaW4oXCJcXG5cIil9YCxcbiAgICAgICk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHBvc3RUcmFuc2xhdGlvbk11dGF0aW9uKHRyYW5zbGF0aW9uKTtcbiAgfTtcbn1cblxudHlwZSBDb252ZXJ0T3B0aW9ucyA9IHtcbiAgLyoqXG4gICAqIFRoZSBsYW5ndWFnZSB0byBjb252ZXJ0IHRvXG4gICAqL1xuICBsYW5ndWFnZToga2V5b2YgdHlwZW9mIHRyYW5zbGF0b3JzIHwgXCJ0eXBlc2NyaXB0XCI7XG4gIC8qKlxuICAgKiBUaGUgcHJvdmlkZXIgc2NoZW1hIHRvIHVzZSBmb3IgY29udmVyc2lvblxuICAgKi9cbiAgcHJvdmlkZXJTY2hlbWE6IFByb3ZpZGVyU2NoZW1hO1xuICAvKipcbiAgICogVGhlIGJhc2UgY2xhc3MgdG8gZXh0ZW5kIGZyb20uIERlZmF1bHRzIHRvIGBjb25zdHJ1Y3RzLkNvbnN0cnVjdGBcbiAgICovXG4gIGNvZGVDb250YWluZXI/OiBzdHJpbmc7XG4gIC8qKlxuICAgKiBXaGV0aGVyIHRvIHRocm93IGFuIGVycm9yIGlmIHRoZSB0cmFuc2xhdGlvbiBmYWlsc1xuICAgKiBEZWZhdWx0cyB0byBmYWxzZVxuICAgKi9cbiAgdGhyb3dPblRyYW5zbGF0aW9uRXJyb3I/OiBib29sZWFuO1xufTtcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNvbnZlcnQoXG4gIGhjbDogc3RyaW5nLFxuICB7XG4gICAgbGFuZ3VhZ2UsXG4gICAgcHJvdmlkZXJTY2hlbWEsXG4gICAgdGhyb3dPblRyYW5zbGF0aW9uRXJyb3IgPSBmYWxzZSxcbiAgICBjb2RlQ29udGFpbmVyID0gXCJjZGt0bi5UZXJyYWZvcm1TdGFja1wiLFxuICB9OiBDb252ZXJ0T3B0aW9ucyxcbikge1xuICBjb25zdCBmaWxlTmFtZSA9IFwidGVycmFmb3JtLnRmXCI7XG4gIGNvbnN0IHRyYW5zbGF0ZXIgPVxuICAgIGxhbmd1YWdlID09PSBcInR5cGVzY3JpcHRcIlxuICAgICAgPyAoZmlsZTogRmlsZSwgX3Rocm93T25UcmFuc2xhdGlvbkVycm9yOiBib29sZWFuKSA9PiBmaWxlLmNvbnRlbnRzXG4gICAgICA6IHRyYW5zbGF0b3JGb3JMYW5ndWFnZShsYW5ndWFnZSk7XG5cbiAgaWYgKCF0cmFuc2xhdGVyKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFwiVW5zdXBwb3J0ZWQgbGFuZ3VhZ2UgdXNlZDogXCIgKyBsYW5ndWFnZSk7XG4gIH1cblxuICBjb25zdCB0c0NvZGUgPSBhd2FpdCBjb252ZXJ0VG9UeXBlc2NyaXB0KGhjbCwgcHJvdmlkZXJTY2hlbWEsIGNvZGVDb250YWluZXIpO1xuXG4gIHJldHVybiB7XG4gICAgLi4udHNDb2RlLFxuICAgIGFsbDogdHJhbnNsYXRlcihcbiAgICAgIHsgZmlsZU5hbWUsIGNvbnRlbnRzOiB0c0NvZGUuYWxsIH0sXG4gICAgICB0aHJvd09uVHJhbnNsYXRpb25FcnJvcixcbiAgICApLFxuICAgIGltcG9ydHM6IHRyYW5zbGF0ZXIoeyBmaWxlTmFtZSwgY29udGVudHM6IHRzQ29kZS5pbXBvcnRzIH0sIGZhbHNlKSxcbiAgICBjb2RlOiB0cmFuc2xhdGVyKHsgZmlsZU5hbWUsIGNvbnRlbnRzOiB0c0NvZGUuY29kZSB9LCBmYWxzZSksXG4gICAgc3RhdHM6IHsgLi4udHNDb2RlLnN0YXRzLCBsYW5ndWFnZSB9LFxuICB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0VGVycmFmb3JtQ29uZmlnRnJvbURpcihpbXBvcnRQYXRoOiBzdHJpbmcpIHtcbiAgY29uc3QgYWJzUGF0aCA9IHBhdGgucmVzb2x2ZShpbXBvcnRQYXRoKTtcbiAgY29uc3QgZmlsZUNvbnRlbnRzID0gZ2xvYlxuICAgIC5zeW5jKFwiLi8qLnRmXCIsIHsgY3dkOiBhYnNQYXRoIH0pXG4gICAgLm1hcCgocCkgPT4gZnMucmVhZEZpbGVTeW5jKHBhdGgucmVzb2x2ZShhYnNQYXRoLCBwKSwgXCJ1dGY4XCIpKTtcblxuICByZXR1cm4gZmlsZUNvbnRlbnRzLmpvaW4oXCJcXG5cIik7XG59XG5cbnR5cGUgQ2RrdGZKc29uID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj4gJiB7XG4gIHRlcnJhZm9ybVByb3ZpZGVyczogYW55W107XG4gIHRlcnJhZm9ybU1vZHVsZXM6IGFueVtdO1xufTtcbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjb252ZXJ0UHJvamVjdChcbiAgY29tYmluZWRIY2w6IHN0cmluZyxcbiAgeyBsYW5ndWFnZSwgcHJvdmlkZXJTY2hlbWEgfTogQ29udmVydE9wdGlvbnMsXG4pIHtcbiAgaWYgKGxhbmd1YWdlICE9PSBcInR5cGVzY3JpcHRcIikge1xuICAgIHRocm93IG5ldyBFcnJvcihcIlVuc3VwcG9ydGVkIGxhbmd1YWdlIHVzZWQ6IFwiICsgbGFuZ3VhZ2UpO1xuICB9XG5cbiAgY29uc3Qge1xuICAgIGltcG9ydHMsXG4gICAgY29kZSxcbiAgICBwcm92aWRlcnMsXG4gICAgbW9kdWxlczogdGZNb2R1bGVzLFxuICAgIHN0YXRzLFxuICB9ID0gYXdhaXQgY29udmVydChjb21iaW5lZEhjbCwge1xuICAgIGxhbmd1YWdlLFxuICAgIHByb3ZpZGVyU2NoZW1hLFxuICB9KTtcblxuICByZXR1cm4ge1xuICAgIGNvZGU6IChpbnB1dE1haW5GaWxlOiBzdHJpbmcpID0+IHtcbiAgICAgIGNvbnN0IGltcG9ydE1haW5GaWxlID0gW2ltcG9ydHMsIGlucHV0TWFpbkZpbGVdLmpvaW4oXCJcXG5cIik7XG4gICAgICBjb25zdCBvdXRwdXRNYWluRmlsZSA9IGltcG9ydE1haW5GaWxlLnJlcGxhY2UoQ09ERV9NQVJLRVIsIGNvZGUpO1xuICAgICAgcmV0dXJuIHByZXR0aWVyLmZvcm1hdChvdXRwdXRNYWluRmlsZSwgeyBwYXJzZXI6IFwiYmFiZWxcIiB9KTtcbiAgICB9LFxuICAgIGNka3RmSnNvbjogKGlucHV0Q2RrdGZKc29uOiBDZGt0Zkpzb24pID0+IHtcbiAgICAgIGNvbnN0IGNka3RmSnNvbiA9IHsgLi4uaW5wdXRDZGt0Zkpzb24gfTtcbiAgICAgIGNka3RmSnNvbi50ZXJyYWZvcm1Qcm92aWRlcnMgPSBwcm92aWRlcnM7XG4gICAgICBjZGt0Zkpzb24udGVycmFmb3JtTW9kdWxlcyA9IHRmTW9kdWxlcztcbiAgICAgIHJldHVybiBjZGt0Zkpzb247XG4gICAgfSxcbiAgICBzdGF0cyxcbiAgfTtcbn1cblxuZXhwb3J0IHsgaXNSZWdpc3RyeU1vZHVsZSwgYXR0cmlidXRlTmFtZVRvQ2RrdGZOYW1lIH07XG4iXX0=
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { DirectedGraph } from "graphology";
|
|
2
|
+
import { ProgramScope } from "./types";
|
|
3
|
+
import { Import } from "./schema";
|
|
4
|
+
export declare function forEachGlobal<T, R>(scope: ProgramScope, prefix: string, record: Record<string, T> | undefined, iterator: (scope: ProgramScope, key: string, id: string, value: T, graph: DirectedGraph) => Promise<R>): Record<string, {
|
|
5
|
+
code: (graph: DirectedGraph) => Promise<R>;
|
|
6
|
+
value: T;
|
|
7
|
+
}>;
|
|
8
|
+
export declare function forEachImport<R>(scope: ProgramScope, prefix: string, record: Import[] | undefined, iterator: (scope: ProgramScope, id: string, value: Import, graph: DirectedGraph) => Promise<R>): Record<string, {
|
|
9
|
+
code: (graph: DirectedGraph) => Promise<R>;
|
|
10
|
+
value: Import;
|
|
11
|
+
}>;
|
|
12
|
+
export declare function forEachProvider<T extends {
|
|
13
|
+
alias?: string;
|
|
14
|
+
}, R>(scope: ProgramScope, record: Record<string, T[]> | undefined, iterator: (scope: ProgramScope, key: string, id: string, value: T, graph: DirectedGraph) => Promise<R>): Record<string, {
|
|
15
|
+
code: (graph: DirectedGraph) => Promise<R>;
|
|
16
|
+
value: T;
|
|
17
|
+
}>;
|
|
18
|
+
export declare function forEachNamespaced<T, R>(scope: ProgramScope, record: Record<string, Record<string, T>> | undefined, iterator: (scope: ProgramScope, type: string, key: string, id: string, value: T, graph: DirectedGraph) => Promise<R>, prefix?: string): Record<string, {
|
|
19
|
+
code: (graph: DirectedGraph) => Promise<R>;
|
|
20
|
+
value: T;
|
|
21
|
+
}>;
|
|
22
|
+
export declare function resourceStats(obj: Record<string, Record<string, unknown>>): Record<string, Record<string, number>>;
|
|
23
|
+
//# sourceMappingURL=iteration.d.ts.map
|