@cdktn/provider-generator 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/{lib → build}/__tests__/edge-provider-schema/builder.d.ts +1 -0
- package/build/__tests__/edge-provider-schema/builder.js +106 -0
- package/build/__tests__/edge-provider-schema/cli.d.ts +2 -0
- package/build/__tests__/edge-provider-schema/cli.js +67 -0
- package/{lib → build}/__tests__/edge-provider-schema/index.d.ts +1 -0
- package/build/__tests__/edge-provider-schema/index.js +154 -0
- package/build/__tests__/edge-provider-schema.test.d.ts +2 -0
- package/build/__tests__/edge-provider-schema.test.js +46 -0
- package/build/__tests__/provider.test.d.ts +2 -0
- package/build/__tests__/provider.test.js +158 -0
- package/build/get/__tests__/constructs-maker.test.d.ts +2 -0
- package/build/get/__tests__/constructs-maker.test.js +113 -0
- package/build/get/__tests__/generator/complex-computed-types.test.d.ts +2 -0
- package/build/get/__tests__/generator/complex-computed-types.test.js +42 -0
- package/build/get/__tests__/generator/deep-nested-attributes.test.d.ts +2 -0
- package/build/get/__tests__/generator/deep-nested-attributes.test.js +50 -0
- package/build/get/__tests__/generator/description-escaping.test.d.ts +2 -0
- package/build/get/__tests__/generator/description-escaping.test.js +65 -0
- package/build/get/__tests__/generator/empty-provider-resources.test.d.ts +2 -0
- package/build/get/__tests__/generator/empty-provider-resources.test.js +40 -0
- package/build/get/__tests__/generator/export-sharding.test.d.ts +2 -0
- package/build/get/__tests__/generator/export-sharding.test.js +82 -0
- package/build/get/__tests__/generator/module-generator.test.d.ts +2 -0
- package/build/get/__tests__/generator/module-generator.test.js +476 -0
- package/build/get/__tests__/generator/nested-types.test.d.ts +2 -0
- package/build/get/__tests__/generator/nested-types.test.js +51 -0
- package/build/get/__tests__/generator/provider.test.d.ts +2 -0
- package/build/get/__tests__/generator/provider.test.js +51 -0
- package/build/get/__tests__/generator/resource-types.test.d.ts +2 -0
- package/build/get/__tests__/generator/resource-types.test.js +78 -0
- package/build/get/__tests__/generator/skipped-attributes.test.d.ts +2 -0
- package/build/get/__tests__/generator/skipped-attributes.test.js +55 -0
- package/build/get/__tests__/generator/types.test.d.ts +2 -0
- package/build/get/__tests__/generator/types.test.js +289 -0
- package/build/get/__tests__/generator/versions-file.test.d.ts +2 -0
- package/build/get/__tests__/generator/versions-file.test.js +74 -0
- package/{lib → build}/get/__tests__/util.d.ts +1 -0
- package/build/get/__tests__/util.js +59 -0
- package/{lib → build}/get/constructs-maker.d.ts +2 -1
- package/{lib → build}/get/constructs-maker.js +5 -2
- package/{lib → build}/get/generator/custom-defaults.d.ts +1 -0
- package/build/get/generator/custom-defaults.js +403 -0
- package/{lib → build}/get/generator/emitter/attributes-emitter.d.ts +1 -0
- package/build/get/generator/emitter/attributes-emitter.js +161 -0
- package/{lib → build}/get/generator/emitter/index.d.ts +1 -0
- package/{lib → build}/get/generator/emitter/index.js +1 -1
- package/{lib → build}/get/generator/emitter/resource-emitter.d.ts +1 -0
- package/build/get/generator/emitter/resource-emitter.js +147 -0
- package/{lib → build}/get/generator/emitter/struct-emitter.d.ts +1 -0
- package/build/get/generator/emitter/struct-emitter.js +500 -0
- package/{lib → build}/get/generator/loop-detection.d.ts +1 -0
- package/build/get/generator/loop-detection.js +56 -0
- package/{lib → build}/get/generator/models/attribute-model.d.ts +1 -0
- package/build/get/generator/models/attribute-model.js +160 -0
- package/{lib → build}/get/generator/models/attribute-type-model.d.ts +1 -0
- package/build/get/generator/models/attribute-type-model.js +376 -0
- package/{lib → build}/get/generator/models/index.d.ts +1 -0
- package/{lib → build}/get/generator/models/index.js +1 -1
- package/{lib → build}/get/generator/models/resource-model.d.ts +1 -0
- package/build/get/generator/models/resource-model.js +106 -0
- package/{lib → build}/get/generator/models/scope.d.ts +1 -0
- package/build/get/generator/models/scope.js +36 -0
- package/{lib → build}/get/generator/models/struct.d.ts +1 -0
- package/build/get/generator/models/struct.js +90 -0
- package/{lib → build}/get/generator/module-generator.d.ts +1 -0
- package/build/get/generator/module-generator.js +184 -0
- package/{lib → build}/get/generator/provider-generator.d.ts +1 -0
- package/build/get/generator/provider-generator.js +225 -0
- package/{lib → build}/get/generator/resource-parser.d.ts +1 -0
- package/build/get/generator/resource-parser.js +497 -0
- package/{lib → build}/get/generator/sanitized-comments.d.ts +1 -0
- package/build/get/generator/sanitized-comments.js +44 -0
- package/{lib → build}/get/generator/skipped-attributes.d.ts +1 -0
- package/build/get/generator/skipped-attributes.js +29 -0
- package/{lib → build}/index.d.ts +1 -0
- package/{lib → build}/index.js +1 -1
- package/{lib → build}/util.d.ts +1 -0
- package/build/util.js +30 -0
- package/package.json +13 -15
- package/{lib → src}/__tests__/__snapshots__/edge-provider-schema.test.ts.snap +1 -1
- package/{lib → src}/__tests__/__snapshots__/provider.test.ts.snap +14953 -890
- package/{lib → src}/get/__tests__/generator/__snapshots__/complex-computed-types.test.ts.snap +1 -1
- package/{lib → src}/get/__tests__/generator/__snapshots__/description-escaping.test.ts.snap +1 -1
- package/{lib → src}/get/__tests__/generator/__snapshots__/export-sharding.test.ts.snap +1 -1
- package/{lib → src}/get/__tests__/generator/__snapshots__/module-generator.test.ts.snap +10 -10
- package/{lib → src}/get/__tests__/generator/__snapshots__/nested-types.test.ts.snap +1 -1
- package/{lib → src}/get/__tests__/generator/__snapshots__/provider.test.ts.snap +1 -1
- package/{lib → src}/get/__tests__/generator/__snapshots__/resource-types.test.ts.snap +1 -1
- package/{lib → src}/get/__tests__/generator/__snapshots__/skipped-attributes.test.ts.snap +1 -1
- package/{lib → src}/get/__tests__/generator/__snapshots__/types.test.ts.snap +92 -1
- package/src/get/__tests__/generator/fixtures/map-of-map-of-string.test.fixture.json +27 -0
- package/tsconfig.json +6 -3
- package/lib/__tests__/edge-provider-schema/builder.js +0 -106
- package/lib/__tests__/edge-provider-schema/cli.d.ts +0 -1
- package/lib/__tests__/edge-provider-schema/cli.js +0 -67
- package/lib/__tests__/edge-provider-schema/index.js +0 -154
- package/lib/__tests__/edge-provider-schema.test.d.ts +0 -1
- package/lib/__tests__/edge-provider-schema.test.js +0 -46
- package/lib/__tests__/provider.test.d.ts +0 -1
- package/lib/__tests__/provider.test.js +0 -158
- package/lib/get/__tests__/constructs-maker.test.d.ts +0 -1
- package/lib/get/__tests__/constructs-maker.test.js +0 -113
- package/lib/get/__tests__/generator/complex-computed-types.test.d.ts +0 -1
- package/lib/get/__tests__/generator/complex-computed-types.test.js +0 -42
- package/lib/get/__tests__/generator/deep-nested-attributes.test.d.ts +0 -1
- package/lib/get/__tests__/generator/deep-nested-attributes.test.js +0 -50
- package/lib/get/__tests__/generator/description-escaping.test.d.ts +0 -1
- package/lib/get/__tests__/generator/description-escaping.test.js +0 -65
- package/lib/get/__tests__/generator/empty-provider-resources.test.d.ts +0 -1
- package/lib/get/__tests__/generator/empty-provider-resources.test.js +0 -40
- package/lib/get/__tests__/generator/export-sharding.test.d.ts +0 -1
- package/lib/get/__tests__/generator/export-sharding.test.js +0 -82
- package/lib/get/__tests__/generator/module-generator.test.d.ts +0 -1
- package/lib/get/__tests__/generator/module-generator.test.js +0 -476
- package/lib/get/__tests__/generator/nested-types.test.d.ts +0 -1
- package/lib/get/__tests__/generator/nested-types.test.js +0 -51
- package/lib/get/__tests__/generator/provider.test.d.ts +0 -1
- package/lib/get/__tests__/generator/provider.test.js +0 -51
- package/lib/get/__tests__/generator/resource-types.test.d.ts +0 -1
- package/lib/get/__tests__/generator/resource-types.test.js +0 -78
- package/lib/get/__tests__/generator/skipped-attributes.test.d.ts +0 -1
- package/lib/get/__tests__/generator/skipped-attributes.test.js +0 -55
- package/lib/get/__tests__/generator/types.test.d.ts +0 -1
- package/lib/get/__tests__/generator/types.test.js +0 -280
- package/lib/get/__tests__/generator/versions-file.test.d.ts +0 -1
- package/lib/get/__tests__/generator/versions-file.test.js +0 -74
- package/lib/get/__tests__/util.js +0 -59
- package/lib/get/generator/custom-defaults.js +0 -403
- package/lib/get/generator/emitter/attributes-emitter.js +0 -161
- package/lib/get/generator/emitter/resource-emitter.js +0 -144
- package/lib/get/generator/emitter/struct-emitter.js +0 -500
- package/lib/get/generator/loop-detection.js +0 -56
- package/lib/get/generator/models/attribute-model.js +0 -160
- package/lib/get/generator/models/attribute-type-model.js +0 -376
- package/lib/get/generator/models/resource-model.js +0 -106
- package/lib/get/generator/models/scope.js +0 -36
- package/lib/get/generator/models/struct.js +0 -90
- package/lib/get/generator/module-generator.js +0 -184
- package/lib/get/generator/provider-generator.js +0 -225
- package/lib/get/generator/resource-parser.js +0 -497
- package/lib/get/generator/sanitized-comments.js +0 -44
- package/lib/get/generator/skipped-attributes.js +0 -29
- package/lib/util.js +0 -30
- /package/{lib → src}/get/__tests__/generator/fixtures/aws-provider.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/aws_acm_certificate.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/aws_cloudfront_distribution.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/aws_fms_admin_account.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/aws_quicksight_template.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/aws_s3_bucket.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/aws_security_group.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/aws_wafv2_web_acl.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/block-type-nested-computed-list.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/block-type-set-list.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/boolean-list.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/boolean-map.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/comment-ending-sequence.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/computed-complex-nested.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/computed-complex.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/computed-optional-complex.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/data_aws_quicksight_analysis.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/datadog_dashboard.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/datadog_spans_metric.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/deep-attributes.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/deeply-nested-block-types.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/description-escaping.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/elasticstack-provider.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/empty-provider-resources.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/ignored-attributes.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/incompatible-attribute-names.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/incompatible-resource-names.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/list-list-object.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/list-list-string.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/list-of-string-map.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/map-of-string-list.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/markdown-description-with-code-blocks.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/module-get-x.test.fixture.tf +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/module-no-newline-1.test.fixture.tf +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/module-no-newline-2.test.fixture.tf +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/module-no-outputs.test.fixture.tf +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/module-no-variable-type.test.fixture.tf +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/module-with-star-default.test.fixture.tf +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/name-conflict.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/nested-type-without-attributes.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/nested-types.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/number-list.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/number-map.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/primitive-boolean.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/primitive-dynamic.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/primitive-number.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/primitive-string.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/single-block-type.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/string-list.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/string-map.test.fixture.json +0 -0
- /package/{lib → src}/get/__tests__/generator/fixtures/stripe-schema.test.fixture.json +0 -0
|
@@ -0,0 +1,497 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ResourceParser = exports.sanitizeClassOrNamespaceName = void 0;
|
|
4
|
+
// Copyright (c) HashiCorp, Inc
|
|
5
|
+
// SPDX-License-Identifier: MPL-2.0
|
|
6
|
+
const codemaker_1 = require("codemaker");
|
|
7
|
+
const commons_1 = require("@cdktn/commons");
|
|
8
|
+
const provider_schema_1 = require("@cdktn/provider-schema");
|
|
9
|
+
const models_1 = require("./models");
|
|
10
|
+
const loop_detection_1 = require("./loop-detection");
|
|
11
|
+
const skipped_attributes_1 = require("./skipped-attributes");
|
|
12
|
+
// Can't be used in expressions like "export * as <keyword> from ... "
|
|
13
|
+
// filtered from all keywords from: https://github.com/microsoft/TypeScript/blob/503604c884bd0557c851b11b699ef98cdb65b93b/src/compiler/types.ts#L114-L197
|
|
14
|
+
const RESERVED_KEYWORDS_FOR_NAMESPACES = [
|
|
15
|
+
"implements",
|
|
16
|
+
"interface",
|
|
17
|
+
"let",
|
|
18
|
+
"package",
|
|
19
|
+
"private",
|
|
20
|
+
"protected",
|
|
21
|
+
"public",
|
|
22
|
+
"static",
|
|
23
|
+
"yield",
|
|
24
|
+
"await",
|
|
25
|
+
];
|
|
26
|
+
const COLLIDING_NAMESPACE_NAMES = [
|
|
27
|
+
// e.g. hashicorp/consul – collides with the LICENSE file on case-insensitive filesystems in the Go package (#2627)
|
|
28
|
+
"license",
|
|
29
|
+
// collides for Go packages
|
|
30
|
+
"version",
|
|
31
|
+
];
|
|
32
|
+
const isReservedClassOrNamespaceName = (className) => {
|
|
33
|
+
return [
|
|
34
|
+
"string",
|
|
35
|
+
"object",
|
|
36
|
+
"function",
|
|
37
|
+
...RESERVED_KEYWORDS_FOR_NAMESPACES,
|
|
38
|
+
...COLLIDING_NAMESPACE_NAMES,
|
|
39
|
+
].includes(className.toLowerCase());
|
|
40
|
+
};
|
|
41
|
+
const isReservedStructClassName = (className) => {
|
|
42
|
+
return className.toLowerCase().endsWith("list");
|
|
43
|
+
};
|
|
44
|
+
const getFileName = (provider, baseName) => {
|
|
45
|
+
if (baseName === "index") {
|
|
46
|
+
return "index-resource/index.ts";
|
|
47
|
+
}
|
|
48
|
+
if (baseName === `${provider}_provider`) {
|
|
49
|
+
return "provider/index.ts";
|
|
50
|
+
}
|
|
51
|
+
return `${(0, codemaker_1.toSnakeCase)(baseName).replace(/_/g, "-")}/index.ts`;
|
|
52
|
+
};
|
|
53
|
+
function sanitizeClassOrNamespaceName(baseName, isProvider = false) {
|
|
54
|
+
const resourceIsNamedProvider = !isProvider && baseName === "provider";
|
|
55
|
+
if (isReservedClassOrNamespaceName(baseName) || resourceIsNamedProvider) {
|
|
56
|
+
return `${baseName}_resource`;
|
|
57
|
+
}
|
|
58
|
+
else {
|
|
59
|
+
return baseName;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
exports.sanitizeClassOrNamespaceName = sanitizeClassOrNamespaceName;
|
|
63
|
+
/**
|
|
64
|
+
* Remove attributes that may conflict after being snake cased
|
|
65
|
+
* Example: oci_core_ipsec_connection_tunnel_management (hashicorp/oci@=5.21.0) has bgp_ipv6_state and bgp_ipv6state
|
|
66
|
+
* (which both result in "bgpIpv6State" when camel-cased, with the second one being deprecated: true)
|
|
67
|
+
* As we currently don't handle any deprecated ones at all, we'll just delete one of the two attributes for now
|
|
68
|
+
* @param attributes
|
|
69
|
+
*/
|
|
70
|
+
function deduplicateAttributesWithSameName(attributes) {
|
|
71
|
+
return attributes.filter((attr, idx) => {
|
|
72
|
+
const hasOtherWithSameName = attributes
|
|
73
|
+
.slice(idx + 1) // only search after the index of the current attribute to avoid deleting both
|
|
74
|
+
.some((other) => other.name === attr.name && other !== attr);
|
|
75
|
+
return !hasOtherWithSameName;
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
class Parser {
|
|
79
|
+
constructor(classNames) {
|
|
80
|
+
this.classNames = classNames;
|
|
81
|
+
this.structs = new Array();
|
|
82
|
+
}
|
|
83
|
+
uniqueClassName(className) {
|
|
84
|
+
if (this.classNames.includes(className)) {
|
|
85
|
+
className = `${className}A`;
|
|
86
|
+
}
|
|
87
|
+
this.classNames.push(className);
|
|
88
|
+
return className;
|
|
89
|
+
}
|
|
90
|
+
resourceFrom(fqpn, type, schema, terraformSchemaType, constraint) {
|
|
91
|
+
let baseName = type;
|
|
92
|
+
const providerNameFromConstraint = constraint
|
|
93
|
+
? constraint.name
|
|
94
|
+
: undefined;
|
|
95
|
+
const providerNameFromFQPN = (0, provider_schema_1.parseFQPN)(fqpn).name;
|
|
96
|
+
if (baseName.startsWith(`${providerNameFromFQPN}_`)) {
|
|
97
|
+
baseName = baseName.substr(providerNameFromFQPN.length + 1);
|
|
98
|
+
}
|
|
99
|
+
const providerName = providerNameFromConstraint
|
|
100
|
+
? providerNameFromConstraint
|
|
101
|
+
: providerNameFromFQPN;
|
|
102
|
+
const isProvider = terraformSchemaType === "provider";
|
|
103
|
+
if (isProvider) {
|
|
104
|
+
baseName = `${providerName}_${baseName}`;
|
|
105
|
+
if (!("attributes" in schema.block)) {
|
|
106
|
+
schema.block = {
|
|
107
|
+
attributes: {},
|
|
108
|
+
block_types: schema.block.block_types || {},
|
|
109
|
+
};
|
|
110
|
+
}
|
|
111
|
+
// somehow missing from provider schema
|
|
112
|
+
schema.block.attributes["alias"] = {
|
|
113
|
+
type: "string",
|
|
114
|
+
description: "Alias name",
|
|
115
|
+
optional: true,
|
|
116
|
+
computed: false,
|
|
117
|
+
};
|
|
118
|
+
}
|
|
119
|
+
baseName = sanitizeClassOrNamespaceName(baseName, isProvider);
|
|
120
|
+
const className = this.uniqueClassName((0, codemaker_1.toPascalCase)(baseName));
|
|
121
|
+
// avoid naming collision - see https://github.com/hashicorp/terraform-cdk/issues/299
|
|
122
|
+
const configStructName = this.uniqueClassName(`${className}Config`);
|
|
123
|
+
const fileName = getFileName(providerName, baseName);
|
|
124
|
+
const filePath = `providers/${(0, codemaker_1.toSnakeCase)(providerName)}/${fileName}`;
|
|
125
|
+
let attributes = this.renderAttributesForBlock(new models_1.Scope({
|
|
126
|
+
name: baseName,
|
|
127
|
+
isProvider,
|
|
128
|
+
parent: isProvider
|
|
129
|
+
? undefined
|
|
130
|
+
: new models_1.Scope({ name: providerName, isProvider: true }),
|
|
131
|
+
}), schema.block);
|
|
132
|
+
function getStructAttribute(attributes, path) {
|
|
133
|
+
const [first, ...rest] = path.split(".");
|
|
134
|
+
const attribute = attributes.find((att) => {
|
|
135
|
+
return att.terraformName === first;
|
|
136
|
+
});
|
|
137
|
+
if (!attribute)
|
|
138
|
+
throw new Error(`Expected to find recursive attribute at path: ${path}`);
|
|
139
|
+
if (!attribute.type.struct)
|
|
140
|
+
throw new Error(`Expected to find struct type attribute at path: ${path} but got ${attribute.type.storedClassType}`);
|
|
141
|
+
if (rest.length === 0)
|
|
142
|
+
return attribute;
|
|
143
|
+
return getStructAttribute(attribute.type.struct.attributes, rest.join("."));
|
|
144
|
+
}
|
|
145
|
+
// Introduce recursion for some attributes
|
|
146
|
+
const recursiveAttributePaths = (0, loop_detection_1.detectAttributeLoops)(attributes);
|
|
147
|
+
Object.entries(recursiveAttributePaths).forEach(([attributePath, structPath]) => {
|
|
148
|
+
// TODO: build this to be a bit more defensive (e.g. remove ! operator)
|
|
149
|
+
const recursionTargetStructAttribute = getStructAttribute(attributes, structPath);
|
|
150
|
+
const parts = attributePath.split(".");
|
|
151
|
+
const attributeName = parts.pop();
|
|
152
|
+
const parentAttribute = getStructAttribute(attributes, parts.join("."));
|
|
153
|
+
const indexToReplace = parentAttribute.type.struct.attributes.findIndex((att) => att.terraformName === attributeName);
|
|
154
|
+
if (indexToReplace === -1)
|
|
155
|
+
throw new Error("Can't find attribute at path " + attributePath);
|
|
156
|
+
const previousAttribute = parentAttribute.type.struct.attributes[indexToReplace];
|
|
157
|
+
parentAttribute.type.struct.attributes[indexToReplace] =
|
|
158
|
+
recursionTargetStructAttribute; // introduce recursion
|
|
159
|
+
// ugly, pls c̶a̶l̶l̶ refactor me maybe
|
|
160
|
+
// we store all structs in this.structs – now we need to dispose all structs that are part of previousAttribute
|
|
161
|
+
const disposeStructs = (attr) => {
|
|
162
|
+
if (attr.type.struct) {
|
|
163
|
+
attr.type.struct.attributes.forEach(disposeStructs);
|
|
164
|
+
this.structs = this.structs.filter((s) => s !== attr.type.struct);
|
|
165
|
+
}
|
|
166
|
+
};
|
|
167
|
+
disposeStructs(previousAttribute);
|
|
168
|
+
});
|
|
169
|
+
attributes = deduplicateAttributesWithSameName(attributes);
|
|
170
|
+
const resourceModel = new models_1.ResourceModel({
|
|
171
|
+
terraformType: type,
|
|
172
|
+
baseName,
|
|
173
|
+
fileName,
|
|
174
|
+
filePath,
|
|
175
|
+
className,
|
|
176
|
+
schema,
|
|
177
|
+
fqpn,
|
|
178
|
+
attributes,
|
|
179
|
+
terraformSchemaType,
|
|
180
|
+
structs: this.structs,
|
|
181
|
+
configStructName,
|
|
182
|
+
});
|
|
183
|
+
return resourceModel;
|
|
184
|
+
}
|
|
185
|
+
renderAttributeType(scope, attributeType, parentKind) {
|
|
186
|
+
const parent = scope[scope.length - 1];
|
|
187
|
+
if ((0, skipped_attributes_1.shouldSkipAttribute)(parent.baseName)) {
|
|
188
|
+
return new models_1.MapAttributeTypeModel(new models_1.SimpleAttributeTypeModel("any"));
|
|
189
|
+
}
|
|
190
|
+
if (typeof attributeType === "string") {
|
|
191
|
+
switch (attributeType) {
|
|
192
|
+
case "bool":
|
|
193
|
+
return new models_1.SimpleAttributeTypeModel("boolean");
|
|
194
|
+
case "string":
|
|
195
|
+
return new models_1.SimpleAttributeTypeModel("string");
|
|
196
|
+
case "number":
|
|
197
|
+
return new models_1.SimpleAttributeTypeModel("number");
|
|
198
|
+
case "dynamic":
|
|
199
|
+
return new models_1.MapAttributeTypeModel(new models_1.SimpleAttributeTypeModel("any"));
|
|
200
|
+
default:
|
|
201
|
+
throw new Error(`invalid primitive type ${attributeType}`);
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
if (Array.isArray(attributeType)) {
|
|
205
|
+
if (attributeType.length !== 2) {
|
|
206
|
+
throw new Error(`unexpected array`);
|
|
207
|
+
}
|
|
208
|
+
const [kind, type] = attributeType;
|
|
209
|
+
if (kind === "set" || kind === "list") {
|
|
210
|
+
const elementType = this.renderAttributeType(scope, type, [kind, parentKind].join(""));
|
|
211
|
+
return kind === "list"
|
|
212
|
+
? new models_1.ListAttributeTypeModel(elementType, false, false)
|
|
213
|
+
: new models_1.SetAttributeTypeModel(elementType, false, false);
|
|
214
|
+
}
|
|
215
|
+
if (kind === "map") {
|
|
216
|
+
const valueType = this.renderAttributeType(scope, type, [kind, parentKind].join(""));
|
|
217
|
+
return new models_1.MapAttributeTypeModel(valueType);
|
|
218
|
+
}
|
|
219
|
+
if (kind === "object") {
|
|
220
|
+
const objAttributes = type;
|
|
221
|
+
const attributes = {};
|
|
222
|
+
for (const [name, type] of Object.entries(objAttributes)) {
|
|
223
|
+
attributes[name] = { type };
|
|
224
|
+
}
|
|
225
|
+
const struct = this.addAnonymousStruct(scope, attributes, parentKind !== null && parentKind !== void 0 ? parentKind : kind);
|
|
226
|
+
return new models_1.StructAttributeTypeModel(struct);
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
if ((0, commons_1.isAttributeNestedType)(attributeType)) {
|
|
230
|
+
let struct = undefined;
|
|
231
|
+
let typeModel = undefined;
|
|
232
|
+
switch (attributeType.nesting_mode) {
|
|
233
|
+
case "list":
|
|
234
|
+
struct = this.addAnonymousStruct(scope, attributeType.attributes, attributeType.nesting_mode);
|
|
235
|
+
typeModel = new models_1.ListAttributeTypeModel(new models_1.StructAttributeTypeModel(struct), false, false);
|
|
236
|
+
break;
|
|
237
|
+
case "set":
|
|
238
|
+
struct = this.addAnonymousStruct(scope, attributeType.attributes, attributeType.nesting_mode);
|
|
239
|
+
typeModel = new models_1.SetAttributeTypeModel(new models_1.StructAttributeTypeModel(struct), false, false);
|
|
240
|
+
break;
|
|
241
|
+
case "map":
|
|
242
|
+
struct = this.addAnonymousStruct(scope, attributeType.attributes, attributeType.nesting_mode);
|
|
243
|
+
typeModel = new models_1.MapAttributeTypeModel(new models_1.StructAttributeTypeModel(struct));
|
|
244
|
+
break;
|
|
245
|
+
case "single":
|
|
246
|
+
struct = this.addAnonymousStruct(scope, attributeType.attributes, attributeType.nesting_mode);
|
|
247
|
+
typeModel = new models_1.StructAttributeTypeModel(struct);
|
|
248
|
+
break;
|
|
249
|
+
default: {
|
|
250
|
+
throw new Error(`nested_type with nesting_mode "${attributeType.nesting_mode}" not supported (attribute scope: ${scope
|
|
251
|
+
.map((s) => s.fullName)
|
|
252
|
+
.join(",")}`);
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
return typeModel;
|
|
256
|
+
}
|
|
257
|
+
throw new Error(`unknown type ${JSON.stringify(attributeType)}`);
|
|
258
|
+
}
|
|
259
|
+
renderAttributesForBlock(parentType, block) {
|
|
260
|
+
const attributes = new Array();
|
|
261
|
+
for (const [terraformAttributeName, att] of Object.entries(block.attributes || {})) {
|
|
262
|
+
let type;
|
|
263
|
+
let forcePlainGetterType = false;
|
|
264
|
+
if ((0, skipped_attributes_1.shouldSkipAttribute)(parentType.fullName(terraformAttributeName))) {
|
|
265
|
+
type = new models_1.SkippedAttributeTypeModel();
|
|
266
|
+
forcePlainGetterType = true;
|
|
267
|
+
}
|
|
268
|
+
else {
|
|
269
|
+
type = this.renderAttributeType([
|
|
270
|
+
parentType,
|
|
271
|
+
new models_1.Scope({
|
|
272
|
+
name: terraformAttributeName,
|
|
273
|
+
parent: parentType,
|
|
274
|
+
isProvider: parentType.isProvider,
|
|
275
|
+
isComputed: !!att.computed,
|
|
276
|
+
isOptional: !!att.optional,
|
|
277
|
+
isRequired: !!att.required,
|
|
278
|
+
isNestedType: (0, commons_1.isNestedTypeAttribute)(att),
|
|
279
|
+
}),
|
|
280
|
+
], att.type || att.nested_type);
|
|
281
|
+
}
|
|
282
|
+
const name = (0, codemaker_1.toCamelCase)(terraformAttributeName);
|
|
283
|
+
attributes.push(new models_1.AttributeModel({
|
|
284
|
+
terraformFullName: parentType.fullName(terraformAttributeName),
|
|
285
|
+
description: att.description,
|
|
286
|
+
name,
|
|
287
|
+
storageName: `_${name}`,
|
|
288
|
+
computed: !!att.computed,
|
|
289
|
+
optional: !!att.optional,
|
|
290
|
+
terraformName: terraformAttributeName,
|
|
291
|
+
type,
|
|
292
|
+
provider: parentType.isProvider,
|
|
293
|
+
required: !!att.required,
|
|
294
|
+
forcePlainGetterType,
|
|
295
|
+
}));
|
|
296
|
+
}
|
|
297
|
+
for (const [blockTypeName, blockType] of Object.entries(block.block_types || {})) {
|
|
298
|
+
if ((0, skipped_attributes_1.shouldSkipAttribute)(parentType.fullName(blockTypeName))) {
|
|
299
|
+
const name = (0, codemaker_1.toCamelCase)(blockTypeName);
|
|
300
|
+
const parent = new models_1.Scope({
|
|
301
|
+
name: blockTypeName,
|
|
302
|
+
parent: parentType,
|
|
303
|
+
isProvider: parentType.isProvider,
|
|
304
|
+
});
|
|
305
|
+
attributes.push(new models_1.AttributeModel({
|
|
306
|
+
name,
|
|
307
|
+
terraformName: blockTypeName,
|
|
308
|
+
terraformFullName: parent.fullName(blockTypeName),
|
|
309
|
+
type: new models_1.SkippedAttributeTypeModel(),
|
|
310
|
+
description: `${blockTypeName} block`,
|
|
311
|
+
storageName: `_${name}`,
|
|
312
|
+
optional: true,
|
|
313
|
+
computed: false,
|
|
314
|
+
provider: parentType.isProvider,
|
|
315
|
+
required: false,
|
|
316
|
+
}));
|
|
317
|
+
continue;
|
|
318
|
+
}
|
|
319
|
+
// create a struct for this block
|
|
320
|
+
let blockAttributes = this.renderAttributesForBlock(new models_1.Scope({
|
|
321
|
+
name: `${parentType.name}_${blockTypeName}`,
|
|
322
|
+
parent: parentType,
|
|
323
|
+
isProvider: parentType.isProvider,
|
|
324
|
+
inBlockType: true,
|
|
325
|
+
}), blockType.block);
|
|
326
|
+
blockAttributes = deduplicateAttributesWithSameName(blockAttributes);
|
|
327
|
+
const blockStruct = this.addStruct([
|
|
328
|
+
parentType,
|
|
329
|
+
new models_1.Scope({
|
|
330
|
+
name: blockTypeName,
|
|
331
|
+
parent: parentType,
|
|
332
|
+
isProvider: parentType.isProvider,
|
|
333
|
+
}),
|
|
334
|
+
], blockAttributes, blockType.nesting_mode, (blockType.nesting_mode === "list" ||
|
|
335
|
+
blockType.nesting_mode === "set") &&
|
|
336
|
+
blockType.max_items === 1);
|
|
337
|
+
// define the attribute
|
|
338
|
+
attributes.push(attributeForBlockType(blockTypeName, blockType, blockStruct, parentType.isProvider, parentType));
|
|
339
|
+
}
|
|
340
|
+
return attributes;
|
|
341
|
+
function attributeForBlockType(terraformName, blockType, struct, isProvider, parent) {
|
|
342
|
+
const name = (0, codemaker_1.toCamelCase)(terraformName);
|
|
343
|
+
let optional;
|
|
344
|
+
let required;
|
|
345
|
+
switch (blockType.nesting_mode) {
|
|
346
|
+
case "single":
|
|
347
|
+
optional = !struct.attributes.some((x) => !x.optional);
|
|
348
|
+
required = !struct.attributes.some((x) => !x.required);
|
|
349
|
+
// This is for bug #3570 as both optional and required evaluate to false under some circumstances
|
|
350
|
+
// (this then causes the computed block to not be part of assignableAttributes and thus skipped in the generated code)
|
|
351
|
+
// Hence: If both optional and required are false, set optional to true IF at least one
|
|
352
|
+
// attribute in the block has optional = true or required = true, as this would mean that at least something can be set
|
|
353
|
+
// and the block is not all computed.
|
|
354
|
+
if (!optional && !required) {
|
|
355
|
+
optional = struct.attributes.some((x) => x.optional || x.required);
|
|
356
|
+
}
|
|
357
|
+
return new models_1.AttributeModel({
|
|
358
|
+
name,
|
|
359
|
+
terraformName,
|
|
360
|
+
terraformFullName: parent.fullName(terraformName),
|
|
361
|
+
type: new models_1.StructAttributeTypeModel(struct),
|
|
362
|
+
description: `${terraformName} block`,
|
|
363
|
+
storageName: `_${name}`,
|
|
364
|
+
optional,
|
|
365
|
+
computed: false,
|
|
366
|
+
provider: isProvider,
|
|
367
|
+
required,
|
|
368
|
+
});
|
|
369
|
+
case "map":
|
|
370
|
+
return new models_1.AttributeModel({
|
|
371
|
+
name,
|
|
372
|
+
terraformName,
|
|
373
|
+
terraformFullName: parent.fullName(terraformName),
|
|
374
|
+
type: new models_1.MapAttributeTypeModel(new models_1.StructAttributeTypeModel(struct)),
|
|
375
|
+
description: `${terraformName} block`,
|
|
376
|
+
storageName: `_${name}`,
|
|
377
|
+
optional: false,
|
|
378
|
+
computed: false,
|
|
379
|
+
provider: isProvider,
|
|
380
|
+
required: false,
|
|
381
|
+
});
|
|
382
|
+
case "list":
|
|
383
|
+
case "set":
|
|
384
|
+
optional =
|
|
385
|
+
blockType.min_items === undefined ? true : blockType.min_items < 1;
|
|
386
|
+
required =
|
|
387
|
+
blockType.min_items === undefined ? false : blockType.min_items > 0;
|
|
388
|
+
return new models_1.AttributeModel({
|
|
389
|
+
name,
|
|
390
|
+
terraformName: terraformName,
|
|
391
|
+
terraformFullName: parent.fullName(terraformName),
|
|
392
|
+
type: blockType.nesting_mode === "list"
|
|
393
|
+
? new models_1.ListAttributeTypeModel(new models_1.StructAttributeTypeModel(struct), blockType.max_items === 1, true)
|
|
394
|
+
: new models_1.SetAttributeTypeModel(new models_1.StructAttributeTypeModel(struct), blockType.max_items === 1, true),
|
|
395
|
+
description: `${terraformName} block`,
|
|
396
|
+
storageName: `_${name}`,
|
|
397
|
+
optional,
|
|
398
|
+
computed: false,
|
|
399
|
+
provider: isProvider,
|
|
400
|
+
required,
|
|
401
|
+
});
|
|
402
|
+
}
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
addAnonymousStruct(scope, attrs, nesting_mode) {
|
|
406
|
+
let attributes = new Array();
|
|
407
|
+
const parent = scope[scope.length - 1];
|
|
408
|
+
if (attrs) {
|
|
409
|
+
for (const [terraformName, att] of Object.entries(attrs || {})) {
|
|
410
|
+
// nested types support computed, optional and required on attribute level
|
|
411
|
+
// if parent is computed, child always is computed as well
|
|
412
|
+
const computed = !!parent.isComputed || (parent.isNestedType && !!att.computed);
|
|
413
|
+
const optional = parent.isNestedType
|
|
414
|
+
? !!att.optional
|
|
415
|
+
: !!parent.isOptional;
|
|
416
|
+
const required = parent.isNestedType
|
|
417
|
+
? !!att.required
|
|
418
|
+
: !!parent.isRequired;
|
|
419
|
+
const name = (0, codemaker_1.toCamelCase)(terraformName);
|
|
420
|
+
const type = this.renderAttributeType([
|
|
421
|
+
...scope,
|
|
422
|
+
new models_1.Scope({
|
|
423
|
+
name: terraformName,
|
|
424
|
+
parent,
|
|
425
|
+
isProvider: parent.isProvider,
|
|
426
|
+
isComputed: computed,
|
|
427
|
+
isOptional: optional,
|
|
428
|
+
isRequired: required,
|
|
429
|
+
isNestedType: parent.isNestedType,
|
|
430
|
+
}),
|
|
431
|
+
], att.type || att.nested_type);
|
|
432
|
+
attributes.push(new models_1.AttributeModel({
|
|
433
|
+
name,
|
|
434
|
+
storageName: `_${name}`,
|
|
435
|
+
computed: computed,
|
|
436
|
+
description: att.description,
|
|
437
|
+
optional: optional,
|
|
438
|
+
terraformName,
|
|
439
|
+
terraformFullName: parent.fullName(terraformName),
|
|
440
|
+
type,
|
|
441
|
+
provider: parent.isProvider,
|
|
442
|
+
required: required,
|
|
443
|
+
}));
|
|
444
|
+
}
|
|
445
|
+
}
|
|
446
|
+
attributes = deduplicateAttributesWithSameName(attributes);
|
|
447
|
+
return this.addStruct(scope, attributes, nesting_mode);
|
|
448
|
+
}
|
|
449
|
+
addStruct(scope, attributes, nesting_mode, isSingleItem = false) {
|
|
450
|
+
const possibleName = (0, codemaker_1.toPascalCase)(scope.map((x) => (0, codemaker_1.toSnakeCase)(x.name)).join("_"));
|
|
451
|
+
const name = this.uniqueClassName(isReservedStructClassName(possibleName)
|
|
452
|
+
? `${possibleName}Struct`
|
|
453
|
+
: possibleName);
|
|
454
|
+
const parent = scope[scope.length - 1];
|
|
455
|
+
// blockType.nesting_mode => list/set & blockType.max_items === 1,
|
|
456
|
+
const isClass = (parent.isComputed && !parent.isOptional) || isSingleItem;
|
|
457
|
+
const isAnonymous = true;
|
|
458
|
+
const s = new models_1.Struct(name, attributes, isClass, isAnonymous, isSingleItem, nesting_mode);
|
|
459
|
+
this.structs.push(s);
|
|
460
|
+
return s;
|
|
461
|
+
}
|
|
462
|
+
}
|
|
463
|
+
class ResourceParser {
|
|
464
|
+
constructor() {
|
|
465
|
+
this.uniqueClassnames = [];
|
|
466
|
+
this.resources = {};
|
|
467
|
+
}
|
|
468
|
+
parse(fqpn, type, schema, terraformType, constraint) {
|
|
469
|
+
if (this.resources[type]) {
|
|
470
|
+
return this.resources[type];
|
|
471
|
+
}
|
|
472
|
+
const parser = new Parser(this.uniqueClassnames);
|
|
473
|
+
const resource = parser.resourceFrom(fqpn, type, schema, terraformType, constraint);
|
|
474
|
+
this.resources[type] = resource;
|
|
475
|
+
return resource;
|
|
476
|
+
}
|
|
477
|
+
// Used by convert to determine the right name for a class
|
|
478
|
+
getClassNameForResource(terraformType) {
|
|
479
|
+
const resource = this.resources[terraformType];
|
|
480
|
+
return resource ? resource.className : "";
|
|
481
|
+
}
|
|
482
|
+
// Used by convert to determine the right name for a namespace
|
|
483
|
+
getNamespaceNameForResource(terraformType) {
|
|
484
|
+
// Special case external provider since the name of resource doesn't have a prefix
|
|
485
|
+
if (terraformType === "data_external_") {
|
|
486
|
+
terraformType = "data_external";
|
|
487
|
+
}
|
|
488
|
+
const resource = this.resources[terraformType];
|
|
489
|
+
if (!resource) {
|
|
490
|
+
return "";
|
|
491
|
+
}
|
|
492
|
+
const folder = `providers/${resource.provider}`;
|
|
493
|
+
return resource.filePath.replace(`${folder}/`, "").replace("/index.ts", "");
|
|
494
|
+
}
|
|
495
|
+
}
|
|
496
|
+
exports.ResourceParser = ResourceParser;
|
|
497
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource-parser.js","sourceRoot":"","sources":["../../../src/get/generator/resource-parser.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright (c) HashiCorp, Inc\n// SPDX-License-Identifier: MPL-2.0\nimport { toCamelCase, toPascalCase, toSnakeCase } from \"codemaker\";\nimport {\n  Attribute,\n  AttributeNestedType,\n  AttributeType,\n  Block,\n  BlockType,\n  ConstructsMakerTarget,\n  isAttributeNestedType,\n  isNestedTypeAttribute,\n  Schema,\n} from \"@cdktn/commons\";\nimport { ProviderName, FQPN, parseFQPN } from \"@cdktn/provider-schema\";\nimport {\n  ResourceModel,\n  AttributeTypeModel,\n  Struct,\n  Scope,\n  AttributeModel,\n  SimpleAttributeTypeModel,\n  ListAttributeTypeModel,\n  SetAttributeTypeModel,\n  MapAttributeTypeModel,\n  StructAttributeTypeModel,\n  SkippedAttributeTypeModel,\n} from \"./models\";\nimport { detectAttributeLoops } from \"./loop-detection\";\nimport { shouldSkipAttribute } from \"./skipped-attributes\";\n\n// Can't be used in expressions like \"export * as <keyword> from ... \"\n// filtered from all keywords from: https://github.com/microsoft/TypeScript/blob/503604c884bd0557c851b11b699ef98cdb65b93b/src/compiler/types.ts#L114-L197\nconst RESERVED_KEYWORDS_FOR_NAMESPACES = [\n  \"implements\",\n  \"interface\",\n  \"let\",\n  \"package\",\n  \"private\",\n  \"protected\",\n  \"public\",\n  \"static\",\n  \"yield\",\n  \"await\",\n];\n\nconst COLLIDING_NAMESPACE_NAMES = [\n  // e.g. hashicorp/consul – collides with the LICENSE file on case-insensitive filesystems in the Go package (#2627)\n  \"license\",\n  // collides for Go packages\n  \"version\",\n];\n\nconst isReservedClassOrNamespaceName = (className: string): boolean => {\n  return [\n    \"string\",\n    \"object\",\n    \"function\",\n    ...RESERVED_KEYWORDS_FOR_NAMESPACES,\n    ...COLLIDING_NAMESPACE_NAMES,\n  ].includes(className.toLowerCase());\n};\n\nconst isReservedStructClassName = (className: string): boolean => {\n  return className.toLowerCase().endsWith(\"list\");\n};\n\nconst getFileName = (provider: ProviderName, baseName: string): string => {\n  if (baseName === \"index\") {\n    return \"index-resource/index.ts\";\n  }\n\n  if (baseName === `${provider}_provider`) {\n    return \"provider/index.ts\";\n  }\n\n  return `${toSnakeCase(baseName).replace(/_/g, \"-\")}/index.ts`;\n};\n\nexport function sanitizeClassOrNamespaceName(\n  baseName: string,\n  isProvider = false,\n) {\n  const resourceIsNamedProvider = !isProvider && baseName === \"provider\";\n\n  if (isReservedClassOrNamespaceName(baseName) || resourceIsNamedProvider) {\n    return `${baseName}_resource`;\n  } else {\n    return baseName;\n  }\n}\n\n/**\n * Remove attributes that may conflict after being snake cased\n * Example: oci_core_ipsec_connection_tunnel_management (hashicorp/oci@=5.21.0) has bgp_ipv6_state and bgp_ipv6state\n * (which both result in \"bgpIpv6State\" when camel-cased, with the second one being deprecated: true)\n * As we currently don't handle any deprecated ones at all, we'll just delete one of the two attributes for now\n * @param attributes\n */\nfunction deduplicateAttributesWithSameName(\n  attributes: AttributeModel[],\n): AttributeModel[] {\n  return attributes.filter((attr, idx) => {\n    const hasOtherWithSameName = attributes\n      .slice(idx + 1) // only search after the index of the current attribute to avoid deleting both\n      .some((other) => other.name === attr.name && other !== attr);\n    return !hasOtherWithSameName;\n  });\n}\n\nclass Parser {\n  private structs = new Array<Struct>();\n\n  constructor(private classNames: string[]) {}\n\n  private uniqueClassName(className: string): string {\n    if (this.classNames.includes(className)) {\n      className = `${className}A`;\n    }\n    this.classNames.push(className);\n    return className;\n  }\n\n  public resourceFrom(\n    fqpn: FQPN,\n    type: string,\n    schema: Schema,\n    terraformSchemaType: string,\n    constraint?: ConstructsMakerTarget,\n  ): ResourceModel {\n    let baseName = type;\n\n    const providerNameFromConstraint = constraint\n      ? (constraint.name as ProviderName)\n      : undefined;\n    const providerNameFromFQPN = parseFQPN(fqpn).name;\n\n    if (baseName.startsWith(`${providerNameFromFQPN}_`)) {\n      baseName = baseName.substr(providerNameFromFQPN.length + 1);\n    }\n\n    const providerName = providerNameFromConstraint\n      ? providerNameFromConstraint\n      : providerNameFromFQPN;\n\n    const isProvider = terraformSchemaType === \"provider\";\n    if (isProvider) {\n      baseName = `${providerName}_${baseName}`;\n      if (!(\"attributes\" in schema.block)) {\n        schema.block = {\n          attributes: {},\n          block_types: (schema.block as Block).block_types || {},\n        };\n      }\n      // somehow missing from provider schema\n      schema.block.attributes[\"alias\"] = {\n        type: \"string\",\n        description: \"Alias name\",\n        optional: true,\n        computed: false,\n      };\n    }\n\n    baseName = sanitizeClassOrNamespaceName(baseName, isProvider);\n\n    const className = this.uniqueClassName(toPascalCase(baseName));\n    // avoid naming collision - see https://github.com/hashicorp/terraform-cdk/issues/299\n    const configStructName = this.uniqueClassName(`${className}Config`);\n    const fileName = getFileName(providerName, baseName);\n\n    const filePath = `providers/${toSnakeCase(providerName)}/${fileName}`;\n    let attributes = this.renderAttributesForBlock(\n      new Scope({\n        name: baseName,\n        isProvider,\n        parent: isProvider\n          ? undefined\n          : new Scope({ name: providerName, isProvider: true }),\n      }),\n      schema.block,\n    );\n\n    function getStructAttribute(\n      attributes: AttributeModel[],\n      path: string,\n    ): AttributeModel {\n      const [first, ...rest] = path.split(\".\");\n      const attribute = attributes.find((att) => {\n        return att.terraformName === first;\n      });\n      if (!attribute)\n        throw new Error(\n          `Expected to find recursive attribute at path: ${path}`,\n        );\n      if (!attribute.type.struct)\n        throw new Error(\n          `Expected to find struct type attribute at path: ${path} but got ${attribute.type.storedClassType}`,\n        );\n      if (rest.length === 0) return attribute;\n      return getStructAttribute(\n        attribute.type.struct.attributes,\n        rest.join(\".\"),\n      );\n    }\n\n    // Introduce recursion for some attributes\n    const recursiveAttributePaths = detectAttributeLoops(attributes);\n\n    Object.entries(recursiveAttributePaths).forEach(\n      ([attributePath, structPath]) => {\n        // TODO: build this to be a bit more defensive (e.g. remove ! operator)\n        const recursionTargetStructAttribute = getStructAttribute(\n          attributes,\n          structPath,\n        );\n        const parts = attributePath.split(\".\");\n        const attributeName = parts.pop();\n        const parentAttribute = getStructAttribute(attributes, parts.join(\".\"));\n        const indexToReplace =\n          parentAttribute.type.struct!.attributes.findIndex(\n            (att) => att.terraformName === attributeName,\n          );\n        if (indexToReplace === -1)\n          throw new Error(\"Can't find attribute at path \" + attributePath);\n        const previousAttribute =\n          parentAttribute.type.struct!.attributes[indexToReplace];\n\n        parentAttribute.type.struct!.attributes[indexToReplace] =\n          recursionTargetStructAttribute; // introduce recursion\n\n        // ugly, pls c̶a̶l̶l̶ refactor me maybe\n        // we store all structs in this.structs – now we need to dispose all structs that are part of previousAttribute\n        const disposeStructs = (attr: AttributeModel) => {\n          if (attr.type.struct) {\n            attr.type.struct.attributes.forEach(disposeStructs);\n            this.structs = this.structs.filter((s) => s !== attr.type.struct);\n          }\n        };\n\n        disposeStructs(previousAttribute);\n      },\n    );\n\n    attributes = deduplicateAttributesWithSameName(attributes);\n\n    const resourceModel = new ResourceModel({\n      terraformType: type,\n      baseName,\n      fileName,\n      filePath,\n      className,\n      schema,\n      fqpn,\n      attributes,\n      terraformSchemaType,\n      structs: this.structs,\n      configStructName,\n    });\n\n    return resourceModel;\n  }\n\n  private renderAttributeType(\n    scope: Scope[],\n    attributeType: AttributeType | AttributeNestedType,\n    parentKind?: string,\n  ): AttributeTypeModel {\n    const parent = scope[scope.length - 1];\n    if (shouldSkipAttribute(parent.baseName)) {\n      return new MapAttributeTypeModel(new SimpleAttributeTypeModel(\"any\"));\n    }\n\n    if (typeof attributeType === \"string\") {\n      switch (attributeType) {\n        case \"bool\":\n          return new SimpleAttributeTypeModel(\"boolean\");\n        case \"string\":\n          return new SimpleAttributeTypeModel(\"string\");\n        case \"number\":\n          return new SimpleAttributeTypeModel(\"number\");\n        case \"dynamic\":\n          return new MapAttributeTypeModel(new SimpleAttributeTypeModel(\"any\"));\n        default:\n          throw new Error(`invalid primitive type ${attributeType}`);\n      }\n    }\n\n    if (Array.isArray(attributeType)) {\n      if (attributeType.length !== 2) {\n        throw new Error(`unexpected array`);\n      }\n\n      const [kind, type] = attributeType;\n\n      if (kind === \"set\" || kind === \"list\") {\n        const elementType = this.renderAttributeType(\n          scope,\n          type as AttributeType,\n          [kind, parentKind].join(\"\"),\n        );\n        return kind === \"list\"\n          ? new ListAttributeTypeModel(elementType, false, false)\n          : new SetAttributeTypeModel(elementType, false, false);\n      }\n\n      if (kind === \"map\") {\n        const valueType = this.renderAttributeType(\n          scope,\n          type as AttributeType,\n          [kind, parentKind].join(\"\"),\n        );\n        return new MapAttributeTypeModel(valueType);\n      }\n\n      if (kind === \"object\") {\n        const objAttributes = type as { [name: string]: AttributeType };\n        const attributes: { [name: string]: Attribute } = {};\n        for (const [name, type] of Object.entries(objAttributes)) {\n          attributes[name] = { type };\n        }\n        const struct = this.addAnonymousStruct(\n          scope,\n          attributes,\n          parentKind ?? kind,\n        );\n        return new StructAttributeTypeModel(struct);\n      }\n    }\n\n    if (isAttributeNestedType(attributeType)) {\n      let struct = undefined;\n      let typeModel = undefined;\n      switch (attributeType.nesting_mode) {\n        case \"list\":\n          struct = this.addAnonymousStruct(\n            scope,\n            attributeType.attributes,\n            attributeType.nesting_mode,\n          );\n          typeModel = new ListAttributeTypeModel(\n            new StructAttributeTypeModel(struct),\n            false,\n            false,\n          );\n          break;\n        case \"set\":\n          struct = this.addAnonymousStruct(\n            scope,\n            attributeType.attributes,\n            attributeType.nesting_mode,\n          );\n          typeModel = new SetAttributeTypeModel(\n            new StructAttributeTypeModel(struct),\n            false,\n            false,\n          );\n          break;\n        case \"map\":\n          struct = this.addAnonymousStruct(\n            scope,\n            attributeType.attributes,\n            attributeType.nesting_mode,\n          );\n          typeModel = new MapAttributeTypeModel(\n            new StructAttributeTypeModel(struct),\n          );\n          break;\n        case \"single\":\n          struct = this.addAnonymousStruct(\n            scope,\n            attributeType.attributes,\n            attributeType.nesting_mode,\n          );\n          typeModel = new StructAttributeTypeModel(struct);\n          break;\n        default: {\n          throw new Error(\n            `nested_type with nesting_mode \"${\n              attributeType.nesting_mode\n            }\" not supported (attribute scope: ${scope\n              .map((s) => s.fullName)\n              .join(\",\")}`,\n          );\n        }\n      }\n      return typeModel;\n    }\n\n    throw new Error(`unknown type ${JSON.stringify(attributeType)}`);\n  }\n\n  public renderAttributesForBlock(parentType: Scope, block: Block) {\n    const attributes = new Array<AttributeModel>();\n\n    for (const [terraformAttributeName, att] of Object.entries(\n      block.attributes || {},\n    )) {\n      let type: AttributeTypeModel;\n      let forcePlainGetterType = false;\n      if (shouldSkipAttribute(parentType.fullName(terraformAttributeName))) {\n        type = new SkippedAttributeTypeModel();\n        forcePlainGetterType = true;\n      } else {\n        type = this.renderAttributeType(\n          [\n            parentType,\n            new Scope({\n              name: terraformAttributeName,\n              parent: parentType,\n              isProvider: parentType.isProvider,\n              isComputed: !!att.computed,\n              isOptional: !!att.optional,\n              isRequired: !!att.required,\n              isNestedType: isNestedTypeAttribute(att),\n            }),\n          ],\n          att.type || att.nested_type,\n        );\n      }\n\n      const name = toCamelCase(terraformAttributeName);\n\n      attributes.push(\n        new AttributeModel({\n          terraformFullName: parentType.fullName(terraformAttributeName),\n          description: att.description,\n          name,\n          storageName: `_${name}`,\n          computed: !!att.computed,\n          optional: !!att.optional,\n          terraformName: terraformAttributeName,\n          type,\n          provider: parentType.isProvider,\n          required: !!att.required,\n          forcePlainGetterType,\n        }),\n      );\n    }\n\n    for (const [blockTypeName, blockType] of Object.entries(\n      block.block_types || {},\n    )) {\n      if (shouldSkipAttribute(parentType.fullName(blockTypeName))) {\n        const name = toCamelCase(blockTypeName);\n        const parent = new Scope({\n          name: blockTypeName,\n          parent: parentType,\n          isProvider: parentType.isProvider,\n        });\n        attributes.push(\n          new AttributeModel({\n            name,\n            terraformName: blockTypeName,\n            terraformFullName: parent.fullName(blockTypeName),\n            type: new SkippedAttributeTypeModel(),\n            description: `${blockTypeName} block`,\n            storageName: `_${name}`,\n            optional: true,\n            computed: false,\n            provider: parentType.isProvider,\n            required: false,\n          }),\n        );\n        continue;\n      }\n      // create a struct for this block\n      let blockAttributes = this.renderAttributesForBlock(\n        new Scope({\n          name: `${parentType.name}_${blockTypeName}`,\n          parent: parentType,\n          isProvider: parentType.isProvider,\n          inBlockType: true,\n        }),\n        blockType.block,\n      );\n\n      blockAttributes = deduplicateAttributesWithSameName(blockAttributes);\n\n      const blockStruct = this.addStruct(\n        [\n          parentType,\n          new Scope({\n            name: blockTypeName,\n            parent: parentType,\n            isProvider: parentType.isProvider,\n          }),\n        ],\n        blockAttributes,\n        blockType.nesting_mode,\n        (blockType.nesting_mode === \"list\" ||\n          blockType.nesting_mode === \"set\") &&\n          blockType.max_items === 1,\n      );\n\n      // define the attribute\n      attributes.push(\n        attributeForBlockType(\n          blockTypeName,\n          blockType,\n          blockStruct,\n          parentType.isProvider,\n          parentType,\n        ),\n      );\n    }\n\n    return attributes;\n\n    function attributeForBlockType(\n      terraformName: string,\n      blockType: BlockType,\n      struct: Struct,\n      isProvider: boolean,\n      parent: Scope,\n    ): AttributeModel {\n      const name = toCamelCase(terraformName);\n      let optional: boolean;\n      let required: boolean;\n\n      switch (blockType.nesting_mode) {\n        case \"single\":\n          optional = !struct.attributes.some((x) => !x.optional);\n          required = !struct.attributes.some((x) => !x.required);\n\n          // This is for bug #3570 as both optional and required evaluate to false under some circumstances\n          // (this then causes the computed block to not be part of assignableAttributes and thus skipped in the generated code)\n          // Hence: If both optional and required are false, set optional to true IF at least one\n          // attribute in the block has optional = true or required = true, as this would mean that at least something can be set\n          // and the block is not all computed.\n          if (!optional && !required) {\n            optional = struct.attributes.some((x) => x.optional || x.required);\n          }\n\n          return new AttributeModel({\n            name,\n            terraformName,\n            terraformFullName: parent.fullName(terraformName),\n            type: new StructAttributeTypeModel(struct),\n            description: `${terraformName} block`,\n            storageName: `_${name}`,\n            optional,\n            computed: false,\n            provider: isProvider,\n            required,\n          });\n\n        case \"map\":\n          return new AttributeModel({\n            name,\n            terraformName,\n            terraformFullName: parent.fullName(terraformName),\n            type: new MapAttributeTypeModel(\n              new StructAttributeTypeModel(struct),\n            ),\n            description: `${terraformName} block`,\n            storageName: `_${name}`,\n            optional: false,\n            computed: false,\n            provider: isProvider,\n            required: false,\n          });\n\n        case \"list\":\n        case \"set\":\n          optional =\n            blockType.min_items === undefined ? true : blockType.min_items < 1;\n          required =\n            blockType.min_items === undefined ? false : blockType.min_items > 0;\n          return new AttributeModel({\n            name,\n            terraformName: terraformName,\n            terraformFullName: parent.fullName(terraformName),\n            type:\n              blockType.nesting_mode === \"list\"\n                ? new ListAttributeTypeModel(\n                    new StructAttributeTypeModel(struct),\n                    blockType.max_items === 1,\n                    true,\n                  )\n                : new SetAttributeTypeModel(\n                    new StructAttributeTypeModel(struct),\n                    blockType.max_items === 1,\n                    true,\n                  ),\n            description: `${terraformName} block`,\n            storageName: `_${name}`,\n            optional,\n            computed: false,\n            provider: isProvider,\n            required,\n          });\n      }\n    }\n  }\n  private addAnonymousStruct(\n    scope: Scope[],\n    attrs: { [name: string]: Attribute } | undefined,\n    nesting_mode: string,\n  ) {\n    let attributes = new Array<AttributeModel>();\n    const parent = scope[scope.length - 1];\n    if (attrs) {\n      for (const [terraformName, att] of Object.entries(attrs || {})) {\n        // nested types support computed, optional and required on attribute level\n        // if parent is computed, child always is computed as well\n        const computed =\n          !!parent.isComputed || (parent.isNestedType && !!att.computed);\n        const optional = parent.isNestedType\n          ? !!att.optional\n          : !!parent.isOptional;\n        const required = parent.isNestedType\n          ? !!att.required\n          : !!parent.isRequired;\n        const name = toCamelCase(terraformName);\n        const type = this.renderAttributeType(\n          [\n            ...scope,\n            new Scope({\n              name: terraformName,\n              parent,\n              isProvider: parent.isProvider,\n              isComputed: computed,\n              isOptional: optional,\n              isRequired: required,\n              isNestedType: parent.isNestedType,\n            }),\n          ],\n          att.type || att.nested_type,\n        );\n        attributes.push(\n          new AttributeModel({\n            name,\n            storageName: `_${name}`,\n            computed: computed,\n            description: att.description,\n            optional: optional,\n            terraformName,\n            terraformFullName: parent.fullName(terraformName),\n            type,\n            provider: parent.isProvider,\n            required: required,\n          }),\n        );\n      }\n    }\n\n    attributes = deduplicateAttributesWithSameName(attributes);\n\n    return this.addStruct(scope, attributes, nesting_mode);\n  }\n\n  private addStruct(\n    scope: Scope[],\n    attributes: AttributeModel[],\n    nesting_mode: string,\n    isSingleItem = false,\n  ) {\n    const possibleName = toPascalCase(\n      scope.map((x) => toSnakeCase(x.name)).join(\"_\"),\n    );\n    const name = this.uniqueClassName(\n      isReservedStructClassName(possibleName)\n        ? `${possibleName}Struct`\n        : possibleName,\n    );\n\n    const parent = scope[scope.length - 1];\n    // blockType.nesting_mode => list/set & blockType.max_items === 1,\n    const isClass = (parent.isComputed && !parent.isOptional) || isSingleItem;\n    const isAnonymous = true;\n    const s = new Struct(\n      name,\n      attributes,\n      isClass,\n      isAnonymous,\n      isSingleItem,\n      nesting_mode,\n    );\n    this.structs.push(s);\n    return s;\n  }\n}\n\nexport class ResourceParser {\n  private uniqueClassnames: string[] = [];\n  private resources: Record<string, ResourceModel> = {};\n\n  public parse(\n    fqpn: FQPN,\n    type: string,\n    schema: Schema,\n    terraformType: string,\n    constraint?: ConstructsMakerTarget,\n  ): ResourceModel {\n    if (this.resources[type]) {\n      return this.resources[type];\n    }\n\n    const parser = new Parser(this.uniqueClassnames);\n    const resource = parser.resourceFrom(\n      fqpn,\n      type,\n      schema,\n      terraformType,\n      constraint,\n    );\n    this.resources[type] = resource;\n    return resource;\n  }\n\n  // Used by convert to determine the right name for a class\n  public getClassNameForResource(terraformType: string) {\n    const resource = this.resources[terraformType];\n    return resource ? resource.className : \"\";\n  }\n\n  // Used by convert to determine the right name for a namespace\n  public getNamespaceNameForResource(terraformType: string) {\n    // Special case external provider since the name of resource doesn't have a prefix\n    if (terraformType === \"data_external_\") {\n      terraformType = \"data_external\";\n    }\n\n    const resource = this.resources[terraformType];\n    if (!resource) {\n      return \"\";\n    }\n    const folder = `providers/${resource.provider}`;\n    return resource.filePath.replace(`${folder}/`, \"\").replace(\"/index.ts\", \"\");\n  }\n}\n"]}
|