@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
|
@@ -0,0 +1,229 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Cleanup and verification utilities for integration tests
|
|
3
|
+
*
|
|
4
|
+
* Provides functions for verifying resource deletion via Azure Resource Graph
|
|
5
|
+
* and finding/cleaning orphaned resources with comprehensive safety mechanisms.
|
|
6
|
+
*/
|
|
7
|
+
/**
|
|
8
|
+
* Metadata for an individual resource created during testing
|
|
9
|
+
*/
|
|
10
|
+
export interface ResourceMetadata {
|
|
11
|
+
/**
|
|
12
|
+
* Azure resource ID
|
|
13
|
+
*/
|
|
14
|
+
readonly resourceId: string;
|
|
15
|
+
/**
|
|
16
|
+
* Azure resource type (e.g., 'Microsoft.Resources/resourceGroups')
|
|
17
|
+
*/
|
|
18
|
+
readonly resourceType: string;
|
|
19
|
+
/**
|
|
20
|
+
* Resource name
|
|
21
|
+
*/
|
|
22
|
+
readonly name: string;
|
|
23
|
+
/**
|
|
24
|
+
* Resource location (Azure region)
|
|
25
|
+
*/
|
|
26
|
+
readonly location?: string;
|
|
27
|
+
/**
|
|
28
|
+
* Resource tags
|
|
29
|
+
*/
|
|
30
|
+
readonly tags: Record<string, string>;
|
|
31
|
+
/**
|
|
32
|
+
* Test run ID that created this resource
|
|
33
|
+
*/
|
|
34
|
+
readonly testRunId: string;
|
|
35
|
+
/**
|
|
36
|
+
* Timestamp when resource was created
|
|
37
|
+
*/
|
|
38
|
+
readonly createdAt: Date;
|
|
39
|
+
/**
|
|
40
|
+
* Whether this resource was successfully destroyed
|
|
41
|
+
*/
|
|
42
|
+
readonly destroyed?: boolean;
|
|
43
|
+
/**
|
|
44
|
+
* Timestamp when resource was destroyed
|
|
45
|
+
*/
|
|
46
|
+
readonly destroyedAt?: Date;
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* Result of resource cleanup verification
|
|
50
|
+
*/
|
|
51
|
+
export interface VerificationResult {
|
|
52
|
+
/**
|
|
53
|
+
* Whether verification succeeded (all resources deleted)
|
|
54
|
+
*/
|
|
55
|
+
readonly success: boolean;
|
|
56
|
+
/**
|
|
57
|
+
* Human-readable message describing the result
|
|
58
|
+
*/
|
|
59
|
+
readonly message: string;
|
|
60
|
+
/**
|
|
61
|
+
* List of orphaned resource IDs (if any)
|
|
62
|
+
*/
|
|
63
|
+
readonly orphanedResources: string[];
|
|
64
|
+
/**
|
|
65
|
+
* Number of resources expected to be deleted
|
|
66
|
+
*/
|
|
67
|
+
readonly expectedCount?: number;
|
|
68
|
+
/**
|
|
69
|
+
* Number of resources actually found
|
|
70
|
+
*/
|
|
71
|
+
readonly foundCount?: number;
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Options for cleanup operations
|
|
75
|
+
*/
|
|
76
|
+
export interface CleanupOptions {
|
|
77
|
+
/**
|
|
78
|
+
* Whether to perform dry-run (no actual deletion)
|
|
79
|
+
*/
|
|
80
|
+
readonly dryRun: boolean;
|
|
81
|
+
/**
|
|
82
|
+
* Minimum age in hours before resource can be cleaned up
|
|
83
|
+
*/
|
|
84
|
+
readonly minAgeHours: number;
|
|
85
|
+
/**
|
|
86
|
+
* Azure subscription ID to target (optional)
|
|
87
|
+
*/
|
|
88
|
+
readonly subscription?: string;
|
|
89
|
+
/**
|
|
90
|
+
* Specific resource groups to target (optional)
|
|
91
|
+
*/
|
|
92
|
+
readonly resourceGroups?: string[];
|
|
93
|
+
/**
|
|
94
|
+
* Maximum number of resources to clean up in one operation
|
|
95
|
+
*/
|
|
96
|
+
readonly maxResources?: number;
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* Orphaned resource information
|
|
100
|
+
*/
|
|
101
|
+
export interface OrphanedResource {
|
|
102
|
+
/**
|
|
103
|
+
* Azure resource ID
|
|
104
|
+
*/
|
|
105
|
+
readonly resourceId: string;
|
|
106
|
+
/**
|
|
107
|
+
* Azure resource type
|
|
108
|
+
*/
|
|
109
|
+
readonly resourceType: string;
|
|
110
|
+
/**
|
|
111
|
+
* Resource name
|
|
112
|
+
*/
|
|
113
|
+
readonly name: string;
|
|
114
|
+
/**
|
|
115
|
+
* Resource location
|
|
116
|
+
*/
|
|
117
|
+
readonly location: string;
|
|
118
|
+
/**
|
|
119
|
+
* Resource group name
|
|
120
|
+
*/
|
|
121
|
+
readonly resourceGroup: string;
|
|
122
|
+
/**
|
|
123
|
+
* Test run ID from tags
|
|
124
|
+
*/
|
|
125
|
+
readonly testRunId: string;
|
|
126
|
+
/**
|
|
127
|
+
* Creation timestamp from tags
|
|
128
|
+
*/
|
|
129
|
+
readonly createdAt: Date;
|
|
130
|
+
/**
|
|
131
|
+
* Cleanup after timestamp from tags
|
|
132
|
+
*/
|
|
133
|
+
readonly cleanupAfter: Date;
|
|
134
|
+
/**
|
|
135
|
+
* Test name from tags
|
|
136
|
+
*/
|
|
137
|
+
readonly testName: string;
|
|
138
|
+
/**
|
|
139
|
+
* Age in hours since creation
|
|
140
|
+
*/
|
|
141
|
+
readonly ageHours: number;
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Result of cleanup operation
|
|
145
|
+
*/
|
|
146
|
+
export interface CleanupResult {
|
|
147
|
+
/**
|
|
148
|
+
* Number of resources successfully deleted
|
|
149
|
+
*/
|
|
150
|
+
readonly deleted: number;
|
|
151
|
+
/**
|
|
152
|
+
* Number of resources that failed to delete
|
|
153
|
+
*/
|
|
154
|
+
readonly failed: number;
|
|
155
|
+
/**
|
|
156
|
+
* Number of resources skipped (e.g., too young)
|
|
157
|
+
*/
|
|
158
|
+
readonly skipped: number;
|
|
159
|
+
/**
|
|
160
|
+
* Detailed error messages for failures
|
|
161
|
+
*/
|
|
162
|
+
readonly errors?: string[];
|
|
163
|
+
}
|
|
164
|
+
/**
|
|
165
|
+
* Main cleanup service class
|
|
166
|
+
*/
|
|
167
|
+
export declare class ResourceCleanupService {
|
|
168
|
+
/**
|
|
169
|
+
* Checks if Azure CLI is available
|
|
170
|
+
*/
|
|
171
|
+
checkAzureCliAvailable(): boolean;
|
|
172
|
+
/**
|
|
173
|
+
* Executes an Azure CLI command with error handling
|
|
174
|
+
*/
|
|
175
|
+
private executeAzureCommand;
|
|
176
|
+
/**
|
|
177
|
+
* Queries Azure Resource Graph for resources by tags
|
|
178
|
+
*
|
|
179
|
+
* Returns empty array on errors for graceful degradation.
|
|
180
|
+
*/
|
|
181
|
+
queryAzureResourcesByTags(tags: Record<string, string>, subscription?: string): any[];
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* Verifies that all resources with a specific test run ID are deleted
|
|
185
|
+
*
|
|
186
|
+
* Performs Azure Resource Graph query to check for remaining resources.
|
|
187
|
+
* Implements retry logic to handle Azure's eventual consistency.
|
|
188
|
+
*
|
|
189
|
+
* @param runId - Test run UUID to verify
|
|
190
|
+
* @param expectedResources - List of resources that should be deleted
|
|
191
|
+
* @param subscription - Optional subscription ID to query
|
|
192
|
+
* @returns Verification result with details
|
|
193
|
+
*/
|
|
194
|
+
export declare function verifyResourcesDeleted(runId: string, expectedResources: ResourceMetadata[], subscription?: string): VerificationResult;
|
|
195
|
+
/**
|
|
196
|
+
* Finds orphaned test resources that are past their TTL
|
|
197
|
+
*
|
|
198
|
+
* Queries Azure for resources with test tags that are older than
|
|
199
|
+
* their cleanup-after timestamp.
|
|
200
|
+
*
|
|
201
|
+
* @param options - Cleanup options
|
|
202
|
+
* @returns List of orphaned resources
|
|
203
|
+
*/
|
|
204
|
+
export declare function findOrphanedResources(options: CleanupOptions): Promise<OrphanedResource[]>;
|
|
205
|
+
/**
|
|
206
|
+
* Cleans up orphaned test resources
|
|
207
|
+
*
|
|
208
|
+
* Deletes resource groups containing orphaned test resources.
|
|
209
|
+
* Implements safety checks and dry-run mode.
|
|
210
|
+
*
|
|
211
|
+
* @param resources - List of orphaned resources to clean up
|
|
212
|
+
* @param options - Cleanup options
|
|
213
|
+
* @returns Cleanup result with statistics
|
|
214
|
+
*/
|
|
215
|
+
export declare function cleanupOrphanedResources(resources: OrphanedResource[], options: CleanupOptions): Promise<CleanupResult>;
|
|
216
|
+
/**
|
|
217
|
+
* Verifies resource cleanup with retry logic for eventual consistency
|
|
218
|
+
*
|
|
219
|
+
* Wrapper around verifyResourcesDeleted that implements retry logic
|
|
220
|
+
* to handle Azure's eventual consistency delays.
|
|
221
|
+
*
|
|
222
|
+
* @param runId - Test run UUID to verify
|
|
223
|
+
* @param expectedResources - List of resources that should be deleted
|
|
224
|
+
* @param subscription - Optional subscription ID to query
|
|
225
|
+
* @param maxRetries - Maximum number of retry attempts
|
|
226
|
+
* @param retryWaitMs - Wait time between retries in milliseconds
|
|
227
|
+
* @returns Final verification result
|
|
228
|
+
*/
|
|
229
|
+
export declare function verifyResourcesDeletedWithRetry(runId: string, expectedResources: ResourceMetadata[], subscription?: string, maxRetries?: number, retryWaitMs?: number): Promise<VerificationResult>;
|
|
@@ -0,0 +1,331 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.ResourceCleanupService = void 0;
|
|
5
|
+
exports.verifyResourcesDeleted = verifyResourcesDeleted;
|
|
6
|
+
exports.findOrphanedResources = findOrphanedResources;
|
|
7
|
+
exports.cleanupOrphanedResources = cleanupOrphanedResources;
|
|
8
|
+
exports.verifyResourcesDeletedWithRetry = verifyResourcesDeletedWithRetry;
|
|
9
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
10
|
+
/**
|
|
11
|
+
* Cleanup and verification utilities for integration tests
|
|
12
|
+
*
|
|
13
|
+
* Provides functions for verifying resource deletion via Azure Resource Graph
|
|
14
|
+
* and finding/cleaning orphaned resources with comprehensive safety mechanisms.
|
|
15
|
+
*/
|
|
16
|
+
const child_process_1 = require("child_process");
|
|
17
|
+
/**
|
|
18
|
+
* Minimum age threshold (hours) before cleanup is allowed
|
|
19
|
+
*/
|
|
20
|
+
const MINIMUM_AGE_HOURS = 2;
|
|
21
|
+
/**
|
|
22
|
+
* Maximum retry attempts for eventual consistency
|
|
23
|
+
*/
|
|
24
|
+
const MAX_RETRIES = 5;
|
|
25
|
+
/**
|
|
26
|
+
* Wait time between retries (milliseconds)
|
|
27
|
+
*/
|
|
28
|
+
const RETRY_WAIT_MS = 30000; // 30 seconds
|
|
29
|
+
/**
|
|
30
|
+
* Main cleanup service class
|
|
31
|
+
*/
|
|
32
|
+
class ResourceCleanupService {
|
|
33
|
+
/**
|
|
34
|
+
* Checks if Azure CLI is available
|
|
35
|
+
*/
|
|
36
|
+
checkAzureCliAvailable() {
|
|
37
|
+
try {
|
|
38
|
+
(0, child_process_1.execSync)("az --version", { stdio: "pipe" });
|
|
39
|
+
return true;
|
|
40
|
+
}
|
|
41
|
+
catch (error) {
|
|
42
|
+
return false;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Executes an Azure CLI command with error handling
|
|
47
|
+
*/
|
|
48
|
+
executeAzureCommand(command) {
|
|
49
|
+
try {
|
|
50
|
+
return (0, child_process_1.execSync)(command, {
|
|
51
|
+
encoding: "utf-8",
|
|
52
|
+
stdio: "pipe",
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
catch (error) {
|
|
56
|
+
throw new Error(`Azure CLI command failed: ${error.message}\n` +
|
|
57
|
+
`Command: ${command}\n` +
|
|
58
|
+
`Stderr: ${error.stderr?.toString() || "N/A"}`);
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Queries Azure Resource Graph for resources by tags
|
|
63
|
+
*
|
|
64
|
+
* Returns empty array on errors for graceful degradation.
|
|
65
|
+
*/
|
|
66
|
+
queryAzureResourcesByTags(tags, subscription) {
|
|
67
|
+
// Build tag filter conditions
|
|
68
|
+
const tagConditions = Object.entries(tags)
|
|
69
|
+
.map(([key, value]) => `tags['${key}'] == '${value}'`)
|
|
70
|
+
.join(" and ");
|
|
71
|
+
const query = `
|
|
72
|
+
Resources
|
|
73
|
+
| where ${tagConditions}
|
|
74
|
+
| project id, type, name, location, resourceGroup, tags
|
|
75
|
+
`;
|
|
76
|
+
const subscriptionArg = subscription
|
|
77
|
+
? `--subscriptions "${subscription}"`
|
|
78
|
+
: "";
|
|
79
|
+
const command = `az graph query -q "${query.replace(/\n/g, " ")}" ${subscriptionArg} --output json`;
|
|
80
|
+
try {
|
|
81
|
+
const result = this.executeAzureCommand(command);
|
|
82
|
+
const parsed = JSON.parse(result);
|
|
83
|
+
return parsed.data || [];
|
|
84
|
+
}
|
|
85
|
+
catch (error) {
|
|
86
|
+
console.error("Failed to query Azure Resource Graph:", error.message);
|
|
87
|
+
return [];
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
exports.ResourceCleanupService = ResourceCleanupService;
|
|
92
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
93
|
+
ResourceCleanupService[_a] = { fqn: "@microsoft/terraform-cdk-constructs.ResourceCleanupService", version: "1.2.0" };
|
|
94
|
+
/**
|
|
95
|
+
* Verifies that all resources with a specific test run ID are deleted
|
|
96
|
+
*
|
|
97
|
+
* Performs Azure Resource Graph query to check for remaining resources.
|
|
98
|
+
* Implements retry logic to handle Azure's eventual consistency.
|
|
99
|
+
*
|
|
100
|
+
* @param runId - Test run UUID to verify
|
|
101
|
+
* @param expectedResources - List of resources that should be deleted
|
|
102
|
+
* @param subscription - Optional subscription ID to query
|
|
103
|
+
* @returns Verification result with details
|
|
104
|
+
*/
|
|
105
|
+
function verifyResourcesDeleted(runId, expectedResources, subscription) {
|
|
106
|
+
const service = new ResourceCleanupService();
|
|
107
|
+
// Check if Azure CLI is available before attempting verification
|
|
108
|
+
if (!service.checkAzureCliAvailable()) {
|
|
109
|
+
return {
|
|
110
|
+
success: false,
|
|
111
|
+
message: "Verification failed: Azure CLI is not available",
|
|
112
|
+
orphanedResources: [],
|
|
113
|
+
expectedCount: expectedResources.length,
|
|
114
|
+
foundCount: -1,
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
try {
|
|
118
|
+
const orphanedResources = service.queryAzureResourcesByTags({ "test:run-id": runId }, subscription);
|
|
119
|
+
if (orphanedResources.length > 0) {
|
|
120
|
+
const resourceIds = orphanedResources.map((r) => r.id);
|
|
121
|
+
return {
|
|
122
|
+
success: false,
|
|
123
|
+
message: `Found ${orphanedResources.length} orphaned resources after destroy`,
|
|
124
|
+
orphanedResources: resourceIds,
|
|
125
|
+
expectedCount: expectedResources.length,
|
|
126
|
+
foundCount: orphanedResources.length,
|
|
127
|
+
};
|
|
128
|
+
}
|
|
129
|
+
return {
|
|
130
|
+
success: true,
|
|
131
|
+
message: "All resources successfully deleted",
|
|
132
|
+
orphanedResources: [],
|
|
133
|
+
expectedCount: expectedResources.length,
|
|
134
|
+
foundCount: 0,
|
|
135
|
+
};
|
|
136
|
+
}
|
|
137
|
+
catch (error) {
|
|
138
|
+
return {
|
|
139
|
+
success: false,
|
|
140
|
+
message: `Verification failed: ${error.message}`,
|
|
141
|
+
orphanedResources: [],
|
|
142
|
+
expectedCount: expectedResources.length,
|
|
143
|
+
foundCount: -1,
|
|
144
|
+
};
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
/**
|
|
148
|
+
* Finds orphaned test resources that are past their TTL
|
|
149
|
+
*
|
|
150
|
+
* Queries Azure for resources with test tags that are older than
|
|
151
|
+
* their cleanup-after timestamp.
|
|
152
|
+
*
|
|
153
|
+
* @param options - Cleanup options
|
|
154
|
+
* @returns List of orphaned resources
|
|
155
|
+
*/
|
|
156
|
+
async function findOrphanedResources(options) {
|
|
157
|
+
const now = new Date();
|
|
158
|
+
console.log(`[Cleanup] Searching for orphaned test resources...`);
|
|
159
|
+
console.log(`[Cleanup] Minimum age: ${options.minAgeHours} hours`);
|
|
160
|
+
// Query for test resources
|
|
161
|
+
const query = `
|
|
162
|
+
Resources
|
|
163
|
+
| where tags['test:managed-by'] == 'terraform-cdk-constructs-tests'
|
|
164
|
+
| where tags['test:auto-cleanup'] == 'true'
|
|
165
|
+
| project id, type, name, location, resourceGroup,
|
|
166
|
+
testRunId = tags['test:run-id'],
|
|
167
|
+
createdAt = tags['test:created-at'],
|
|
168
|
+
cleanupAfter = tags['test:cleanup-after'],
|
|
169
|
+
testName = tags['test:name']
|
|
170
|
+
| order by tostring(createdAt) asc
|
|
171
|
+
`;
|
|
172
|
+
try {
|
|
173
|
+
const subscriptionArg = options.subscription
|
|
174
|
+
? `--subscriptions "${options.subscription}"`
|
|
175
|
+
: "";
|
|
176
|
+
const command = `az graph query -q "${query.replace(/\n/g, " ")}" ${subscriptionArg} --output json`;
|
|
177
|
+
const result = (0, child_process_1.execSync)(command, {
|
|
178
|
+
encoding: "utf-8",
|
|
179
|
+
stdio: "pipe",
|
|
180
|
+
});
|
|
181
|
+
const parsed = JSON.parse(result);
|
|
182
|
+
const resources = parsed.data || [];
|
|
183
|
+
console.log(`[Cleanup] Found ${resources.length} test resources`);
|
|
184
|
+
// Filter by age and cleanup eligibility
|
|
185
|
+
const orphaned = [];
|
|
186
|
+
for (const resource of resources) {
|
|
187
|
+
const createdAt = new Date(resource.createdAt);
|
|
188
|
+
const cleanupAfter = new Date(resource.cleanupAfter);
|
|
189
|
+
const ageHours = (now.getTime() - createdAt.getTime()) / (60 * 60 * 1000);
|
|
190
|
+
// Apply safety checks
|
|
191
|
+
if (ageHours < MINIMUM_AGE_HOURS) {
|
|
192
|
+
console.log(`[Cleanup] Skipping ${resource.name}: too young (${ageHours.toFixed(1)}h < ${MINIMUM_AGE_HOURS}h)`);
|
|
193
|
+
continue;
|
|
194
|
+
}
|
|
195
|
+
if (now < cleanupAfter) {
|
|
196
|
+
console.log(`[Cleanup] Skipping ${resource.name}: not yet eligible for cleanup`);
|
|
197
|
+
continue;
|
|
198
|
+
}
|
|
199
|
+
if (ageHours < options.minAgeHours) {
|
|
200
|
+
console.log(`[Cleanup] Skipping ${resource.name}: younger than min age (${ageHours.toFixed(1)}h < ${options.minAgeHours}h)`);
|
|
201
|
+
continue;
|
|
202
|
+
}
|
|
203
|
+
orphaned.push({
|
|
204
|
+
resourceId: resource.id,
|
|
205
|
+
resourceType: resource.type,
|
|
206
|
+
name: resource.name,
|
|
207
|
+
location: resource.location,
|
|
208
|
+
resourceGroup: resource.resourceGroup,
|
|
209
|
+
testRunId: resource.testRunId,
|
|
210
|
+
createdAt,
|
|
211
|
+
cleanupAfter,
|
|
212
|
+
testName: resource.testName,
|
|
213
|
+
ageHours,
|
|
214
|
+
});
|
|
215
|
+
}
|
|
216
|
+
console.log(`[Cleanup] Found ${orphaned.length} orphaned resources eligible for cleanup`);
|
|
217
|
+
// Limit resources if specified
|
|
218
|
+
if (options.maxResources && orphaned.length > options.maxResources) {
|
|
219
|
+
console.log(`[Cleanup] Limiting to ${options.maxResources} resources`);
|
|
220
|
+
return orphaned.slice(0, options.maxResources);
|
|
221
|
+
}
|
|
222
|
+
return orphaned;
|
|
223
|
+
}
|
|
224
|
+
catch (error) {
|
|
225
|
+
console.error("[Cleanup] Failed to find orphaned resources:", error.message);
|
|
226
|
+
throw error;
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* Cleans up orphaned test resources
|
|
231
|
+
*
|
|
232
|
+
* Deletes resource groups containing orphaned test resources.
|
|
233
|
+
* Implements safety checks and dry-run mode.
|
|
234
|
+
*
|
|
235
|
+
* @param resources - List of orphaned resources to clean up
|
|
236
|
+
* @param options - Cleanup options
|
|
237
|
+
* @returns Cleanup result with statistics
|
|
238
|
+
*/
|
|
239
|
+
async function cleanupOrphanedResources(resources, options) {
|
|
240
|
+
if (resources.length === 0) {
|
|
241
|
+
console.log("[Cleanup] No resources to clean up");
|
|
242
|
+
return { deleted: 0, failed: 0, skipped: 0 };
|
|
243
|
+
}
|
|
244
|
+
if (options.dryRun) {
|
|
245
|
+
console.log("[Cleanup] DRY RUN - Would delete the following resources:");
|
|
246
|
+
resources.forEach((r) => {
|
|
247
|
+
console.log(` - ${r.name} (${r.resourceType}) in ${r.resourceGroup} - Age: ${r.ageHours.toFixed(1)}h`);
|
|
248
|
+
});
|
|
249
|
+
return { deleted: 0, failed: 0, skipped: resources.length };
|
|
250
|
+
}
|
|
251
|
+
// Group resources by resource group for efficient deletion
|
|
252
|
+
const byResourceGroup = new Map();
|
|
253
|
+
for (const resource of resources) {
|
|
254
|
+
const group = byResourceGroup.get(resource.resourceGroup) || [];
|
|
255
|
+
group.push(resource);
|
|
256
|
+
byResourceGroup.set(resource.resourceGroup, group);
|
|
257
|
+
}
|
|
258
|
+
console.log(`[Cleanup] Deleting ${byResourceGroup.size} resource groups containing ${resources.length} resources`);
|
|
259
|
+
let deleted = 0;
|
|
260
|
+
let failed = 0;
|
|
261
|
+
const errors = [];
|
|
262
|
+
for (const [rgName, rgResources] of byResourceGroup) {
|
|
263
|
+
try {
|
|
264
|
+
console.log(`[Cleanup] Deleting resource group: ${rgName} (${rgResources.length} resources)`);
|
|
265
|
+
// Validate that all resources in the group are test resources
|
|
266
|
+
const allTestResources = rgResources.every((r) => r.testRunId && r.ageHours >= MINIMUM_AGE_HOURS);
|
|
267
|
+
if (!allTestResources) {
|
|
268
|
+
console.warn(`[Cleanup] Skipping ${rgName}: not all resources are valid test resources`);
|
|
269
|
+
failed += rgResources.length;
|
|
270
|
+
errors.push(`Resource group ${rgName} contains non-test resources`);
|
|
271
|
+
continue;
|
|
272
|
+
}
|
|
273
|
+
// Delete resource group (async, no-wait)
|
|
274
|
+
(0, child_process_1.execSync)(`az group delete --name "${rgName}" --yes --no-wait`, {
|
|
275
|
+
encoding: "utf-8",
|
|
276
|
+
stdio: "pipe",
|
|
277
|
+
});
|
|
278
|
+
deleted += rgResources.length;
|
|
279
|
+
console.log(`[Cleanup] ✓ Initiated deletion of ${rgName}`);
|
|
280
|
+
}
|
|
281
|
+
catch (error) {
|
|
282
|
+
console.error(`[Cleanup] ✗ Failed to delete ${rgName}:`, error.message);
|
|
283
|
+
failed += rgResources.length;
|
|
284
|
+
errors.push(`Failed to delete ${rgName}: ${error.message}`);
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
const result = {
|
|
288
|
+
deleted,
|
|
289
|
+
failed,
|
|
290
|
+
skipped: 0,
|
|
291
|
+
errors: errors.length > 0 ? errors : undefined,
|
|
292
|
+
};
|
|
293
|
+
console.log("[Cleanup] Summary:");
|
|
294
|
+
console.log(` - Deleted: ${result.deleted} resources`);
|
|
295
|
+
console.log(` - Failed: ${result.failed} resources`);
|
|
296
|
+
if (result.errors) {
|
|
297
|
+
console.log(` - Errors: ${result.errors.length}`);
|
|
298
|
+
}
|
|
299
|
+
return result;
|
|
300
|
+
}
|
|
301
|
+
/**
|
|
302
|
+
* Verifies resource cleanup with retry logic for eventual consistency
|
|
303
|
+
*
|
|
304
|
+
* Wrapper around verifyResourcesDeleted that implements retry logic
|
|
305
|
+
* to handle Azure's eventual consistency delays.
|
|
306
|
+
*
|
|
307
|
+
* @param runId - Test run UUID to verify
|
|
308
|
+
* @param expectedResources - List of resources that should be deleted
|
|
309
|
+
* @param subscription - Optional subscription ID to query
|
|
310
|
+
* @param maxRetries - Maximum number of retry attempts
|
|
311
|
+
* @param retryWaitMs - Wait time between retries in milliseconds
|
|
312
|
+
* @returns Final verification result
|
|
313
|
+
*/
|
|
314
|
+
async function verifyResourcesDeletedWithRetry(runId, expectedResources, subscription, maxRetries = MAX_RETRIES, retryWaitMs = RETRY_WAIT_MS) {
|
|
315
|
+
let lastResult;
|
|
316
|
+
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
|
317
|
+
console.log(`[Verification] Attempt ${attempt}/${maxRetries}`);
|
|
318
|
+
lastResult = verifyResourcesDeleted(runId, expectedResources, subscription);
|
|
319
|
+
if (lastResult.success) {
|
|
320
|
+
console.log("[Verification] ✓ All resources confirmed deleted");
|
|
321
|
+
return lastResult;
|
|
322
|
+
}
|
|
323
|
+
if (attempt < maxRetries) {
|
|
324
|
+
console.log(`[Verification] Found ${lastResult.foundCount} resources, retrying in ${retryWaitMs / 1000}s...`);
|
|
325
|
+
await new Promise((resolve) => setTimeout(resolve, retryWaitMs));
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
console.error(`[Verification] ✗ Failed after ${maxRetries} attempts: ${lastResult.message}`);
|
|
329
|
+
return lastResult;
|
|
330
|
+
}
|
|
331
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cleanup.js","sourceRoot":"","sources":["../../../src/testing/lib/cleanup.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Cleanup and verification utilities for integration tests\n *\n * Provides functions for verifying resource deletion via Azure Resource Graph\n * and finding/cleaning orphaned resources with comprehensive safety mechanisms.\n */\n\nimport { execSync } from \"child_process\";\n\n/**\n * Metadata for an individual resource created during testing\n */\nexport interface ResourceMetadata {\n  /**\n   * Azure resource ID\n   */\n  readonly resourceId: string;\n\n  /**\n   * Azure resource type (e.g., 'Microsoft.Resources/resourceGroups')\n   */\n  readonly resourceType: string;\n\n  /**\n   * Resource name\n   */\n  readonly name: string;\n\n  /**\n   * Resource location (Azure region)\n   */\n  readonly location?: string;\n\n  /**\n   * Resource tags\n   */\n  readonly tags: Record<string, string>;\n\n  /**\n   * Test run ID that created this resource\n   */\n  readonly testRunId: string;\n\n  /**\n   * Timestamp when resource was created\n   */\n  readonly createdAt: Date;\n\n  /**\n   * Whether this resource was successfully destroyed\n   */\n  readonly destroyed?: boolean;\n\n  /**\n   * Timestamp when resource was destroyed\n   */\n  readonly destroyedAt?: Date;\n}\n\n/**\n * Result of resource cleanup verification\n */\nexport interface VerificationResult {\n  /**\n   * Whether verification succeeded (all resources deleted)\n   */\n  readonly success: boolean;\n\n  /**\n   * Human-readable message describing the result\n   */\n  readonly message: string;\n\n  /**\n   * List of orphaned resource IDs (if any)\n   */\n  readonly orphanedResources: string[];\n\n  /**\n   * Number of resources expected to be deleted\n   */\n  readonly expectedCount?: number;\n\n  /**\n   * Number of resources actually found\n   */\n  readonly foundCount?: number;\n}\n\n/**\n * Options for cleanup operations\n */\nexport interface CleanupOptions {\n  /**\n   * Whether to perform dry-run (no actual deletion)\n   */\n  readonly dryRun: boolean;\n\n  /**\n   * Minimum age in hours before resource can be cleaned up\n   */\n  readonly minAgeHours: number;\n\n  /**\n   * Azure subscription ID to target (optional)\n   */\n  readonly subscription?: string;\n\n  /**\n   * Specific resource groups to target (optional)\n   */\n  readonly resourceGroups?: string[];\n\n  /**\n   * Maximum number of resources to clean up in one operation\n   */\n  readonly maxResources?: number;\n}\n\n/**\n * Orphaned resource information\n */\nexport interface OrphanedResource {\n  /**\n   * Azure resource ID\n   */\n  readonly resourceId: string;\n\n  /**\n   * Azure resource type\n   */\n  readonly resourceType: string;\n\n  /**\n   * Resource name\n   */\n  readonly name: string;\n\n  /**\n   * Resource location\n   */\n  readonly location: string;\n\n  /**\n   * Resource group name\n   */\n  readonly resourceGroup: string;\n\n  /**\n   * Test run ID from tags\n   */\n  readonly testRunId: string;\n\n  /**\n   * Creation timestamp from tags\n   */\n  readonly createdAt: Date;\n\n  /**\n   * Cleanup after timestamp from tags\n   */\n  readonly cleanupAfter: Date;\n\n  /**\n   * Test name from tags\n   */\n  readonly testName: string;\n\n  /**\n   * Age in hours since creation\n   */\n  readonly ageHours: number;\n}\n\n/**\n * Result of cleanup operation\n */\nexport interface CleanupResult {\n  /**\n   * Number of resources successfully deleted\n   */\n  readonly deleted: number;\n\n  /**\n   * Number of resources that failed to delete\n   */\n  readonly failed: number;\n\n  /**\n   * Number of resources skipped (e.g., too young)\n   */\n  readonly skipped: number;\n\n  /**\n   * Detailed error messages for failures\n   */\n  readonly errors?: string[];\n}\n\n/**\n * Minimum age threshold (hours) before cleanup is allowed\n */\nconst MINIMUM_AGE_HOURS = 2;\n\n/**\n * Maximum retry attempts for eventual consistency\n */\nconst MAX_RETRIES = 5;\n\n/**\n * Wait time between retries (milliseconds)\n */\nconst RETRY_WAIT_MS = 30000; // 30 seconds\n\n/**\n * Main cleanup service class\n */\nexport class ResourceCleanupService {\n  /**\n   * Checks if Azure CLI is available\n   */\n  public checkAzureCliAvailable(): boolean {\n    try {\n      execSync(\"az --version\", { stdio: \"pipe\" });\n      return true;\n    } catch (error) {\n      return false;\n    }\n  }\n\n  /**\n   * Executes an Azure CLI command with error handling\n   */\n  private executeAzureCommand(command: string): string {\n    try {\n      return execSync(command, {\n        encoding: \"utf-8\",\n        stdio: \"pipe\",\n      });\n    } catch (error: any) {\n      throw new Error(\n        `Azure CLI command failed: ${error.message}\\n` +\n          `Command: ${command}\\n` +\n          `Stderr: ${error.stderr?.toString() || \"N/A\"}`,\n      );\n    }\n  }\n\n  /**\n   * Queries Azure Resource Graph for resources by tags\n   *\n   * Returns empty array on errors for graceful degradation.\n   */\n  public queryAzureResourcesByTags(\n    tags: Record<string, string>,\n    subscription?: string,\n  ): any[] {\n    // Build tag filter conditions\n    const tagConditions = Object.entries(tags)\n      .map(([key, value]) => `tags['${key}'] == '${value}'`)\n      .join(\" and \");\n\n    const query = `\n      Resources\n      | where ${tagConditions}\n      | project id, type, name, location, resourceGroup, tags\n    `;\n\n    const subscriptionArg = subscription\n      ? `--subscriptions \"${subscription}\"`\n      : \"\";\n\n    const command = `az graph query -q \"${query.replace(/\\n/g, \" \")}\" ${subscriptionArg} --output json`;\n\n    try {\n      const result = this.executeAzureCommand(command);\n      const parsed = JSON.parse(result);\n      return parsed.data || [];\n    } catch (error: any) {\n      console.error(\"Failed to query Azure Resource Graph:\", error.message);\n      return [];\n    }\n  }\n}\n\n/**\n * Verifies that all resources with a specific test run ID are deleted\n *\n * Performs Azure Resource Graph query to check for remaining resources.\n * Implements retry logic to handle Azure's eventual consistency.\n *\n * @param runId - Test run UUID to verify\n * @param expectedResources - List of resources that should be deleted\n * @param subscription - Optional subscription ID to query\n * @returns Verification result with details\n */\nexport function verifyResourcesDeleted(\n  runId: string,\n  expectedResources: ResourceMetadata[],\n  subscription?: string,\n): VerificationResult {\n  const service = new ResourceCleanupService();\n\n  // Check if Azure CLI is available before attempting verification\n  if (!service.checkAzureCliAvailable()) {\n    return {\n      success: false,\n      message: \"Verification failed: Azure CLI is not available\",\n      orphanedResources: [],\n      expectedCount: expectedResources.length,\n      foundCount: -1,\n    };\n  }\n\n  try {\n    const orphanedResources = service.queryAzureResourcesByTags(\n      { \"test:run-id\": runId },\n      subscription,\n    );\n\n    if (orphanedResources.length > 0) {\n      const resourceIds = orphanedResources.map((r: any) => r.id);\n      return {\n        success: false,\n        message: `Found ${orphanedResources.length} orphaned resources after destroy`,\n        orphanedResources: resourceIds,\n        expectedCount: expectedResources.length,\n        foundCount: orphanedResources.length,\n      };\n    }\n\n    return {\n      success: true,\n      message: \"All resources successfully deleted\",\n      orphanedResources: [],\n      expectedCount: expectedResources.length,\n      foundCount: 0,\n    };\n  } catch (error: any) {\n    return {\n      success: false,\n      message: `Verification failed: ${error.message}`,\n      orphanedResources: [],\n      expectedCount: expectedResources.length,\n      foundCount: -1,\n    };\n  }\n}\n\n/**\n * Finds orphaned test resources that are past their TTL\n *\n * Queries Azure for resources with test tags that are older than\n * their cleanup-after timestamp.\n *\n * @param options - Cleanup options\n * @returns List of orphaned resources\n */\nexport async function findOrphanedResources(\n  options: CleanupOptions,\n): Promise<OrphanedResource[]> {\n  const now = new Date();\n\n  console.log(`[Cleanup] Searching for orphaned test resources...`);\n  console.log(`[Cleanup] Minimum age: ${options.minAgeHours} hours`);\n\n  // Query for test resources\n  const query = `\n    Resources\n    | where tags['test:managed-by'] == 'terraform-cdk-constructs-tests'\n    | where tags['test:auto-cleanup'] == 'true'\n    | project id, type, name, location, resourceGroup,\n              testRunId = tags['test:run-id'],\n              createdAt = tags['test:created-at'],\n              cleanupAfter = tags['test:cleanup-after'],\n              testName = tags['test:name']\n    | order by tostring(createdAt) asc\n  `;\n\n  try {\n    const subscriptionArg = options.subscription\n      ? `--subscriptions \"${options.subscription}\"`\n      : \"\";\n\n    const command = `az graph query -q \"${query.replace(/\\n/g, \" \")}\" ${subscriptionArg} --output json`;\n\n    const result = execSync(command, {\n      encoding: \"utf-8\",\n      stdio: \"pipe\",\n    });\n\n    const parsed = JSON.parse(result);\n    const resources = parsed.data || [];\n\n    console.log(`[Cleanup] Found ${resources.length} test resources`);\n\n    // Filter by age and cleanup eligibility\n    const orphaned: OrphanedResource[] = [];\n\n    for (const resource of resources) {\n      const createdAt = new Date(resource.createdAt);\n      const cleanupAfter = new Date(resource.cleanupAfter);\n      const ageHours = (now.getTime() - createdAt.getTime()) / (60 * 60 * 1000);\n\n      // Apply safety checks\n      if (ageHours < MINIMUM_AGE_HOURS) {\n        console.log(\n          `[Cleanup] Skipping ${resource.name}: too young (${ageHours.toFixed(1)}h < ${MINIMUM_AGE_HOURS}h)`,\n        );\n        continue;\n      }\n\n      if (now < cleanupAfter) {\n        console.log(\n          `[Cleanup] Skipping ${resource.name}: not yet eligible for cleanup`,\n        );\n        continue;\n      }\n\n      if (ageHours < options.minAgeHours) {\n        console.log(\n          `[Cleanup] Skipping ${resource.name}: younger than min age (${ageHours.toFixed(1)}h < ${options.minAgeHours}h)`,\n        );\n        continue;\n      }\n\n      orphaned.push({\n        resourceId: resource.id,\n        resourceType: resource.type,\n        name: resource.name,\n        location: resource.location,\n        resourceGroup: resource.resourceGroup,\n        testRunId: resource.testRunId,\n        createdAt,\n        cleanupAfter,\n        testName: resource.testName,\n        ageHours,\n      });\n    }\n\n    console.log(\n      `[Cleanup] Found ${orphaned.length} orphaned resources eligible for cleanup`,\n    );\n\n    // Limit resources if specified\n    if (options.maxResources && orphaned.length > options.maxResources) {\n      console.log(`[Cleanup] Limiting to ${options.maxResources} resources`);\n      return orphaned.slice(0, options.maxResources);\n    }\n\n    return orphaned;\n  } catch (error: any) {\n    console.error(\n      \"[Cleanup] Failed to find orphaned resources:\",\n      error.message,\n    );\n    throw error;\n  }\n}\n\n/**\n * Cleans up orphaned test resources\n *\n * Deletes resource groups containing orphaned test resources.\n * Implements safety checks and dry-run mode.\n *\n * @param resources - List of orphaned resources to clean up\n * @param options - Cleanup options\n * @returns Cleanup result with statistics\n */\nexport async function cleanupOrphanedResources(\n  resources: OrphanedResource[],\n  options: CleanupOptions,\n): Promise<CleanupResult> {\n  if (resources.length === 0) {\n    console.log(\"[Cleanup] No resources to clean up\");\n    return { deleted: 0, failed: 0, skipped: 0 };\n  }\n\n  if (options.dryRun) {\n    console.log(\"[Cleanup] DRY RUN - Would delete the following resources:\");\n    resources.forEach((r) => {\n      console.log(\n        `  - ${r.name} (${r.resourceType}) in ${r.resourceGroup} - Age: ${r.ageHours.toFixed(1)}h`,\n      );\n    });\n    return { deleted: 0, failed: 0, skipped: resources.length };\n  }\n\n  // Group resources by resource group for efficient deletion\n  const byResourceGroup = new Map<string, OrphanedResource[]>();\n  for (const resource of resources) {\n    const group = byResourceGroup.get(resource.resourceGroup) || [];\n    group.push(resource);\n    byResourceGroup.set(resource.resourceGroup, group);\n  }\n\n  console.log(\n    `[Cleanup] Deleting ${byResourceGroup.size} resource groups containing ${resources.length} resources`,\n  );\n\n  let deleted = 0;\n  let failed = 0;\n  const errors: string[] = [];\n\n  for (const [rgName, rgResources] of byResourceGroup) {\n    try {\n      console.log(\n        `[Cleanup] Deleting resource group: ${rgName} (${rgResources.length} resources)`,\n      );\n\n      // Validate that all resources in the group are test resources\n      const allTestResources = rgResources.every(\n        (r) => r.testRunId && r.ageHours >= MINIMUM_AGE_HOURS,\n      );\n\n      if (!allTestResources) {\n        console.warn(\n          `[Cleanup] Skipping ${rgName}: not all resources are valid test resources`,\n        );\n        failed += rgResources.length;\n        errors.push(`Resource group ${rgName} contains non-test resources`);\n        continue;\n      }\n\n      // Delete resource group (async, no-wait)\n      execSync(`az group delete --name \"${rgName}\" --yes --no-wait`, {\n        encoding: \"utf-8\",\n        stdio: \"pipe\",\n      });\n\n      deleted += rgResources.length;\n      console.log(`[Cleanup] ✓ Initiated deletion of ${rgName}`);\n    } catch (error: any) {\n      console.error(`[Cleanup] ✗ Failed to delete ${rgName}:`, error.message);\n      failed += rgResources.length;\n      errors.push(`Failed to delete ${rgName}: ${error.message}`);\n    }\n  }\n\n  const result: CleanupResult = {\n    deleted,\n    failed,\n    skipped: 0,\n    errors: errors.length > 0 ? errors : undefined,\n  };\n\n  console.log(\"[Cleanup] Summary:\");\n  console.log(`  - Deleted: ${result.deleted} resources`);\n  console.log(`  - Failed: ${result.failed} resources`);\n  if (result.errors) {\n    console.log(`  - Errors: ${result.errors.length}`);\n  }\n\n  return result;\n}\n\n/**\n * Verifies resource cleanup with retry logic for eventual consistency\n *\n * Wrapper around verifyResourcesDeleted that implements retry logic\n * to handle Azure's eventual consistency delays.\n *\n * @param runId - Test run UUID to verify\n * @param expectedResources - List of resources that should be deleted\n * @param subscription - Optional subscription ID to query\n * @param maxRetries - Maximum number of retry attempts\n * @param retryWaitMs - Wait time between retries in milliseconds\n * @returns Final verification result\n */\nexport async function verifyResourcesDeletedWithRetry(\n  runId: string,\n  expectedResources: ResourceMetadata[],\n  subscription?: string,\n  maxRetries: number = MAX_RETRIES,\n  retryWaitMs: number = RETRY_WAIT_MS,\n): Promise<VerificationResult> {\n  let lastResult: VerificationResult;\n\n  for (let attempt = 1; attempt <= maxRetries; attempt++) {\n    console.log(`[Verification] Attempt ${attempt}/${maxRetries}`);\n\n    lastResult = verifyResourcesDeleted(runId, expectedResources, subscription);\n\n    if (lastResult.success) {\n      console.log(\"[Verification] ✓ All resources confirmed deleted\");\n      return lastResult;\n    }\n\n    if (attempt < maxRetries) {\n      console.log(\n        `[Verification] Found ${lastResult.foundCount} resources, retrying in ${retryWaitMs / 1000}s...`,\n      );\n      await new Promise((resolve) => setTimeout(resolve, retryWaitMs));\n    }\n  }\n\n  console.error(\n    `[Verification] ✗ Failed after ${maxRetries} attempts: ${lastResult!.message}`,\n  );\n  return lastResult!;\n}\n"]}
|