@microsoft/terraform-cdk-constructs 1.7.1 → 1.9.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (121) hide show
  1. package/.jsii +46476 -27231
  2. package/API.md +68443 -28286
  3. package/lib/azure-actiongroup/lib/action-group.js +1 -1
  4. package/lib/azure-activitylogalert/lib/activity-log-alert.js +1 -1
  5. package/lib/azure-aks/lib/aks-cluster.js +1 -1
  6. package/lib/azure-diagnosticsettings/lib/diagnostic-settings.js +1 -1
  7. package/lib/azure-dnsforwardingruleset/lib/dns-forwarding-ruleset.js +1 -1
  8. package/lib/azure-dnsforwardingruleset/lib/forwarding-rule.js +1 -1
  9. package/lib/azure-dnsforwardingruleset/lib/virtual-network-link.js +1 -1
  10. package/lib/azure-dnsresolver/lib/dns-resolver.js +1 -1
  11. package/lib/azure-dnsresolver/lib/inbound-endpoint.js +1 -1
  12. package/lib/azure-dnsresolver/lib/outbound-endpoint.js +1 -1
  13. package/lib/azure-dnszone/lib/dns-zone.js +1 -1
  14. package/lib/azure-dnszone/lib/index.d.ts +1 -0
  15. package/lib/azure-dnszone/lib/index.js +2 -1
  16. package/lib/azure-dnszone/lib/records/dns-record-schemas.d.ts +68 -0
  17. package/lib/azure-dnszone/lib/records/dns-record-schemas.js +813 -0
  18. package/lib/azure-dnszone/lib/records/dns-records.d.ts +688 -0
  19. package/lib/azure-dnszone/lib/records/dns-records.js +924 -0
  20. package/lib/azure-dnszone/lib/records/index.d.ts +19 -0
  21. package/lib/azure-dnszone/lib/records/index.js +38 -0
  22. package/lib/azure-dnszone/test/dns-records.integ.d.ts +21 -0
  23. package/lib/azure-dnszone/test/dns-records.integ.js +321 -0
  24. package/lib/azure-dnszone/test/dns-records.spec.d.ts +20 -0
  25. package/lib/azure-dnszone/test/dns-records.spec.js +950 -0
  26. package/lib/azure-loganalyticsworkspace/index.d.ts +6 -0
  27. package/lib/azure-loganalyticsworkspace/index.js +23 -0
  28. package/lib/azure-loganalyticsworkspace/lib/index.d.ts +5 -0
  29. package/lib/azure-loganalyticsworkspace/lib/index.js +22 -0
  30. package/lib/azure-loganalyticsworkspace/lib/log-analytics-workspace-schemas.d.ts +51 -0
  31. package/lib/azure-loganalyticsworkspace/lib/log-analytics-workspace-schemas.js +255 -0
  32. package/lib/azure-loganalyticsworkspace/lib/log-analytics-workspace.d.ts +301 -0
  33. package/lib/azure-loganalyticsworkspace/lib/log-analytics-workspace.js +213 -0
  34. package/lib/azure-loganalyticsworkspace/test/log-analytics-workspace.integ.d.ts +9 -0
  35. package/lib/azure-loganalyticsworkspace/test/log-analytics-workspace.integ.js +71 -0
  36. package/lib/azure-loganalyticsworkspace/test/log-analytics-workspace.spec.d.ts +8 -0
  37. package/lib/azure-loganalyticsworkspace/test/log-analytics-workspace.spec.js +504 -0
  38. package/lib/azure-metricalert/lib/metric-alert.js +1 -1
  39. package/lib/azure-networkinterface/lib/network-interface.js +1 -1
  40. package/lib/azure-networksecuritygroup/lib/network-security-group.js +1 -1
  41. package/lib/azure-networkwatcher/index.d.ts +14 -0
  42. package/lib/azure-networkwatcher/index.js +31 -0
  43. package/lib/azure-networkwatcher/lib/index.d.ts +5 -0
  44. package/lib/azure-networkwatcher/lib/index.js +22 -0
  45. package/lib/azure-networkwatcher/lib/network-watcher-schemas.d.ts +47 -0
  46. package/lib/azure-networkwatcher/lib/network-watcher-schemas.js +167 -0
  47. package/lib/azure-networkwatcher/lib/network-watcher.d.ts +181 -0
  48. package/lib/azure-networkwatcher/lib/network-watcher.js +187 -0
  49. package/lib/azure-networkwatcher/test/network-watcher.integ.d.ts +12 -0
  50. package/lib/azure-networkwatcher/test/network-watcher.integ.js +84 -0
  51. package/lib/azure-networkwatcher/test/network-watcher.spec.d.ts +8 -0
  52. package/lib/azure-networkwatcher/test/network-watcher.spec.js +312 -0
  53. package/lib/azure-policyassignment/lib/policy-assignment.js +1 -1
  54. package/lib/azure-policydefinition/lib/policy-definition.js +1 -1
  55. package/lib/azure-policysetdefinition/index.d.ts +10 -0
  56. package/lib/azure-policysetdefinition/index.js +27 -0
  57. package/lib/azure-policysetdefinition/lib/index.d.ts +5 -0
  58. package/lib/azure-policysetdefinition/lib/index.js +22 -0
  59. package/lib/azure-policysetdefinition/lib/policy-set-definition-schemas.d.ts +50 -0
  60. package/lib/azure-policysetdefinition/lib/policy-set-definition-schemas.js +255 -0
  61. package/lib/azure-policysetdefinition/lib/policy-set-definition.d.ts +426 -0
  62. package/lib/azure-policysetdefinition/lib/policy-set-definition.js +255 -0
  63. package/lib/azure-policysetdefinition/test/policy-set-definition.integ.d.ts +9 -0
  64. package/lib/azure-policysetdefinition/test/policy-set-definition.integ.js +56 -0
  65. package/lib/azure-policysetdefinition/test/policy-set-definition.spec.d.ts +8 -0
  66. package/lib/azure-policysetdefinition/test/policy-set-definition.spec.js +745 -0
  67. package/lib/azure-privatednszone/lib/index.d.ts +1 -0
  68. package/lib/azure-privatednszone/lib/index.js +2 -1
  69. package/lib/azure-privatednszone/lib/private-dns-zone.d.ts +0 -2
  70. package/lib/azure-privatednszone/lib/private-dns-zone.js +6 -13
  71. package/lib/azure-privatednszone/lib/records/index.d.ts +7 -0
  72. package/lib/azure-privatednszone/lib/records/index.js +26 -0
  73. package/lib/azure-privatednszone/lib/records/private-dns-record-schemas.d.ts +52 -0
  74. package/lib/azure-privatednszone/lib/records/private-dns-record-schemas.js +683 -0
  75. package/lib/azure-privatednszone/lib/records/private-dns-records.d.ts +523 -0
  76. package/lib/azure-privatednszone/lib/records/private-dns-records.js +739 -0
  77. package/lib/azure-privatednszone/test/private-dns-records.integ.d.ts +19 -0
  78. package/lib/azure-privatednszone/test/private-dns-records.integ.js +245 -0
  79. package/lib/azure-privatednszone/test/private-dns-records.spec.d.ts +18 -0
  80. package/lib/azure-privatednszone/test/private-dns-records.spec.js +756 -0
  81. package/lib/azure-privatednszone/test/private-dns-zone.spec.js +5 -5
  82. package/lib/azure-privatednszonelink/lib/private-dns-zone-link.js +1 -1
  83. package/lib/azure-publicipaddress/lib/public-ip-address.js +1 -1
  84. package/lib/azure-resourcegroup/lib/resource-group.js +1 -1
  85. package/lib/azure-roleassignment/lib/role-assignment.js +1 -1
  86. package/lib/azure-roledefinition/lib/role-definition.js +1 -1
  87. package/lib/azure-storageaccount/lib/storage-account.js +1 -1
  88. package/lib/azure-subnet/lib/subnet.js +1 -1
  89. package/lib/azure-virtualmachine/lib/virtual-machine.js +1 -1
  90. package/lib/azure-virtualnetwork/lib/virtual-network.js +1 -1
  91. package/lib/azure-virtualnetworkgateway/lib/virtual-network-gateway-schemas.js +2 -2
  92. package/lib/azure-virtualnetworkgateway/lib/virtual-network-gateway.d.ts +4 -2
  93. package/lib/azure-virtualnetworkgateway/lib/virtual-network-gateway.js +8 -5
  94. package/lib/azure-virtualnetworkgateway/test/virtual-network-gateway.spec.js +109 -1
  95. package/lib/azure-virtualnetworkgatewayconnection/lib/virtual-network-gateway-connection.js +1 -1
  96. package/lib/azure-virtualnetworkmanager/lib/connectivity-configuration.js +1 -1
  97. package/lib/azure-virtualnetworkmanager/lib/ipam-pool-static-cidr.js +1 -1
  98. package/lib/azure-virtualnetworkmanager/lib/ipam-pool.js +1 -1
  99. package/lib/azure-virtualnetworkmanager/lib/network-group-static-member.js +1 -1
  100. package/lib/azure-virtualnetworkmanager/lib/network-group.js +1 -1
  101. package/lib/azure-virtualnetworkmanager/lib/security-admin-configuration.js +1 -1
  102. package/lib/azure-virtualnetworkmanager/lib/security-admin-rule-collection.js +1 -1
  103. package/lib/azure-virtualnetworkmanager/lib/security-admin-rule.js +1 -1
  104. package/lib/azure-virtualnetworkmanager/lib/virtual-network-manager.js +1 -1
  105. package/lib/azure-vmss/lib/virtual-machine-scale-set.js +1 -1
  106. package/lib/core-azure/lib/azapi/azapi-resource.js +4 -4
  107. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-client-config/index.js +2 -2
  108. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource/index.js +5 -5
  109. package/lib/core-azure/lib/azapi/providers-azapi/provider/index.js +1 -1
  110. package/lib/core-azure/lib/azapi/providers-azapi/resource/index.js +5 -5
  111. package/lib/core-azure/lib/azapi/providers-azapi/resource-action/index.js +3 -3
  112. package/lib/core-azure/lib/azapi/providers-azapi/update-resource/index.js +3 -3
  113. package/lib/core-azure/lib/azapi/schema-mapper/schema-mapper.js +1 -1
  114. package/lib/core-azure/lib/version-manager/api-version-manager.js +1 -1
  115. package/lib/core-azure/lib/version-manager/interfaces/version-interfaces.js +7 -7
  116. package/lib/index.d.ts +23 -0
  117. package/lib/index.js +25 -2
  118. package/lib/testing/index.js +2 -2
  119. package/lib/testing/lib/cleanup.js +1 -1
  120. package/lib/testing/lib/metadata.js +1 -1
  121. package/package.json +1 -1
