@microsoft/terraform-cdk-constructs 1.0.0 → 1.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +60999 -15780
- package/API.md +54947 -10610
- package/README.md +72 -36
- package/docs/monitoring-guide.md +206 -0
- package/docs/testing.md +84 -1
- package/lib/azure-actiongroup/index.d.ts +11 -0
- package/lib/azure-actiongroup/index.js +28 -0
- package/lib/azure-actiongroup/lib/action-group-schemas.d.ts +24 -0
- package/lib/azure-actiongroup/lib/action-group-schemas.js +201 -0
- package/lib/azure-actiongroup/lib/action-group.d.ts +312 -0
- package/lib/azure-actiongroup/lib/action-group.js +201 -0
- package/lib/azure-actiongroup/lib/index.d.ts +5 -0
- package/lib/azure-actiongroup/lib/index.js +22 -0
- package/lib/azure-actiongroup/test/action-group.integ.d.ts +9 -0
- package/lib/azure-actiongroup/test/action-group.integ.js +79 -0
- package/lib/azure-actiongroup/test/action-group.spec.d.ts +8 -0
- package/lib/azure-actiongroup/test/action-group.spec.js +602 -0
- package/lib/azure-activitylogalert/index.d.ts +11 -0
- package/lib/azure-activitylogalert/index.js +28 -0
- package/lib/azure-activitylogalert/lib/activity-log-alert-schemas.d.ts +24 -0
- package/lib/azure-activitylogalert/lib/activity-log-alert-schemas.js +182 -0
- package/lib/azure-activitylogalert/lib/activity-log-alert.d.ts +236 -0
- package/lib/azure-activitylogalert/lib/activity-log-alert.js +212 -0
- package/lib/azure-activitylogalert/lib/index.d.ts +5 -0
- package/lib/azure-activitylogalert/lib/index.js +22 -0
- package/lib/azure-activitylogalert/test/activity-log-alert.integ.d.ts +9 -0
- package/lib/azure-activitylogalert/test/activity-log-alert.integ.js +98 -0
- package/lib/azure-activitylogalert/test/activity-log-alert.spec.d.ts +9 -0
- package/lib/azure-activitylogalert/test/activity-log-alert.spec.js +1101 -0
- package/lib/azure-aks/index.d.ts +9 -0
- package/lib/azure-aks/index.js +28 -0
- package/lib/azure-aks/lib/aks-cluster-schemas.d.ts +505 -0
- package/lib/azure-aks/lib/aks-cluster-schemas.js +572 -0
- package/lib/azure-aks/lib/aks-cluster.d.ts +395 -0
- package/lib/azure-aks/lib/aks-cluster.js +489 -0
- package/lib/azure-aks/lib/index.d.ts +11 -0
- package/lib/azure-aks/lib/index.js +38 -0
- package/lib/azure-aks/test/aks-cluster.integ.d.ts +9 -0
- package/lib/azure-aks/test/aks-cluster.integ.js +260 -0
- package/lib/azure-aks/test/aks-cluster.spec.d.ts +8 -0
- package/lib/azure-aks/test/aks-cluster.spec.js +1396 -0
- package/lib/azure-diagnosticsettings/index.d.ts +7 -0
- package/lib/azure-diagnosticsettings/index.js +24 -0
- package/lib/azure-diagnosticsettings/lib/diagnostic-settings-schemas.d.ts +50 -0
- package/lib/azure-diagnosticsettings/lib/diagnostic-settings-schemas.js +207 -0
- package/lib/azure-diagnosticsettings/lib/diagnostic-settings.d.ts +193 -0
- package/lib/azure-diagnosticsettings/lib/diagnostic-settings.js +200 -0
- package/lib/azure-diagnosticsettings/lib/index.d.ts +8 -0
- package/lib/azure-diagnosticsettings/lib/index.js +25 -0
- package/lib/azure-diagnosticsettings/test/diagnostic-settings.spec.d.ts +8 -0
- package/lib/azure-diagnosticsettings/test/diagnostic-settings.spec.js +432 -0
- package/lib/azure-metricalert/index.d.ts +11 -0
- package/lib/azure-metricalert/index.js +28 -0
- package/lib/azure-metricalert/lib/index.d.ts +5 -0
- package/lib/azure-metricalert/lib/index.js +22 -0
- package/lib/azure-metricalert/lib/metric-alert-schemas.d.ts +24 -0
- package/lib/azure-metricalert/lib/metric-alert-schemas.js +242 -0
- package/lib/azure-metricalert/lib/metric-alert.d.ts +344 -0
- package/lib/azure-metricalert/lib/metric-alert.js +252 -0
- package/lib/azure-metricalert/test/metric-alert.integ.d.ts +9 -0
- package/lib/azure-metricalert/test/metric-alert.integ.js +123 -0
- package/lib/azure-metricalert/test/metric-alert.spec.d.ts +9 -0
- package/lib/azure-metricalert/test/metric-alert.spec.js +1266 -0
- package/lib/azure-networkinterface/index.d.ts +4 -0
- package/lib/azure-networkinterface/index.js +21 -0
- package/lib/azure-networkinterface/lib/index.d.ts +6 -0
- package/lib/azure-networkinterface/lib/index.js +22 -0
- package/lib/azure-networkinterface/lib/network-interface-schemas.d.ts +32 -0
- package/lib/azure-networkinterface/lib/network-interface-schemas.js +222 -0
- package/lib/azure-networkinterface/lib/network-interface.d.ts +265 -0
- package/lib/azure-networkinterface/lib/network-interface.js +228 -0
- package/lib/azure-networkinterface/test/network-interface.integ.d.ts +9 -0
- package/lib/azure-networkinterface/test/network-interface.integ.js +102 -0
- package/lib/azure-networkinterface/test/network-interface.spec.d.ts +8 -0
- package/lib/azure-networkinterface/test/network-interface.spec.js +655 -0
- package/lib/azure-networksecuritygroup/index.d.ts +6 -0
- package/lib/azure-networksecuritygroup/index.js +23 -0
- package/lib/azure-networksecuritygroup/lib/index.d.ts +5 -0
- package/lib/azure-networksecuritygroup/lib/index.js +16 -0
- package/lib/azure-networksecuritygroup/lib/network-security-group-schemas.d.ts +32 -0
- package/lib/azure-networksecuritygroup/lib/network-security-group-schemas.js +198 -0
- package/lib/azure-networksecuritygroup/lib/network-security-group.d.ts +268 -0
- package/lib/azure-networksecuritygroup/lib/network-security-group.js +248 -0
- package/lib/azure-networksecuritygroup/test/network-security-group.integ.d.ts +9 -0
- package/lib/azure-networksecuritygroup/test/network-security-group.integ.js +151 -0
- package/lib/azure-networksecuritygroup/test/network-security-group.spec.d.ts +8 -0
- package/lib/azure-networksecuritygroup/test/network-security-group.spec.js +652 -0
- package/lib/azure-publicipaddress/index.d.ts +6 -0
- package/lib/azure-publicipaddress/index.js +23 -0
- package/lib/azure-publicipaddress/lib/index.d.ts +9 -0
- package/lib/azure-publicipaddress/lib/index.js +19 -0
- package/lib/azure-publicipaddress/lib/public-ip-address-schemas.d.ts +32 -0
- package/lib/azure-publicipaddress/lib/public-ip-address-schemas.js +227 -0
- package/lib/azure-publicipaddress/lib/public-ip-address.d.ts +256 -0
- package/lib/azure-publicipaddress/lib/public-ip-address.js +267 -0
- package/lib/azure-publicipaddress/test/public-ip-address.integ.d.ts +9 -0
- package/lib/azure-publicipaddress/test/public-ip-address.integ.js +75 -0
- package/lib/azure-publicipaddress/test/public-ip-address.spec.d.ts +15 -0
- package/lib/azure-publicipaddress/test/public-ip-address.spec.js +498 -0
- package/lib/azure-resourcegroup/lib/resource-group.js +1 -1
- package/lib/azure-resourcegroup/test/resource-group.integ.js +23 -11
- package/lib/azure-storageaccount/lib/storage-account-schemas.d.ts +98 -0
- package/lib/azure-storageaccount/lib/storage-account-schemas.js +1 -1
- package/lib/azure-storageaccount/lib/storage-account.d.ts +38 -1
- package/lib/azure-storageaccount/lib/storage-account.js +138 -2
- package/lib/azure-storageaccount/test/storage-account.integ.js +91 -12
- package/lib/azure-subnet/index.d.ts +4 -0
- package/lib/azure-subnet/index.js +21 -0
- package/lib/azure-subnet/lib/index.d.ts +5 -0
- package/lib/azure-subnet/lib/index.js +22 -0
- package/lib/azure-subnet/lib/subnet-schemas.d.ts +32 -0
- package/lib/azure-subnet/lib/subnet-schemas.js +237 -0
- package/lib/azure-subnet/lib/subnet.d.ts +285 -0
- package/lib/azure-subnet/lib/subnet.js +254 -0
- package/lib/azure-subnet/test/subnet.integ.d.ts +11 -0
- package/lib/azure-subnet/test/subnet.integ.js +80 -0
- package/lib/azure-subnet/test/subnet.spec.d.ts +8 -0
- package/lib/azure-subnet/test/subnet.spec.js +396 -0
- package/lib/azure-virtualmachine/index.d.ts +7 -0
- package/lib/azure-virtualmachine/index.js +24 -0
- package/lib/azure-virtualmachine/lib/index.d.ts +11 -0
- package/lib/azure-virtualmachine/lib/index.js +38 -0
- package/lib/azure-virtualmachine/lib/virtual-machine-schemas.d.ts +327 -0
- package/lib/azure-virtualmachine/lib/virtual-machine-schemas.js +544 -0
- package/lib/azure-virtualmachine/lib/virtual-machine.d.ts +377 -0
- package/lib/azure-virtualmachine/lib/virtual-machine.js +462 -0
- package/lib/azure-virtualmachine/test/virtual-machine.integ.d.ts +9 -0
- package/lib/azure-virtualmachine/test/virtual-machine.integ.js +220 -0
- package/lib/azure-virtualmachine/test/virtual-machine.spec.d.ts +9 -0
- package/lib/azure-virtualmachine/test/virtual-machine.spec.js +1032 -0
- package/lib/azure-virtualnetwork/index.d.ts +10 -0
- package/lib/azure-virtualnetwork/index.js +28 -0
- package/lib/azure-virtualnetwork/lib/index.d.ts +9 -0
- package/lib/azure-virtualnetwork/lib/index.js +22 -0
- package/lib/azure-virtualnetwork/lib/virtual-network-schemas.d.ts +32 -0
- package/lib/azure-virtualnetwork/lib/virtual-network-schemas.js +236 -0
- package/lib/azure-virtualnetwork/lib/virtual-network.d.ts +213 -0
- package/lib/azure-virtualnetwork/lib/virtual-network.js +245 -0
- package/lib/azure-virtualnetwork/test/virtual-network.integ.d.ts +9 -0
- package/lib/azure-virtualnetwork/test/virtual-network.integ.js +91 -0
- package/lib/azure-virtualnetwork/test/virtual-network.spec.d.ts +8 -0
- package/lib/azure-virtualnetwork/test/virtual-network.spec.js +398 -0
- package/lib/azure-vmss/index.d.ts +6 -0
- package/lib/azure-vmss/index.js +23 -0
- package/lib/azure-vmss/lib/index.d.ts +9 -0
- package/lib/azure-vmss/lib/index.js +39 -0
- package/lib/azure-vmss/lib/virtual-machine-scale-set.d.ts +258 -0
- package/lib/azure-vmss/lib/virtual-machine-scale-set.js +503 -0
- package/lib/azure-vmss/lib/vmss-schemas.d.ts +472 -0
- package/lib/azure-vmss/lib/vmss-schemas.js +444 -0
- package/lib/azure-vmss/test/virtual-machine-scale-set.integ.d.ts +15 -0
- package/lib/azure-vmss/test/virtual-machine-scale-set.integ.js +261 -0
- package/lib/azure-vmss/test/virtual-machine-scale-set.spec.d.ts +8 -0
- package/lib/azure-vmss/test/virtual-machine-scale-set.spec.js +997 -0
- package/lib/core-azure/lib/azapi/azapi-resource.d.ts +95 -27
- package/lib/core-azure/lib/azapi/azapi-resource.js +95 -58
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-client-config/index.d.ts +33 -21
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-client-config/index.js +69 -27
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource/index.d.ts +134 -120
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource/index.js +209 -59
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-action/index.d.ts +141 -128
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-action/index.js +183 -36
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-id/index.d.ts +55 -43
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-id/index.js +94 -26
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-list/index.d.ts +129 -116
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-list/index.js +155 -32
- package/lib/core-azure/lib/azapi/providers-azapi/data-plane-resource/index.d.ts +187 -174
- package/lib/core-azure/lib/azapi/providers-azapi/data-plane-resource/index.js +267 -48
- package/lib/core-azure/lib/azapi/providers-azapi/index.d.ts +10 -10
- package/lib/core-azure/lib/azapi/providers-azapi/index.js +1 -1
- package/lib/core-azure/lib/azapi/providers-azapi/lazy-index.d.ts +0 -0
- package/lib/core-azure/lib/azapi/providers-azapi/lazy-index.js +13 -0
- package/lib/core-azure/lib/azapi/providers-azapi/provider/index.d.ts +174 -162
- package/lib/core-azure/lib/azapi/providers-azapi/provider/index.js +274 -14
- package/lib/core-azure/lib/azapi/providers-azapi/resource/index.d.ts +239 -225
- package/lib/core-azure/lib/azapi/providers-azapi/resource/index.js +360 -76
- package/lib/core-azure/lib/azapi/providers-azapi/resource-action/index.d.ts +163 -150
- package/lib/core-azure/lib/azapi/providers-azapi/resource-action/index.js +221 -46
- package/lib/core-azure/lib/azapi/providers-azapi/update-resource/index.d.ts +162 -149
- package/lib/core-azure/lib/azapi/providers-azapi/update-resource/index.js +249 -50
- package/lib/core-azure/lib/azapi/schema-mapper/schema-mapper.js +4 -2
- 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 +37 -0
- package/lib/index.js +40 -2
- package/lib/testing/index.d.ts +92 -2
- package/lib/testing/index.js +258 -18
- package/lib/testing/lib/cleanup.d.ts +229 -0
- package/lib/testing/lib/cleanup.js +331 -0
- package/lib/testing/lib/metadata.d.ts +151 -0
- package/lib/testing/lib/metadata.js +180 -0
- package/lib/testing/lib/naming.d.ts +96 -0
- package/lib/testing/lib/naming.js +336 -0
- package/node_modules/uuid/LICENSE.md +9 -0
- package/node_modules/uuid/README.md +510 -0
- package/node_modules/uuid/dist/cjs/index.d.ts +15 -0
- package/node_modules/uuid/dist/cjs/index.js +31 -0
- package/node_modules/uuid/dist/cjs/max.d.ts +2 -0
- package/node_modules/uuid/dist/cjs/max.js +3 -0
- package/node_modules/uuid/dist/cjs/md5.d.ts +4 -0
- package/node_modules/uuid/dist/cjs/md5.js +13 -0
- package/node_modules/uuid/dist/cjs/native.d.ts +6 -0
- package/node_modules/uuid/dist/cjs/native.js +4 -0
- package/node_modules/uuid/dist/cjs/nil.d.ts +2 -0
- package/node_modules/uuid/dist/cjs/nil.js +3 -0
- package/node_modules/uuid/dist/cjs/package.json +1 -0
- package/node_modules/uuid/dist/cjs/parse.d.ts +2 -0
- package/node_modules/uuid/dist/cjs/parse.js +11 -0
- package/node_modules/uuid/dist/cjs/regex.d.ts +2 -0
- package/node_modules/uuid/dist/cjs/regex.js +3 -0
- package/node_modules/uuid/dist/cjs/rng.d.ts +1 -0
- package/node_modules/uuid/dist/cjs/rng.js +13 -0
- package/node_modules/uuid/dist/cjs/sha1.d.ts +4 -0
- package/node_modules/uuid/dist/cjs/sha1.js +13 -0
- package/node_modules/uuid/dist/cjs/stringify.d.ts +3 -0
- package/node_modules/uuid/dist/cjs/stringify.js +39 -0
- package/node_modules/uuid/dist/cjs/types.d.ts +21 -0
- package/node_modules/uuid/dist/cjs/types.js +2 -0
- package/node_modules/uuid/dist/cjs/uuid-bin.d.ts +1 -0
- package/node_modules/uuid/dist/cjs/uuid-bin.js +72 -0
- package/node_modules/uuid/dist/cjs/v1.d.ts +11 -0
- package/node_modules/uuid/dist/cjs/v1.js +87 -0
- package/node_modules/uuid/dist/cjs/v1ToV6.d.ts +2 -0
- package/node_modules/uuid/dist/cjs/v1ToV6.js +13 -0
- package/node_modules/uuid/dist/cjs/v3.d.ts +9 -0
- package/node_modules/uuid/dist/cjs/v3.js +14 -0
- package/node_modules/uuid/dist/cjs/v35.d.ts +7 -0
- package/node_modules/uuid/dist/cjs/v35.js +41 -0
- package/node_modules/uuid/dist/cjs/v4.d.ts +4 -0
- package/node_modules/uuid/dist/cjs/v4.js +29 -0
- package/node_modules/uuid/dist/cjs/v5.d.ts +9 -0
- package/node_modules/uuid/dist/cjs/v5.js +14 -0
- package/node_modules/uuid/dist/cjs/v6.d.ts +4 -0
- package/node_modules/uuid/dist/cjs/v6.js +19 -0
- package/node_modules/uuid/dist/cjs/v6ToV1.d.ts +2 -0
- package/node_modules/uuid/dist/cjs/v6ToV1.js +13 -0
- package/node_modules/uuid/dist/cjs/v7.d.ts +9 -0
- package/node_modules/uuid/dist/cjs/v7.js +69 -0
- package/node_modules/uuid/dist/cjs/validate.d.ts +2 -0
- package/node_modules/uuid/dist/cjs/validate.js +7 -0
- package/node_modules/uuid/dist/cjs/version.d.ts +2 -0
- package/node_modules/uuid/dist/cjs/version.js +10 -0
- package/node_modules/uuid/dist/cjs-browser/index.d.ts +15 -0
- package/node_modules/uuid/dist/cjs-browser/index.js +31 -0
- package/node_modules/uuid/dist/cjs-browser/max.d.ts +2 -0
- package/node_modules/uuid/dist/cjs-browser/max.js +3 -0
- package/node_modules/uuid/dist/cjs-browser/md5.d.ts +2 -0
- package/node_modules/uuid/dist/cjs-browser/md5.js +137 -0
- package/node_modules/uuid/dist/cjs-browser/native.d.ts +4 -0
- package/node_modules/uuid/dist/cjs-browser/native.js +4 -0
- package/node_modules/uuid/dist/cjs-browser/nil.d.ts +2 -0
- package/node_modules/uuid/dist/cjs-browser/nil.js +3 -0
- package/node_modules/uuid/dist/cjs-browser/package.json +1 -0
- package/node_modules/uuid/dist/cjs-browser/parse.d.ts +2 -0
- package/node_modules/uuid/dist/cjs-browser/parse.js +11 -0
- package/node_modules/uuid/dist/cjs-browser/regex.d.ts +2 -0
- package/node_modules/uuid/dist/cjs-browser/regex.js +3 -0
- package/node_modules/uuid/dist/cjs-browser/rng.d.ts +1 -0
- package/node_modules/uuid/dist/cjs-browser/rng.js +14 -0
- package/node_modules/uuid/dist/cjs-browser/sha1.d.ts +2 -0
- package/node_modules/uuid/dist/cjs-browser/sha1.js +72 -0
- package/node_modules/uuid/dist/cjs-browser/stringify.d.ts +3 -0
- package/node_modules/uuid/dist/cjs-browser/stringify.js +39 -0
- package/node_modules/uuid/dist/cjs-browser/types.d.ts +21 -0
- package/node_modules/uuid/dist/cjs-browser/types.js +2 -0
- package/node_modules/uuid/dist/cjs-browser/uuid-bin.d.ts +1 -0
- package/node_modules/uuid/dist/cjs-browser/uuid-bin.js +72 -0
- package/node_modules/uuid/dist/cjs-browser/v1.d.ts +11 -0
- package/node_modules/uuid/dist/cjs-browser/v1.js +87 -0
- package/node_modules/uuid/dist/cjs-browser/v1ToV6.d.ts +2 -0
- package/node_modules/uuid/dist/cjs-browser/v1ToV6.js +13 -0
- package/node_modules/uuid/dist/cjs-browser/v3.d.ts +9 -0
- package/node_modules/uuid/dist/cjs-browser/v3.js +14 -0
- package/node_modules/uuid/dist/cjs-browser/v35.d.ts +7 -0
- package/node_modules/uuid/dist/cjs-browser/v35.js +41 -0
- package/node_modules/uuid/dist/cjs-browser/v4.d.ts +4 -0
- package/node_modules/uuid/dist/cjs-browser/v4.js +29 -0
- package/node_modules/uuid/dist/cjs-browser/v5.d.ts +9 -0
- package/node_modules/uuid/dist/cjs-browser/v5.js +14 -0
- package/node_modules/uuid/dist/cjs-browser/v6.d.ts +4 -0
- package/node_modules/uuid/dist/cjs-browser/v6.js +19 -0
- package/node_modules/uuid/dist/cjs-browser/v6ToV1.d.ts +2 -0
- package/node_modules/uuid/dist/cjs-browser/v6ToV1.js +13 -0
- package/node_modules/uuid/dist/cjs-browser/v7.d.ts +9 -0
- package/node_modules/uuid/dist/cjs-browser/v7.js +69 -0
- package/node_modules/uuid/dist/cjs-browser/validate.d.ts +2 -0
- package/node_modules/uuid/dist/cjs-browser/validate.js +7 -0
- package/node_modules/uuid/dist/cjs-browser/version.d.ts +2 -0
- package/node_modules/uuid/dist/cjs-browser/version.js +10 -0
- package/node_modules/uuid/dist/esm/bin/uuid +2 -0
- package/node_modules/uuid/dist/esm/index.d.ts +15 -0
- package/node_modules/uuid/dist/esm/index.js +14 -0
- package/node_modules/uuid/dist/esm/max.d.ts +2 -0
- package/node_modules/uuid/dist/esm/max.js +1 -0
- package/node_modules/uuid/dist/esm/md5.d.ts +4 -0
- package/node_modules/uuid/dist/esm/md5.js +11 -0
- package/node_modules/uuid/dist/esm/native.d.ts +6 -0
- package/node_modules/uuid/dist/esm/native.js +2 -0
- package/node_modules/uuid/dist/esm/nil.d.ts +2 -0
- package/node_modules/uuid/dist/esm/nil.js +1 -0
- package/node_modules/uuid/dist/esm/parse.d.ts +2 -0
- package/node_modules/uuid/dist/esm/parse.js +9 -0
- package/node_modules/uuid/dist/esm/regex.d.ts +2 -0
- package/node_modules/uuid/dist/esm/regex.js +1 -0
- package/node_modules/uuid/dist/esm/rng.d.ts +1 -0
- package/node_modules/uuid/dist/esm/rng.js +10 -0
- package/node_modules/uuid/dist/esm/sha1.d.ts +4 -0
- package/node_modules/uuid/dist/esm/sha1.js +11 -0
- package/node_modules/uuid/dist/esm/stringify.d.ts +3 -0
- package/node_modules/uuid/dist/esm/stringify.js +35 -0
- package/node_modules/uuid/dist/esm/types.d.ts +21 -0
- package/node_modules/uuid/dist/esm/types.js +1 -0
- package/node_modules/uuid/dist/esm/uuid-bin.d.ts +1 -0
- package/node_modules/uuid/dist/esm/uuid-bin.js +70 -0
- package/node_modules/uuid/dist/esm/v1.d.ts +11 -0
- package/node_modules/uuid/dist/esm/v1.js +83 -0
- package/node_modules/uuid/dist/esm/v1ToV6.d.ts +2 -0
- package/node_modules/uuid/dist/esm/v1ToV6.js +10 -0
- package/node_modules/uuid/dist/esm/v3.d.ts +9 -0
- package/node_modules/uuid/dist/esm/v3.js +9 -0
- package/node_modules/uuid/dist/esm/v35.d.ts +7 -0
- package/node_modules/uuid/dist/esm/v35.js +36 -0
- package/node_modules/uuid/dist/esm/v4.d.ts +4 -0
- package/node_modules/uuid/dist/esm/v4.js +27 -0
- package/node_modules/uuid/dist/esm/v5.d.ts +9 -0
- package/node_modules/uuid/dist/esm/v5.js +9 -0
- package/node_modules/uuid/dist/esm/v6.d.ts +4 -0
- package/node_modules/uuid/dist/esm/v6.js +17 -0
- package/node_modules/uuid/dist/esm/v6ToV1.d.ts +2 -0
- package/node_modules/uuid/dist/esm/v6ToV1.js +10 -0
- package/node_modules/uuid/dist/esm/v7.d.ts +9 -0
- package/node_modules/uuid/dist/esm/v7.js +65 -0
- package/node_modules/uuid/dist/esm/validate.d.ts +2 -0
- package/node_modules/uuid/dist/esm/validate.js +5 -0
- package/node_modules/uuid/dist/esm/version.d.ts +2 -0
- package/node_modules/uuid/dist/esm/version.js +8 -0
- package/node_modules/uuid/dist/esm-browser/index.d.ts +15 -0
- package/node_modules/uuid/dist/esm-browser/index.js +14 -0
- package/node_modules/uuid/dist/esm-browser/max.d.ts +2 -0
- package/node_modules/uuid/dist/esm-browser/max.js +1 -0
- package/node_modules/uuid/dist/esm-browser/md5.d.ts +2 -0
- package/node_modules/uuid/dist/esm-browser/md5.js +135 -0
- package/node_modules/uuid/dist/esm-browser/native.d.ts +4 -0
- package/node_modules/uuid/dist/esm-browser/native.js +2 -0
- package/node_modules/uuid/dist/esm-browser/nil.d.ts +2 -0
- package/node_modules/uuid/dist/esm-browser/nil.js +1 -0
- package/node_modules/uuid/dist/esm-browser/parse.d.ts +2 -0
- package/node_modules/uuid/dist/esm-browser/parse.js +9 -0
- package/node_modules/uuid/dist/esm-browser/regex.d.ts +2 -0
- package/node_modules/uuid/dist/esm-browser/regex.js +1 -0
- package/node_modules/uuid/dist/esm-browser/rng.d.ts +1 -0
- package/node_modules/uuid/dist/esm-browser/rng.js +11 -0
- package/node_modules/uuid/dist/esm-browser/sha1.d.ts +2 -0
- package/node_modules/uuid/dist/esm-browser/sha1.js +70 -0
- package/node_modules/uuid/dist/esm-browser/stringify.d.ts +3 -0
- package/node_modules/uuid/dist/esm-browser/stringify.js +35 -0
- package/node_modules/uuid/dist/esm-browser/types.d.ts +21 -0
- package/node_modules/uuid/dist/esm-browser/types.js +1 -0
- package/node_modules/uuid/dist/esm-browser/uuid-bin.d.ts +1 -0
- package/node_modules/uuid/dist/esm-browser/uuid-bin.js +70 -0
- package/node_modules/uuid/dist/esm-browser/v1.d.ts +11 -0
- package/node_modules/uuid/dist/esm-browser/v1.js +83 -0
- package/node_modules/uuid/dist/esm-browser/v1ToV6.d.ts +2 -0
- package/node_modules/uuid/dist/esm-browser/v1ToV6.js +10 -0
- package/node_modules/uuid/dist/esm-browser/v3.d.ts +9 -0
- package/node_modules/uuid/dist/esm-browser/v3.js +9 -0
- package/node_modules/uuid/dist/esm-browser/v35.d.ts +7 -0
- package/node_modules/uuid/dist/esm-browser/v35.js +36 -0
- package/node_modules/uuid/dist/esm-browser/v4.d.ts +4 -0
- package/node_modules/uuid/dist/esm-browser/v4.js +27 -0
- package/node_modules/uuid/dist/esm-browser/v5.d.ts +9 -0
- package/node_modules/uuid/dist/esm-browser/v5.js +9 -0
- package/node_modules/uuid/dist/esm-browser/v6.d.ts +4 -0
- package/node_modules/uuid/dist/esm-browser/v6.js +17 -0
- package/node_modules/uuid/dist/esm-browser/v6ToV1.d.ts +2 -0
- package/node_modules/uuid/dist/esm-browser/v6ToV1.js +10 -0
- package/node_modules/uuid/dist/esm-browser/v7.d.ts +9 -0
- package/node_modules/uuid/dist/esm-browser/v7.js +65 -0
- package/node_modules/uuid/dist/esm-browser/validate.d.ts +2 -0
- package/node_modules/uuid/dist/esm-browser/validate.js +5 -0
- package/node_modules/uuid/dist/esm-browser/version.d.ts +2 -0
- package/node_modules/uuid/dist/esm-browser/version.js +8 -0
- package/node_modules/uuid/package.json +132 -0
- package/package.json +17 -12
- package/scripts/cleanup-test-resources.ts +369 -0
package/lib/testing/index.d.ts
CHANGED
|
@@ -1,9 +1,72 @@
|
|
|
1
1
|
import { ExecSyncOptionsWithStringEncoding } from "child_process";
|
|
2
2
|
import { TerraformStack } from "cdktf";
|
|
3
3
|
import { Construct } from "constructs";
|
|
4
|
+
import { ResourceMetadata } from "./lib/cleanup";
|
|
5
|
+
import { TestRunMetadata, TestRunOptions } from "./lib/metadata";
|
|
6
|
+
export { TestRunOptions, TestRunMetadata, CIContext } from "./lib/metadata";
|
|
7
|
+
/**
|
|
8
|
+
* Options for BaseTestStack constructor
|
|
9
|
+
*/
|
|
10
|
+
export interface BaseTestStackOptions {
|
|
11
|
+
/**
|
|
12
|
+
* Test run configuration options
|
|
13
|
+
*/
|
|
14
|
+
readonly testRunOptions?: TestRunOptions;
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Base stack for integration tests with optional metadata support
|
|
18
|
+
*
|
|
19
|
+
* When metadata is enabled, automatically generates unique test run IDs,
|
|
20
|
+
* injects system tags, and provides utilities for resource naming.
|
|
21
|
+
*/
|
|
4
22
|
export declare class BaseTestStack extends TerraformStack {
|
|
5
23
|
readonly name: string;
|
|
6
|
-
|
|
24
|
+
/**
|
|
25
|
+
* Optional test run metadata (only present when using BaseTestStackOptions)
|
|
26
|
+
*/
|
|
27
|
+
readonly metadata?: TestRunMetadata;
|
|
28
|
+
/**
|
|
29
|
+
* Registry of resources created in this stack for tracking and cleanup verification
|
|
30
|
+
*/
|
|
31
|
+
private readonly resourceRegistry;
|
|
32
|
+
constructor(scope: Construct, id: string, options?: BaseTestStackOptions);
|
|
33
|
+
/**
|
|
34
|
+
* Registers a resource for tracking and cleanup verification
|
|
35
|
+
*
|
|
36
|
+
* @param resourceId - Azure resource ID
|
|
37
|
+
* @param resourceType - Azure resource type
|
|
38
|
+
* @param name - Resource name
|
|
39
|
+
* @param location - Optional resource location
|
|
40
|
+
* @param tags - Optional resource tags
|
|
41
|
+
*/
|
|
42
|
+
registerResource(resourceId: string, resourceType: string, name: string, location?: string, tags?: Record<string, string>): void;
|
|
43
|
+
/**
|
|
44
|
+
* Retrieves all registered resources
|
|
45
|
+
*
|
|
46
|
+
* @returns Array of registered resource metadata
|
|
47
|
+
*/
|
|
48
|
+
registeredResources(): ResourceMetadata[];
|
|
49
|
+
/**
|
|
50
|
+
* Generates system tags for resources (only available when metadata is present)
|
|
51
|
+
*
|
|
52
|
+
* @returns Integration test system tags
|
|
53
|
+
* @throws Error if metadata is not initialized
|
|
54
|
+
*/
|
|
55
|
+
systemTags(): Record<string, string>;
|
|
56
|
+
/**
|
|
57
|
+
* Generates a unique resource name with proper Azure compliance
|
|
58
|
+
*
|
|
59
|
+
* @param resourceType - Azure resource type (e.g., 'Microsoft.Resources/resourceGroups')
|
|
60
|
+
* @param customIdentifier - Optional custom identifier (defaults to test name)
|
|
61
|
+
* @returns Unique, Azure-compliant resource name
|
|
62
|
+
* @throws Error if metadata is not initialized
|
|
63
|
+
*
|
|
64
|
+
* @example
|
|
65
|
+
* const stack = new BaseTestStack(app, 'test-storage', { testRunOptions: {} });
|
|
66
|
+
* const rgName = stack.generateResourceName('Microsoft.Resources/resourceGroups');
|
|
67
|
+
* // Returns: 'rg-test-storage-a1b2c3'
|
|
68
|
+
*/
|
|
69
|
+
generateResourceName(resourceType: string, customIdentifier?: string): string;
|
|
7
70
|
}
|
|
8
71
|
export declare function execTerraformCommand(command: string, opts: ExecSyncOptionsWithStringEncoding, streamOutput: boolean): {
|
|
9
72
|
stdout: string;
|
|
@@ -24,7 +87,23 @@ export declare function TerraformPlanExitCode(stack: any, streamOutput?: boolean
|
|
|
24
87
|
export declare function TerraformIdempotentCheck(stack: any): AssertionReturn;
|
|
25
88
|
export declare function TerraformDestroy(stack: any, streamOutput?: boolean): AssertionReturn;
|
|
26
89
|
export declare function TerraformApplyAndCheckIdempotency(stack: any, streamOutput?: boolean): AssertionReturn;
|
|
27
|
-
export declare function TerraformApplyCheckAndDestroy(stack: any
|
|
90
|
+
export declare function TerraformApplyCheckAndDestroy(stack: any, options?: {
|
|
91
|
+
verifyCleanup?: boolean;
|
|
92
|
+
}): void;
|
|
93
|
+
/**
|
|
94
|
+
* Enhanced terraform destroy with cleanup verification
|
|
95
|
+
*
|
|
96
|
+
* Performs terraform destroy and verifies resource cleanup via
|
|
97
|
+
* Azure Resource Graph queries. Implements retry logic for
|
|
98
|
+
* eventual consistency.
|
|
99
|
+
*
|
|
100
|
+
* @param stack - Terraform stack directory path
|
|
101
|
+
* @param metadata - Test run metadata
|
|
102
|
+
* @param resources - List of resources to verify deletion
|
|
103
|
+
* @param streamOutput - Whether to stream command output
|
|
104
|
+
* @returns Assertion result
|
|
105
|
+
*/
|
|
106
|
+
export declare function TerraformDestroyWithVerification(stack: any, metadata: TestRunMetadata, resources: ResourceMetadata[], streamOutput?: boolean): AssertionReturn;
|
|
28
107
|
export declare function TerraformPlan(stack: any): AssertionReturn;
|
|
29
108
|
export declare class AssertionReturn {
|
|
30
109
|
readonly message: string;
|
|
@@ -36,4 +115,15 @@ export declare class AssertionReturn {
|
|
|
36
115
|
*/
|
|
37
116
|
constructor(message: string, success: boolean);
|
|
38
117
|
}
|
|
118
|
+
/**
|
|
119
|
+
* Cleanup function for removing all CDKTF output directories in /tmp.
|
|
120
|
+
*
|
|
121
|
+
* WARNING: This function should NOT be used in afterAll() hooks when running
|
|
122
|
+
* tests in parallel, as it will delete directories that other running tests
|
|
123
|
+
* may still be using. Instead, TerraformApplyCheckAndDestroy now cleans up
|
|
124
|
+
* its own directory automatically.
|
|
125
|
+
*
|
|
126
|
+
* This function is kept for manual cleanup or single-threaded test scenarios.
|
|
127
|
+
*/
|
|
39
128
|
export declare function cleanupCdkTfOutDirs(): void;
|
|
129
|
+
export { ResourceMetadata, VerificationResult, CleanupOptions, OrphanedResource, CleanupResult, ResourceCleanupService, verifyResourcesDeleted, verifyResourcesDeletedWithRetry, findOrphanedResources, cleanupOrphanedResources, } from "./lib/cleanup";
|
package/lib/testing/index.js
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
var _a, _b;
|
|
3
3
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
-
exports.AssertionReturn = exports.BaseTestStack = void 0;
|
|
4
|
+
exports.cleanupOrphanedResources = exports.findOrphanedResources = exports.verifyResourcesDeletedWithRetry = exports.verifyResourcesDeleted = exports.ResourceCleanupService = exports.AssertionReturn = exports.BaseTestStack = exports.TestRunMetadata = void 0;
|
|
5
5
|
exports.execTerraformCommand = execTerraformCommand;
|
|
6
6
|
exports.TerraformApply = TerraformApply;
|
|
7
7
|
exports.TerraformPlanExitCode = TerraformPlanExitCode;
|
|
@@ -9,6 +9,7 @@ exports.TerraformIdempotentCheck = TerraformIdempotentCheck;
|
|
|
9
9
|
exports.TerraformDestroy = TerraformDestroy;
|
|
10
10
|
exports.TerraformApplyAndCheckIdempotency = TerraformApplyAndCheckIdempotency;
|
|
11
11
|
exports.TerraformApplyCheckAndDestroy = TerraformApplyCheckAndDestroy;
|
|
12
|
+
exports.TerraformDestroyWithVerification = TerraformDestroyWithVerification;
|
|
12
13
|
exports.TerraformPlan = TerraformPlan;
|
|
13
14
|
exports.cleanupCdkTfOutDirs = cleanupCdkTfOutDirs;
|
|
14
15
|
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
@@ -17,20 +18,121 @@ const fs = require("fs");
|
|
|
17
18
|
const path = require("path");
|
|
18
19
|
const cdktf = require("cdktf");
|
|
19
20
|
const cdktf_1 = require("cdktf");
|
|
21
|
+
const cleanup_1 = require("./lib/cleanup");
|
|
22
|
+
const metadata_1 = require("./lib/metadata");
|
|
23
|
+
const naming_1 = require("./lib/naming");
|
|
24
|
+
// Re-export types used in public APIs for jsii compliance
|
|
25
|
+
var metadata_2 = require("./lib/metadata");
|
|
26
|
+
Object.defineProperty(exports, "TestRunMetadata", { enumerable: true, get: function () { return metadata_2.TestRunMetadata; } });
|
|
27
|
+
/**
|
|
28
|
+
* Base stack for integration tests with optional metadata support
|
|
29
|
+
*
|
|
30
|
+
* When metadata is enabled, automatically generates unique test run IDs,
|
|
31
|
+
* injects system tags, and provides utilities for resource naming.
|
|
32
|
+
*/
|
|
20
33
|
class BaseTestStack extends cdktf_1.TerraformStack {
|
|
21
|
-
constructor(scope, id) {
|
|
34
|
+
constructor(scope, id, options) {
|
|
22
35
|
super(scope, id);
|
|
36
|
+
/**
|
|
37
|
+
* Registry of resources created in this stack for tracking and cleanup verification
|
|
38
|
+
*/
|
|
39
|
+
this.resourceRegistry = new Map();
|
|
40
|
+
// Generate metadata if options are provided
|
|
41
|
+
if (options?.testRunOptions) {
|
|
42
|
+
this.metadata = new metadata_1.TestRunMetadata(id, options.testRunOptions);
|
|
43
|
+
// Log test run information
|
|
44
|
+
console.log(`[Test Run] ID: ${this.metadata.runId}`);
|
|
45
|
+
console.log(`[Test Run] Name: ${this.metadata.testName}`);
|
|
46
|
+
console.log(`[Test Run] Created: ${this.metadata.createdAt.toISOString()}`);
|
|
47
|
+
console.log(`[Test Run] Cleanup After: ${this.metadata.cleanupAfter.toISOString()}`);
|
|
48
|
+
if (this.metadata.ciContext) {
|
|
49
|
+
console.log(`[Test Run] CI Platform: ${this.metadata.ciContext.platform}`);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
23
52
|
const name = new cdktf.TerraformVariable(this, "name", {
|
|
24
53
|
type: "string",
|
|
25
|
-
default: "test",
|
|
54
|
+
default: this.metadata?.testName || "test",
|
|
26
55
|
description: "System name used to randomize the resources",
|
|
27
56
|
});
|
|
28
57
|
this.name = name.value;
|
|
29
58
|
}
|
|
59
|
+
/**
|
|
60
|
+
* Registers a resource for tracking and cleanup verification
|
|
61
|
+
*
|
|
62
|
+
* @param resourceId - Azure resource ID
|
|
63
|
+
* @param resourceType - Azure resource type
|
|
64
|
+
* @param name - Resource name
|
|
65
|
+
* @param location - Optional resource location
|
|
66
|
+
* @param tags - Optional resource tags
|
|
67
|
+
*/
|
|
68
|
+
registerResource(resourceId, resourceType, name, location, tags) {
|
|
69
|
+
if (!this.metadata) {
|
|
70
|
+
console.warn("[BaseTestStack] Resource registration requires metadata. " +
|
|
71
|
+
"Initialize with testRunOptions to enable resource tracking.");
|
|
72
|
+
return;
|
|
73
|
+
}
|
|
74
|
+
const metadata = {
|
|
75
|
+
resourceId,
|
|
76
|
+
resourceType,
|
|
77
|
+
name,
|
|
78
|
+
location,
|
|
79
|
+
tags: tags || {},
|
|
80
|
+
testRunId: this.metadata.runId,
|
|
81
|
+
createdAt: this.metadata.createdAt,
|
|
82
|
+
};
|
|
83
|
+
this.resourceRegistry.set(resourceId, metadata);
|
|
84
|
+
console.log(`[BaseTestStack] Registered resource: ${name} (${resourceType})`);
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Retrieves all registered resources
|
|
88
|
+
*
|
|
89
|
+
* @returns Array of registered resource metadata
|
|
90
|
+
*/
|
|
91
|
+
registeredResources() {
|
|
92
|
+
return Array.from(this.resourceRegistry.values());
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Generates system tags for resources (only available when metadata is present)
|
|
96
|
+
*
|
|
97
|
+
* @returns Integration test system tags
|
|
98
|
+
* @throws Error if metadata is not initialized
|
|
99
|
+
*/
|
|
100
|
+
systemTags() {
|
|
101
|
+
if (!this.metadata) {
|
|
102
|
+
throw new Error("System tags are only available when BaseTestStack is initialized with testRunOptions. " +
|
|
103
|
+
"Pass { testRunOptions: {} } to the constructor to enable metadata.");
|
|
104
|
+
}
|
|
105
|
+
return this.metadata.generateSystemTags();
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Generates a unique resource name with proper Azure compliance
|
|
109
|
+
*
|
|
110
|
+
* @param resourceType - Azure resource type (e.g., 'Microsoft.Resources/resourceGroups')
|
|
111
|
+
* @param customIdentifier - Optional custom identifier (defaults to test name)
|
|
112
|
+
* @returns Unique, Azure-compliant resource name
|
|
113
|
+
* @throws Error if metadata is not initialized
|
|
114
|
+
*
|
|
115
|
+
* @example
|
|
116
|
+
* const stack = new BaseTestStack(app, 'test-storage', { testRunOptions: {} });
|
|
117
|
+
* const rgName = stack.generateResourceName('Microsoft.Resources/resourceGroups');
|
|
118
|
+
* // Returns: 'rg-test-storage-a1b2c3'
|
|
119
|
+
*/
|
|
120
|
+
generateResourceName(resourceType, customIdentifier) {
|
|
121
|
+
if (!this.metadata) {
|
|
122
|
+
throw new Error("Resource name generation is only available when BaseTestStack is initialized with testRunOptions. " +
|
|
123
|
+
"Pass { testRunOptions: {} } to the constructor to enable metadata.");
|
|
124
|
+
}
|
|
125
|
+
return (0, naming_1.generateResourceName)({
|
|
126
|
+
resourceType,
|
|
127
|
+
testName: this.metadata.testName,
|
|
128
|
+
runId: this.metadata.runId,
|
|
129
|
+
customIdentifier,
|
|
130
|
+
});
|
|
131
|
+
}
|
|
30
132
|
}
|
|
31
133
|
exports.BaseTestStack = BaseTestStack;
|
|
32
134
|
_a = JSII_RTTI_SYMBOL_1;
|
|
33
|
-
BaseTestStack[_a] = { fqn: "@microsoft/terraform-cdk-constructs.
|
|
135
|
+
BaseTestStack[_a] = { fqn: "@microsoft/terraform-cdk-constructs.BaseTestStack", version: "1.2.0" };
|
|
34
136
|
function execTerraformCommand(command, opts, streamOutput) {
|
|
35
137
|
try {
|
|
36
138
|
if (streamOutput) {
|
|
@@ -50,7 +152,13 @@ function execTerraformCommand(command, opts, streamOutput) {
|
|
|
50
152
|
}
|
|
51
153
|
function TerraformApply(stack, streamOutput = false) {
|
|
52
154
|
try {
|
|
53
|
-
const
|
|
155
|
+
const manifestPath = path.resolve(stack, "manifest.json");
|
|
156
|
+
// Safety check: verify manifest exists before reading
|
|
157
|
+
if (!fs.existsSync(manifestPath)) {
|
|
158
|
+
throw new Error(`Manifest file not found at ${manifestPath}. ` +
|
|
159
|
+
`Ensure Testing.fullSynth() completed successfully and the output directory still exists.`);
|
|
160
|
+
}
|
|
161
|
+
const manifest = JSON.parse(fs.readFileSync(manifestPath, "utf8"));
|
|
54
162
|
const stacks = Object.entries(manifest.stacks);
|
|
55
163
|
for (const [, stackDetails] of stacks) {
|
|
56
164
|
const opts = {
|
|
@@ -79,7 +187,13 @@ function TerraformApply(stack, streamOutput = false) {
|
|
|
79
187
|
}
|
|
80
188
|
function TerraformPlanExitCode(stack, streamOutput = false) {
|
|
81
189
|
try {
|
|
82
|
-
const
|
|
190
|
+
const manifestPath = path.resolve(stack, "manifest.json");
|
|
191
|
+
// Safety check: verify manifest exists before reading
|
|
192
|
+
if (!fs.existsSync(manifestPath)) {
|
|
193
|
+
throw new Error(`Manifest file not found at ${manifestPath}. ` +
|
|
194
|
+
`Ensure Testing.fullSynth() completed successfully and the output directory still exists.`);
|
|
195
|
+
}
|
|
196
|
+
const manifest = JSON.parse(fs.readFileSync(manifestPath, "utf8"));
|
|
83
197
|
const stacks = Object.entries(manifest.stacks);
|
|
84
198
|
for (const [, stackDetails] of stacks) {
|
|
85
199
|
const opts = {
|
|
@@ -112,7 +226,13 @@ function TerraformPlanExitCode(stack, streamOutput = false) {
|
|
|
112
226
|
}
|
|
113
227
|
function TerraformIdempotentCheck(stack) {
|
|
114
228
|
try {
|
|
115
|
-
const
|
|
229
|
+
const manifestPath = path.resolve(stack, "manifest.json");
|
|
230
|
+
// Safety check: verify manifest exists before reading
|
|
231
|
+
if (!fs.existsSync(manifestPath)) {
|
|
232
|
+
throw new Error(`Manifest file not found at ${manifestPath}. ` +
|
|
233
|
+
`Ensure Testing.fullSynth() completed successfully and the output directory still exists.`);
|
|
234
|
+
}
|
|
235
|
+
const manifest = JSON.parse(fs.readFileSync(manifestPath, "utf8"));
|
|
116
236
|
const stacks = Object.entries(manifest.stacks);
|
|
117
237
|
for (const [, stackDetails] of stacks) {
|
|
118
238
|
const opts = {
|
|
@@ -144,7 +264,13 @@ function TerraformIdempotentCheck(stack) {
|
|
|
144
264
|
}
|
|
145
265
|
function TerraformDestroy(stack, streamOutput = false) {
|
|
146
266
|
try {
|
|
147
|
-
const
|
|
267
|
+
const manifestPath = path.resolve(stack, "manifest.json");
|
|
268
|
+
// Safety check: verify manifest exists before reading
|
|
269
|
+
if (!fs.existsSync(manifestPath)) {
|
|
270
|
+
throw new Error(`Manifest file not found at ${manifestPath}. ` +
|
|
271
|
+
`Ensure Testing.fullSynth() completed successfully and the output directory still exists.`);
|
|
272
|
+
}
|
|
273
|
+
const manifest = JSON.parse(fs.readFileSync(manifestPath, "utf8"));
|
|
148
274
|
const stacks = Object.entries(manifest.stacks);
|
|
149
275
|
for (const [, stackDetails] of stacks) {
|
|
150
276
|
const opts = {
|
|
@@ -180,9 +306,14 @@ function TerraformApplyAndCheckIdempotency(stack, streamOutput = false) {
|
|
|
180
306
|
}
|
|
181
307
|
return new AssertionReturn(`Terraform apply and idempotency check completed successfully`, true);
|
|
182
308
|
}
|
|
183
|
-
function TerraformApplyCheckAndDestroy(stack) {
|
|
309
|
+
function TerraformApplyCheckAndDestroy(stack, options) {
|
|
310
|
+
const streamOutput = process.env.STREAM_OUTPUT === "true";
|
|
311
|
+
const verifyCleanup = options?.verifyCleanup ?? false;
|
|
312
|
+
// Extract metadata and resources if available
|
|
313
|
+
const metadata = stack.metadata;
|
|
314
|
+
const hasVerification = verifyCleanup && metadata;
|
|
184
315
|
try {
|
|
185
|
-
const applyAndCheckResult = TerraformApplyAndCheckIdempotency(stack);
|
|
316
|
+
const applyAndCheckResult = TerraformApplyAndCheckIdempotency(stack, streamOutput);
|
|
186
317
|
if (!applyAndCheckResult.success) {
|
|
187
318
|
throw new Error(applyAndCheckResult.message);
|
|
188
319
|
}
|
|
@@ -193,20 +324,113 @@ function TerraformApplyCheckAndDestroy(stack) {
|
|
|
193
324
|
}
|
|
194
325
|
finally {
|
|
195
326
|
try {
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
327
|
+
if (hasVerification) {
|
|
328
|
+
// Use enhanced destroy with verification
|
|
329
|
+
const resources = typeof stack.registeredResources === "function"
|
|
330
|
+
? stack.registeredResources()
|
|
331
|
+
: [];
|
|
332
|
+
const destroyResult = TerraformDestroyWithVerification(stack, metadata, resources, streamOutput);
|
|
333
|
+
if (!destroyResult.success) {
|
|
334
|
+
throw new Error(`Destroy verification failed: ${destroyResult.message}`);
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
else {
|
|
338
|
+
// Use standard destroy (backward compatible)
|
|
339
|
+
const destroyResult = TerraformDestroy(stack, streamOutput);
|
|
340
|
+
if (!destroyResult.success) {
|
|
341
|
+
console.error("Error during Terraform destroy:", destroyResult.message);
|
|
342
|
+
}
|
|
199
343
|
}
|
|
200
344
|
}
|
|
201
345
|
catch (destroyError) {
|
|
202
346
|
console.error("Error during Terraform destroy:", destroyError);
|
|
347
|
+
throw destroyError; // Re-throw to fail the test
|
|
348
|
+
}
|
|
349
|
+
finally {
|
|
350
|
+
// Clean up this specific test's output directory
|
|
351
|
+
try {
|
|
352
|
+
if (fs.existsSync(stack)) {
|
|
353
|
+
(0, child_process_1.execSync)(`rm -rf "${stack}"`);
|
|
354
|
+
}
|
|
355
|
+
}
|
|
356
|
+
catch (cleanupError) {
|
|
357
|
+
// Ignore cleanup errors - directory may already be gone
|
|
358
|
+
}
|
|
203
359
|
}
|
|
204
360
|
}
|
|
205
361
|
}
|
|
362
|
+
/**
|
|
363
|
+
* Enhanced terraform destroy with cleanup verification
|
|
364
|
+
*
|
|
365
|
+
* Performs terraform destroy and verifies resource cleanup via
|
|
366
|
+
* Azure Resource Graph queries. Implements retry logic for
|
|
367
|
+
* eventual consistency.
|
|
368
|
+
*
|
|
369
|
+
* @param stack - Terraform stack directory path
|
|
370
|
+
* @param metadata - Test run metadata
|
|
371
|
+
* @param resources - List of resources to verify deletion
|
|
372
|
+
* @param streamOutput - Whether to stream command output
|
|
373
|
+
* @returns Assertion result
|
|
374
|
+
*/
|
|
375
|
+
function TerraformDestroyWithVerification(stack, metadata, resources, streamOutput = false) {
|
|
376
|
+
try {
|
|
377
|
+
console.log("[Destroy] Starting terraform destroy...");
|
|
378
|
+
// Step 1: Run terraform destroy
|
|
379
|
+
const destroyResult = TerraformDestroy(stack, streamOutput);
|
|
380
|
+
if (!destroyResult.success) {
|
|
381
|
+
throw new Error(`Terraform destroy failed: ${destroyResult.message}`);
|
|
382
|
+
}
|
|
383
|
+
console.log("[Destroy] Terraform destroy completed successfully");
|
|
384
|
+
// Step 2: If no resources to verify, skip verification
|
|
385
|
+
if (resources.length === 0) {
|
|
386
|
+
console.log("[Destroy] No resources to verify (none registered)");
|
|
387
|
+
return new AssertionReturn("Terraform destroy completed (no verification needed)", true);
|
|
388
|
+
}
|
|
389
|
+
console.log(`[Destroy] Verifying cleanup of ${resources.length} resources...`);
|
|
390
|
+
console.log("[Destroy] Waiting 30s for Azure eventual consistency...");
|
|
391
|
+
// Step 3: Wait for eventual consistency
|
|
392
|
+
const sleep = (ms) => {
|
|
393
|
+
(0, child_process_1.execSync)(`sleep ${ms / 1000}`, { stdio: "ignore" });
|
|
394
|
+
};
|
|
395
|
+
sleep(30000);
|
|
396
|
+
// Step 4: Verify resources are deleted with retry logic
|
|
397
|
+
console.log("[Destroy] Starting verification...");
|
|
398
|
+
const verificationResult = (0, cleanup_1.verifyResourcesDeleted)(metadata.runId, resources);
|
|
399
|
+
if (!verificationResult.success) {
|
|
400
|
+
// Step 5: Retry verification after additional wait
|
|
401
|
+
console.warn("[Destroy] Initial verification failed, retrying...");
|
|
402
|
+
sleep(30000);
|
|
403
|
+
const retryResult = (0, cleanup_1.verifyResourcesDeleted)(metadata.runId, resources);
|
|
404
|
+
if (!retryResult.success) {
|
|
405
|
+
const errorMessage = `Resource cleanup verification failed:\n` +
|
|
406
|
+
` Expected: ${retryResult.expectedCount} resources\n` +
|
|
407
|
+
` Found: ${retryResult.foundCount} orphaned resources\n` +
|
|
408
|
+
` Orphaned IDs:\n` +
|
|
409
|
+
retryResult.orphanedResources.map((id) => ` - ${id}`).join("\n");
|
|
410
|
+
console.error("[Destroy] ✗ Verification failed:", errorMessage);
|
|
411
|
+
throw new Error(errorMessage);
|
|
412
|
+
}
|
|
413
|
+
console.log("[Destroy] ✓ Verification succeeded on retry");
|
|
414
|
+
}
|
|
415
|
+
else {
|
|
416
|
+
console.log("[Destroy] ✓ All resources verified deleted");
|
|
417
|
+
}
|
|
418
|
+
return new AssertionReturn("Terraform destroy and verification completed successfully", true);
|
|
419
|
+
}
|
|
420
|
+
catch (error) {
|
|
421
|
+
console.error("[Destroy] Error during Terraform destroy and verification:", error);
|
|
422
|
+
return new AssertionReturn(`Terraform destroy and verification failed: ${error.message}`, false);
|
|
423
|
+
}
|
|
424
|
+
}
|
|
206
425
|
function TerraformPlan(stack) {
|
|
207
426
|
try {
|
|
208
|
-
|
|
209
|
-
|
|
427
|
+
const manifestPath = path.resolve(stack, "manifest.json");
|
|
428
|
+
// Safety check: verify manifest exists before reading
|
|
429
|
+
if (!fs.existsSync(manifestPath)) {
|
|
430
|
+
throw new Error(`Manifest file not found at ${manifestPath}. ` +
|
|
431
|
+
`Ensure Testing.fullSynth() completed successfully and the output directory still exists.`);
|
|
432
|
+
}
|
|
433
|
+
const manifest = JSON.parse(fs.readFileSync(manifestPath, "utf8"));
|
|
210
434
|
const stacks = Object.entries(manifest.stacks);
|
|
211
435
|
stacks.forEach(([, stackDetails]) => {
|
|
212
436
|
const opts = {
|
|
@@ -241,8 +465,17 @@ class AssertionReturn {
|
|
|
241
465
|
}
|
|
242
466
|
exports.AssertionReturn = AssertionReturn;
|
|
243
467
|
_b = JSII_RTTI_SYMBOL_1;
|
|
244
|
-
AssertionReturn[_b] = { fqn: "@microsoft/terraform-cdk-constructs.
|
|
245
|
-
|
|
468
|
+
AssertionReturn[_b] = { fqn: "@microsoft/terraform-cdk-constructs.AssertionReturn", version: "1.2.0" };
|
|
469
|
+
/**
|
|
470
|
+
* Cleanup function for removing all CDKTF output directories in /tmp.
|
|
471
|
+
*
|
|
472
|
+
* WARNING: This function should NOT be used in afterAll() hooks when running
|
|
473
|
+
* tests in parallel, as it will delete directories that other running tests
|
|
474
|
+
* may still be using. Instead, TerraformApplyCheckAndDestroy now cleans up
|
|
475
|
+
* its own directory automatically.
|
|
476
|
+
*
|
|
477
|
+
* This function is kept for manual cleanup or single-threaded test scenarios.
|
|
478
|
+
*/
|
|
246
479
|
function cleanupCdkTfOutDirs() {
|
|
247
480
|
try {
|
|
248
481
|
(0, child_process_1.execSync)('find /tmp -name "cdktf.outdir.*" -type d -exec rm -rf {} +');
|
|
@@ -251,4 +484,11 @@ function cleanupCdkTfOutDirs() {
|
|
|
251
484
|
console.error("Error during cleanup:", error);
|
|
252
485
|
}
|
|
253
486
|
}
|
|
254
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/testing/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { execSync, ExecSyncOptionsWithStringEncoding } from \"child_process\";\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as cdktf from \"cdktf\";\nimport { TerraformStack } from \"cdktf\";\nimport { Construct } from \"constructs\";\n\nexport class BaseTestStack extends TerraformStack {\n  public readonly name: string;\n\n  constructor(scope: Construct, id: string) {\n    super(scope, id);\n\n    const name = new cdktf.TerraformVariable(this, \"name\", {\n      type: \"string\",\n      default: \"test\",\n      description: \"System name used to randomize the resources\",\n    });\n\n    this.name = name.value;\n  }\n}\n\nexport function execTerraformCommand(\n  command: string,\n  opts: ExecSyncOptionsWithStringEncoding,\n  streamOutput: boolean,\n): { stdout: string; stderr: string; exitCode: number } {\n  try {\n    if (streamOutput) {\n      execSync(command, { ...opts, stdio: \"inherit\" });\n      return { stdout: \"\", stderr: \"\", exitCode: 0 };\n    } else {\n      const stdout = execSync(command, { ...opts, stdio: \"pipe\" }).toString();\n      return { stdout, stderr: \"\", exitCode: 0 };\n    }\n  } catch (error: any) {\n    const stdout = error.stdout ? error.stdout.toString() : \"\";\n    const stderr = error.stderr ? error.stderr.toString() : \"\";\n    return { stdout, stderr, exitCode: error.status || 1 };\n  }\n}\n\nexport function TerraformApply(\n  stack: any,\n  streamOutput: boolean = false,\n): { stdout: string; stderr: string; error: any } {\n  try {\n    const manifest = JSON.parse(\n      fs.readFileSync(path.resolve(stack, \"manifest.json\"), \"utf8\"),\n    );\n    const stacks = Object.entries(manifest.stacks);\n\n    for (const [, stackDetails] of stacks) {\n      const opts: ExecSyncOptionsWithStringEncoding = {\n        cwd: path.resolve(stack, (stackDetails as any).workingDirectory),\n        env: process.env,\n        encoding: \"utf-8\",\n      };\n\n      const initResult = execTerraformCommand(\n        `terraform init`,\n        opts,\n        streamOutput,\n      );\n      if (initResult.exitCode !== 0)\n        throw new Error(\n          `Terraform init failed: ${initResult.stdout} ${initResult.stderr}`,\n        );\n\n      const applyResult = execTerraformCommand(\n        `terraform apply -auto-approve`,\n        opts,\n        streamOutput,\n      );\n      if (applyResult.exitCode !== 0) {\n        return {\n          stdout: applyResult.stdout,\n          stderr: applyResult.stderr,\n          error: new Error(\n            `Terraform apply failed: ${applyResult.stdout} ${applyResult.stderr}`,\n          ),\n        };\n      }\n    }\n\n    return { stdout: \"Terraform apply succeeded\", stderr: \"\", error: null };\n  } catch (error: any) {\n    console.error(\"Error during Terraform init or apply:\", error);\n    return { stdout: \"\", stderr: \"\", error };\n  }\n}\n\nexport function TerraformPlanExitCode(\n  stack: any,\n  streamOutput: boolean = false,\n): { stdout: string; stderr: string; exitCode: number; error: any } {\n  try {\n    const manifest = JSON.parse(\n      fs.readFileSync(path.resolve(stack, \"manifest.json\"), \"utf8\"),\n    );\n    const stacks = Object.entries(manifest.stacks);\n\n    for (const [, stackDetails] of stacks) {\n      const opts: ExecSyncOptionsWithStringEncoding = {\n        cwd: path.resolve(stack, (stackDetails as any).workingDirectory),\n        env: process.env,\n        encoding: \"utf-8\",\n      };\n\n      const initResult = execTerraformCommand(\n        `terraform init`,\n        opts,\n        streamOutput,\n      );\n      if (initResult.exitCode !== 0)\n        throw new Error(\n          `Terraform init failed: ${initResult.stdout} ${initResult.stderr}`,\n        );\n\n      const planResult = execTerraformCommand(\n        `terraform plan -input=false -lock=false -detailed-exitcode`,\n        opts,\n        streamOutput,\n      );\n      return {\n        stdout: planResult.stdout,\n        stderr: planResult.stderr,\n        exitCode: planResult.exitCode,\n        error: null,\n      };\n    }\n\n    return {\n      stdout: \"Terraform plan succeeded\",\n      stderr: \"\",\n      exitCode: 0,\n      error: null,\n    };\n  } catch (error: any) {\n    console.error(\"Error during Terraform init or plan:\", error);\n    return { stdout: \"\", stderr: \"\", exitCode: 1, error };\n  }\n}\nexport function TerraformIdempotentCheck(stack: any): AssertionReturn {\n  try {\n    const manifest = JSON.parse(\n      fs.readFileSync(path.resolve(stack, \"manifest.json\"), \"utf8\"),\n    );\n    const stacks = Object.entries(manifest.stacks);\n\n    for (const [, stackDetails] of stacks) {\n      const opts = {\n        cwd: path.resolve(stack, (stackDetails as any).workingDirectory),\n        env: process.env,\n        stdio: \"pipe\",\n      } as any;\n\n      execSync(`terraform init`, opts);\n\n      let planOutput: string;\n      try {\n        planOutput = execSync(\n          `terraform plan -input=false -detailed-exitcode`,\n          opts,\n        ).toString();\n      } catch (error: any) {\n        planOutput = error.stdout ? error.stdout.toString() : \"\";\n      }\n\n      if (\n        planOutput.includes(\n          \"No changes. Your infrastructure matches the configuration.\",\n        )\n      ) {\n        return new AssertionReturn(\n          `Expected subject to be idempotent with no changes`,\n          true,\n        );\n      } else {\n        throw new Error(`Plan resulted in changes:\\n${planOutput}`);\n      }\n    }\n\n    return new AssertionReturn(\n      `Expected subject to be idempotent with no changes`,\n      true,\n    );\n  } catch (error) {\n    console.error(\"Error during Terraform init or plan:\", error);\n    throw error;\n  }\n}\n\nexport function TerraformDestroy(\n  stack: any,\n  streamOutput: boolean = false,\n): AssertionReturn {\n  try {\n    const manifest = JSON.parse(\n      fs.readFileSync(path.resolve(stack, \"manifest.json\"), \"utf8\"),\n    );\n    const stacks = Object.entries(manifest.stacks);\n\n    for (const [, stackDetails] of stacks) {\n      const opts: ExecSyncOptionsWithStringEncoding = {\n        cwd: path.resolve(stack, (stackDetails as any).workingDirectory),\n        env: process.env,\n        encoding: \"utf-8\",\n      };\n\n      const initResult = execTerraformCommand(\n        `terraform init`,\n        opts,\n        streamOutput,\n      );\n      if (initResult.exitCode !== 0)\n        throw new Error(\n          `Terraform init failed: ${initResult.stdout} ${initResult.stderr}`,\n        );\n\n      const destroyResult = execTerraformCommand(\n        `terraform destroy -auto-approve`,\n        opts,\n        streamOutput,\n      );\n      if (destroyResult.exitCode !== 0)\n        throw new Error(\n          `Terraform destroy failed: ${destroyResult.stdout} ${destroyResult.stderr}`,\n        );\n    }\n\n    return new AssertionReturn(\n      `Expected subject to destroy successfully`,\n      true,\n    );\n  } catch (error: any) {\n    console.error(\"Error during Terraform init or destroy:\", error);\n    return new AssertionReturn(\n      `Expected subject to destroy successfully`,\n      false,\n    );\n  }\n}\n\nexport function TerraformApplyAndCheckIdempotency(\n  stack: any,\n  streamOutput: boolean = false,\n): AssertionReturn {\n  const applyResult = TerraformApply(stack, streamOutput);\n  if (applyResult.error) {\n    return new AssertionReturn(applyResult.error.message, false);\n  }\n\n  const planResult = TerraformPlanExitCode(stack, streamOutput);\n  if (planResult.error) {\n    return new AssertionReturn(planResult.error.message, false);\n  }\n\n  if (planResult.exitCode !== 0) {\n    return new AssertionReturn(\n      `Terraform configuration not idempotent:\\n${planResult.stdout}`,\n      false,\n    );\n  }\n\n  return new AssertionReturn(\n    `Terraform apply and idempotency check completed successfully`,\n    true,\n  );\n}\n\nexport function TerraformApplyCheckAndDestroy(stack: any): void {\n  try {\n    const applyAndCheckResult = TerraformApplyAndCheckIdempotency(stack);\n    if (!applyAndCheckResult.success) {\n      throw new Error(applyAndCheckResult.message);\n    }\n  } catch (error) {\n    console.error(\"Error during Terraform apply and idempotency check:\", error);\n    throw error; // Re-throw the error to ensure the test fails\n  } finally {\n    try {\n      const destroyResult = TerraformDestroy(stack);\n      if (!destroyResult.success) {\n        console.error(\"Error during Terraform destroy:\", destroyResult.message);\n      }\n    } catch (destroyError) {\n      console.error(\"Error during Terraform destroy:\", destroyError);\n    }\n  }\n}\n\nexport function TerraformPlan(stack: any): AssertionReturn {\n  try {\n    // Assuming the presence of a received variable pointing to the relevant directory\n    const manifest = JSON.parse(\n      fs.readFileSync(path.resolve(stack, \"manifest.json\"), \"utf8\"),\n    );\n    const stacks = Object.entries(manifest.stacks);\n    stacks.forEach(([, stackDetails]) => {\n      const opts = {\n        cwd: path.resolve(stack, (stackDetails as any).workingDirectory),\n        env: process.env,\n        stdio: \"pipe\",\n      } as any;\n      execSync(`terraform init`, opts);\n      execSync(`terraform plan -input=false -lock=false`, opts);\n    });\n\n    return new AssertionReturn(\n      `Expected subject not to plan successfully`,\n      true,\n    );\n  } catch (error) {\n    console.error(\"Error during Terraform init or plan:\", error);\n    throw error;\n    return new AssertionReturn(\n      \"Expected subject to plan successfully, false\",\n      false,\n    );\n  }\n}\n\nexport class AssertionReturn {\n  /**\n   * Create an AssertionReturn\n   * @param message - String message containing information about the result of the assertion\n   * @param success - Boolean success denoting the success of the assertion\n   */\n  constructor(\n    public readonly message: string,\n    public readonly success: boolean,\n  ) {\n    if (!success) {\n      throw new Error(message);\n    }\n  }\n}\n\n// Define a cleanup function that can be exported and used in tests.\nexport function cleanupCdkTfOutDirs() {\n  try {\n    execSync('find /tmp -name \"cdktf.outdir.*\" -type d -exec rm -rf {} +');\n  } catch (error) {\n    console.error(\"Error during cleanup:\", error);\n  }\n}\n"]}
|
|
487
|
+
// Export cleanup utilities
|
|
488
|
+
var cleanup_2 = require("./lib/cleanup");
|
|
489
|
+
Object.defineProperty(exports, "ResourceCleanupService", { enumerable: true, get: function () { return cleanup_2.ResourceCleanupService; } });
|
|
490
|
+
Object.defineProperty(exports, "verifyResourcesDeleted", { enumerable: true, get: function () { return cleanup_2.verifyResourcesDeleted; } });
|
|
491
|
+
Object.defineProperty(exports, "verifyResourcesDeletedWithRetry", { enumerable: true, get: function () { return cleanup_2.verifyResourcesDeletedWithRetry; } });
|
|
492
|
+
Object.defineProperty(exports, "findOrphanedResources", { enumerable: true, get: function () { return cleanup_2.findOrphanedResources; } });
|
|
493
|
+
Object.defineProperty(exports, "cleanupOrphanedResources", { enumerable: true, get: function () { return cleanup_2.cleanupOrphanedResources; } });
|
|
494
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/testing/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { execSync, ExecSyncOptionsWithStringEncoding } from \"child_process\";\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as cdktf from \"cdktf\";\nimport { TerraformStack } from \"cdktf\";\nimport { Construct } from \"constructs\";\nimport { ResourceMetadata, verifyResourcesDeleted } from \"./lib/cleanup\";\nimport { TestRunMetadata, TestRunOptions } from \"./lib/metadata\";\nimport { generateResourceName } from \"./lib/naming\";\n\n// Re-export types used in public APIs for jsii compliance\nexport { TestRunOptions, TestRunMetadata, CIContext } from \"./lib/metadata\";\n\n/**\n * Options for BaseTestStack constructor\n */\nexport interface BaseTestStackOptions {\n  /**\n   * Test run configuration options\n   */\n  readonly testRunOptions?: TestRunOptions;\n}\n\n/**\n * Base stack for integration tests with optional metadata support\n *\n * When metadata is enabled, automatically generates unique test run IDs,\n * injects system tags, and provides utilities for resource naming.\n */\nexport class BaseTestStack extends TerraformStack {\n  public readonly name: string;\n\n  /**\n   * Optional test run metadata (only present when using BaseTestStackOptions)\n   */\n  public readonly metadata?: TestRunMetadata;\n\n  /**\n   * Registry of resources created in this stack for tracking and cleanup verification\n   */\n  private readonly resourceRegistry: Map<string, ResourceMetadata> = new Map();\n\n  constructor(scope: Construct, id: string, options?: BaseTestStackOptions) {\n    super(scope, id);\n\n    // Generate metadata if options are provided\n    if (options?.testRunOptions) {\n      this.metadata = new TestRunMetadata(id, options.testRunOptions);\n\n      // Log test run information\n      console.log(`[Test Run] ID: ${this.metadata.runId}`);\n      console.log(`[Test Run] Name: ${this.metadata.testName}`);\n      console.log(\n        `[Test Run] Created: ${this.metadata.createdAt.toISOString()}`,\n      );\n      console.log(\n        `[Test Run] Cleanup After: ${this.metadata.cleanupAfter.toISOString()}`,\n      );\n      if (this.metadata.ciContext) {\n        console.log(\n          `[Test Run] CI Platform: ${this.metadata.ciContext.platform}`,\n        );\n      }\n    }\n\n    const name = new cdktf.TerraformVariable(this, \"name\", {\n      type: \"string\",\n      default: this.metadata?.testName || \"test\",\n      description: \"System name used to randomize the resources\",\n    });\n\n    this.name = name.value;\n  }\n\n  /**\n   * Registers a resource for tracking and cleanup verification\n   *\n   * @param resourceId - Azure resource ID\n   * @param resourceType - Azure resource type\n   * @param name - Resource name\n   * @param location - Optional resource location\n   * @param tags - Optional resource tags\n   */\n  public registerResource(\n    resourceId: string,\n    resourceType: string,\n    name: string,\n    location?: string,\n    tags?: Record<string, string>,\n  ): void {\n    if (!this.metadata) {\n      console.warn(\n        \"[BaseTestStack] Resource registration requires metadata. \" +\n          \"Initialize with testRunOptions to enable resource tracking.\",\n      );\n      return;\n    }\n\n    const metadata: ResourceMetadata = {\n      resourceId,\n      resourceType,\n      name,\n      location,\n      tags: tags || {},\n      testRunId: this.metadata.runId,\n      createdAt: this.metadata.createdAt,\n    };\n\n    this.resourceRegistry.set(resourceId, metadata);\n    console.log(\n      `[BaseTestStack] Registered resource: ${name} (${resourceType})`,\n    );\n  }\n\n  /**\n   * Retrieves all registered resources\n   *\n   * @returns Array of registered resource metadata\n   */\n  public registeredResources(): ResourceMetadata[] {\n    return Array.from(this.resourceRegistry.values());\n  }\n\n  /**\n   * Generates system tags for resources (only available when metadata is present)\n   *\n   * @returns Integration test system tags\n   * @throws Error if metadata is not initialized\n   */\n  public systemTags(): Record<string, string> {\n    if (!this.metadata) {\n      throw new Error(\n        \"System tags are only available when BaseTestStack is initialized with testRunOptions. \" +\n          \"Pass { testRunOptions: {} } to the constructor to enable metadata.\",\n      );\n    }\n    return this.metadata.generateSystemTags();\n  }\n\n  /**\n   * Generates a unique resource name with proper Azure compliance\n   *\n   * @param resourceType - Azure resource type (e.g., 'Microsoft.Resources/resourceGroups')\n   * @param customIdentifier - Optional custom identifier (defaults to test name)\n   * @returns Unique, Azure-compliant resource name\n   * @throws Error if metadata is not initialized\n   *\n   * @example\n   * const stack = new BaseTestStack(app, 'test-storage', { testRunOptions: {} });\n   * const rgName = stack.generateResourceName('Microsoft.Resources/resourceGroups');\n   * // Returns: 'rg-test-storage-a1b2c3'\n   */\n  public generateResourceName(\n    resourceType: string,\n    customIdentifier?: string,\n  ): string {\n    if (!this.metadata) {\n      throw new Error(\n        \"Resource name generation is only available when BaseTestStack is initialized with testRunOptions. \" +\n          \"Pass { testRunOptions: {} } to the constructor to enable metadata.\",\n      );\n    }\n\n    return generateResourceName({\n      resourceType,\n      testName: this.metadata.testName,\n      runId: this.metadata.runId,\n      customIdentifier,\n    });\n  }\n}\n\nexport function execTerraformCommand(\n  command: string,\n  opts: ExecSyncOptionsWithStringEncoding,\n  streamOutput: boolean,\n): { stdout: string; stderr: string; exitCode: number } {\n  try {\n    if (streamOutput) {\n      execSync(command, { ...opts, stdio: \"inherit\" });\n      return { stdout: \"\", stderr: \"\", exitCode: 0 };\n    } else {\n      const stdout = execSync(command, { ...opts, stdio: \"pipe\" }).toString();\n      return { stdout, stderr: \"\", exitCode: 0 };\n    }\n  } catch (error: any) {\n    const stdout = error.stdout ? error.stdout.toString() : \"\";\n    const stderr = error.stderr ? error.stderr.toString() : \"\";\n    return { stdout, stderr, exitCode: error.status || 1 };\n  }\n}\n\nexport function TerraformApply(\n  stack: any,\n  streamOutput: boolean = false,\n): { stdout: string; stderr: string; error: any } {\n  try {\n    const manifestPath = path.resolve(stack, \"manifest.json\");\n\n    // Safety check: verify manifest exists before reading\n    if (!fs.existsSync(manifestPath)) {\n      throw new Error(\n        `Manifest file not found at ${manifestPath}. ` +\n          `Ensure Testing.fullSynth() completed successfully and the output directory still exists.`,\n      );\n    }\n\n    const manifest = JSON.parse(fs.readFileSync(manifestPath, \"utf8\"));\n    const stacks = Object.entries(manifest.stacks);\n\n    for (const [, stackDetails] of stacks) {\n      const opts: ExecSyncOptionsWithStringEncoding = {\n        cwd: path.resolve(stack, (stackDetails as any).workingDirectory),\n        env: process.env,\n        encoding: \"utf-8\",\n      };\n\n      const initResult = execTerraformCommand(\n        `terraform init`,\n        opts,\n        streamOutput,\n      );\n      if (initResult.exitCode !== 0)\n        throw new Error(\n          `Terraform init failed: ${initResult.stdout} ${initResult.stderr}`,\n        );\n\n      const applyResult = execTerraformCommand(\n        `terraform apply -auto-approve`,\n        opts,\n        streamOutput,\n      );\n      if (applyResult.exitCode !== 0) {\n        return {\n          stdout: applyResult.stdout,\n          stderr: applyResult.stderr,\n          error: new Error(\n            `Terraform apply failed: ${applyResult.stdout} ${applyResult.stderr}`,\n          ),\n        };\n      }\n    }\n\n    return { stdout: \"Terraform apply succeeded\", stderr: \"\", error: null };\n  } catch (error: any) {\n    console.error(\"Error during Terraform init or apply:\", error);\n    return { stdout: \"\", stderr: \"\", error };\n  }\n}\n\nexport function TerraformPlanExitCode(\n  stack: any,\n  streamOutput: boolean = false,\n): { stdout: string; stderr: string; exitCode: number; error: any } {\n  try {\n    const manifestPath = path.resolve(stack, \"manifest.json\");\n\n    // Safety check: verify manifest exists before reading\n    if (!fs.existsSync(manifestPath)) {\n      throw new Error(\n        `Manifest file not found at ${manifestPath}. ` +\n          `Ensure Testing.fullSynth() completed successfully and the output directory still exists.`,\n      );\n    }\n\n    const manifest = JSON.parse(fs.readFileSync(manifestPath, \"utf8\"));\n    const stacks = Object.entries(manifest.stacks);\n\n    for (const [, stackDetails] of stacks) {\n      const opts: ExecSyncOptionsWithStringEncoding = {\n        cwd: path.resolve(stack, (stackDetails as any).workingDirectory),\n        env: process.env,\n        encoding: \"utf-8\",\n      };\n\n      const initResult = execTerraformCommand(\n        `terraform init`,\n        opts,\n        streamOutput,\n      );\n      if (initResult.exitCode !== 0)\n        throw new Error(\n          `Terraform init failed: ${initResult.stdout} ${initResult.stderr}`,\n        );\n\n      const planResult = execTerraformCommand(\n        `terraform plan -input=false -lock=false -detailed-exitcode`,\n        opts,\n        streamOutput,\n      );\n      return {\n        stdout: planResult.stdout,\n        stderr: planResult.stderr,\n        exitCode: planResult.exitCode,\n        error: null,\n      };\n    }\n\n    return {\n      stdout: \"Terraform plan succeeded\",\n      stderr: \"\",\n      exitCode: 0,\n      error: null,\n    };\n  } catch (error: any) {\n    console.error(\"Error during Terraform init or plan:\", error);\n    return { stdout: \"\", stderr: \"\", exitCode: 1, error };\n  }\n}\nexport function TerraformIdempotentCheck(stack: any): AssertionReturn {\n  try {\n    const manifestPath = path.resolve(stack, \"manifest.json\");\n\n    // Safety check: verify manifest exists before reading\n    if (!fs.existsSync(manifestPath)) {\n      throw new Error(\n        `Manifest file not found at ${manifestPath}. ` +\n          `Ensure Testing.fullSynth() completed successfully and the output directory still exists.`,\n      );\n    }\n\n    const manifest = JSON.parse(fs.readFileSync(manifestPath, \"utf8\"));\n    const stacks = Object.entries(manifest.stacks);\n\n    for (const [, stackDetails] of stacks) {\n      const opts = {\n        cwd: path.resolve(stack, (stackDetails as any).workingDirectory),\n        env: process.env,\n        stdio: \"pipe\",\n      } as any;\n\n      execSync(`terraform init`, opts);\n\n      let planOutput: string;\n      try {\n        planOutput = execSync(\n          `terraform plan -input=false -detailed-exitcode`,\n          opts,\n        ).toString();\n      } catch (error: any) {\n        planOutput = error.stdout ? error.stdout.toString() : \"\";\n      }\n\n      if (\n        planOutput.includes(\n          \"No changes. Your infrastructure matches the configuration.\",\n        )\n      ) {\n        return new AssertionReturn(\n          `Expected subject to be idempotent with no changes`,\n          true,\n        );\n      } else {\n        throw new Error(`Plan resulted in changes:\\n${planOutput}`);\n      }\n    }\n\n    return new AssertionReturn(\n      `Expected subject to be idempotent with no changes`,\n      true,\n    );\n  } catch (error) {\n    console.error(\"Error during Terraform init or plan:\", error);\n    throw error;\n  }\n}\n\nexport function TerraformDestroy(\n  stack: any,\n  streamOutput: boolean = false,\n): AssertionReturn {\n  try {\n    const manifestPath = path.resolve(stack, \"manifest.json\");\n\n    // Safety check: verify manifest exists before reading\n    if (!fs.existsSync(manifestPath)) {\n      throw new Error(\n        `Manifest file not found at ${manifestPath}. ` +\n          `Ensure Testing.fullSynth() completed successfully and the output directory still exists.`,\n      );\n    }\n\n    const manifest = JSON.parse(fs.readFileSync(manifestPath, \"utf8\"));\n    const stacks = Object.entries(manifest.stacks);\n\n    for (const [, stackDetails] of stacks) {\n      const opts: ExecSyncOptionsWithStringEncoding = {\n        cwd: path.resolve(stack, (stackDetails as any).workingDirectory),\n        env: process.env,\n        encoding: \"utf-8\",\n      };\n\n      const initResult = execTerraformCommand(\n        `terraform init`,\n        opts,\n        streamOutput,\n      );\n      if (initResult.exitCode !== 0)\n        throw new Error(\n          `Terraform init failed: ${initResult.stdout} ${initResult.stderr}`,\n        );\n\n      const destroyResult = execTerraformCommand(\n        `terraform destroy -auto-approve`,\n        opts,\n        streamOutput,\n      );\n      if (destroyResult.exitCode !== 0)\n        throw new Error(\n          `Terraform destroy failed: ${destroyResult.stdout} ${destroyResult.stderr}`,\n        );\n    }\n\n    return new AssertionReturn(\n      `Expected subject to destroy successfully`,\n      true,\n    );\n  } catch (error: any) {\n    console.error(\"Error during Terraform init or destroy:\", error);\n    return new AssertionReturn(\n      `Expected subject to destroy successfully`,\n      false,\n    );\n  }\n}\n\nexport function TerraformApplyAndCheckIdempotency(\n  stack: any,\n  streamOutput: boolean = false,\n): AssertionReturn {\n  const applyResult = TerraformApply(stack, streamOutput);\n  if (applyResult.error) {\n    return new AssertionReturn(applyResult.error.message, false);\n  }\n\n  const planResult = TerraformPlanExitCode(stack, streamOutput);\n  if (planResult.error) {\n    return new AssertionReturn(planResult.error.message, false);\n  }\n\n  if (planResult.exitCode !== 0) {\n    return new AssertionReturn(\n      `Terraform configuration not idempotent:\\n${planResult.stdout}`,\n      false,\n    );\n  }\n\n  return new AssertionReturn(\n    `Terraform apply and idempotency check completed successfully`,\n    true,\n  );\n}\n\nexport function TerraformApplyCheckAndDestroy(\n  stack: any,\n  options?: { verifyCleanup?: boolean },\n): void {\n  const streamOutput = process.env.STREAM_OUTPUT === \"true\";\n  const verifyCleanup = options?.verifyCleanup ?? false;\n\n  // Extract metadata and resources if available\n  const metadata: TestRunMetadata | undefined = (stack as any).metadata;\n  const hasVerification = verifyCleanup && metadata;\n\n  try {\n    const applyAndCheckResult = TerraformApplyAndCheckIdempotency(\n      stack,\n      streamOutput,\n    );\n    if (!applyAndCheckResult.success) {\n      throw new Error(applyAndCheckResult.message);\n    }\n  } catch (error) {\n    console.error(\"Error during Terraform apply and idempotency check:\", error);\n    throw error; // Re-throw the error to ensure the test fails\n  } finally {\n    try {\n      if (hasVerification) {\n        // Use enhanced destroy with verification\n        const resources: ResourceMetadata[] =\n          typeof (stack as any).registeredResources === \"function\"\n            ? (stack as any).registeredResources()\n            : [];\n\n        const destroyResult = TerraformDestroyWithVerification(\n          stack,\n          metadata!,\n          resources,\n          streamOutput,\n        );\n\n        if (!destroyResult.success) {\n          throw new Error(\n            `Destroy verification failed: ${destroyResult.message}`,\n          );\n        }\n      } else {\n        // Use standard destroy (backward compatible)\n        const destroyResult = TerraformDestroy(stack, streamOutput);\n        if (!destroyResult.success) {\n          console.error(\n            \"Error during Terraform destroy:\",\n            destroyResult.message,\n          );\n        }\n      }\n    } catch (destroyError) {\n      console.error(\"Error during Terraform destroy:\", destroyError);\n      throw destroyError; // Re-throw to fail the test\n    } finally {\n      // Clean up this specific test's output directory\n      try {\n        if (fs.existsSync(stack)) {\n          execSync(`rm -rf \"${stack}\"`);\n        }\n      } catch (cleanupError) {\n        // Ignore cleanup errors - directory may already be gone\n      }\n    }\n  }\n}\n\n/**\n * Enhanced terraform destroy with cleanup verification\n *\n * Performs terraform destroy and verifies resource cleanup via\n * Azure Resource Graph queries. Implements retry logic for\n * eventual consistency.\n *\n * @param stack - Terraform stack directory path\n * @param metadata - Test run metadata\n * @param resources - List of resources to verify deletion\n * @param streamOutput - Whether to stream command output\n * @returns Assertion result\n */\nexport function TerraformDestroyWithVerification(\n  stack: any,\n  metadata: TestRunMetadata,\n  resources: ResourceMetadata[],\n  streamOutput: boolean = false,\n): AssertionReturn {\n  try {\n    console.log(\"[Destroy] Starting terraform destroy...\");\n\n    // Step 1: Run terraform destroy\n    const destroyResult = TerraformDestroy(stack, streamOutput);\n\n    if (!destroyResult.success) {\n      throw new Error(`Terraform destroy failed: ${destroyResult.message}`);\n    }\n\n    console.log(\"[Destroy] Terraform destroy completed successfully\");\n\n    // Step 2: If no resources to verify, skip verification\n    if (resources.length === 0) {\n      console.log(\"[Destroy] No resources to verify (none registered)\");\n      return new AssertionReturn(\n        \"Terraform destroy completed (no verification needed)\",\n        true,\n      );\n    }\n\n    console.log(\n      `[Destroy] Verifying cleanup of ${resources.length} resources...`,\n    );\n    console.log(\"[Destroy] Waiting 30s for Azure eventual consistency...\");\n\n    // Step 3: Wait for eventual consistency\n    const sleep = (ms: number) => {\n      execSync(`sleep ${ms / 1000}`, { stdio: \"ignore\" });\n    };\n    sleep(30000);\n\n    // Step 4: Verify resources are deleted with retry logic\n    console.log(\"[Destroy] Starting verification...\");\n    const verificationResult = verifyResourcesDeleted(\n      metadata.runId,\n      resources,\n    );\n\n    if (!verificationResult.success) {\n      // Step 5: Retry verification after additional wait\n      console.warn(\"[Destroy] Initial verification failed, retrying...\");\n      sleep(30000);\n\n      const retryResult = verifyResourcesDeleted(metadata.runId, resources);\n\n      if (!retryResult.success) {\n        const errorMessage =\n          `Resource cleanup verification failed:\\n` +\n          `  Expected: ${retryResult.expectedCount} resources\\n` +\n          `  Found: ${retryResult.foundCount} orphaned resources\\n` +\n          `  Orphaned IDs:\\n` +\n          retryResult.orphanedResources.map((id) => `    - ${id}`).join(\"\\n\");\n\n        console.error(\"[Destroy] ✗ Verification failed:\", errorMessage);\n        throw new Error(errorMessage);\n      }\n\n      console.log(\"[Destroy] ✓ Verification succeeded on retry\");\n    } else {\n      console.log(\"[Destroy] ✓ All resources verified deleted\");\n    }\n\n    return new AssertionReturn(\n      \"Terraform destroy and verification completed successfully\",\n      true,\n    );\n  } catch (error: any) {\n    console.error(\n      \"[Destroy] Error during Terraform destroy and verification:\",\n      error,\n    );\n    return new AssertionReturn(\n      `Terraform destroy and verification failed: ${error.message}`,\n      false,\n    );\n  }\n}\n\nexport function TerraformPlan(stack: any): AssertionReturn {\n  try {\n    const manifestPath = path.resolve(stack, \"manifest.json\");\n\n    // Safety check: verify manifest exists before reading\n    if (!fs.existsSync(manifestPath)) {\n      throw new Error(\n        `Manifest file not found at ${manifestPath}. ` +\n          `Ensure Testing.fullSynth() completed successfully and the output directory still exists.`,\n      );\n    }\n\n    const manifest = JSON.parse(fs.readFileSync(manifestPath, \"utf8\"));\n    const stacks = Object.entries(manifest.stacks);\n    stacks.forEach(([, stackDetails]) => {\n      const opts = {\n        cwd: path.resolve(stack, (stackDetails as any).workingDirectory),\n        env: process.env,\n        stdio: \"pipe\",\n      } as any;\n      execSync(`terraform init`, opts);\n      execSync(`terraform plan -input=false -lock=false`, opts);\n    });\n\n    return new AssertionReturn(\n      `Expected subject not to plan successfully`,\n      true,\n    );\n  } catch (error) {\n    console.error(\"Error during Terraform init or plan:\", error);\n    throw error;\n    return new AssertionReturn(\n      \"Expected subject to plan successfully, false\",\n      false,\n    );\n  }\n}\n\nexport class AssertionReturn {\n  /**\n   * Create an AssertionReturn\n   * @param message - String message containing information about the result of the assertion\n   * @param success - Boolean success denoting the success of the assertion\n   */\n  constructor(\n    public readonly message: string,\n    public readonly success: boolean,\n  ) {\n    if (!success) {\n      throw new Error(message);\n    }\n  }\n}\n\n/**\n * Cleanup function for removing all CDKTF output directories in /tmp.\n *\n * WARNING: This function should NOT be used in afterAll() hooks when running\n * tests in parallel, as it will delete directories that other running tests\n * may still be using. Instead, TerraformApplyCheckAndDestroy now cleans up\n * its own directory automatically.\n *\n * This function is kept for manual cleanup or single-threaded test scenarios.\n */\nexport function cleanupCdkTfOutDirs() {\n  try {\n    execSync('find /tmp -name \"cdktf.outdir.*\" -type d -exec rm -rf {} +');\n  } catch (error) {\n    console.error(\"Error during cleanup:\", error);\n  }\n}\n\n// Export cleanup utilities\nexport {\n  ResourceMetadata,\n  VerificationResult,\n  CleanupOptions,\n  OrphanedResource,\n  CleanupResult,\n  ResourceCleanupService,\n  verifyResourcesDeleted,\n  verifyResourcesDeletedWithRetry,\n  findOrphanedResources,\n  cleanupOrphanedResources,\n} from \"./lib/cleanup\";\n"]}
|