@microsoft/terraform-cdk-constructs 1.2.0 → 1.3.1
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/.jsii +54630 -26185
- package/API.md +72091 -23784
- package/lib/azure-actiongroup/index.d.ts +0 -10
- package/lib/azure-actiongroup/index.js +1 -11
- package/lib/azure-actiongroup/lib/action-group.d.ts +0 -10
- package/lib/azure-actiongroup/lib/action-group.js +6 -32
- package/lib/azure-actiongroup/lib/index.d.ts +1 -4
- package/lib/azure-actiongroup/lib/index.js +2 -5
- package/lib/azure-activitylogalert/index.d.ts +0 -10
- package/lib/azure-activitylogalert/index.js +1 -11
- package/lib/azure-activitylogalert/lib/activity-log-alert.d.ts +0 -10
- package/lib/azure-activitylogalert/lib/activity-log-alert.js +6 -32
- package/lib/azure-activitylogalert/lib/index.d.ts +1 -4
- package/lib/azure-activitylogalert/lib/index.js +2 -5
- package/lib/azure-aks/index.d.ts +0 -8
- package/lib/azure-aks/index.js +1 -11
- package/lib/azure-aks/lib/aks-cluster.d.ts +4 -13
- package/lib/azure-aks/lib/aks-cluster.js +13 -36
- package/lib/azure-aks/lib/index.d.ts +1 -10
- package/lib/azure-aks/lib/index.js +2 -21
- package/lib/azure-diagnosticsettings/index.d.ts +0 -6
- package/lib/azure-diagnosticsettings/index.js +1 -7
- package/lib/azure-diagnosticsettings/lib/diagnostic-settings.d.ts +0 -10
- package/lib/azure-diagnosticsettings/lib/diagnostic-settings.js +6 -32
- package/lib/azure-diagnosticsettings/lib/index.d.ts +1 -7
- package/lib/azure-diagnosticsettings/lib/index.js +2 -8
- package/lib/azure-dnsforwardingruleset/index.d.ts +1 -0
- package/lib/azure-dnsforwardingruleset/index.js +18 -0
- package/lib/azure-dnsforwardingruleset/lib/dns-forwarding-ruleset-schemas.d.ts +24 -0
- package/lib/azure-dnsforwardingruleset/lib/dns-forwarding-ruleset-schemas.js +206 -0
- package/lib/azure-dnsforwardingruleset/lib/dns-forwarding-ruleset.d.ts +174 -0
- package/lib/azure-dnsforwardingruleset/lib/dns-forwarding-ruleset.js +214 -0
- package/lib/azure-dnsforwardingruleset/lib/forwarding-rule-schemas.d.ts +24 -0
- package/lib/azure-dnsforwardingruleset/lib/forwarding-rule-schemas.js +242 -0
- package/lib/azure-dnsforwardingruleset/lib/forwarding-rule.d.ts +193 -0
- package/lib/azure-dnsforwardingruleset/lib/forwarding-rule.js +193 -0
- package/lib/azure-dnsforwardingruleset/lib/index.d.ts +6 -0
- package/lib/azure-dnsforwardingruleset/lib/index.js +23 -0
- package/lib/azure-dnsforwardingruleset/lib/virtual-network-link-schemas.d.ts +24 -0
- package/lib/azure-dnsforwardingruleset/lib/virtual-network-link-schemas.js +199 -0
- package/lib/azure-dnsforwardingruleset/lib/virtual-network-link.d.ts +160 -0
- package/lib/azure-dnsforwardingruleset/lib/virtual-network-link.js +178 -0
- package/lib/azure-dnsforwardingruleset/test/dns-forwarding-ruleset.integ.d.ts +9 -0
- package/lib/azure-dnsforwardingruleset/test/dns-forwarding-ruleset.integ.js +133 -0
- package/lib/azure-dnsforwardingruleset/test/dns-forwarding-ruleset.spec.d.ts +9 -0
- package/lib/azure-dnsforwardingruleset/test/dns-forwarding-ruleset.spec.js +350 -0
- package/lib/azure-dnsforwardingruleset/test/forwarding-rule.spec.d.ts +9 -0
- package/lib/azure-dnsforwardingruleset/test/forwarding-rule.spec.js +397 -0
- package/lib/azure-dnsforwardingruleset/test/virtual-network-link.spec.d.ts +9 -0
- package/lib/azure-dnsforwardingruleset/test/virtual-network-link.spec.js +285 -0
- package/lib/azure-dnsresolver/index.d.ts +1 -0
- package/lib/azure-dnsresolver/index.js +18 -0
- package/lib/azure-dnsresolver/lib/dns-resolver-schemas.d.ts +24 -0
- package/lib/azure-dnsresolver/lib/dns-resolver-schemas.js +218 -0
- package/lib/azure-dnsresolver/lib/dns-resolver.d.ts +170 -0
- package/lib/azure-dnsresolver/lib/dns-resolver.js +236 -0
- package/lib/azure-dnsresolver/lib/inbound-endpoint-schemas.d.ts +24 -0
- package/lib/azure-dnsresolver/lib/inbound-endpoint-schemas.js +261 -0
- package/lib/azure-dnsresolver/lib/inbound-endpoint.d.ts +189 -0
- package/lib/azure-dnsresolver/lib/inbound-endpoint.js +243 -0
- package/lib/azure-dnsresolver/lib/index.d.ts +6 -0
- package/lib/azure-dnsresolver/lib/index.js +23 -0
- package/lib/azure-dnsresolver/lib/outbound-endpoint-schemas.d.ts +24 -0
- package/lib/azure-dnsresolver/lib/outbound-endpoint-schemas.js +231 -0
- package/lib/azure-dnsresolver/lib/outbound-endpoint.d.ts +175 -0
- package/lib/azure-dnsresolver/lib/outbound-endpoint.js +234 -0
- package/lib/azure-dnsresolver/test/dns-resolver.integ.d.ts +14 -0
- package/lib/azure-dnsresolver/test/dns-resolver.integ.js +117 -0
- package/lib/azure-dnsresolver/test/dns-resolver.spec.d.ts +9 -0
- package/lib/azure-dnsresolver/test/dns-resolver.spec.js +353 -0
- package/lib/azure-dnsresolver/test/inbound-endpoint.integ.d.ts +9 -0
- package/lib/azure-dnsresolver/test/inbound-endpoint.integ.js +151 -0
- package/lib/azure-dnsresolver/test/inbound-endpoint.spec.d.ts +9 -0
- package/lib/azure-dnsresolver/test/inbound-endpoint.spec.js +441 -0
- package/lib/azure-dnsresolver/test/outbound-endpoint.integ.d.ts +9 -0
- package/lib/azure-dnsresolver/test/outbound-endpoint.integ.js +149 -0
- package/lib/azure-dnsresolver/test/outbound-endpoint.spec.d.ts +9 -0
- package/lib/azure-dnsresolver/test/outbound-endpoint.spec.js +301 -0
- package/lib/azure-dnszone/index.d.ts +1 -0
- package/lib/azure-dnszone/index.js +18 -0
- package/lib/azure-dnszone/lib/dns-zone-schemas.d.ts +24 -0
- package/lib/azure-dnszone/lib/dns-zone-schemas.js +191 -0
- package/lib/azure-dnszone/lib/dns-zone.d.ts +182 -0
- package/lib/azure-dnszone/lib/dns-zone.js +228 -0
- package/lib/azure-dnszone/lib/index.d.ts +2 -0
- package/lib/azure-dnszone/lib/index.js +19 -0
- package/lib/azure-dnszone/test/dns-zone.integ.d.ts +9 -0
- package/lib/azure-dnszone/test/dns-zone.integ.js +85 -0
- package/lib/azure-dnszone/test/dns-zone.spec.d.ts +9 -0
- package/lib/azure-dnszone/test/dns-zone.spec.js +285 -0
- package/lib/azure-metricalert/index.d.ts +0 -10
- package/lib/azure-metricalert/index.js +1 -11
- package/lib/azure-metricalert/lib/index.d.ts +1 -4
- package/lib/azure-metricalert/lib/index.js +2 -5
- package/lib/azure-metricalert/lib/metric-alert.d.ts +0 -10
- package/lib/azure-metricalert/lib/metric-alert.js +6 -32
- package/lib/azure-networkinterface/index.d.ts +0 -3
- package/lib/azure-networkinterface/index.js +1 -4
- package/lib/azure-networkinterface/lib/index.d.ts +1 -5
- package/lib/azure-networkinterface/lib/index.js +2 -5
- package/lib/azure-networkinterface/lib/network-interface.d.ts +4 -9
- package/lib/azure-networkinterface/lib/network-interface.js +14 -29
- package/lib/azure-networkinterface/test/network-interface.integ.js +20 -6
- package/lib/azure-networksecuritygroup/index.d.ts +0 -5
- package/lib/azure-networksecuritygroup/index.js +1 -6
- package/lib/azure-networksecuritygroup/lib/index.d.ts +2 -5
- package/lib/azure-networksecuritygroup/lib/index.js +17 -14
- package/lib/azure-networksecuritygroup/lib/network-security-group.d.ts +4 -13
- package/lib/azure-networksecuritygroup/lib/network-security-group.js +14 -36
- package/lib/azure-policyassignment/index.d.ts +1 -0
- package/lib/azure-policyassignment/index.js +18 -0
- package/lib/azure-policyassignment/lib/index.d.ts +2 -0
- package/lib/azure-policyassignment/lib/index.js +19 -0
- package/lib/azure-policyassignment/lib/policy-assignment-schemas.d.ts +25 -0
- package/lib/azure-policyassignment/lib/policy-assignment-schemas.js +260 -0
- package/lib/azure-policyassignment/lib/policy-assignment.d.ts +349 -0
- package/lib/azure-policyassignment/lib/policy-assignment.js +237 -0
- package/lib/azure-policyassignment/test/policy-assignment.integ.d.ts +13 -0
- package/lib/azure-policyassignment/test/policy-assignment.integ.js +153 -0
- package/lib/azure-policyassignment/test/policy-assignment.spec.d.ts +9 -0
- package/lib/azure-policyassignment/test/policy-assignment.spec.js +651 -0
- package/lib/azure-policydefinition/index.d.ts +1 -0
- package/lib/azure-policydefinition/index.js +18 -0
- package/lib/azure-policydefinition/lib/index.d.ts +2 -0
- package/lib/azure-policydefinition/lib/index.js +19 -0
- package/lib/azure-policydefinition/lib/policy-definition-schemas.d.ts +25 -0
- package/lib/azure-policydefinition/lib/policy-definition-schemas.js +210 -0
- package/lib/azure-policydefinition/lib/policy-definition.d.ts +281 -0
- package/lib/azure-policydefinition/lib/policy-definition.js +236 -0
- package/lib/azure-policydefinition/test/policy-definition.integ.d.ts +9 -0
- package/lib/azure-policydefinition/test/policy-definition.integ.js +137 -0
- package/lib/azure-policydefinition/test/policy-definition.spec.d.ts +9 -0
- package/lib/azure-policydefinition/test/policy-definition.spec.js +806 -0
- package/lib/azure-privatednszone/index.d.ts +1 -0
- package/lib/azure-privatednszone/index.js +18 -0
- package/lib/azure-privatednszone/lib/index.d.ts +2 -0
- package/lib/azure-privatednszone/lib/index.js +19 -0
- package/lib/azure-privatednszone/lib/private-dns-zone-schemas.d.ts +24 -0
- package/lib/azure-privatednszone/lib/private-dns-zone-schemas.js +254 -0
- package/lib/azure-privatednszone/lib/private-dns-zone.d.ts +178 -0
- package/lib/azure-privatednszone/lib/private-dns-zone.js +272 -0
- package/lib/azure-privatednszone/test/private-dns-zone.integ.d.ts +9 -0
- package/lib/azure-privatednszone/test/private-dns-zone.integ.js +84 -0
- package/lib/azure-privatednszone/test/private-dns-zone.spec.d.ts +9 -0
- package/lib/azure-privatednszone/test/private-dns-zone.spec.js +341 -0
- package/lib/azure-privatednszonelink/index.d.ts +1 -0
- package/lib/azure-privatednszonelink/index.js +18 -0
- package/lib/azure-privatednszonelink/lib/index.d.ts +2 -0
- package/lib/azure-privatednszonelink/lib/index.js +19 -0
- package/lib/azure-privatednszonelink/lib/private-dns-zone-link-schemas.d.ts +24 -0
- package/lib/azure-privatednszonelink/lib/private-dns-zone-link-schemas.js +262 -0
- package/lib/azure-privatednszonelink/lib/private-dns-zone-link.d.ts +202 -0
- package/lib/azure-privatednszonelink/lib/private-dns-zone-link.js +250 -0
- package/lib/azure-privatednszonelink/test/private-dns-zone-link.integ.d.ts +9 -0
- package/lib/azure-privatednszonelink/test/private-dns-zone-link.integ.js +110 -0
- package/lib/azure-privatednszonelink/test/private-dns-zone-link.spec.d.ts +9 -0
- package/lib/azure-privatednszonelink/test/private-dns-zone-link.spec.js +465 -0
- package/lib/azure-publicipaddress/index.d.ts +0 -5
- package/lib/azure-publicipaddress/index.js +1 -6
- package/lib/azure-publicipaddress/lib/index.d.ts +2 -9
- package/lib/azure-publicipaddress/lib/index.js +17 -17
- package/lib/azure-publicipaddress/lib/public-ip-address.d.ts +4 -13
- package/lib/azure-publicipaddress/lib/public-ip-address.js +14 -36
- package/lib/azure-resourcegroup/index.d.ts +0 -37
- package/lib/azure-resourcegroup/index.js +1 -39
- package/lib/azure-resourcegroup/lib/index.d.ts +1 -44
- package/lib/azure-resourcegroup/lib/index.js +2 -43
- package/lib/azure-resourcegroup/lib/resource-group.d.ts +9 -23
- package/lib/azure-resourcegroup/lib/resource-group.js +23 -56
- package/lib/azure-resourcegroup/test/resource-group.spec.js +13 -19
- package/lib/azure-roleassignment/index.d.ts +1 -0
- package/lib/azure-roleassignment/index.js +18 -0
- package/lib/azure-roleassignment/lib/index.d.ts +2 -0
- package/lib/azure-roleassignment/lib/index.js +19 -0
- package/lib/azure-roleassignment/lib/role-assignment-schemas.d.ts +25 -0
- package/lib/azure-roleassignment/lib/role-assignment-schemas.js +238 -0
- package/lib/azure-roleassignment/lib/role-assignment.d.ts +294 -0
- package/lib/azure-roleassignment/lib/role-assignment.js +257 -0
- package/lib/azure-roleassignment/test/role-assignment.integ.d.ts +12 -0
- package/lib/azure-roleassignment/test/role-assignment.integ.js +101 -0
- package/lib/azure-roleassignment/test/role-assignment.spec.d.ts +9 -0
- package/lib/azure-roleassignment/test/role-assignment.spec.js +633 -0
- package/lib/azure-roledefinition/index.d.ts +1 -0
- package/lib/azure-roledefinition/index.js +18 -0
- package/lib/azure-roledefinition/lib/index.d.ts +2 -0
- package/lib/azure-roledefinition/lib/index.js +19 -0
- package/lib/azure-roledefinition/lib/role-definition-schemas.d.ts +25 -0
- package/lib/azure-roledefinition/lib/role-definition-schemas.js +195 -0
- package/lib/azure-roledefinition/lib/role-definition.d.ts +236 -0
- package/lib/azure-roledefinition/lib/role-definition.js +192 -0
- package/lib/azure-roledefinition/test/role-definition.integ.d.ts +12 -0
- package/lib/azure-roledefinition/test/role-definition.integ.js +142 -0
- package/lib/azure-roledefinition/test/role-definition.spec.d.ts +9 -0
- package/lib/azure-roledefinition/test/role-definition.spec.js +946 -0
- package/lib/azure-storageaccount/index.d.ts +0 -19
- package/lib/azure-storageaccount/index.js +1 -20
- package/lib/azure-storageaccount/lib/index.d.ts +1 -7
- package/lib/azure-storageaccount/lib/index.js +2 -8
- package/lib/azure-storageaccount/lib/storage-account.d.ts +4 -9
- package/lib/azure-storageaccount/lib/storage-account.js +15 -29
- package/lib/azure-subnet/index.d.ts +0 -3
- package/lib/azure-subnet/index.js +1 -4
- package/lib/azure-subnet/lib/index.d.ts +1 -4
- package/lib/azure-subnet/lib/index.js +2 -5
- package/lib/azure-subnet/lib/subnet.d.ts +3 -14
- package/lib/azure-subnet/lib/subnet.js +28 -47
- package/lib/azure-subnet/test/subnet.integ.js +19 -7
- package/lib/azure-subnet/test/subnet.spec.js +1 -2
- package/lib/azure-virtualmachine/index.d.ts +0 -6
- package/lib/azure-virtualmachine/index.js +1 -7
- package/lib/azure-virtualmachine/lib/index.d.ts +1 -10
- package/lib/azure-virtualmachine/lib/index.js +2 -21
- package/lib/azure-virtualmachine/lib/virtual-machine.d.ts +4 -13
- package/lib/azure-virtualmachine/lib/virtual-machine.js +15 -36
- package/lib/azure-virtualnetwork/index.d.ts +0 -9
- package/lib/azure-virtualnetwork/index.js +1 -11
- package/lib/azure-virtualnetwork/lib/index.d.ts +2 -9
- package/lib/azure-virtualnetwork/lib/index.js +17 -20
- package/lib/azure-virtualnetwork/lib/virtual-network.d.ts +4 -13
- package/lib/azure-virtualnetwork/lib/virtual-network.js +15 -36
- package/lib/azure-virtualnetworkmanager/index.d.ts +1 -0
- package/lib/azure-virtualnetworkmanager/index.js +18 -0
- package/lib/azure-virtualnetworkmanager/lib/connectivity-configuration-schemas.d.ts +48 -0
- package/lib/azure-virtualnetworkmanager/lib/connectivity-configuration-schemas.js +265 -0
- package/lib/azure-virtualnetworkmanager/lib/connectivity-configuration.d.ts +185 -0
- package/lib/azure-virtualnetworkmanager/lib/connectivity-configuration.js +206 -0
- package/lib/azure-virtualnetworkmanager/lib/index.d.ts +14 -0
- package/lib/azure-virtualnetworkmanager/lib/index.js +31 -0
- package/lib/azure-virtualnetworkmanager/lib/network-group-schemas.d.ts +32 -0
- package/lib/azure-virtualnetworkmanager/lib/network-group-schemas.js +189 -0
- package/lib/azure-virtualnetworkmanager/lib/network-group-static-member-schemas.d.ts +32 -0
- package/lib/azure-virtualnetworkmanager/lib/network-group-static-member-schemas.js +201 -0
- package/lib/azure-virtualnetworkmanager/lib/network-group-static-member.d.ts +135 -0
- package/lib/azure-virtualnetworkmanager/lib/network-group-static-member.js +163 -0
- package/lib/azure-virtualnetworkmanager/lib/network-group.d.ts +139 -0
- package/lib/azure-virtualnetworkmanager/lib/network-group.js +158 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-configuration-schemas.d.ts +32 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-configuration-schemas.js +182 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-configuration.d.ts +144 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-configuration.js +164 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-rule-collection-schemas.d.ts +38 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-rule-collection-schemas.js +206 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-rule-collection.d.ts +142 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-rule-collection.js +162 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-rule-schemas.d.ts +39 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-rule-schemas.js +359 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-rule.d.ts +221 -0
- package/lib/azure-virtualnetworkmanager/lib/security-admin-rule.js +204 -0
- package/lib/azure-virtualnetworkmanager/lib/virtual-network-manager-schemas.d.ts +32 -0
- package/lib/azure-virtualnetworkmanager/lib/virtual-network-manager-schemas.js +236 -0
- package/lib/azure-virtualnetworkmanager/lib/virtual-network-manager.d.ts +337 -0
- package/lib/azure-virtualnetworkmanager/lib/virtual-network-manager.js +283 -0
- package/lib/azure-virtualnetworkmanager/test/virtual-network-manager.integ.d.ts +25 -0
- package/lib/azure-virtualnetworkmanager/test/virtual-network-manager.integ.js +402 -0
- package/lib/azure-virtualnetworkmanager/test/virtual-network-manager.spec.d.ts +9 -0
- package/lib/azure-virtualnetworkmanager/test/virtual-network-manager.spec.js +1652 -0
- package/lib/azure-vmss/index.d.ts +0 -5
- package/lib/azure-vmss/index.js +1 -6
- package/lib/azure-vmss/lib/index.d.ts +0 -7
- package/lib/azure-vmss/lib/index.js +1 -21
- package/lib/azure-vmss/lib/virtual-machine-scale-set.d.ts +5 -13
- package/lib/azure-vmss/lib/virtual-machine-scale-set.js +29 -53
- package/lib/core-azure/lib/azapi/azapi-resource-tags.spec.d.ts +10 -0
- package/lib/core-azure/lib/azapi/azapi-resource-tags.spec.js +218 -0
- package/lib/core-azure/lib/azapi/azapi-resource.d.ts +206 -26
- package/lib/core-azure/lib/azapi/azapi-resource.js +379 -91
- package/lib/core-azure/lib/azapi/azapi-resource.spec.js +2 -2
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-client-config/index.js +2 -2
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource/index.js +5 -5
- package/lib/core-azure/lib/azapi/providers-azapi/provider/index.js +5 -3
- package/lib/core-azure/lib/azapi/providers-azapi/resource/index.js +5 -5
- package/lib/core-azure/lib/azapi/providers-azapi/resource-action/index.js +3 -3
- package/lib/core-azure/lib/azapi/providers-azapi/update-resource/index.js +3 -3
- package/lib/core-azure/lib/azapi/resource-schema-validator.d.ts +118 -0
- package/lib/core-azure/lib/azapi/resource-schema-validator.js +236 -0
- package/lib/core-azure/lib/azapi/resource-version-manager.d.ts +103 -0
- package/lib/core-azure/lib/azapi/resource-version-manager.js +168 -0
- package/lib/core-azure/lib/azapi/schema-mapper/schema-mapper.js +11 -6
- package/lib/core-azure/lib/index.d.ts +2 -2
- package/lib/core-azure/lib/index.js +5 -5
- package/lib/core-azure/lib/version-manager/api-version-manager.js +1 -1
- package/lib/core-azure/lib/version-manager/interfaces/version-interfaces.js +7 -7
- package/lib/index.d.ts +19 -0
- package/lib/index.js +21 -2
- package/lib/testing/index.js +2 -2
- package/lib/testing/lib/cleanup.d.ts +1 -0
- package/lib/testing/lib/cleanup.js +19 -12
- package/lib/testing/lib/metadata.js +19 -16
- package/package.json +1 -1
- package/scripts/cleanup-test-resources.ts +22 -4
- package/scripts/generate-index.js +25 -8
|
@@ -0,0 +1,633 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Comprehensive tests for the unified RoleAssignment implementation
|
|
4
|
+
*
|
|
5
|
+
* This test suite validates the unified RoleAssignment class that uses
|
|
6
|
+
* the VersionedAzapiResource framework. Tests cover automatic version resolution,
|
|
7
|
+
* explicit version pinning, schema validation, property transformation, and
|
|
8
|
+
* role assignment-specific functionality.
|
|
9
|
+
*/
|
|
10
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
11
|
+
const cdktf_1 = require("cdktf");
|
|
12
|
+
const cdktf = require("cdktf");
|
|
13
|
+
const api_version_manager_1 = require("../../core-azure/lib/version-manager/api-version-manager");
|
|
14
|
+
const version_interfaces_1 = require("../../core-azure/lib/version-manager/interfaces/version-interfaces");
|
|
15
|
+
const role_assignment_1 = require("../lib/role-assignment");
|
|
16
|
+
const role_assignment_schemas_1 = require("../lib/role-assignment-schemas");
|
|
17
|
+
describe("RoleAssignment - Unified Implementation", () => {
|
|
18
|
+
let app;
|
|
19
|
+
let stack;
|
|
20
|
+
let manager;
|
|
21
|
+
// Test constants
|
|
22
|
+
const TEST_ROLE_DEF_ID = "/providers/Microsoft.Authorization/roleDefinitions/acdd72a7-3385-48ef-bd42-f606fba81ae7"; // Reader
|
|
23
|
+
const TEST_PRINCIPAL_ID = "00000000-0000-0000-0000-000000000001";
|
|
24
|
+
const TEST_SUBSCRIPTION_SCOPE = "/subscriptions/00000000-0000-0000-0000-000000000000";
|
|
25
|
+
beforeEach(() => {
|
|
26
|
+
app = cdktf_1.Testing.app();
|
|
27
|
+
stack = new cdktf.TerraformStack(app, "TestStack");
|
|
28
|
+
manager = api_version_manager_1.ApiVersionManager.instance();
|
|
29
|
+
// Ensure Role Assignment schemas are registered
|
|
30
|
+
try {
|
|
31
|
+
manager.registerResourceType(role_assignment_schemas_1.ROLE_ASSIGNMENT_TYPE, role_assignment_schemas_1.ALL_ROLE_ASSIGNMENT_VERSIONS);
|
|
32
|
+
}
|
|
33
|
+
catch (error) {
|
|
34
|
+
// Ignore if already registered
|
|
35
|
+
}
|
|
36
|
+
});
|
|
37
|
+
describe("Constructor and Basic Properties", () => {
|
|
38
|
+
it("should create role assignment with automatic latest version resolution", () => {
|
|
39
|
+
const props = {
|
|
40
|
+
name: "test-assignment",
|
|
41
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
42
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
43
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
44
|
+
};
|
|
45
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", props);
|
|
46
|
+
expect(roleAssignment).toBeInstanceOf(role_assignment_1.RoleAssignment);
|
|
47
|
+
expect(roleAssignment.resolvedApiVersion).toBe("2022-04-01"); // Latest version
|
|
48
|
+
expect(roleAssignment.props).toBe(props);
|
|
49
|
+
// Name is a deterministic UUID based on scope, roleDefinitionId, and principalId
|
|
50
|
+
expect(roleAssignment.name).toMatch(/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/);
|
|
51
|
+
});
|
|
52
|
+
it("should create role assignment with explicit version pinning", () => {
|
|
53
|
+
const props = {
|
|
54
|
+
name: "test-assignment-pinned",
|
|
55
|
+
apiVersion: "2022-04-01",
|
|
56
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
57
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
58
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
59
|
+
};
|
|
60
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", props);
|
|
61
|
+
expect(roleAssignment.resolvedApiVersion).toBe("2022-04-01");
|
|
62
|
+
});
|
|
63
|
+
it("should create role assignment with all optional properties", () => {
|
|
64
|
+
const props = {
|
|
65
|
+
name: "test-assignment-full",
|
|
66
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
67
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
68
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
69
|
+
principalType: "User",
|
|
70
|
+
description: "Test assignment for unit testing",
|
|
71
|
+
condition: "@Resource[Microsoft.Storage/storageAccounts:name] StringEquals 'test'",
|
|
72
|
+
conditionVersion: "2.0",
|
|
73
|
+
ignoreChanges: ["description"],
|
|
74
|
+
enableValidation: true,
|
|
75
|
+
enableMigrationAnalysis: true,
|
|
76
|
+
enableTransformation: true,
|
|
77
|
+
};
|
|
78
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", props);
|
|
79
|
+
expect(roleAssignment.props.roleDefinitionId).toBe(TEST_ROLE_DEF_ID);
|
|
80
|
+
expect(roleAssignment.props.principalId).toBe(TEST_PRINCIPAL_ID);
|
|
81
|
+
expect(roleAssignment.props.scope).toBe(TEST_SUBSCRIPTION_SCOPE);
|
|
82
|
+
expect(roleAssignment.props.principalType).toBe("User");
|
|
83
|
+
expect(roleAssignment.props.description).toBe("Test assignment for unit testing");
|
|
84
|
+
expect(roleAssignment.props.condition).toBeDefined();
|
|
85
|
+
expect(roleAssignment.props.conditionVersion).toBe("2.0");
|
|
86
|
+
});
|
|
87
|
+
it("should use default name when name is not provided", () => {
|
|
88
|
+
const props = {
|
|
89
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
90
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
91
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
92
|
+
};
|
|
93
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", props);
|
|
94
|
+
// Name is a deterministic UUID, not the construct ID
|
|
95
|
+
expect(roleAssignment.name).toMatch(/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/);
|
|
96
|
+
});
|
|
97
|
+
it("should require roleDefinitionId to be provided", () => {
|
|
98
|
+
const props = {
|
|
99
|
+
name: "test-assignment",
|
|
100
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
101
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
102
|
+
};
|
|
103
|
+
expect(() => {
|
|
104
|
+
new role_assignment_1.RoleAssignment(stack, "TestAssignment", props);
|
|
105
|
+
}).toThrow("Required property 'roleDefinitionId' is missing");
|
|
106
|
+
});
|
|
107
|
+
it("should require principalId to be provided", () => {
|
|
108
|
+
const props = {
|
|
109
|
+
name: "test-assignment",
|
|
110
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
111
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
112
|
+
};
|
|
113
|
+
expect(() => {
|
|
114
|
+
new role_assignment_1.RoleAssignment(stack, "TestAssignment", props);
|
|
115
|
+
}).toThrow("Required property 'principalId' is missing");
|
|
116
|
+
});
|
|
117
|
+
it("should require scope to be provided", () => {
|
|
118
|
+
const props = {
|
|
119
|
+
name: "test-assignment",
|
|
120
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
121
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
122
|
+
};
|
|
123
|
+
expect(() => {
|
|
124
|
+
new role_assignment_1.RoleAssignment(stack, "TestAssignment", props);
|
|
125
|
+
}).toThrow("Required property 'scope' is missing");
|
|
126
|
+
});
|
|
127
|
+
});
|
|
128
|
+
describe("Framework Integration", () => {
|
|
129
|
+
it("should resolve latest API version automatically", () => {
|
|
130
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
131
|
+
name: "test-assignment",
|
|
132
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
133
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
134
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
135
|
+
});
|
|
136
|
+
expect(roleAssignment.resolvedApiVersion).toBe("2022-04-01");
|
|
137
|
+
expect(roleAssignment.latestVersion()).toBe("2022-04-01");
|
|
138
|
+
});
|
|
139
|
+
it("should support all registered API versions", () => {
|
|
140
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
141
|
+
name: "test-assignment",
|
|
142
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
143
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
144
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
145
|
+
});
|
|
146
|
+
const supportedVersions = roleAssignment.supportedVersions();
|
|
147
|
+
expect(supportedVersions).toContain("2022-04-01");
|
|
148
|
+
});
|
|
149
|
+
it("should validate version support", () => {
|
|
150
|
+
// Valid version
|
|
151
|
+
expect(() => {
|
|
152
|
+
new role_assignment_1.RoleAssignment(stack, "ValidVersion", {
|
|
153
|
+
name: "test-assignment",
|
|
154
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
155
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
156
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
157
|
+
apiVersion: "2022-04-01",
|
|
158
|
+
});
|
|
159
|
+
}).not.toThrow();
|
|
160
|
+
// Invalid version
|
|
161
|
+
expect(() => {
|
|
162
|
+
new role_assignment_1.RoleAssignment(stack, "InvalidVersion", {
|
|
163
|
+
name: "test-assignment",
|
|
164
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
165
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
166
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
167
|
+
apiVersion: "2020-01-01",
|
|
168
|
+
});
|
|
169
|
+
}).toThrow("Unsupported API version '2020-01-01'");
|
|
170
|
+
});
|
|
171
|
+
it("should load correct schema for resolved version", () => {
|
|
172
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
173
|
+
name: "test-assignment",
|
|
174
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
175
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
176
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
177
|
+
apiVersion: "2022-04-01",
|
|
178
|
+
});
|
|
179
|
+
expect(roleAssignment.schema).toBeDefined();
|
|
180
|
+
expect(roleAssignment.schema.resourceType).toBe(role_assignment_schemas_1.ROLE_ASSIGNMENT_TYPE);
|
|
181
|
+
expect(roleAssignment.schema.version).toBe("2022-04-01");
|
|
182
|
+
expect(roleAssignment.schema.properties).toBeDefined();
|
|
183
|
+
});
|
|
184
|
+
it("should load version configuration correctly", () => {
|
|
185
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
186
|
+
name: "test-assignment",
|
|
187
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
188
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
189
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
190
|
+
});
|
|
191
|
+
expect(roleAssignment.versionConfig).toBeDefined();
|
|
192
|
+
expect(roleAssignment.versionConfig.version).toBe("2022-04-01");
|
|
193
|
+
expect(roleAssignment.versionConfig.supportLevel).toBe(version_interfaces_1.VersionSupportLevel.ACTIVE);
|
|
194
|
+
});
|
|
195
|
+
});
|
|
196
|
+
describe("Property Validation", () => {
|
|
197
|
+
it("should validate properties when validation is enabled", () => {
|
|
198
|
+
const props = {
|
|
199
|
+
name: "test-assignment",
|
|
200
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
201
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
202
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
203
|
+
enableValidation: true,
|
|
204
|
+
};
|
|
205
|
+
expect(() => {
|
|
206
|
+
new role_assignment_1.RoleAssignment(stack, "TestAssignment", props);
|
|
207
|
+
}).not.toThrow();
|
|
208
|
+
});
|
|
209
|
+
it("should have validation results for valid properties", () => {
|
|
210
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
211
|
+
name: "valid-assignment",
|
|
212
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
213
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
214
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
215
|
+
description: "A valid role assignment",
|
|
216
|
+
enableValidation: true,
|
|
217
|
+
});
|
|
218
|
+
expect(roleAssignment.validationResult).toBeDefined();
|
|
219
|
+
expect(roleAssignment.validationResult.valid).toBe(true);
|
|
220
|
+
expect(roleAssignment.validationResult.errors).toHaveLength(0);
|
|
221
|
+
});
|
|
222
|
+
it("should skip validation when disabled", () => {
|
|
223
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
224
|
+
name: "test-assignment",
|
|
225
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
226
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
227
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
228
|
+
enableValidation: false,
|
|
229
|
+
});
|
|
230
|
+
expect(roleAssignment).toBeDefined();
|
|
231
|
+
});
|
|
232
|
+
});
|
|
233
|
+
describe("Migration Analysis", () => {
|
|
234
|
+
it("should skip migration analysis for single version", () => {
|
|
235
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
236
|
+
name: "test-assignment",
|
|
237
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
238
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
239
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
240
|
+
apiVersion: "2022-04-01",
|
|
241
|
+
});
|
|
242
|
+
// Since there's only one version, migration analysis should be skipped
|
|
243
|
+
expect(roleAssignment).toBeDefined();
|
|
244
|
+
});
|
|
245
|
+
it("should skip migration analysis when disabled", () => {
|
|
246
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
247
|
+
name: "test-assignment",
|
|
248
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
249
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
250
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
251
|
+
enableMigrationAnalysis: false,
|
|
252
|
+
});
|
|
253
|
+
expect(roleAssignment.migrationAnalysis).toBeUndefined();
|
|
254
|
+
});
|
|
255
|
+
});
|
|
256
|
+
describe("Resource Creation and Body", () => {
|
|
257
|
+
it("should create correct resource body with minimal properties", () => {
|
|
258
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
259
|
+
name: "test-assignment",
|
|
260
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
261
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
262
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
263
|
+
});
|
|
264
|
+
expect(roleAssignment).toBeDefined();
|
|
265
|
+
expect(roleAssignment.props.roleDefinitionId).toBeDefined();
|
|
266
|
+
expect(roleAssignment.props.principalId).toBeDefined();
|
|
267
|
+
expect(roleAssignment.props.scope).toBeDefined();
|
|
268
|
+
});
|
|
269
|
+
it("should create correct resource body with all properties", () => {
|
|
270
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
271
|
+
name: "test-assignment",
|
|
272
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
273
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
274
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
275
|
+
principalType: "ServicePrincipal",
|
|
276
|
+
description: "Test assignment with all properties",
|
|
277
|
+
condition: "@Resource[Microsoft.Storage/storageAccounts:name] StringEquals 'test'",
|
|
278
|
+
conditionVersion: "2.0",
|
|
279
|
+
});
|
|
280
|
+
expect(roleAssignment).toBeDefined();
|
|
281
|
+
expect(roleAssignment.props.roleDefinitionId).toBe(TEST_ROLE_DEF_ID);
|
|
282
|
+
expect(roleAssignment.props.principalId).toBe(TEST_PRINCIPAL_ID);
|
|
283
|
+
expect(roleAssignment.props.principalType).toBe("ServicePrincipal");
|
|
284
|
+
expect(roleAssignment.props.description).toBe("Test assignment with all properties");
|
|
285
|
+
});
|
|
286
|
+
it("should create Terraform outputs", () => {
|
|
287
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
288
|
+
name: "test-assignment-outputs",
|
|
289
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
290
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
291
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
292
|
+
});
|
|
293
|
+
expect(roleAssignment.idOutput).toBeInstanceOf(cdktf.TerraformOutput);
|
|
294
|
+
expect(roleAssignment.nameOutput).toBeInstanceOf(cdktf.TerraformOutput);
|
|
295
|
+
});
|
|
296
|
+
});
|
|
297
|
+
describe("Principal Types", () => {
|
|
298
|
+
it("should support User principal type", () => {
|
|
299
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "UserAssignment", {
|
|
300
|
+
name: "user-assignment",
|
|
301
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
302
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
303
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
304
|
+
principalType: "User",
|
|
305
|
+
});
|
|
306
|
+
expect(roleAssignment.principalType).toBe("User");
|
|
307
|
+
});
|
|
308
|
+
it("should support Group principal type", () => {
|
|
309
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "GroupAssignment", {
|
|
310
|
+
name: "group-assignment",
|
|
311
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
312
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
313
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
314
|
+
principalType: "Group",
|
|
315
|
+
});
|
|
316
|
+
expect(roleAssignment.principalType).toBe("Group");
|
|
317
|
+
});
|
|
318
|
+
it("should support ServicePrincipal principal type", () => {
|
|
319
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "SPAssignment", {
|
|
320
|
+
name: "sp-assignment",
|
|
321
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
322
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
323
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
324
|
+
principalType: "ServicePrincipal",
|
|
325
|
+
});
|
|
326
|
+
expect(roleAssignment.principalType).toBe("ServicePrincipal");
|
|
327
|
+
});
|
|
328
|
+
it("should support ForeignGroup principal type", () => {
|
|
329
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "ForeignGroupAssignment", {
|
|
330
|
+
name: "foreign-group-assignment",
|
|
331
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
332
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
333
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
334
|
+
principalType: "ForeignGroup",
|
|
335
|
+
});
|
|
336
|
+
expect(roleAssignment.principalType).toBe("ForeignGroup");
|
|
337
|
+
});
|
|
338
|
+
it("should support Device principal type", () => {
|
|
339
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "DeviceAssignment", {
|
|
340
|
+
name: "device-assignment",
|
|
341
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
342
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
343
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
344
|
+
principalType: "Device",
|
|
345
|
+
});
|
|
346
|
+
expect(roleAssignment.principalType).toBe("Device");
|
|
347
|
+
});
|
|
348
|
+
it("should work without principalType specified", () => {
|
|
349
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "NoPrincipalType", {
|
|
350
|
+
name: "no-principal-type",
|
|
351
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
352
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
353
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
354
|
+
});
|
|
355
|
+
expect(roleAssignment.principalType).toBeUndefined();
|
|
356
|
+
});
|
|
357
|
+
});
|
|
358
|
+
describe("Conditional Assignments (ABAC)", () => {
|
|
359
|
+
it("should support conditional assignments", () => {
|
|
360
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "ConditionalAssignment", {
|
|
361
|
+
name: "conditional-assignment",
|
|
362
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
363
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
364
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
365
|
+
condition: "@Resource[Microsoft.Storage/storageAccounts/blobServices/containers:name] StringEquals 'logs'",
|
|
366
|
+
conditionVersion: "2.0",
|
|
367
|
+
});
|
|
368
|
+
expect(roleAssignment.props.condition).toBeDefined();
|
|
369
|
+
expect(roleAssignment.props.conditionVersion).toBe("2.0");
|
|
370
|
+
});
|
|
371
|
+
it("should support complex ABAC conditions", () => {
|
|
372
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "ComplexCondition", {
|
|
373
|
+
name: "complex-condition",
|
|
374
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
375
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
376
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
377
|
+
condition: "(@Resource[Microsoft.Storage/storageAccounts:name] StringEquals 'production' AND @Request[subOperation] StringEqualsIgnoreCase 'Blob.Read.WithTagConditions')",
|
|
378
|
+
conditionVersion: "2.0",
|
|
379
|
+
description: "Conditional access to production storage",
|
|
380
|
+
});
|
|
381
|
+
expect(roleAssignment.props.condition).toBeDefined();
|
|
382
|
+
expect(roleAssignment.props.conditionVersion).toBe("2.0");
|
|
383
|
+
});
|
|
384
|
+
});
|
|
385
|
+
describe("Delegated Managed Identity", () => {
|
|
386
|
+
it("should support delegated managed identity for group assignments", () => {
|
|
387
|
+
const managedIdentityId = "/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/rg/providers/Microsoft.ManagedIdentity/userAssignedIdentities/identity";
|
|
388
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "DelegatedMI", {
|
|
389
|
+
name: "delegated-mi-assignment",
|
|
390
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
391
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
392
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
393
|
+
principalType: "Group",
|
|
394
|
+
delegatedManagedIdentityResourceId: managedIdentityId,
|
|
395
|
+
});
|
|
396
|
+
expect(roleAssignment.props.delegatedManagedIdentityResourceId).toBe(managedIdentityId);
|
|
397
|
+
expect(roleAssignment.props.principalType).toBe("Group");
|
|
398
|
+
});
|
|
399
|
+
});
|
|
400
|
+
describe("Scope Configurations", () => {
|
|
401
|
+
it("should support subscription scope", () => {
|
|
402
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "SubscriptionScope", {
|
|
403
|
+
name: "subscription-assignment",
|
|
404
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
405
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
406
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
407
|
+
});
|
|
408
|
+
expect(roleAssignment.assignmentScope).toContain("/subscriptions/");
|
|
409
|
+
});
|
|
410
|
+
it("should support resource group scope", () => {
|
|
411
|
+
const rgScope = "/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/test-rg";
|
|
412
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "ResourceGroupScope", {
|
|
413
|
+
name: "rg-assignment",
|
|
414
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
415
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
416
|
+
scope: rgScope,
|
|
417
|
+
});
|
|
418
|
+
expect(roleAssignment.assignmentScope).toContain("/resourceGroups/");
|
|
419
|
+
});
|
|
420
|
+
it("should support resource scope", () => {
|
|
421
|
+
const resourceScope = "/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/test-rg/providers/Microsoft.Storage/storageAccounts/testaccount";
|
|
422
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "ResourceScope", {
|
|
423
|
+
name: "resource-assignment",
|
|
424
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
425
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
426
|
+
scope: resourceScope,
|
|
427
|
+
});
|
|
428
|
+
expect(roleAssignment.assignmentScope).toContain("/providers/Microsoft.Storage/");
|
|
429
|
+
});
|
|
430
|
+
});
|
|
431
|
+
describe("Public Methods and Properties", () => {
|
|
432
|
+
let roleAssignment;
|
|
433
|
+
beforeEach(() => {
|
|
434
|
+
roleAssignment = new role_assignment_1.RoleAssignment(stack, "TestAssignment", {
|
|
435
|
+
name: "test-assignment",
|
|
436
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
437
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
438
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
439
|
+
principalType: "User",
|
|
440
|
+
description: "A test assignment",
|
|
441
|
+
});
|
|
442
|
+
});
|
|
443
|
+
it("should have correct id format", () => {
|
|
444
|
+
expect(roleAssignment.id).toMatch(/^\$\{.*\.id\}$/);
|
|
445
|
+
});
|
|
446
|
+
it("should have resourceId property matching id", () => {
|
|
447
|
+
expect(roleAssignment.resourceId).toBe(roleAssignment.id);
|
|
448
|
+
});
|
|
449
|
+
it("should return correct roleDefinitionId", () => {
|
|
450
|
+
expect(roleAssignment.roleDefinitionId).toBe(TEST_ROLE_DEF_ID);
|
|
451
|
+
});
|
|
452
|
+
it("should return correct principalId", () => {
|
|
453
|
+
expect(roleAssignment.principalId).toBe(TEST_PRINCIPAL_ID);
|
|
454
|
+
});
|
|
455
|
+
it("should return correct assignmentScope", () => {
|
|
456
|
+
expect(roleAssignment.assignmentScope).toBe(TEST_SUBSCRIPTION_SCOPE);
|
|
457
|
+
});
|
|
458
|
+
it("should return correct principalType", () => {
|
|
459
|
+
expect(roleAssignment.principalType).toBe("User");
|
|
460
|
+
});
|
|
461
|
+
});
|
|
462
|
+
describe("Ignore Changes Configuration", () => {
|
|
463
|
+
it("should apply ignore changes lifecycle rules", () => {
|
|
464
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "IgnoreChanges", {
|
|
465
|
+
name: "ignore-changes-assignment",
|
|
466
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
467
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
468
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
469
|
+
ignoreChanges: ["description"],
|
|
470
|
+
});
|
|
471
|
+
expect(roleAssignment).toBeInstanceOf(role_assignment_1.RoleAssignment);
|
|
472
|
+
});
|
|
473
|
+
it("should handle empty ignore changes array", () => {
|
|
474
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "EmptyIgnore", {
|
|
475
|
+
name: "empty-ignore-assignment",
|
|
476
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
477
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
478
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
479
|
+
ignoreChanges: [],
|
|
480
|
+
});
|
|
481
|
+
expect(roleAssignment).toBeInstanceOf(role_assignment_1.RoleAssignment);
|
|
482
|
+
});
|
|
483
|
+
});
|
|
484
|
+
describe("Error Handling", () => {
|
|
485
|
+
it("should handle invalid API versions gracefully", () => {
|
|
486
|
+
expect(() => {
|
|
487
|
+
new role_assignment_1.RoleAssignment(stack, "InvalidAPI", {
|
|
488
|
+
name: "invalid-api-assignment",
|
|
489
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
490
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
491
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
492
|
+
apiVersion: "invalid-version",
|
|
493
|
+
});
|
|
494
|
+
}).toThrow("Unsupported API version 'invalid-version'");
|
|
495
|
+
});
|
|
496
|
+
it("should validate properties when validation is enabled", () => {
|
|
497
|
+
// Role assignments generate deterministic UUIDs for names, so name validation
|
|
498
|
+
// doesn't apply the same way. Test that valid role assignments pass validation.
|
|
499
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "ValidationTest", {
|
|
500
|
+
name: "test-assignment", // This is ignored in favor of generated UUID
|
|
501
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
502
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
503
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
504
|
+
enableValidation: true,
|
|
505
|
+
});
|
|
506
|
+
expect(roleAssignment.validationResult).toBeDefined();
|
|
507
|
+
expect(roleAssignment.validationResult.valid).toBe(true);
|
|
508
|
+
});
|
|
509
|
+
it("should handle schema registration errors gracefully", () => {
|
|
510
|
+
expect(() => {
|
|
511
|
+
new role_assignment_1.RoleAssignment(stack, "SchemaTest", {
|
|
512
|
+
name: "schema-test-assignment",
|
|
513
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
514
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
515
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
516
|
+
});
|
|
517
|
+
}).not.toThrow();
|
|
518
|
+
});
|
|
519
|
+
});
|
|
520
|
+
describe("JSII Compliance", () => {
|
|
521
|
+
it("should have JSII-compliant constructor", () => {
|
|
522
|
+
expect(() => {
|
|
523
|
+
new role_assignment_1.RoleAssignment(stack, "JsiiTest", {
|
|
524
|
+
name: "jsii-test",
|
|
525
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
526
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
527
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
528
|
+
});
|
|
529
|
+
}).not.toThrow();
|
|
530
|
+
});
|
|
531
|
+
it("should have JSII-compliant properties", () => {
|
|
532
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "JsiiProps", {
|
|
533
|
+
name: "jsii-props",
|
|
534
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
535
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
536
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
537
|
+
});
|
|
538
|
+
expect(typeof roleAssignment.id).toBe("string");
|
|
539
|
+
expect(typeof roleAssignment.name).toBe("string");
|
|
540
|
+
expect(typeof roleAssignment.resolvedApiVersion).toBe("string");
|
|
541
|
+
expect(typeof roleAssignment.roleDefinitionId).toBe("string");
|
|
542
|
+
expect(typeof roleAssignment.principalId).toBe("string");
|
|
543
|
+
expect(typeof roleAssignment.assignmentScope).toBe("string");
|
|
544
|
+
});
|
|
545
|
+
it("should have JSII-compliant methods", () => {
|
|
546
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "JsiiMethods", {
|
|
547
|
+
name: "jsii-methods",
|
|
548
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
549
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
550
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
551
|
+
});
|
|
552
|
+
expect(typeof roleAssignment.latestVersion).toBe("function");
|
|
553
|
+
expect(typeof roleAssignment.supportedVersions).toBe("function");
|
|
554
|
+
});
|
|
555
|
+
it("should serialize complex objects correctly", () => {
|
|
556
|
+
const roleAssignment = new role_assignment_1.RoleAssignment(stack, "JsiiSerialization", {
|
|
557
|
+
name: "jsii-serialization",
|
|
558
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
559
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
560
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
561
|
+
principalType: "User",
|
|
562
|
+
});
|
|
563
|
+
expect(() => JSON.stringify(roleAssignment.validationResult)).not.toThrow();
|
|
564
|
+
expect(() => JSON.stringify(roleAssignment.schema)).not.toThrow();
|
|
565
|
+
expect(() => JSON.stringify(roleAssignment.versionConfig)).not.toThrow();
|
|
566
|
+
});
|
|
567
|
+
});
|
|
568
|
+
describe("CDK Terraform Integration", () => {
|
|
569
|
+
it("should synthesize to valid Terraform configuration", () => {
|
|
570
|
+
new role_assignment_1.RoleAssignment(stack, "SynthTest", {
|
|
571
|
+
name: "synth-test",
|
|
572
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
573
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
574
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
575
|
+
});
|
|
576
|
+
const synthesized = cdktf_1.Testing.synth(stack);
|
|
577
|
+
expect(synthesized).toBeDefined();
|
|
578
|
+
const stackConfig = JSON.parse(synthesized);
|
|
579
|
+
expect(stackConfig.resource).toBeDefined();
|
|
580
|
+
});
|
|
581
|
+
it("should work in complex CDK constructs", () => {
|
|
582
|
+
class ComplexConstruct extends cdktf.TerraformStack {
|
|
583
|
+
constructor(scope, id) {
|
|
584
|
+
super(scope, id);
|
|
585
|
+
const assignment1 = new role_assignment_1.RoleAssignment(this, "Assignment1", {
|
|
586
|
+
name: "assignment-1",
|
|
587
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
588
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
589
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
590
|
+
});
|
|
591
|
+
const assignment2 = new role_assignment_1.RoleAssignment(this, "Assignment2", {
|
|
592
|
+
name: "assignment-2",
|
|
593
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
594
|
+
principalId: "00000000-0000-0000-0000-000000000002",
|
|
595
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
596
|
+
principalType: "Group",
|
|
597
|
+
apiVersion: "2022-04-01",
|
|
598
|
+
});
|
|
599
|
+
new cdktf.TerraformOutput(this, "Assignment1Id", {
|
|
600
|
+
value: assignment1.id,
|
|
601
|
+
});
|
|
602
|
+
new cdktf.TerraformOutput(this, "Assignment2Id", {
|
|
603
|
+
value: assignment2.id,
|
|
604
|
+
});
|
|
605
|
+
}
|
|
606
|
+
}
|
|
607
|
+
expect(() => {
|
|
608
|
+
new ComplexConstruct(app, "ComplexStack");
|
|
609
|
+
}).not.toThrow();
|
|
610
|
+
});
|
|
611
|
+
it("should handle multiple role assignments in the same stack", () => {
|
|
612
|
+
const assignment1 = new role_assignment_1.RoleAssignment(stack, "Assignment1", {
|
|
613
|
+
name: "assignment-1",
|
|
614
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
615
|
+
principalId: TEST_PRINCIPAL_ID,
|
|
616
|
+
scope: TEST_SUBSCRIPTION_SCOPE,
|
|
617
|
+
});
|
|
618
|
+
const assignment2 = new role_assignment_1.RoleAssignment(stack, "Assignment2", {
|
|
619
|
+
name: "assignment-2",
|
|
620
|
+
roleDefinitionId: TEST_ROLE_DEF_ID,
|
|
621
|
+
principalId: "00000000-0000-0000-0000-000000000002",
|
|
622
|
+
scope: "/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/test-rg",
|
|
623
|
+
principalType: "ServicePrincipal",
|
|
624
|
+
apiVersion: "2022-04-01",
|
|
625
|
+
});
|
|
626
|
+
expect(assignment1.resolvedApiVersion).toBe("2022-04-01");
|
|
627
|
+
expect(assignment2.resolvedApiVersion).toBe("2022-04-01");
|
|
628
|
+
const synthesized = cdktf_1.Testing.synth(stack);
|
|
629
|
+
expect(synthesized).toBeDefined();
|
|
630
|
+
});
|
|
631
|
+
});
|
|
632
|
+
});
|
|
633
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"role-assignment.spec.js","sourceRoot":"","sources":["../../../src/azure-roleassignment/test/role-assignment.spec.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Comprehensive tests for the unified RoleAssignment implementation\n *\n * This test suite validates the unified RoleAssignment class that uses\n * the VersionedAzapiResource framework. Tests cover automatic version resolution,\n * explicit version pinning, schema validation, property transformation, and\n * role assignment-specific functionality.\n */\n\nimport { Testing } from \"cdktf\";\nimport * as cdktf from \"cdktf\";\nimport { ApiVersionManager } from \"../../core-azure/lib/version-manager/api-version-manager\";\nimport { VersionSupportLevel } from \"../../core-azure/lib/version-manager/interfaces/version-interfaces\";\nimport { RoleAssignment, RoleAssignmentProps } from \"../lib/role-assignment\";\nimport {\n  ALL_ROLE_ASSIGNMENT_VERSIONS,\n  ROLE_ASSIGNMENT_TYPE,\n} from \"../lib/role-assignment-schemas\";\n\ndescribe(\"RoleAssignment - Unified Implementation\", () => {\n  let app: cdktf.App;\n  let stack: cdktf.TerraformStack;\n  let manager: ApiVersionManager;\n\n  // Test constants\n  const TEST_ROLE_DEF_ID =\n    \"/providers/Microsoft.Authorization/roleDefinitions/acdd72a7-3385-48ef-bd42-f606fba81ae7\"; // Reader\n  const TEST_PRINCIPAL_ID = \"00000000-0000-0000-0000-000000000001\";\n  const TEST_SUBSCRIPTION_SCOPE =\n    \"/subscriptions/00000000-0000-0000-0000-000000000000\";\n\n  beforeEach(() => {\n    app = Testing.app();\n    stack = new cdktf.TerraformStack(app, \"TestStack\");\n    manager = ApiVersionManager.instance();\n\n    // Ensure Role Assignment schemas are registered\n    try {\n      manager.registerResourceType(\n        ROLE_ASSIGNMENT_TYPE,\n        ALL_ROLE_ASSIGNMENT_VERSIONS,\n      );\n    } catch (error) {\n      // Ignore if already registered\n    }\n  });\n\n  describe(\"Constructor and Basic Properties\", () => {\n    it(\"should create role assignment with automatic latest version resolution\", () => {\n      const props: RoleAssignmentProps = {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      };\n\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", props);\n\n      expect(roleAssignment).toBeInstanceOf(RoleAssignment);\n      expect(roleAssignment.resolvedApiVersion).toBe(\"2022-04-01\"); // Latest version\n      expect(roleAssignment.props).toBe(props);\n      // Name is a deterministic UUID based on scope, roleDefinitionId, and principalId\n      expect(roleAssignment.name).toMatch(\n        /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/,\n      );\n    });\n\n    it(\"should create role assignment with explicit version pinning\", () => {\n      const props: RoleAssignmentProps = {\n        name: \"test-assignment-pinned\",\n        apiVersion: \"2022-04-01\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      };\n\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", props);\n\n      expect(roleAssignment.resolvedApiVersion).toBe(\"2022-04-01\");\n    });\n\n    it(\"should create role assignment with all optional properties\", () => {\n      const props: RoleAssignmentProps = {\n        name: \"test-assignment-full\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        principalType: \"User\",\n        description: \"Test assignment for unit testing\",\n        condition:\n          \"@Resource[Microsoft.Storage/storageAccounts:name] StringEquals 'test'\",\n        conditionVersion: \"2.0\",\n        ignoreChanges: [\"description\"],\n        enableValidation: true,\n        enableMigrationAnalysis: true,\n        enableTransformation: true,\n      };\n\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", props);\n\n      expect(roleAssignment.props.roleDefinitionId).toBe(TEST_ROLE_DEF_ID);\n      expect(roleAssignment.props.principalId).toBe(TEST_PRINCIPAL_ID);\n      expect(roleAssignment.props.scope).toBe(TEST_SUBSCRIPTION_SCOPE);\n      expect(roleAssignment.props.principalType).toBe(\"User\");\n      expect(roleAssignment.props.description).toBe(\n        \"Test assignment for unit testing\",\n      );\n      expect(roleAssignment.props.condition).toBeDefined();\n      expect(roleAssignment.props.conditionVersion).toBe(\"2.0\");\n    });\n\n    it(\"should use default name when name is not provided\", () => {\n      const props: RoleAssignmentProps = {\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      };\n\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", props);\n\n      // Name is a deterministic UUID, not the construct ID\n      expect(roleAssignment.name).toMatch(\n        /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/,\n      );\n    });\n\n    it(\"should require roleDefinitionId to be provided\", () => {\n      const props: any = {\n        name: \"test-assignment\",\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      };\n\n      expect(() => {\n        new RoleAssignment(stack, \"TestAssignment\", props);\n      }).toThrow(\"Required property 'roleDefinitionId' is missing\");\n    });\n\n    it(\"should require principalId to be provided\", () => {\n      const props: any = {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      };\n\n      expect(() => {\n        new RoleAssignment(stack, \"TestAssignment\", props);\n      }).toThrow(\"Required property 'principalId' is missing\");\n    });\n\n    it(\"should require scope to be provided\", () => {\n      const props: any = {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n      };\n\n      expect(() => {\n        new RoleAssignment(stack, \"TestAssignment\", props);\n      }).toThrow(\"Required property 'scope' is missing\");\n    });\n  });\n\n  describe(\"Framework Integration\", () => {\n    it(\"should resolve latest API version automatically\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      });\n\n      expect(roleAssignment.resolvedApiVersion).toBe(\"2022-04-01\");\n      expect(roleAssignment.latestVersion()).toBe(\"2022-04-01\");\n    });\n\n    it(\"should support all registered API versions\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      });\n\n      const supportedVersions = roleAssignment.supportedVersions();\n      expect(supportedVersions).toContain(\"2022-04-01\");\n    });\n\n    it(\"should validate version support\", () => {\n      // Valid version\n      expect(() => {\n        new RoleAssignment(stack, \"ValidVersion\", {\n          name: \"test-assignment\",\n          roleDefinitionId: TEST_ROLE_DEF_ID,\n          principalId: TEST_PRINCIPAL_ID,\n          scope: TEST_SUBSCRIPTION_SCOPE,\n          apiVersion: \"2022-04-01\",\n        });\n      }).not.toThrow();\n\n      // Invalid version\n      expect(() => {\n        new RoleAssignment(stack, \"InvalidVersion\", {\n          name: \"test-assignment\",\n          roleDefinitionId: TEST_ROLE_DEF_ID,\n          principalId: TEST_PRINCIPAL_ID,\n          scope: TEST_SUBSCRIPTION_SCOPE,\n          apiVersion: \"2020-01-01\",\n        });\n      }).toThrow(\"Unsupported API version '2020-01-01'\");\n    });\n\n    it(\"should load correct schema for resolved version\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        apiVersion: \"2022-04-01\",\n      });\n\n      expect(roleAssignment.schema).toBeDefined();\n      expect(roleAssignment.schema.resourceType).toBe(ROLE_ASSIGNMENT_TYPE);\n      expect(roleAssignment.schema.version).toBe(\"2022-04-01\");\n      expect(roleAssignment.schema.properties).toBeDefined();\n    });\n\n    it(\"should load version configuration correctly\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      });\n\n      expect(roleAssignment.versionConfig).toBeDefined();\n      expect(roleAssignment.versionConfig.version).toBe(\"2022-04-01\");\n      expect(roleAssignment.versionConfig.supportLevel).toBe(\n        VersionSupportLevel.ACTIVE,\n      );\n    });\n  });\n\n  describe(\"Property Validation\", () => {\n    it(\"should validate properties when validation is enabled\", () => {\n      const props: RoleAssignmentProps = {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        enableValidation: true,\n      };\n\n      expect(() => {\n        new RoleAssignment(stack, \"TestAssignment\", props);\n      }).not.toThrow();\n    });\n\n    it(\"should have validation results for valid properties\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"valid-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        description: \"A valid role assignment\",\n        enableValidation: true,\n      });\n\n      expect(roleAssignment.validationResult).toBeDefined();\n      expect(roleAssignment.validationResult!.valid).toBe(true);\n      expect(roleAssignment.validationResult!.errors).toHaveLength(0);\n    });\n\n    it(\"should skip validation when disabled\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        enableValidation: false,\n      });\n\n      expect(roleAssignment).toBeDefined();\n    });\n  });\n\n  describe(\"Migration Analysis\", () => {\n    it(\"should skip migration analysis for single version\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        apiVersion: \"2022-04-01\",\n      });\n\n      // Since there's only one version, migration analysis should be skipped\n      expect(roleAssignment).toBeDefined();\n    });\n\n    it(\"should skip migration analysis when disabled\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        enableMigrationAnalysis: false,\n      });\n\n      expect(roleAssignment.migrationAnalysis).toBeUndefined();\n    });\n  });\n\n  describe(\"Resource Creation and Body\", () => {\n    it(\"should create correct resource body with minimal properties\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      });\n\n      expect(roleAssignment).toBeDefined();\n      expect(roleAssignment.props.roleDefinitionId).toBeDefined();\n      expect(roleAssignment.props.principalId).toBeDefined();\n      expect(roleAssignment.props.scope).toBeDefined();\n    });\n\n    it(\"should create correct resource body with all properties\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        principalType: \"ServicePrincipal\",\n        description: \"Test assignment with all properties\",\n        condition:\n          \"@Resource[Microsoft.Storage/storageAccounts:name] StringEquals 'test'\",\n        conditionVersion: \"2.0\",\n      });\n\n      expect(roleAssignment).toBeDefined();\n      expect(roleAssignment.props.roleDefinitionId).toBe(TEST_ROLE_DEF_ID);\n      expect(roleAssignment.props.principalId).toBe(TEST_PRINCIPAL_ID);\n      expect(roleAssignment.props.principalType).toBe(\"ServicePrincipal\");\n      expect(roleAssignment.props.description).toBe(\n        \"Test assignment with all properties\",\n      );\n    });\n\n    it(\"should create Terraform outputs\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment-outputs\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      });\n\n      expect(roleAssignment.idOutput).toBeInstanceOf(cdktf.TerraformOutput);\n      expect(roleAssignment.nameOutput).toBeInstanceOf(cdktf.TerraformOutput);\n    });\n  });\n\n  describe(\"Principal Types\", () => {\n    it(\"should support User principal type\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"UserAssignment\", {\n        name: \"user-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        principalType: \"User\",\n      });\n\n      expect(roleAssignment.principalType).toBe(\"User\");\n    });\n\n    it(\"should support Group principal type\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"GroupAssignment\", {\n        name: \"group-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        principalType: \"Group\",\n      });\n\n      expect(roleAssignment.principalType).toBe(\"Group\");\n    });\n\n    it(\"should support ServicePrincipal principal type\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"SPAssignment\", {\n        name: \"sp-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        principalType: \"ServicePrincipal\",\n      });\n\n      expect(roleAssignment.principalType).toBe(\"ServicePrincipal\");\n    });\n\n    it(\"should support ForeignGroup principal type\", () => {\n      const roleAssignment = new RoleAssignment(\n        stack,\n        \"ForeignGroupAssignment\",\n        {\n          name: \"foreign-group-assignment\",\n          roleDefinitionId: TEST_ROLE_DEF_ID,\n          principalId: TEST_PRINCIPAL_ID,\n          scope: TEST_SUBSCRIPTION_SCOPE,\n          principalType: \"ForeignGroup\",\n        },\n      );\n\n      expect(roleAssignment.principalType).toBe(\"ForeignGroup\");\n    });\n\n    it(\"should support Device principal type\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"DeviceAssignment\", {\n        name: \"device-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        principalType: \"Device\",\n      });\n\n      expect(roleAssignment.principalType).toBe(\"Device\");\n    });\n\n    it(\"should work without principalType specified\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"NoPrincipalType\", {\n        name: \"no-principal-type\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      });\n\n      expect(roleAssignment.principalType).toBeUndefined();\n    });\n  });\n\n  describe(\"Conditional Assignments (ABAC)\", () => {\n    it(\"should support conditional assignments\", () => {\n      const roleAssignment = new RoleAssignment(\n        stack,\n        \"ConditionalAssignment\",\n        {\n          name: \"conditional-assignment\",\n          roleDefinitionId: TEST_ROLE_DEF_ID,\n          principalId: TEST_PRINCIPAL_ID,\n          scope: TEST_SUBSCRIPTION_SCOPE,\n          condition:\n            \"@Resource[Microsoft.Storage/storageAccounts/blobServices/containers:name] StringEquals 'logs'\",\n          conditionVersion: \"2.0\",\n        },\n      );\n\n      expect(roleAssignment.props.condition).toBeDefined();\n      expect(roleAssignment.props.conditionVersion).toBe(\"2.0\");\n    });\n\n    it(\"should support complex ABAC conditions\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"ComplexCondition\", {\n        name: \"complex-condition\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        condition:\n          \"(@Resource[Microsoft.Storage/storageAccounts:name] StringEquals 'production' AND @Request[subOperation] StringEqualsIgnoreCase 'Blob.Read.WithTagConditions')\",\n        conditionVersion: \"2.0\",\n        description: \"Conditional access to production storage\",\n      });\n\n      expect(roleAssignment.props.condition).toBeDefined();\n      expect(roleAssignment.props.conditionVersion).toBe(\"2.0\");\n    });\n  });\n\n  describe(\"Delegated Managed Identity\", () => {\n    it(\"should support delegated managed identity for group assignments\", () => {\n      const managedIdentityId =\n        \"/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/rg/providers/Microsoft.ManagedIdentity/userAssignedIdentities/identity\";\n\n      const roleAssignment = new RoleAssignment(stack, \"DelegatedMI\", {\n        name: \"delegated-mi-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        principalType: \"Group\",\n        delegatedManagedIdentityResourceId: managedIdentityId,\n      });\n\n      expect(roleAssignment.props.delegatedManagedIdentityResourceId).toBe(\n        managedIdentityId,\n      );\n      expect(roleAssignment.props.principalType).toBe(\"Group\");\n    });\n  });\n\n  describe(\"Scope Configurations\", () => {\n    it(\"should support subscription scope\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"SubscriptionScope\", {\n        name: \"subscription-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      });\n\n      expect(roleAssignment.assignmentScope).toContain(\"/subscriptions/\");\n    });\n\n    it(\"should support resource group scope\", () => {\n      const rgScope =\n        \"/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/test-rg\";\n\n      const roleAssignment = new RoleAssignment(stack, \"ResourceGroupScope\", {\n        name: \"rg-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: rgScope,\n      });\n\n      expect(roleAssignment.assignmentScope).toContain(\"/resourceGroups/\");\n    });\n\n    it(\"should support resource scope\", () => {\n      const resourceScope =\n        \"/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/test-rg/providers/Microsoft.Storage/storageAccounts/testaccount\";\n\n      const roleAssignment = new RoleAssignment(stack, \"ResourceScope\", {\n        name: \"resource-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: resourceScope,\n      });\n\n      expect(roleAssignment.assignmentScope).toContain(\n        \"/providers/Microsoft.Storage/\",\n      );\n    });\n  });\n\n  describe(\"Public Methods and Properties\", () => {\n    let roleAssignment: RoleAssignment;\n\n    beforeEach(() => {\n      roleAssignment = new RoleAssignment(stack, \"TestAssignment\", {\n        name: \"test-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        principalType: \"User\",\n        description: \"A test assignment\",\n      });\n    });\n\n    it(\"should have correct id format\", () => {\n      expect(roleAssignment.id).toMatch(/^\\$\\{.*\\.id\\}$/);\n    });\n\n    it(\"should have resourceId property matching id\", () => {\n      expect(roleAssignment.resourceId).toBe(roleAssignment.id);\n    });\n\n    it(\"should return correct roleDefinitionId\", () => {\n      expect(roleAssignment.roleDefinitionId).toBe(TEST_ROLE_DEF_ID);\n    });\n\n    it(\"should return correct principalId\", () => {\n      expect(roleAssignment.principalId).toBe(TEST_PRINCIPAL_ID);\n    });\n\n    it(\"should return correct assignmentScope\", () => {\n      expect(roleAssignment.assignmentScope).toBe(TEST_SUBSCRIPTION_SCOPE);\n    });\n\n    it(\"should return correct principalType\", () => {\n      expect(roleAssignment.principalType).toBe(\"User\");\n    });\n  });\n\n  describe(\"Ignore Changes Configuration\", () => {\n    it(\"should apply ignore changes lifecycle rules\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"IgnoreChanges\", {\n        name: \"ignore-changes-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        ignoreChanges: [\"description\"],\n      });\n\n      expect(roleAssignment).toBeInstanceOf(RoleAssignment);\n    });\n\n    it(\"should handle empty ignore changes array\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"EmptyIgnore\", {\n        name: \"empty-ignore-assignment\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        ignoreChanges: [],\n      });\n\n      expect(roleAssignment).toBeInstanceOf(RoleAssignment);\n    });\n  });\n\n  describe(\"Error Handling\", () => {\n    it(\"should handle invalid API versions gracefully\", () => {\n      expect(() => {\n        new RoleAssignment(stack, \"InvalidAPI\", {\n          name: \"invalid-api-assignment\",\n          roleDefinitionId: TEST_ROLE_DEF_ID,\n          principalId: TEST_PRINCIPAL_ID,\n          scope: TEST_SUBSCRIPTION_SCOPE,\n          apiVersion: \"invalid-version\",\n        });\n      }).toThrow(\"Unsupported API version 'invalid-version'\");\n    });\n\n    it(\"should validate properties when validation is enabled\", () => {\n      // Role assignments generate deterministic UUIDs for names, so name validation\n      // doesn't apply the same way. Test that valid role assignments pass validation.\n      const roleAssignment = new RoleAssignment(stack, \"ValidationTest\", {\n        name: \"test-assignment\", // This is ignored in favor of generated UUID\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        enableValidation: true,\n      });\n\n      expect(roleAssignment.validationResult).toBeDefined();\n      expect(roleAssignment.validationResult!.valid).toBe(true);\n    });\n\n    it(\"should handle schema registration errors gracefully\", () => {\n      expect(() => {\n        new RoleAssignment(stack, \"SchemaTest\", {\n          name: \"schema-test-assignment\",\n          roleDefinitionId: TEST_ROLE_DEF_ID,\n          principalId: TEST_PRINCIPAL_ID,\n          scope: TEST_SUBSCRIPTION_SCOPE,\n        });\n      }).not.toThrow();\n    });\n  });\n\n  describe(\"JSII Compliance\", () => {\n    it(\"should have JSII-compliant constructor\", () => {\n      expect(() => {\n        new RoleAssignment(stack, \"JsiiTest\", {\n          name: \"jsii-test\",\n          roleDefinitionId: TEST_ROLE_DEF_ID,\n          principalId: TEST_PRINCIPAL_ID,\n          scope: TEST_SUBSCRIPTION_SCOPE,\n        });\n      }).not.toThrow();\n    });\n\n    it(\"should have JSII-compliant properties\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"JsiiProps\", {\n        name: \"jsii-props\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      });\n\n      expect(typeof roleAssignment.id).toBe(\"string\");\n      expect(typeof roleAssignment.name).toBe(\"string\");\n      expect(typeof roleAssignment.resolvedApiVersion).toBe(\"string\");\n      expect(typeof roleAssignment.roleDefinitionId).toBe(\"string\");\n      expect(typeof roleAssignment.principalId).toBe(\"string\");\n      expect(typeof roleAssignment.assignmentScope).toBe(\"string\");\n    });\n\n    it(\"should have JSII-compliant methods\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"JsiiMethods\", {\n        name: \"jsii-methods\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      });\n\n      expect(typeof roleAssignment.latestVersion).toBe(\"function\");\n      expect(typeof roleAssignment.supportedVersions).toBe(\"function\");\n    });\n\n    it(\"should serialize complex objects correctly\", () => {\n      const roleAssignment = new RoleAssignment(stack, \"JsiiSerialization\", {\n        name: \"jsii-serialization\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n        principalType: \"User\",\n      });\n\n      expect(() =>\n        JSON.stringify(roleAssignment.validationResult),\n      ).not.toThrow();\n      expect(() => JSON.stringify(roleAssignment.schema)).not.toThrow();\n      expect(() => JSON.stringify(roleAssignment.versionConfig)).not.toThrow();\n    });\n  });\n\n  describe(\"CDK Terraform Integration\", () => {\n    it(\"should synthesize to valid Terraform configuration\", () => {\n      new RoleAssignment(stack, \"SynthTest\", {\n        name: \"synth-test\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\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 work in complex CDK constructs\", () => {\n      class ComplexConstruct extends cdktf.TerraformStack {\n        constructor(scope: cdktf.App, id: string) {\n          super(scope, id);\n\n          const assignment1 = new RoleAssignment(this, \"Assignment1\", {\n            name: \"assignment-1\",\n            roleDefinitionId: TEST_ROLE_DEF_ID,\n            principalId: TEST_PRINCIPAL_ID,\n            scope: TEST_SUBSCRIPTION_SCOPE,\n          });\n\n          const assignment2 = new RoleAssignment(this, \"Assignment2\", {\n            name: \"assignment-2\",\n            roleDefinitionId: TEST_ROLE_DEF_ID,\n            principalId: \"00000000-0000-0000-0000-000000000002\",\n            scope: TEST_SUBSCRIPTION_SCOPE,\n            principalType: \"Group\",\n            apiVersion: \"2022-04-01\",\n          });\n\n          new cdktf.TerraformOutput(this, \"Assignment1Id\", {\n            value: assignment1.id,\n          });\n\n          new cdktf.TerraformOutput(this, \"Assignment2Id\", {\n            value: assignment2.id,\n          });\n        }\n      }\n\n      expect(() => {\n        new ComplexConstruct(app, \"ComplexStack\");\n      }).not.toThrow();\n    });\n\n    it(\"should handle multiple role assignments in the same stack\", () => {\n      const assignment1 = new RoleAssignment(stack, \"Assignment1\", {\n        name: \"assignment-1\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: TEST_PRINCIPAL_ID,\n        scope: TEST_SUBSCRIPTION_SCOPE,\n      });\n\n      const assignment2 = new RoleAssignment(stack, \"Assignment2\", {\n        name: \"assignment-2\",\n        roleDefinitionId: TEST_ROLE_DEF_ID,\n        principalId: \"00000000-0000-0000-0000-000000000002\",\n        scope:\n          \"/subscriptions/00000000-0000-0000-0000-000000000000/resourceGroups/test-rg\",\n        principalType: \"ServicePrincipal\",\n        apiVersion: \"2022-04-01\",\n      });\n\n      expect(assignment1.resolvedApiVersion).toBe(\"2022-04-01\");\n      expect(assignment2.resolvedApiVersion).toBe(\"2022-04-01\");\n\n      const synthesized = Testing.synth(stack);\n      expect(synthesized).toBeDefined();\n    });\n  });\n});\n"]}
|