@@ -0,0 +1,745 @@
1
+ "use strict";
2
+ /**
3
+ * Comprehensive tests for the PolicySetDefinition implementation
4
+ *
5
+ * This test suite validates the PolicySetDefinition class using the AzapiResource framework.
6
+ * Tests cover automatic version resolution, explicit version pinning, schema validation,
7
+ * property configurations, and resource creation.
8
+ */
9
+ Object.defineProperty(exports, "__esModule", { value: true });
10
+ const cdktf_1 = require("cdktf");
11
+ const cdktf = require("cdktf");
12
+ const policy_set_definition_1 = require("../lib/policy-set-definition");
13
+ describe("PolicySetDefinition - Implementation", () => {
14
+ let app;
15
+ let stack;
16
+ beforeEach(() => {
17
+ app = cdktf_1.Testing.app();
18
+ stack = new cdktf.TerraformStack(app, "TestStack");
19
+ });
20
+ // =============================================================================
21
+ // Constructor and Basic Properties
22
+ // =============================================================================
23
+ describe("Constructor and Basic Properties", () => {
24
+ it("should create a Policy Set Definition with minimal configuration", () => {
25
+ const props = {
26
+ displayName: "Test Initiative",
27
+ scope: "/subscriptions/test-sub",
28
+ policyDefinitions: [
29
+ {
30
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/test-policy-id",
31
+ },
32
+ ],
33
+ };
34
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "TestPolicySet", props);
35
+ expect(policySet).toBeInstanceOf(policy_set_definition_1.PolicySetDefinition);
36
+ expect(policySet.props).toBe(props);
37
+ expect(policySet.displayName).toBe("Test Initiative");
38
+ expect(policySet.policyType).toBe("Custom");
39
+ expect(policySet.policyDefinitionCount).toBe(1);
40
+ });
41
+ it("should create a Policy Set Definition with description", () => {
42
+ const props = {
43
+ displayName: "Security Initiative",
44
+ description: "Initiative for security compliance",
45
+ scope: "/subscriptions/test-sub",
46
+ policyDefinitions: [
47
+ {
48
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
49
+ },
50
+ ],
51
+ };
52
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "SecurityPolicySet", props);
53
+ expect(policySet.props.description).toBe("Initiative for security compliance");
54
+ });
55
+ it("should create a Policy Set Definition with custom name", () => {
56
+ const props = {
57
+ name: "my-custom-initiative",
58
+ displayName: "Custom Named Initiative",
59
+ scope: "/subscriptions/test-sub",
60
+ policyDefinitions: [
61
+ {
62
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
63
+ },
64
+ ],
65
+ };
66
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "CustomNamedPolicySet", props);
67
+ expect(policySet.props.name).toBe("my-custom-initiative");
68
+ });
69
+ it("should create a Policy Set Definition with metadata", () => {
70
+ const props = {
71
+ displayName: "Metadata Initiative",
72
+ scope: "/subscriptions/test-sub",
73
+ metadata: {
74
+ category: "Security Center",
75
+ version: "1.0.0",
76
+ preview: false,
77
+ deprecated: false,
78
+ },
79
+ policyDefinitions: [
80
+ {
81
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
82
+ },
83
+ ],
84
+ };
85
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "MetadataPolicySet", props);
86
+ expect(policySet.props.metadata?.category).toBe("Security Center");
87
+ expect(policySet.props.metadata?.version).toBe("1.0.0");
88
+ expect(policySet.props.metadata?.preview).toBe(false);
89
+ });
90
+ });
91
+ // =============================================================================
92
+ // Version Resolution
93
+ // =============================================================================
94
+ describe("Version Resolution", () => {
95
+ it("should use latest version 2023-04-01 when no version specified", () => {
96
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "DefaultVersion", {
97
+ displayName: "Default Version Initiative",
98
+ scope: "/subscriptions/test-sub",
99
+ policyDefinitions: [
100
+ {
101
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
102
+ },
103
+ ],
104
+ });
105
+ expect(policySet.resolvedApiVersion).toBe("2023-04-01");
106
+ });
107
+ it("should use explicit version when specified", () => {
108
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "PinnedVersion", {
109
+ displayName: "Pinned Version Initiative",
110
+ scope: "/subscriptions/test-sub",
111
+ policyDefinitions: [
112
+ {
113
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
114
+ },
115
+ ],
116
+ apiVersion: "2021-06-01",
117
+ });
118
+ expect(policySet.resolvedApiVersion).toBe("2021-06-01");
119
+ });
120
+ it("should use latest version 2023-04-01 when explicitly specified", () => {
121
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "LatestVersion", {
122
+ displayName: "Latest Version Initiative",
123
+ scope: "/subscriptions/test-sub",
124
+ policyDefinitions: [
125
+ {
126
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
127
+ },
128
+ ],
129
+ apiVersion: "2023-04-01",
130
+ });
131
+ expect(policySet.resolvedApiVersion).toBe("2023-04-01");
132
+ });
133
+ });
134
+ // =============================================================================
135
+ // Validation Tests
136
+ // =============================================================================
137
+ describe("Validation", () => {
138
+ it("should throw error when displayName is empty", () => {
139
+ const props = {
140
+ displayName: "",
141
+ scope: "/subscriptions/test-sub",
142
+ policyDefinitions: [
143
+ {
144
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
145
+ },
146
+ ],
147
+ };
148
+ expect(() => {
149
+ new policy_set_definition_1.PolicySetDefinition(stack, "EmptyDisplayName", props);
150
+ }).toThrow(/displayName is required/);
151
+ });
152
+ it("should throw error when policyDefinitions is empty", () => {
153
+ const props = {
154
+ displayName: "No Policies Initiative",
155
+ scope: "/subscriptions/test-sub",
156
+ policyDefinitions: [],
157
+ };
158
+ expect(() => {
159
+ new policy_set_definition_1.PolicySetDefinition(stack, "NoPolicies", props);
160
+ }).toThrow(/At least one policy definition reference is required/);
161
+ });
162
+ it("should throw error when policyDefinitionId is missing", () => {
163
+ const props = {
164
+ displayName: "Invalid Policy Reference",
165
+ scope: "/subscriptions/test-sub",
166
+ policyDefinitions: [
167
+ {
168
+ policyDefinitionId: "",
169
+ },
170
+ ],
171
+ };
172
+ expect(() => {
173
+ new policy_set_definition_1.PolicySetDefinition(stack, "InvalidPolicyRef", props);
174
+ }).toThrow(/policyDefinitionId is required/);
175
+ });
176
+ it("should throw error for duplicate group names", () => {
177
+ const props = {
178
+ displayName: "Duplicate Groups",
179
+ scope: "/subscriptions/test-sub",
180
+ policyDefinitionGroups: [
181
+ { name: "Security", displayName: "Security Policies" },
182
+ { name: "Security", displayName: "Duplicate Security" },
183
+ ],
184
+ policyDefinitions: [
185
+ {
186
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
187
+ },
188
+ ],
189
+ };
190
+ expect(() => {
191
+ new policy_set_definition_1.PolicySetDefinition(stack, "DuplicateGroups", props);
192
+ }).toThrow(/Duplicate group name 'Security'/);
193
+ });
194
+ it("should throw error for invalid group reference in policy definition", () => {
195
+ const props = {
196
+ displayName: "Invalid Group Reference",
197
+ scope: "/subscriptions/test-sub",
198
+ policyDefinitionGroups: [
199
+ { name: "Security", displayName: "Security Policies" },
200
+ ],
201
+ policyDefinitions: [
202
+ {
203
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
204
+ groupNames: ["NonExistentGroup"],
205
+ },
206
+ ],
207
+ };
208
+ expect(() => {
209
+ new policy_set_definition_1.PolicySetDefinition(stack, "InvalidGroupRef", props);
210
+ }).toThrow(/references unknown group 'NonExistentGroup'/);
211
+ });
212
+ it("should accept valid policy set with groups", () => {
213
+ const props = {
214
+ displayName: "Valid Groups Initiative",
215
+ scope: "/subscriptions/test-sub",
216
+ policyDefinitionGroups: [
217
+ { name: "Security", displayName: "Security Policies" },
218
+ { name: "Compliance", displayName: "Compliance Policies" },
219
+ ],
220
+ policyDefinitions: [
221
+ {
222
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
223
+ groupNames: ["Security"],
224
+ },
225
+ {
226
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-2",
227
+ groupNames: ["Security", "Compliance"],
228
+ },
229
+ ],
230
+ };
231
+ expect(() => {
232
+ new policy_set_definition_1.PolicySetDefinition(stack, "ValidGroups", props);
233
+ }).not.toThrow();
234
+ });
235
+ });
236
+ // =============================================================================
237
+ // Policy Definition References
238
+ // =============================================================================
239
+ describe("Policy Definition References", () => {
240
+ it("should create initiative with single policy definition", () => {
241
+ const props = {
242
+ displayName: "Single Policy Initiative",
243
+ scope: "/subscriptions/test-sub",
244
+ policyDefinitions: [
245
+ {
246
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
247
+ policyDefinitionReferenceId: "allowedLocations",
248
+ },
249
+ ],
250
+ };
251
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "SinglePolicy", props);
252
+ expect(policySet.policyDefinitionCount).toBe(1);
253
+ });
254
+ it("should create initiative with multiple policy definitions", () => {
255
+ const props = {
256
+ displayName: "Multiple Policy Initiative",
257
+ scope: "/subscriptions/test-sub",
258
+ policyDefinitions: [
259
+ {
260
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
261
+ policyDefinitionReferenceId: "policy1",
262
+ },
263
+ {
264
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-2",
265
+ policyDefinitionReferenceId: "policy2",
266
+ },
267
+ {
268
+ policyDefinitionId: "/subscriptions/sub-id/providers/Microsoft.Authorization/policyDefinitions/custom-policy",
269
+ policyDefinitionReferenceId: "customPolicy",
270
+ },
271
+ ],
272
+ };
273
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "MultiplePolicies", props);
274
+ expect(policySet.policyDefinitionCount).toBe(3);
275
+ });
276
+ it("should create initiative with policy parameters", () => {
277
+ const props = {
278
+ displayName: "Parameterized Initiative",
279
+ scope: "/subscriptions/test-sub",
280
+ policyDefinitions: [
281
+ {
282
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
283
+ policyDefinitionReferenceId: "tagPolicy",
284
+ parameters: {
285
+ tagName: { value: "environment" },
286
+ tagValue: { value: "production" },
287
+ },
288
+ },
289
+ ],
290
+ };
291
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "ParameterizedPolicy", props);
292
+ expect(policySet.props.policyDefinitions[0].parameters?.tagName.value).toBe("environment");
293
+ });
294
+ });
295
+ // =============================================================================
296
+ // Initiative Parameters
297
+ // =============================================================================
298
+ describe("Initiative Parameters", () => {
299
+ it("should create initiative with parameter definitions", () => {
300
+ const props = {
301
+ displayName: "Initiative With Parameters",
302
+ scope: "/subscriptions/test-sub",
303
+ parameters: {
304
+ allowedLocations: {
305
+ type: "Array",
306
+ displayName: "Allowed Locations",
307
+ description: "List of allowed Azure regions",
308
+ defaultValue: ["eastus", "westus2"],
309
+ allowedValues: [["eastus"], ["westus2"], ["eastus", "westus2"]],
310
+ },
311
+ requiredTag: {
312
+ type: "String",
313
+ displayName: "Required Tag",
314
+ description: "Name of the required tag",
315
+ defaultValue: "environment",
316
+ },
317
+ },
318
+ policyDefinitions: [
319
+ {
320
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
321
+ parameters: {
322
+ listOfAllowedLocations: {
323
+ value: "[parameters('allowedLocations')]",
324
+ },
325
+ },
326
+ },
327
+ ],
328
+ };
329
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "WithParams", props);
330
+ expect(policySet.props.parameters?.allowedLocations.type).toBe("Array");
331
+ expect(policySet.props.parameters?.requiredTag.type).toBe("String");
332
+ });
333
+ it("should create initiative with strong typed parameter", () => {
334
+ const props = {
335
+ displayName: "Strong Typed Parameters",
336
+ scope: "/subscriptions/test-sub",
337
+ parameters: {
338
+ storageAccountSku: {
339
+ type: "String",
340
+ displayName: "Storage Account SKU",
341
+ metadata: {
342
+ strongType: "storageSkus",
343
+ displayName: "Storage SKU",
344
+ },
345
+ },
346
+ },
347
+ policyDefinitions: [
348
+ {
349
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
350
+ },
351
+ ],
352
+ };
353
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "StrongTypedParams", props);
354
+ expect(policySet.props.parameters?.storageAccountSku.metadata?.strongType).toBe("storageSkus");
355
+ });
356
+ });
357
+ // =============================================================================
358
+ // Policy Definition Groups
359
+ // =============================================================================
360
+ describe("Policy Definition Groups", () => {
361
+ it("should create initiative with groups", () => {
362
+ const props = {
363
+ displayName: "Grouped Policies Initiative",
364
+ scope: "/subscriptions/test-sub",
365
+ policyDefinitionGroups: [
366
+ {
367
+ name: "Security",
368
+ displayName: "Security Policies",
369
+ category: "Security Center",
370
+ description: "Policies related to security",
371
+ },
372
+ {
373
+ name: "Compliance",
374
+ displayName: "Compliance Policies",
375
+ category: "Regulatory Compliance",
376
+ },
377
+ ],
378
+ policyDefinitions: [
379
+ {
380
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
381
+ groupNames: ["Security"],
382
+ },
383
+ {
384
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-2",
385
+ groupNames: ["Compliance"],
386
+ },
387
+ ],
388
+ };
389
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "GroupedPolicies", props);
390
+ expect(policySet.props.policyDefinitionGroups?.length).toBe(2);
391
+ expect(policySet.props.policyDefinitionGroups?.[0].name).toBe("Security");
392
+ });
393
+ it("should allow policy definitions in multiple groups", () => {
394
+ const props = {
395
+ displayName: "Multi-Group Policy Initiative",
396
+ scope: "/subscriptions/test-sub",
397
+ policyDefinitionGroups: [
398
+ { name: "Security" },
399
+ { name: "Compliance" },
400
+ { name: "Operations" },
401
+ ],
402
+ policyDefinitions: [
403
+ {
404
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
405
+ groupNames: ["Security", "Compliance"],
406
+ },
407
+ ],
408
+ };
409
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "MultiGroupPolicy", props);
410
+ expect(policySet.props.policyDefinitions[0].groupNames?.length).toBe(2);
411
+ });
412
+ });
413
+ // =============================================================================
414
+ // Property Transformation
415
+ // =============================================================================
416
+ describe("Property Transformation", () => {
417
+ it("should generate correct Azure API format via createResourceBody", () => {
418
+ const props = {
419
+ displayName: "Transform Test Initiative",
420
+ description: "Test transformation",
421
+ scope: "/subscriptions/test-sub",
422
+ policyDefinitions: [
423
+ {
424
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
425
+ policyDefinitionReferenceId: "policy1",
426
+ },
427
+ ],
428
+ };
429
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "TransformTest", props);
430
+ // Access the protected createResourceBody method
431
+ const resourceBody = policySet.createResourceBody(props);
432
+ expect(resourceBody).toHaveProperty("properties");
433
+ expect(resourceBody.properties).toHaveProperty("displayName", "Transform Test Initiative");
434
+ expect(resourceBody.properties).toHaveProperty("description", "Test transformation");
435
+ expect(resourceBody.properties).toHaveProperty("policyType", "Custom");
436
+ expect(resourceBody.properties.policyDefinitions).toHaveLength(1);
437
+ });
438
+ it("should include metadata when specified", () => {
439
+ const props = {
440
+ displayName: "Metadata Test",
441
+ scope: "/subscriptions/test-sub",
442
+ metadata: {
443
+ category: "Testing",
444
+ version: "2.0.0",
445
+ },
446
+ policyDefinitions: [
447
+ {
448
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
449
+ },
450
+ ],
451
+ };
452
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "MetadataTest", props);
453
+ const resourceBody = policySet.createResourceBody(props);
454
+ expect(resourceBody.properties.metadata).toBeDefined();
455
+ expect(resourceBody.properties.metadata.category).toBe("Testing");
456
+ });
457
+ it("should include parameters when specified", () => {
458
+ const props = {
459
+ displayName: "Parameters Test",
460
+ scope: "/subscriptions/test-sub",
461
+ parameters: {
462
+ testParam: {
463
+ type: "String",
464
+ defaultValue: "test",
465
+ },
466
+ },
467
+ policyDefinitions: [
468
+ {
469
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
470
+ },
471
+ ],
472
+ };
473
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "ParametersTest", props);
474
+ const resourceBody = policySet.createResourceBody(props);
475
+ expect(resourceBody.properties.parameters).toBeDefined();
476
+ expect(resourceBody.properties.parameters.testParam.type).toBe("String");
477
+ });
478
+ it("should include policyDefinitionGroups when specified", () => {
479
+ const props = {
480
+ displayName: "Groups Test",
481
+ scope: "/subscriptions/test-sub",
482
+ policyDefinitionGroups: [
483
+ { name: "TestGroup", displayName: "Test Group" },
484
+ ],
485
+ policyDefinitions: [
486
+ {
487
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
488
+ groupNames: ["TestGroup"],
489
+ },
490
+ ],
491
+ };
492
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "GroupsTest", props);
493
+ const resourceBody = policySet.createResourceBody(props);
494
+ expect(resourceBody.properties.policyDefinitionGroups).toBeDefined();
495
+ expect(resourceBody.properties.policyDefinitionGroups).toHaveLength(1);
496
+ });
497
+ });
498
+ // =============================================================================
499
+ // Integration with Base Class
500
+ // =============================================================================
501
+ describe("Integration with Base Class", () => {
502
+ it("should inherit from AzapiResource", () => {
503
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "InheritanceTest", {
504
+ displayName: "Inheritance Test",
505
+ scope: "/subscriptions/test-sub",
506
+ policyDefinitions: [
507
+ {
508
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
509
+ },
510
+ ],
511
+ });
512
+ // Verify it has AzapiResource properties
513
+ expect(policySet).toHaveProperty("terraformResource");
514
+ expect(policySet).toHaveProperty("resolvedApiVersion");
515
+ });
516
+ it("should have correct resourceType", () => {
517
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "ResourceTypeTest", {
518
+ displayName: "Resource Type Test",
519
+ scope: "/subscriptions/test-sub",
520
+ policyDefinitions: [
521
+ {
522
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
523
+ },
524
+ ],
525
+ });
526
+ // Access the protected resourceType method
527
+ const resourceType = policySet.resourceType();
528
+ expect(resourceType).toBe("Microsoft.Authorization/policySetDefinitions");
529
+ });
530
+ it("should resolve parent ID correctly", () => {
531
+ const scope = "/subscriptions/test-sub";
532
+ const props = {
533
+ displayName: "Parent ID Test",
534
+ scope,
535
+ policyDefinitions: [
536
+ {
537
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
538
+ },
539
+ ],
540
+ };
541
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "ParentIdTest", props);
542
+ // Access the protected resolveParentId method
543
+ const parentId = policySet.resolveParentId(props);
544
+ expect(parentId).toBe(scope);
545
+ });
546
+ it("should generate resource outputs", () => {
547
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "OutputsTest", {
548
+ displayName: "Outputs Test",
549
+ scope: "/subscriptions/test-sub",
550
+ policyDefinitions: [
551
+ {
552
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
553
+ },
554
+ ],
555
+ });
556
+ expect(policySet.idOutput).toBeInstanceOf(cdktf.TerraformOutput);
557
+ expect(policySet.nameOutput).toBeInstanceOf(cdktf.TerraformOutput);
558
+ expect(policySet.policySetDefinitionIdOutput).toBeInstanceOf(cdktf.TerraformOutput);
559
+ expect(policySet.id).toMatch(/^\$\{.*\.id\}$/);
560
+ });
561
+ });
562
+ // =============================================================================
563
+ // CDK Terraform Integration
564
+ // =============================================================================
565
+ describe("CDK Terraform Integration", () => {
566
+ it("should synthesize to valid Terraform configuration", () => {
567
+ new policy_set_definition_1.PolicySetDefinition(stack, "SynthTest", {
568
+ displayName: "Synth Test Initiative",
569
+ scope: "/subscriptions/test-sub",
570
+ policyDefinitions: [
571
+ {
572
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
573
+ policyDefinitionReferenceId: "testPolicy",
574
+ },
575
+ ],
576
+ });
577
+ const synthesized = cdktf_1.Testing.synth(stack);
578
+ expect(synthesized).toBeDefined();
579
+ const stackConfig = JSON.parse(synthesized);
580
+ expect(stackConfig.resource).toBeDefined();
581
+ });
582
+ it("should handle multiple policy sets in the same stack", () => {
583
+ const policySet1 = new policy_set_definition_1.PolicySetDefinition(stack, "PolicySet1", {
584
+ displayName: "Policy Set 1",
585
+ scope: "/subscriptions/test-sub",
586
+ policyDefinitions: [
587
+ {
588
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
589
+ },
590
+ ],
591
+ });
592
+ const policySet2 = new policy_set_definition_1.PolicySetDefinition(stack, "PolicySet2", {
593
+ displayName: "Policy Set 2",
594
+ scope: "/subscriptions/test-sub",
595
+ policyDefinitions: [
596
+ {
597
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-2",
598
+ },
599
+ ],
600
+ apiVersion: "2021-06-01",
601
+ });
602
+ expect(policySet1.resolvedApiVersion).toBe("2023-04-01");
603
+ expect(policySet2.resolvedApiVersion).toBe("2021-06-01");
604
+ const synthesized = cdktf_1.Testing.synth(stack);
605
+ expect(synthesized).toBeDefined();
606
+ });
607
+ });
608
+ // =============================================================================
609
+ // Complex Scenarios
610
+ // =============================================================================
611
+ describe("Complex Scenarios", () => {
612
+ it("should handle comprehensive initiative configuration", () => {
613
+ const props = {
614
+ name: "comprehensive-initiative",
615
+ displayName: "Comprehensive Security & Compliance Initiative",
616
+ description: "This initiative enforces security and compliance policies across the organization.",
617
+ scope: "/providers/Microsoft.Management/managementGroups/my-mg",
618
+ policyType: "Custom",
619
+ metadata: {
620
+ category: "Security Center",
621
+ version: "3.0.0",
622
+ preview: false,
623
+ deprecated: false,
624
+ },
625
+ parameters: {
626
+ allowedLocations: {
627
+ type: "Array",
628
+ displayName: "Allowed Locations",
629
+ description: "The list of allowed Azure regions",
630
+ defaultValue: ["eastus", "westus2", "centralus"],
631
+ metadata: {
632
+ strongType: "location",
633
+ displayName: "Locations",
634
+ },
635
+ },
636
+ requiredTagName: {
637
+ type: "String",
638
+ displayName: "Required Tag Name",
639
+ defaultValue: "costCenter",
640
+ },
641
+ effect: {
642
+ type: "String",
643
+ displayName: "Effect",
644
+ description: "The effect of the policy",
645
+ defaultValue: "Audit",
646
+ allowedValues: ["Audit", "Deny", "Disabled"],
647
+ },
648
+ },
649
+ policyDefinitionGroups: [
650
+ {
651
+ name: "Security",
652
+ displayName: "Security Policies",
653
+ category: "Security Center",
654
+ description: "Policies ensuring security best practices",
655
+ },
656
+ {
657
+ name: "Compliance",
658
+ displayName: "Compliance Policies",
659
+ category: "Regulatory Compliance",
660
+ description: "Policies for regulatory compliance",
661
+ },
662
+ {
663
+ name: "Tagging",
664
+ displayName: "Tagging Policies",
665
+ category: "Tags",
666
+ description: "Policies for consistent resource tagging",
667
+ },
668
+ ],
669
+ policyDefinitions: [
670
+ {
671
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/e56962a6-4747-49cd-b67b-bf8b01975c4c",
672
+ policyDefinitionReferenceId: "allowedLocations",
673
+ parameters: {
674
+ listOfAllowedLocations: {
675
+ value: "[parameters('allowedLocations')]",
676
+ },
677
+ },
678
+ groupNames: ["Security", "Compliance"],
679
+ },
680
+ {
681
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/96670d01-0a4d-4649-9c89-2d3abc0a5025",
682
+ policyDefinitionReferenceId: "requireTag",
683
+ parameters: {
684
+ tagName: { value: "[parameters('requiredTagName')]" },
685
+ },
686
+ groupNames: ["Tagging"],
687
+ },
688
+ {
689
+ policyDefinitionId: "/subscriptions/test-sub/providers/Microsoft.Authorization/policyDefinitions/custom-security-policy",
690
+ policyDefinitionReferenceId: "customSecurityPolicy",
691
+ parameters: {
692
+ effect: { value: "[parameters('effect')]" },
693
+ },
694
+ groupNames: ["Security"],
695
+ },
696
+ ],
697
+ tags: {
698
+ environment: "production",
699
+ team: "security",
700
+ managedBy: "terraform-cdk",
701
+ },
702
+ };
703
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "ComprehensiveConfig", props);
704
+ const resourceBody = policySet.createResourceBody(props);
705
+ expect(resourceBody.properties.displayName).toBe("Comprehensive Security & Compliance Initiative");
706
+ expect(resourceBody.properties.policyType).toBe("Custom");
707
+ expect(resourceBody.properties.metadata.category).toBe("Security Center");
708
+ expect(resourceBody.properties.parameters.allowedLocations.type).toBe("Array");
709
+ expect(resourceBody.properties.policyDefinitionGroups).toHaveLength(3);
710
+ expect(resourceBody.properties.policyDefinitions).toHaveLength(3);
711
+ expect(resourceBody.properties.policyDefinitions[0].groupNames).toContain("Security");
712
+ });
713
+ it("should handle management group scope", () => {
714
+ const mgScope = "/providers/Microsoft.Management/managementGroups/my-management-group";
715
+ const props = {
716
+ displayName: "Management Group Initiative",
717
+ scope: mgScope,
718
+ policyDefinitions: [
719
+ {
720
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
721
+ },
722
+ ],
723
+ };
724
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "MgScopeInitiative", props);
725
+ const parentId = policySet.resolveParentId(props);
726
+ expect(parentId).toBe(mgScope);
727
+ });
728
+ it("should handle subscription scope", () => {
729
+ const subScope = "/subscriptions/00000000-0000-0000-0000-000000000000";
730
+ const props = {
731
+ displayName: "Subscription Initiative",
732
+ scope: subScope,
733
+ policyDefinitions: [
734
+ {
735
+ policyDefinitionId: "/providers/Microsoft.Authorization/policyDefinitions/policy-1",
736
+ },
737
+ ],
738
+ };
739
+ const policySet = new policy_set_definition_1.PolicySetDefinition(stack, "SubScopeInitiative", props);
740
+ const parentId = policySet.resolveParentId(props);
741
+ expect(parentId).toBe(subScope);
742
+ });
743
+ });
744
+ });
745
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"policy-set-definition.spec.js","sourceRoot":"","sources":["../../../src/azure-policysetdefinition/test/policy-set-definition.spec.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Comprehensive tests for the PolicySetDefinition implementation\n *\n * This test suite validates the PolicySetDefinition class using the AzapiResource framework.\n * Tests cover automatic version resolution, explicit version pinning, schema validation,\n * property configurations, and resource creation.\n */\n\nimport { Testing } from \"cdktf\";\nimport * as cdktf from \"cdktf\";\nimport {\n  PolicySetDefinition,\n  PolicySetDefinitionProps,\n} from \"../lib/policy-set-definition\";\n\ndescribe(\"PolicySetDefinition - Implementation\", () => {\n  let app: cdktf.App;\n  let stack: cdktf.TerraformStack;\n\n  beforeEach(() => {\n    app = Testing.app();\n    stack = new cdktf.TerraformStack(app, \"TestStack\");\n  });\n\n  // =============================================================================\n  // Constructor and Basic Properties\n  // =============================================================================\n\n  describe(\"Constructor and Basic Properties\", () => {\n    it(\"should create a Policy Set Definition with minimal configuration\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Test Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/test-policy-id\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(stack, \"TestPolicySet\", props);\n\n      expect(policySet).toBeInstanceOf(PolicySetDefinition);\n      expect(policySet.props).toBe(props);\n      expect(policySet.displayName).toBe(\"Test Initiative\");\n      expect(policySet.policyType).toBe(\"Custom\");\n      expect(policySet.policyDefinitionCount).toBe(1);\n    });\n\n    it(\"should create a Policy Set Definition with description\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Security Initiative\",\n        description: \"Initiative for security compliance\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"SecurityPolicySet\",\n        props,\n      );\n\n      expect(policySet.props.description).toBe(\n        \"Initiative for security compliance\",\n      );\n    });\n\n    it(\"should create a Policy Set Definition with custom name\", () => {\n      const props: PolicySetDefinitionProps = {\n        name: \"my-custom-initiative\",\n        displayName: \"Custom Named Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"CustomNamedPolicySet\",\n        props,\n      );\n\n      expect(policySet.props.name).toBe(\"my-custom-initiative\");\n    });\n\n    it(\"should create a Policy Set Definition with metadata\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Metadata Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        metadata: {\n          category: \"Security Center\",\n          version: \"1.0.0\",\n          preview: false,\n          deprecated: false,\n        },\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"MetadataPolicySet\",\n        props,\n      );\n\n      expect(policySet.props.metadata?.category).toBe(\"Security Center\");\n      expect(policySet.props.metadata?.version).toBe(\"1.0.0\");\n      expect(policySet.props.metadata?.preview).toBe(false);\n    });\n  });\n\n  // =============================================================================\n  // Version Resolution\n  // =============================================================================\n\n  describe(\"Version Resolution\", () => {\n    it(\"should use latest version 2023-04-01 when no version specified\", () => {\n      const policySet = new PolicySetDefinition(stack, \"DefaultVersion\", {\n        displayName: \"Default Version Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      });\n\n      expect(policySet.resolvedApiVersion).toBe(\"2023-04-01\");\n    });\n\n    it(\"should use explicit version when specified\", () => {\n      const policySet = new PolicySetDefinition(stack, \"PinnedVersion\", {\n        displayName: \"Pinned Version Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n        apiVersion: \"2021-06-01\",\n      });\n\n      expect(policySet.resolvedApiVersion).toBe(\"2021-06-01\");\n    });\n\n    it(\"should use latest version 2023-04-01 when explicitly specified\", () => {\n      const policySet = new PolicySetDefinition(stack, \"LatestVersion\", {\n        displayName: \"Latest Version Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n        apiVersion: \"2023-04-01\",\n      });\n\n      expect(policySet.resolvedApiVersion).toBe(\"2023-04-01\");\n    });\n  });\n\n  // =============================================================================\n  // Validation Tests\n  // =============================================================================\n\n  describe(\"Validation\", () => {\n    it(\"should throw error when displayName is empty\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      expect(() => {\n        new PolicySetDefinition(stack, \"EmptyDisplayName\", props);\n      }).toThrow(/displayName is required/);\n    });\n\n    it(\"should throw error when policyDefinitions is empty\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"No Policies Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [],\n      };\n\n      expect(() => {\n        new PolicySetDefinition(stack, \"NoPolicies\", props);\n      }).toThrow(/At least one policy definition reference is required/);\n    });\n\n    it(\"should throw error when policyDefinitionId is missing\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Invalid Policy Reference\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId: \"\",\n          },\n        ],\n      };\n\n      expect(() => {\n        new PolicySetDefinition(stack, \"InvalidPolicyRef\", props);\n      }).toThrow(/policyDefinitionId is required/);\n    });\n\n    it(\"should throw error for duplicate group names\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Duplicate Groups\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitionGroups: [\n          { name: \"Security\", displayName: \"Security Policies\" },\n          { name: \"Security\", displayName: \"Duplicate Security\" },\n        ],\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      expect(() => {\n        new PolicySetDefinition(stack, \"DuplicateGroups\", props);\n      }).toThrow(/Duplicate group name 'Security'/);\n    });\n\n    it(\"should throw error for invalid group reference in policy definition\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Invalid Group Reference\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitionGroups: [\n          { name: \"Security\", displayName: \"Security Policies\" },\n        ],\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            groupNames: [\"NonExistentGroup\"],\n          },\n        ],\n      };\n\n      expect(() => {\n        new PolicySetDefinition(stack, \"InvalidGroupRef\", props);\n      }).toThrow(/references unknown group 'NonExistentGroup'/);\n    });\n\n    it(\"should accept valid policy set with groups\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Valid Groups Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitionGroups: [\n          { name: \"Security\", displayName: \"Security Policies\" },\n          { name: \"Compliance\", displayName: \"Compliance Policies\" },\n        ],\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            groupNames: [\"Security\"],\n          },\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-2\",\n            groupNames: [\"Security\", \"Compliance\"],\n          },\n        ],\n      };\n\n      expect(() => {\n        new PolicySetDefinition(stack, \"ValidGroups\", props);\n      }).not.toThrow();\n    });\n  });\n\n  // =============================================================================\n  // Policy Definition References\n  // =============================================================================\n\n  describe(\"Policy Definition References\", () => {\n    it(\"should create initiative with single policy definition\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Single Policy Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            policyDefinitionReferenceId: \"allowedLocations\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(stack, \"SinglePolicy\", props);\n      expect(policySet.policyDefinitionCount).toBe(1);\n    });\n\n    it(\"should create initiative with multiple policy definitions\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Multiple Policy Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            policyDefinitionReferenceId: \"policy1\",\n          },\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-2\",\n            policyDefinitionReferenceId: \"policy2\",\n          },\n          {\n            policyDefinitionId:\n              \"/subscriptions/sub-id/providers/Microsoft.Authorization/policyDefinitions/custom-policy\",\n            policyDefinitionReferenceId: \"customPolicy\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"MultiplePolicies\",\n        props,\n      );\n      expect(policySet.policyDefinitionCount).toBe(3);\n    });\n\n    it(\"should create initiative with policy parameters\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Parameterized Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            policyDefinitionReferenceId: \"tagPolicy\",\n            parameters: {\n              tagName: { value: \"environment\" },\n              tagValue: { value: \"production\" },\n            },\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"ParameterizedPolicy\",\n        props,\n      );\n      expect(\n        policySet.props.policyDefinitions[0].parameters?.tagName.value,\n      ).toBe(\"environment\");\n    });\n  });\n\n  // =============================================================================\n  // Initiative Parameters\n  // =============================================================================\n\n  describe(\"Initiative Parameters\", () => {\n    it(\"should create initiative with parameter definitions\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Initiative With Parameters\",\n        scope: \"/subscriptions/test-sub\",\n        parameters: {\n          allowedLocations: {\n            type: \"Array\",\n            displayName: \"Allowed Locations\",\n            description: \"List of allowed Azure regions\",\n            defaultValue: [\"eastus\", \"westus2\"],\n            allowedValues: [[\"eastus\"], [\"westus2\"], [\"eastus\", \"westus2\"]],\n          },\n          requiredTag: {\n            type: \"String\",\n            displayName: \"Required Tag\",\n            description: \"Name of the required tag\",\n            defaultValue: \"environment\",\n          },\n        },\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            parameters: {\n              listOfAllowedLocations: {\n                value: \"[parameters('allowedLocations')]\",\n              },\n            },\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(stack, \"WithParams\", props);\n      expect(policySet.props.parameters?.allowedLocations.type).toBe(\"Array\");\n      expect(policySet.props.parameters?.requiredTag.type).toBe(\"String\");\n    });\n\n    it(\"should create initiative with strong typed parameter\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Strong Typed Parameters\",\n        scope: \"/subscriptions/test-sub\",\n        parameters: {\n          storageAccountSku: {\n            type: \"String\",\n            displayName: \"Storage Account SKU\",\n            metadata: {\n              strongType: \"storageSkus\",\n              displayName: \"Storage SKU\",\n            },\n          },\n        },\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"StrongTypedParams\",\n        props,\n      );\n      expect(\n        policySet.props.parameters?.storageAccountSku.metadata?.strongType,\n      ).toBe(\"storageSkus\");\n    });\n  });\n\n  // =============================================================================\n  // Policy Definition Groups\n  // =============================================================================\n\n  describe(\"Policy Definition Groups\", () => {\n    it(\"should create initiative with groups\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Grouped Policies Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitionGroups: [\n          {\n            name: \"Security\",\n            displayName: \"Security Policies\",\n            category: \"Security Center\",\n            description: \"Policies related to security\",\n          },\n          {\n            name: \"Compliance\",\n            displayName: \"Compliance Policies\",\n            category: \"Regulatory Compliance\",\n          },\n        ],\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            groupNames: [\"Security\"],\n          },\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-2\",\n            groupNames: [\"Compliance\"],\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"GroupedPolicies\",\n        props,\n      );\n      expect(policySet.props.policyDefinitionGroups?.length).toBe(2);\n      expect(policySet.props.policyDefinitionGroups?.[0].name).toBe(\"Security\");\n    });\n\n    it(\"should allow policy definitions in multiple groups\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Multi-Group Policy Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitionGroups: [\n          { name: \"Security\" },\n          { name: \"Compliance\" },\n          { name: \"Operations\" },\n        ],\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            groupNames: [\"Security\", \"Compliance\"],\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"MultiGroupPolicy\",\n        props,\n      );\n      expect(policySet.props.policyDefinitions[0].groupNames?.length).toBe(2);\n    });\n  });\n\n  // =============================================================================\n  // Property Transformation\n  // =============================================================================\n\n  describe(\"Property Transformation\", () => {\n    it(\"should generate correct Azure API format via createResourceBody\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Transform Test Initiative\",\n        description: \"Test transformation\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            policyDefinitionReferenceId: \"policy1\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(stack, \"TransformTest\", props);\n\n      // Access the protected createResourceBody method\n      const resourceBody = (policySet as any).createResourceBody(props);\n\n      expect(resourceBody).toHaveProperty(\"properties\");\n      expect(resourceBody.properties).toHaveProperty(\n        \"displayName\",\n        \"Transform Test Initiative\",\n      );\n      expect(resourceBody.properties).toHaveProperty(\n        \"description\",\n        \"Test transformation\",\n      );\n      expect(resourceBody.properties).toHaveProperty(\"policyType\", \"Custom\");\n      expect(resourceBody.properties.policyDefinitions).toHaveLength(1);\n    });\n\n    it(\"should include metadata when specified\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Metadata Test\",\n        scope: \"/subscriptions/test-sub\",\n        metadata: {\n          category: \"Testing\",\n          version: \"2.0.0\",\n        },\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(stack, \"MetadataTest\", props);\n      const resourceBody = (policySet as any).createResourceBody(props);\n\n      expect(resourceBody.properties.metadata).toBeDefined();\n      expect(resourceBody.properties.metadata.category).toBe(\"Testing\");\n    });\n\n    it(\"should include parameters when specified\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Parameters Test\",\n        scope: \"/subscriptions/test-sub\",\n        parameters: {\n          testParam: {\n            type: \"String\",\n            defaultValue: \"test\",\n          },\n        },\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(stack, \"ParametersTest\", props);\n      const resourceBody = (policySet as any).createResourceBody(props);\n\n      expect(resourceBody.properties.parameters).toBeDefined();\n      expect(resourceBody.properties.parameters.testParam.type).toBe(\"String\");\n    });\n\n    it(\"should include policyDefinitionGroups when specified\", () => {\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Groups Test\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitionGroups: [\n          { name: \"TestGroup\", displayName: \"Test Group\" },\n        ],\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            groupNames: [\"TestGroup\"],\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(stack, \"GroupsTest\", props);\n      const resourceBody = (policySet as any).createResourceBody(props);\n\n      expect(resourceBody.properties.policyDefinitionGroups).toBeDefined();\n      expect(resourceBody.properties.policyDefinitionGroups).toHaveLength(1);\n    });\n  });\n\n  // =============================================================================\n  // Integration with Base Class\n  // =============================================================================\n\n  describe(\"Integration with Base Class\", () => {\n    it(\"should inherit from AzapiResource\", () => {\n      const policySet = new PolicySetDefinition(stack, \"InheritanceTest\", {\n        displayName: \"Inheritance Test\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      });\n\n      // Verify it has AzapiResource properties\n      expect(policySet).toHaveProperty(\"terraformResource\");\n      expect(policySet).toHaveProperty(\"resolvedApiVersion\");\n    });\n\n    it(\"should have correct resourceType\", () => {\n      const policySet = new PolicySetDefinition(stack, \"ResourceTypeTest\", {\n        displayName: \"Resource Type Test\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      });\n\n      // Access the protected resourceType method\n      const resourceType = (policySet as any).resourceType();\n      expect(resourceType).toBe(\"Microsoft.Authorization/policySetDefinitions\");\n    });\n\n    it(\"should resolve parent ID correctly\", () => {\n      const scope = \"/subscriptions/test-sub\";\n\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Parent ID Test\",\n        scope,\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(stack, \"ParentIdTest\", props);\n\n      // Access the protected resolveParentId method\n      const parentId = (policySet as any).resolveParentId(props);\n      expect(parentId).toBe(scope);\n    });\n\n    it(\"should generate resource outputs\", () => {\n      const policySet = new PolicySetDefinition(stack, \"OutputsTest\", {\n        displayName: \"Outputs Test\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      });\n\n      expect(policySet.idOutput).toBeInstanceOf(cdktf.TerraformOutput);\n      expect(policySet.nameOutput).toBeInstanceOf(cdktf.TerraformOutput);\n      expect(policySet.policySetDefinitionIdOutput).toBeInstanceOf(\n        cdktf.TerraformOutput,\n      );\n      expect(policySet.id).toMatch(/^\\$\\{.*\\.id\\}$/);\n    });\n  });\n\n  // =============================================================================\n  // CDK Terraform Integration\n  // =============================================================================\n\n  describe(\"CDK Terraform Integration\", () => {\n    it(\"should synthesize to valid Terraform configuration\", () => {\n      new PolicySetDefinition(stack, \"SynthTest\", {\n        displayName: \"Synth Test Initiative\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n            policyDefinitionReferenceId: \"testPolicy\",\n          },\n        ],\n      });\n\n      const synthesized = Testing.synth(stack);\n      expect(synthesized).toBeDefined();\n\n      const stackConfig = JSON.parse(synthesized);\n      expect(stackConfig.resource).toBeDefined();\n    });\n\n    it(\"should handle multiple policy sets in the same stack\", () => {\n      const policySet1 = new PolicySetDefinition(stack, \"PolicySet1\", {\n        displayName: \"Policy Set 1\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      });\n\n      const policySet2 = new PolicySetDefinition(stack, \"PolicySet2\", {\n        displayName: \"Policy Set 2\",\n        scope: \"/subscriptions/test-sub\",\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-2\",\n          },\n        ],\n        apiVersion: \"2021-06-01\",\n      });\n\n      expect(policySet1.resolvedApiVersion).toBe(\"2023-04-01\");\n      expect(policySet2.resolvedApiVersion).toBe(\"2021-06-01\");\n\n      const synthesized = Testing.synth(stack);\n      expect(synthesized).toBeDefined();\n    });\n  });\n\n  // =============================================================================\n  // Complex Scenarios\n  // =============================================================================\n\n  describe(\"Complex Scenarios\", () => {\n    it(\"should handle comprehensive initiative configuration\", () => {\n      const props: PolicySetDefinitionProps = {\n        name: \"comprehensive-initiative\",\n        displayName: \"Comprehensive Security & Compliance Initiative\",\n        description:\n          \"This initiative enforces security and compliance policies across the organization.\",\n        scope: \"/providers/Microsoft.Management/managementGroups/my-mg\",\n        policyType: \"Custom\",\n        metadata: {\n          category: \"Security Center\",\n          version: \"3.0.0\",\n          preview: false,\n          deprecated: false,\n        },\n        parameters: {\n          allowedLocations: {\n            type: \"Array\",\n            displayName: \"Allowed Locations\",\n            description: \"The list of allowed Azure regions\",\n            defaultValue: [\"eastus\", \"westus2\", \"centralus\"],\n            metadata: {\n              strongType: \"location\",\n              displayName: \"Locations\",\n            },\n          },\n          requiredTagName: {\n            type: \"String\",\n            displayName: \"Required Tag Name\",\n            defaultValue: \"costCenter\",\n          },\n          effect: {\n            type: \"String\",\n            displayName: \"Effect\",\n            description: \"The effect of the policy\",\n            defaultValue: \"Audit\",\n            allowedValues: [\"Audit\", \"Deny\", \"Disabled\"],\n          },\n        },\n        policyDefinitionGroups: [\n          {\n            name: \"Security\",\n            displayName: \"Security Policies\",\n            category: \"Security Center\",\n            description: \"Policies ensuring security best practices\",\n          },\n          {\n            name: \"Compliance\",\n            displayName: \"Compliance Policies\",\n            category: \"Regulatory Compliance\",\n            description: \"Policies for regulatory compliance\",\n          },\n          {\n            name: \"Tagging\",\n            displayName: \"Tagging Policies\",\n            category: \"Tags\",\n            description: \"Policies for consistent resource tagging\",\n          },\n        ],\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/e56962a6-4747-49cd-b67b-bf8b01975c4c\",\n            policyDefinitionReferenceId: \"allowedLocations\",\n            parameters: {\n              listOfAllowedLocations: {\n                value: \"[parameters('allowedLocations')]\",\n              },\n            },\n            groupNames: [\"Security\", \"Compliance\"],\n          },\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/96670d01-0a4d-4649-9c89-2d3abc0a5025\",\n            policyDefinitionReferenceId: \"requireTag\",\n            parameters: {\n              tagName: { value: \"[parameters('requiredTagName')]\" },\n            },\n            groupNames: [\"Tagging\"],\n          },\n          {\n            policyDefinitionId:\n              \"/subscriptions/test-sub/providers/Microsoft.Authorization/policyDefinitions/custom-security-policy\",\n            policyDefinitionReferenceId: \"customSecurityPolicy\",\n            parameters: {\n              effect: { value: \"[parameters('effect')]\" },\n            },\n            groupNames: [\"Security\"],\n          },\n        ],\n        tags: {\n          environment: \"production\",\n          team: \"security\",\n          managedBy: \"terraform-cdk\",\n        },\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"ComprehensiveConfig\",\n        props,\n      );\n      const resourceBody = (policySet as any).createResourceBody(props);\n\n      expect(resourceBody.properties.displayName).toBe(\n        \"Comprehensive Security & Compliance Initiative\",\n      );\n      expect(resourceBody.properties.policyType).toBe(\"Custom\");\n      expect(resourceBody.properties.metadata.category).toBe(\"Security Center\");\n      expect(resourceBody.properties.parameters.allowedLocations.type).toBe(\n        \"Array\",\n      );\n      expect(resourceBody.properties.policyDefinitionGroups).toHaveLength(3);\n      expect(resourceBody.properties.policyDefinitions).toHaveLength(3);\n      expect(resourceBody.properties.policyDefinitions[0].groupNames).toContain(\n        \"Security\",\n      );\n    });\n\n    it(\"should handle management group scope\", () => {\n      const mgScope =\n        \"/providers/Microsoft.Management/managementGroups/my-management-group\";\n\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Management Group Initiative\",\n        scope: mgScope,\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"MgScopeInitiative\",\n        props,\n      );\n      const parentId = (policySet as any).resolveParentId(props);\n\n      expect(parentId).toBe(mgScope);\n    });\n\n    it(\"should handle subscription scope\", () => {\n      const subScope = \"/subscriptions/00000000-0000-0000-0000-000000000000\";\n\n      const props: PolicySetDefinitionProps = {\n        displayName: \"Subscription Initiative\",\n        scope: subScope,\n        policyDefinitions: [\n          {\n            policyDefinitionId:\n              \"/providers/Microsoft.Authorization/policyDefinitions/policy-1\",\n          },\n        ],\n      };\n\n      const policySet = new PolicySetDefinition(\n        stack,\n        \"SubScopeInitiative\",\n        props,\n      );\n      const parentId = (policySet as any).resolveParentId(props);\n\n      expect(parentId).toBe(subScope);\n    });\n  });\n});\n"]}