@microsoft/terraform-cdk-constructs 1.9.0 → 2.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.
Files changed (246) hide show
  1. package/.jsii +38110 -28389
  2. package/API.md +40683 -30877
  3. package/lib/azure-actiongroup/lib/action-group.d.ts +3 -3
  4. package/lib/azure-actiongroup/lib/action-group.js +5 -5
  5. package/lib/azure-actiongroup/test/action-group.integ.d.ts +1 -1
  6. package/lib/azure-actiongroup/test/action-group.integ.js +5 -5
  7. package/lib/azure-actiongroup/test/action-group.spec.js +9 -9
  8. package/lib/azure-activitylogalert/lib/activity-log-alert.d.ts +3 -3
  9. package/lib/azure-activitylogalert/lib/activity-log-alert.js +5 -5
  10. package/lib/azure-activitylogalert/test/activity-log-alert.integ.d.ts +1 -1
  11. package/lib/azure-activitylogalert/test/activity-log-alert.integ.js +6 -6
  12. package/lib/azure-activitylogalert/test/activity-log-alert.spec.js +9 -9
  13. package/lib/azure-aks/lib/aks-cluster.d.ts +6 -6
  14. package/lib/azure-aks/lib/aks-cluster.js +8 -8
  15. package/lib/azure-aks/test/aks-cluster.integ.d.ts +1 -1
  16. package/lib/azure-aks/test/aks-cluster.integ.js +5 -5
  17. package/lib/azure-aks/test/aks-cluster.spec.js +12 -12
  18. package/lib/azure-containerapps/index.d.ts +1 -0
  19. package/lib/azure-containerapps/index.js +18 -0
  20. package/lib/azure-containerapps/lib/container-app-environment-schemas.d.ts +50 -0
  21. package/lib/azure-containerapps/lib/container-app-environment-schemas.js +522 -0
  22. package/lib/azure-containerapps/lib/container-app-environment.d.ts +374 -0
  23. package/lib/azure-containerapps/lib/container-app-environment.js +310 -0
  24. package/lib/azure-containerapps/lib/container-app-schemas.d.ts +48 -0
  25. package/lib/azure-containerapps/lib/container-app-schemas.js +466 -0
  26. package/lib/azure-containerapps/lib/container-app.d.ts +925 -0
  27. package/lib/azure-containerapps/lib/container-app.js +320 -0
  28. package/lib/azure-containerapps/lib/index.d.ts +4 -0
  29. package/lib/azure-containerapps/lib/index.js +21 -0
  30. package/lib/azure-containerapps/test/container-app-environment.integ.d.ts +10 -0
  31. package/lib/azure-containerapps/test/container-app-environment.integ.js +82 -0
  32. package/lib/azure-containerapps/test/container-app-environment.spec.d.ts +9 -0
  33. package/lib/azure-containerapps/test/container-app-environment.spec.js +412 -0
  34. package/lib/azure-containerapps/test/container-app.integ.d.ts +10 -0
  35. package/lib/azure-containerapps/test/container-app.integ.js +119 -0
  36. package/lib/azure-containerapps/test/container-app.spec.d.ts +9 -0
  37. package/lib/azure-containerapps/test/container-app.spec.js +839 -0
  38. package/lib/azure-diagnosticsettings/lib/diagnostic-settings.d.ts +3 -3
  39. package/lib/azure-diagnosticsettings/lib/diagnostic-settings.js +5 -5
  40. package/lib/azure-diagnosticsettings/test/diagnostic-settings.spec.js +9 -9
  41. package/lib/azure-dnsforwardingruleset/lib/dns-forwarding-ruleset.d.ts +7 -7
  42. package/lib/azure-dnsforwardingruleset/lib/dns-forwarding-ruleset.js +9 -9
  43. package/lib/azure-dnsforwardingruleset/lib/forwarding-rule.d.ts +4 -4
  44. package/lib/azure-dnsforwardingruleset/lib/forwarding-rule.js +6 -6
  45. package/lib/azure-dnsforwardingruleset/lib/virtual-network-link.d.ts +4 -4
  46. package/lib/azure-dnsforwardingruleset/lib/virtual-network-link.js +6 -6
  47. package/lib/azure-dnsforwardingruleset/test/dns-forwarding-ruleset.integ.d.ts +1 -1
  48. package/lib/azure-dnsforwardingruleset/test/dns-forwarding-ruleset.integ.js +5 -5
  49. package/lib/azure-dnsforwardingruleset/test/dns-forwarding-ruleset.spec.js +11 -11
  50. package/lib/azure-dnsforwardingruleset/test/forwarding-rule.spec.js +14 -14
  51. package/lib/azure-dnsforwardingruleset/test/virtual-network-link.spec.js +12 -12
  52. package/lib/azure-dnsresolver/lib/dns-resolver.d.ts +8 -8
  53. package/lib/azure-dnsresolver/lib/dns-resolver.js +10 -10
  54. package/lib/azure-dnsresolver/lib/inbound-endpoint.d.ts +6 -6
  55. package/lib/azure-dnsresolver/lib/inbound-endpoint.js +8 -8
  56. package/lib/azure-dnsresolver/lib/outbound-endpoint.d.ts +7 -7
  57. package/lib/azure-dnsresolver/lib/outbound-endpoint.js +9 -9
  58. package/lib/azure-dnsresolver/test/dns-resolver.integ.d.ts +1 -1
  59. package/lib/azure-dnsresolver/test/dns-resolver.integ.js +5 -5
  60. package/lib/azure-dnsresolver/test/dns-resolver.spec.js +11 -11
  61. package/lib/azure-dnsresolver/test/inbound-endpoint.integ.d.ts +1 -1
  62. package/lib/azure-dnsresolver/test/inbound-endpoint.integ.js +5 -5
  63. package/lib/azure-dnsresolver/test/inbound-endpoint.spec.js +12 -12
  64. package/lib/azure-dnsresolver/test/outbound-endpoint.integ.d.ts +1 -1
  65. package/lib/azure-dnsresolver/test/outbound-endpoint.integ.js +5 -5
  66. package/lib/azure-dnsresolver/test/outbound-endpoint.spec.js +9 -9
  67. package/lib/azure-dnszone/lib/dns-zone.d.ts +6 -6
  68. package/lib/azure-dnszone/lib/dns-zone.js +8 -8
  69. package/lib/azure-dnszone/lib/records/dns-records.d.ts +31 -31
  70. package/lib/azure-dnszone/lib/records/dns-records.js +42 -42
  71. package/lib/azure-dnszone/test/dns-records.integ.d.ts +1 -1
  72. package/lib/azure-dnszone/test/dns-records.integ.js +5 -5
  73. package/lib/azure-dnszone/test/dns-records.spec.js +57 -57
  74. package/lib/azure-dnszone/test/dns-zone.integ.d.ts +1 -1
  75. package/lib/azure-dnszone/test/dns-zone.integ.js +5 -5
  76. package/lib/azure-dnszone/test/dns-zone.spec.js +11 -11
  77. package/lib/azure-loganalyticsworkspace/lib/log-analytics-workspace.d.ts +1 -1
  78. package/lib/azure-loganalyticsworkspace/lib/log-analytics-workspace.js +3 -3
  79. package/lib/azure-loganalyticsworkspace/test/log-analytics-workspace.integ.d.ts +1 -1
  80. package/lib/azure-loganalyticsworkspace/test/log-analytics-workspace.integ.js +6 -6
  81. package/lib/azure-loganalyticsworkspace/test/log-analytics-workspace.spec.js +6 -6
  82. package/lib/azure-metricalert/lib/metric-alert.d.ts +3 -3
  83. package/lib/azure-metricalert/lib/metric-alert.js +5 -5
  84. package/lib/azure-metricalert/test/metric-alert.integ.d.ts +1 -1
  85. package/lib/azure-metricalert/test/metric-alert.integ.js +5 -5
  86. package/lib/azure-metricalert/test/metric-alert.spec.js +9 -9
  87. package/lib/azure-networkinterface/lib/network-interface.d.ts +5 -5
  88. package/lib/azure-networkinterface/lib/network-interface.js +7 -7
  89. package/lib/azure-networkinterface/test/network-interface.integ.d.ts +1 -1
  90. package/lib/azure-networkinterface/test/network-interface.integ.js +5 -5
  91. package/lib/azure-networkinterface/test/network-interface.spec.js +11 -11
  92. package/lib/azure-networksecuritygroup/lib/network-security-group.d.ts +6 -6
  93. package/lib/azure-networksecuritygroup/lib/network-security-group.js +8 -8
  94. package/lib/azure-networksecuritygroup/test/network-security-group.integ.d.ts +1 -1
  95. package/lib/azure-networksecuritygroup/test/network-security-group.integ.js +5 -5
  96. package/lib/azure-networksecuritygroup/test/network-security-group.spec.js +12 -12
  97. package/lib/azure-networkwatcher/lib/network-watcher.d.ts +1 -1
  98. package/lib/azure-networkwatcher/lib/network-watcher.js +3 -3
  99. package/lib/azure-networkwatcher/test/network-watcher.integ.d.ts +1 -1
  100. package/lib/azure-networkwatcher/test/network-watcher.integ.js +6 -6
  101. package/lib/azure-networkwatcher/test/network-watcher.spec.js +6 -6
  102. package/lib/azure-policyassignment/lib/policy-assignment.d.ts +3 -3
  103. package/lib/azure-policyassignment/lib/policy-assignment.js +5 -5
  104. package/lib/azure-policyassignment/test/policy-assignment.integ.d.ts +1 -1
  105. package/lib/azure-policyassignment/test/policy-assignment.integ.js +5 -5
  106. package/lib/azure-policyassignment/test/policy-assignment.spec.js +12 -12
  107. package/lib/azure-policydefinition/lib/policy-definition.d.ts +3 -3
  108. package/lib/azure-policydefinition/lib/policy-definition.js +5 -5
  109. package/lib/azure-policydefinition/test/policy-definition.integ.d.ts +1 -1
  110. package/lib/azure-policydefinition/test/policy-definition.integ.js +5 -5
  111. package/lib/azure-policydefinition/test/policy-definition.spec.js +19 -19
  112. package/lib/azure-policysetdefinition/lib/policy-set-definition.d.ts +1 -1
  113. package/lib/azure-policysetdefinition/lib/policy-set-definition.js +3 -3
  114. package/lib/azure-policysetdefinition/test/policy-set-definition.integ.d.ts +1 -1
  115. package/lib/azure-policysetdefinition/test/policy-set-definition.integ.js +6 -6
  116. package/lib/azure-policysetdefinition/test/policy-set-definition.spec.js +6 -6
  117. package/lib/azure-privatednszone/lib/private-dns-zone.d.ts +10 -10
  118. package/lib/azure-privatednszone/lib/private-dns-zone.js +12 -12
  119. package/lib/azure-privatednszone/lib/records/private-dns-records.d.ts +25 -25
  120. package/lib/azure-privatednszone/lib/records/private-dns-records.js +34 -34
  121. package/lib/azure-privatednszone/test/private-dns-records.integ.d.ts +1 -1
  122. package/lib/azure-privatednszone/test/private-dns-records.integ.js +5 -5
  123. package/lib/azure-privatednszone/test/private-dns-records.spec.js +49 -49
  124. package/lib/azure-privatednszone/test/private-dns-zone.integ.d.ts +1 -1
  125. package/lib/azure-privatednszone/test/private-dns-zone.integ.js +5 -5
  126. package/lib/azure-privatednszone/test/private-dns-zone.spec.js +12 -12
  127. package/lib/azure-privatednszonelink/lib/private-dns-zone-link.d.ts +6 -6
  128. package/lib/azure-privatednszonelink/lib/private-dns-zone-link.js +8 -8
  129. package/lib/azure-privatednszonelink/test/private-dns-zone-link.integ.d.ts +1 -1
  130. package/lib/azure-privatednszonelink/test/private-dns-zone-link.integ.js +5 -5
  131. package/lib/azure-privatednszonelink/test/private-dns-zone-link.spec.js +12 -12
  132. package/lib/azure-publicipaddress/lib/public-ip-address.d.ts +5 -5
  133. package/lib/azure-publicipaddress/lib/public-ip-address.js +7 -7
  134. package/lib/azure-publicipaddress/test/public-ip-address.integ.d.ts +1 -1
  135. package/lib/azure-publicipaddress/test/public-ip-address.integ.js +6 -6
  136. package/lib/azure-publicipaddress/test/public-ip-address.spec.js +11 -11
  137. package/lib/azure-resourcegroup/lib/resource-group.d.ts +5 -5
  138. package/lib/azure-resourcegroup/lib/resource-group.js +7 -7
  139. package/lib/azure-resourcegroup/test/resource-group.integ.d.ts +1 -1
  140. package/lib/azure-resourcegroup/test/resource-group.integ.js +5 -5
  141. package/lib/azure-resourcegroup/test/resource-group.spec.js +14 -14
  142. package/lib/azure-roleassignment/lib/role-assignment.d.ts +3 -3
  143. package/lib/azure-roleassignment/lib/role-assignment.js +5 -5
  144. package/lib/azure-roleassignment/test/role-assignment.integ.d.ts +1 -1
  145. package/lib/azure-roleassignment/test/role-assignment.integ.js +5 -5
  146. package/lib/azure-roleassignment/test/role-assignment.spec.js +12 -12
  147. package/lib/azure-roledefinition/lib/role-definition.d.ts +3 -3
  148. package/lib/azure-roledefinition/lib/role-definition.js +5 -5
  149. package/lib/azure-roledefinition/test/role-definition.integ.d.ts +1 -1
  150. package/lib/azure-roledefinition/test/role-definition.integ.js +5 -5
  151. package/lib/azure-roledefinition/test/role-definition.spec.js +12 -12
  152. package/lib/azure-storageaccount/lib/storage-account.d.ts +6 -6
  153. package/lib/azure-storageaccount/lib/storage-account.js +8 -8
  154. package/lib/azure-storageaccount/test/storage-account.integ.d.ts +1 -1
  155. package/lib/azure-storageaccount/test/storage-account.integ.js +5 -5
  156. package/lib/azure-storageaccount/test/storage-account.spec.js +11 -11
  157. package/lib/azure-subnet/lib/subnet.d.ts +3 -3
  158. package/lib/azure-subnet/lib/subnet.js +5 -5
  159. package/lib/azure-subnet/test/subnet.integ.d.ts +1 -1
  160. package/lib/azure-subnet/test/subnet.integ.js +5 -5
  161. package/lib/azure-subnet/test/subnet.spec.js +9 -9
  162. package/lib/azure-virtualmachine/lib/virtual-machine.d.ts +6 -6
  163. package/lib/azure-virtualmachine/lib/virtual-machine.js +8 -8
  164. package/lib/azure-virtualmachine/test/virtual-machine.integ.d.ts +1 -1
  165. package/lib/azure-virtualmachine/test/virtual-machine.integ.js +5 -5
  166. package/lib/azure-virtualmachine/test/virtual-machine.spec.js +17 -17
  167. package/lib/azure-virtualnetwork/lib/virtual-network.d.ts +6 -6
  168. package/lib/azure-virtualnetwork/lib/virtual-network.js +8 -8
  169. package/lib/azure-virtualnetwork/test/virtual-network.integ.d.ts +1 -1
  170. package/lib/azure-virtualnetwork/test/virtual-network.integ.js +5 -5
  171. package/lib/azure-virtualnetwork/test/virtual-network.spec.js +12 -12
  172. package/lib/azure-virtualnetworkgateway/lib/virtual-network-gateway.d.ts +5 -5
  173. package/lib/azure-virtualnetworkgateway/lib/virtual-network-gateway.js +7 -7
  174. package/lib/azure-virtualnetworkgateway/test/virtual-network-gateway.integ.d.ts +1 -1
  175. package/lib/azure-virtualnetworkgateway/test/virtual-network-gateway.integ.js +5 -5
  176. package/lib/azure-virtualnetworkgateway/test/virtual-network-gateway.spec.js +13 -13
  177. package/lib/azure-virtualnetworkgatewayconnection/lib/virtual-network-gateway-connection.d.ts +5 -5
  178. package/lib/azure-virtualnetworkgatewayconnection/lib/virtual-network-gateway-connection.js +7 -7
  179. package/lib/azure-virtualnetworkgatewayconnection/test/virtual-network-gateway-connection.integ.d.ts +1 -1
  180. package/lib/azure-virtualnetworkgatewayconnection/test/virtual-network-gateway-connection.integ.js +5 -5
  181. package/lib/azure-virtualnetworkgatewayconnection/test/virtual-network-gateway-connection.spec.js +20 -20
  182. package/lib/azure-virtualnetworkmanager/lib/connectivity-configuration.d.ts +4 -4
  183. package/lib/azure-virtualnetworkmanager/lib/connectivity-configuration.js +6 -6
  184. package/lib/azure-virtualnetworkmanager/lib/ipam-pool-static-cidr.d.ts +4 -4
  185. package/lib/azure-virtualnetworkmanager/lib/ipam-pool-static-cidr.js +6 -6
  186. package/lib/azure-virtualnetworkmanager/lib/ipam-pool.d.ts +4 -4
  187. package/lib/azure-virtualnetworkmanager/lib/ipam-pool.js +6 -6
  188. package/lib/azure-virtualnetworkmanager/lib/network-group-static-member.d.ts +4 -4
  189. package/lib/azure-virtualnetworkmanager/lib/network-group-static-member.js +6 -6
  190. package/lib/azure-virtualnetworkmanager/lib/network-group.d.ts +4 -4
  191. package/lib/azure-virtualnetworkmanager/lib/network-group.js +6 -6
  192. package/lib/azure-virtualnetworkmanager/lib/security-admin-configuration.d.ts +4 -4
  193. package/lib/azure-virtualnetworkmanager/lib/security-admin-configuration.js +6 -6
  194. package/lib/azure-virtualnetworkmanager/lib/security-admin-rule-collection.d.ts +4 -4
  195. package/lib/azure-virtualnetworkmanager/lib/security-admin-rule-collection.js +6 -6
  196. package/lib/azure-virtualnetworkmanager/lib/security-admin-rule.d.ts +4 -4
  197. package/lib/azure-virtualnetworkmanager/lib/security-admin-rule.js +6 -6
  198. package/lib/azure-virtualnetworkmanager/lib/virtual-network-manager.d.ts +7 -7
  199. package/lib/azure-virtualnetworkmanager/lib/virtual-network-manager.js +9 -9
  200. package/lib/azure-virtualnetworkmanager/test/ipam-pool-static-cidr.spec.js +17 -17
  201. package/lib/azure-virtualnetworkmanager/test/ipam-pool.spec.js +15 -15
  202. package/lib/azure-virtualnetworkmanager/test/virtual-network-manager.integ.d.ts +1 -1
  203. package/lib/azure-virtualnetworkmanager/test/virtual-network-manager.integ.js +5 -5
  204. package/lib/azure-virtualnetworkmanager/test/virtual-network-manager.spec.js +57 -57
  205. package/lib/azure-vmss/lib/virtual-machine-scale-set.d.ts +6 -6
  206. package/lib/azure-vmss/lib/virtual-machine-scale-set.js +8 -8
  207. package/lib/azure-vmss/test/virtual-machine-scale-set.integ.d.ts +1 -1
  208. package/lib/azure-vmss/test/virtual-machine-scale-set.integ.js +5 -5
  209. package/lib/azure-vmss/test/virtual-machine-scale-set.spec.js +12 -12
  210. package/lib/core-azure/lib/azapi/azapi-resource-tags.spec.js +6 -6
  211. package/lib/core-azure/lib/azapi/azapi-resource.d.ts +7 -7
  212. package/lib/core-azure/lib/azapi/azapi-resource.js +6 -6
  213. package/lib/core-azure/lib/azapi/azapi-resource.spec.js +7 -7
  214. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-client-config/index.d.ts +12 -12
  215. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-client-config/index.js +15 -15
  216. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource/index.d.ts +35 -35
  217. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource/index.js +58 -58
  218. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-action/index.d.ts +25 -25
  219. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-action/index.js +49 -49
  220. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-id/index.d.ts +13 -13
  221. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-id/index.js +22 -22
  222. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-list/index.d.ts +24 -24
  223. package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource-list/index.js +40 -40
  224. package/lib/core-azure/lib/azapi/providers-azapi/data-plane-resource/index.d.ts +44 -44
  225. package/lib/core-azure/lib/azapi/providers-azapi/data-plane-resource/index.js +72 -72
  226. package/lib/core-azure/lib/azapi/providers-azapi/index.js +2 -2
  227. package/lib/core-azure/lib/azapi/providers-azapi/lazy-index.js +2 -2
  228. package/lib/core-azure/lib/azapi/providers-azapi/provider/index.d.ts +54 -54
  229. package/lib/core-azure/lib/azapi/providers-azapi/provider/index.js +86 -86
  230. package/lib/core-azure/lib/azapi/providers-azapi/resource/index.d.ts +65 -65
  231. package/lib/core-azure/lib/azapi/providers-azapi/resource/index.js +102 -102
  232. package/lib/core-azure/lib/azapi/providers-azapi/resource-action/index.d.ts +25 -25
  233. package/lib/core-azure/lib/azapi/providers-azapi/resource-action/index.js +62 -62
  234. package/lib/core-azure/lib/azapi/providers-azapi/update-resource/index.d.ts +36 -36
  235. package/lib/core-azure/lib/azapi/providers-azapi/update-resource/index.js +69 -69
  236. package/lib/core-azure/lib/azapi/schema-mapper/schema-mapper.js +1 -1
  237. package/lib/core-azure/lib/version-manager/api-version-manager.js +1 -1
  238. package/lib/core-azure/lib/version-manager/interfaces/version-interfaces.js +7 -7
  239. package/lib/index.d.ts +3 -0
  240. package/lib/index.js +5 -2
  241. package/lib/testing/index.d.ts +1 -1
  242. package/lib/testing/index.js +8 -8
  243. package/lib/testing/lib/cleanup.js +1 -1
  244. package/lib/testing/lib/metadata.js +1 -1
  245. package/package.json +10 -22
  246. package/setup.js +2 -2
@@ -0,0 +1,320 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.ContainerApp = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ /**
7
+ * Unified Azure Container App implementation using AzapiResource framework
8
+ *
9
+ * This class provides a single, version-aware implementation that automatically handles
10
+ * version management, schema validation, and property transformation across all
11
+ * supported API versions.
12
+ *
13
+ * Supported API Versions:
14
+ * - 2024-03-01 (Active)
15
+ * - 2025-07-01 (Active, Latest)
16
+ *
17
+ * Features:
18
+ * - Automatic latest version resolution when no version is specified
19
+ * - Explicit version pinning for stability requirements
20
+ * - Schema-driven validation and transformation
21
+ * - Full backward compatibility with existing interface
22
+ * - JSII compliance for multi-language support
23
+ * - Ingress with external/internal access, TLS, CORS, sticky sessions
24
+ * - Dapr sidecar integration for microservice patterns
25
+ * - Horizontal scaling with HTTP, TCP, custom (KEDA), and Azure Queue rules
26
+ * - Secrets management with Key Vault references
27
+ * - Managed identity (SystemAssigned and UserAssigned)
28
+ * - Init containers, volume mounts, and health probes
29
+ * - Multiple revision mode for blue-green deployments
30
+ */
31
+ const cdktn = require("cdktn");
32
+ const container_app_schemas_1 = require("./container-app-schemas");
33
+ const azapi_resource_1 = require("../../core-azure/lib/azapi/azapi-resource");
34
+ // =============================================================================
35
+ // MAIN CLASS
36
+ // =============================================================================
37
+ /**
38
+ * Unified Azure Container App implementation
39
+ *
40
+ * Azure Container Apps enable you to run microservices and containerized applications
41
+ * on a serverless platform. They provide built-in support for ingress, scaling,
42
+ * Dapr integration, and more.
43
+ *
44
+ * Key features:
45
+ * - External and internal ingress with TLS, CORS, and sticky sessions
46
+ * - Dapr sidecar integration for microservice patterns
47
+ * - Horizontal auto-scaling with HTTP, TCP, custom (KEDA), and Azure Queue rules
48
+ * - Secrets management with Azure Key Vault references
49
+ * - Managed identity (SystemAssigned, UserAssigned, or both)
50
+ * - Init containers for setup tasks
51
+ * - Volume mounts (Azure Files, NFS, Ephemeral, Secret)
52
+ * - Health probes (Liveness, Readiness, Startup)
53
+ * - Multiple revision mode for blue-green and canary deployments
54
+ * - Service binds for dev services (Redis, Postgres, Kafka)
55
+ *
56
+ * @example
57
+ * // Basic Container App with a single container:
58
+ * const app = new ContainerApp(this, "app", {
59
+ * name: "my-container-app",
60
+ * location: "eastus",
61
+ * resourceGroupId: resourceGroup.id,
62
+ * environmentId: environment.id,
63
+ * template: {
64
+ * containers: [{
65
+ * name: "my-app",
66
+ * image: "mcr.microsoft.com/azuredocs/containerapps-helloworld:latest",
67
+ * resources: { cpu: 0.25, memory: "0.5Gi" },
68
+ * }],
69
+ * },
70
+ * });
71
+ *
72
+ * @example
73
+ * // Container App with external ingress and scaling:
74
+ * const app = new ContainerApp(this, "app", {
75
+ * name: "my-web-app",
76
+ * location: "eastus",
77
+ * resourceGroupId: resourceGroup.id,
78
+ * environmentId: environment.id,
79
+ * template: {
80
+ * containers: [{
81
+ * name: "web",
82
+ * image: "myregistry.azurecr.io/myapp:latest",
83
+ * resources: { cpu: 0.5, memory: "1Gi" },
84
+ * env: [{ name: "PORT", value: "3000" }],
85
+ * }],
86
+ * scale: {
87
+ * minReplicas: 1,
88
+ * maxReplicas: 10,
89
+ * rules: [{
90
+ * name: "http-rule",
91
+ * http: { metadata: { concurrentRequests: "50" } },
92
+ * }],
93
+ * },
94
+ * },
95
+ * configuration: {
96
+ * ingress: {
97
+ * external: true,
98
+ * targetPort: 3000,
99
+ * transport: "http",
100
+ * },
101
+ * },
102
+ * });
103
+ *
104
+ * @stability stable
105
+ */
106
+ class ContainerApp extends azapi_resource_1.AzapiResource {
107
+ /**
108
+ * Creates a new Azure Container App
109
+ *
110
+ * @param scope - The scope in which to define this construct
111
+ * @param id - The unique identifier for this instance
112
+ * @param props - Configuration properties for the Container App
113
+ */
114
+ constructor(scope, id, props) {
115
+ super(scope, id, props);
116
+ this.props = props;
117
+ // Create Terraform outputs
118
+ this.idOutput = new cdktn.TerraformOutput(this, "id", {
119
+ value: this.id,
120
+ description: "The ID of the Container App",
121
+ });
122
+ this.locationOutput = new cdktn.TerraformOutput(this, "location", {
123
+ value: `\${${this.terraformResource.fqn}.location}`,
124
+ description: "The location of the Container App",
125
+ });
126
+ this.nameOutput = new cdktn.TerraformOutput(this, "name", {
127
+ value: `\${${this.terraformResource.fqn}.name}`,
128
+ description: "The name of the Container App",
129
+ });
130
+ this.tagsOutput = new cdktn.TerraformOutput(this, "tags", {
131
+ value: `\${${this.terraformResource.fqn}.tags}`,
132
+ description: "The tags assigned to the Container App",
133
+ });
134
+ this.fqdnOutput = new cdktn.TerraformOutput(this, "fqdn", {
135
+ value: `\${${this.terraformResource.fqn}.output.properties.configuration.ingress.fqdn}`,
136
+ description: "The FQDN of the Container App (from ingress)",
137
+ });
138
+ this.latestRevisionFqdnOutput = new cdktn.TerraformOutput(this, "latest_revision_fqdn", {
139
+ value: `\${${this.terraformResource.fqn}.output.properties.latestRevisionFqdn}`,
140
+ description: "The FQDN of the latest revision of the Container App",
141
+ });
142
+ this.latestReadyRevisionNameOutput = new cdktn.TerraformOutput(this, "latest_ready_revision_name", {
143
+ value: `\${${this.terraformResource.fqn}.output.properties.latestReadyRevisionName}`,
144
+ description: "The name of the latest ready revision of the Container App",
145
+ });
146
+ this.provisioningStateOutput = new cdktn.TerraformOutput(this, "provisioning_state", {
147
+ value: `\${${this.terraformResource.fqn}.output.properties.provisioningState}`,
148
+ description: "The provisioning state of the Container App",
149
+ });
150
+ // Override logical IDs
151
+ this.idOutput.overrideLogicalId("id");
152
+ this.locationOutput.overrideLogicalId("location");
153
+ this.nameOutput.overrideLogicalId("name");
154
+ this.tagsOutput.overrideLogicalId("tags");
155
+ this.fqdnOutput.overrideLogicalId("fqdn");
156
+ this.latestRevisionFqdnOutput.overrideLogicalId("latest_revision_fqdn");
157
+ this.latestReadyRevisionNameOutput.overrideLogicalId("latest_ready_revision_name");
158
+ this.provisioningStateOutput.overrideLogicalId("provisioning_state");
159
+ // Apply ignore changes if specified
160
+ this._applyIgnoreChanges();
161
+ }
162
+ // =============================================================================
163
+ // REQUIRED ABSTRACT METHODS FROM AzapiResource
164
+ // =============================================================================
165
+ /**
166
+ * Gets the default API version to use when no explicit version is specified
167
+ */
168
+ defaultVersion() {
169
+ return "2025-02-02-preview";
170
+ }
171
+ /**
172
+ * Gets the Azure resource type for Container Apps
173
+ */
174
+ resourceType() {
175
+ return container_app_schemas_1.CONTAINER_APP_TYPE;
176
+ }
177
+ /**
178
+ * Gets the API schema for the resolved version
179
+ */
180
+ apiSchema() {
181
+ return this.resolveSchema();
182
+ }
183
+ /**
184
+ * Indicates that location is required for Container Apps
185
+ */
186
+ requiresLocation() {
187
+ return true;
188
+ }
189
+ /**
190
+ * Creates the resource body for the Azure API call
191
+ */
192
+ createResourceBody(props) {
193
+ const typedProps = props;
194
+ const properties = {
195
+ environmentId: typedProps.environmentId,
196
+ template: typedProps.template,
197
+ };
198
+ // Configuration
199
+ if (typedProps.configuration) {
200
+ properties.configuration = typedProps.configuration;
201
+ }
202
+ // Workload profile name
203
+ if (typedProps.workloadProfileName) {
204
+ properties.workloadProfileName = typedProps.workloadProfileName;
205
+ }
206
+ const body = {
207
+ location: this.location,
208
+ tags: this.allTags(),
209
+ properties,
210
+ };
211
+ // Identity (top-level, not inside properties)
212
+ if (typedProps.identity) {
213
+ body.identity = typedProps.identity;
214
+ }
215
+ return body;
216
+ }
217
+ // =============================================================================
218
+ // PUBLIC METHODS - Read-only Properties
219
+ // =============================================================================
220
+ /**
221
+ * Get the FQDN of the Container App (from ingress configuration)
222
+ */
223
+ get fqdn() {
224
+ return `\${${this.terraformResource.fqn}.output.properties.configuration.ingress.fqdn}`;
225
+ }
226
+ /**
227
+ * Get the FQDN of the latest revision
228
+ */
229
+ get latestRevisionFqdn() {
230
+ return `\${${this.terraformResource.fqn}.output.properties.latestRevisionFqdn}`;
231
+ }
232
+ /**
233
+ * Get the name of the latest ready revision
234
+ */
235
+ get latestReadyRevisionName() {
236
+ return `\${${this.terraformResource.fqn}.output.properties.latestReadyRevisionName}`;
237
+ }
238
+ /**
239
+ * Get the provisioning state of the Container App
240
+ */
241
+ get provisioningState() {
242
+ return `\${${this.terraformResource.fqn}.output.properties.provisioningState}`;
243
+ }
244
+ /**
245
+ * Get the running status of the Container App
246
+ */
247
+ get runningStatus() {
248
+ return `\${${this.terraformResource.fqn}.output.properties.runningStatus}`;
249
+ }
250
+ /**
251
+ * Add a tag to the Container App
252
+ */
253
+ addTag(key, value) {
254
+ if (!this.props.tags) {
255
+ this.props.tags = {};
256
+ }
257
+ this.props.tags[key] = value;
258
+ }
259
+ /**
260
+ * Remove a tag from the Container App
261
+ */
262
+ removeTag(key) {
263
+ if (this.props.tags && this.props.tags[key]) {
264
+ delete this.props.tags[key];
265
+ }
266
+ }
267
+ // =============================================================================
268
+ // RESOURCE CONFIG CUSTOMIZATION
269
+ // =============================================================================
270
+ /**
271
+ * Customizes the resource configuration to handle Azure value normalization.
272
+ *
273
+ * Azure normalizes values in API responses:
274
+ * - Location: "eastus" → "East US"
275
+ * - Enum values: "auto" → "Auto", "enabled" → "Enabled"
276
+ *
277
+ * This override:
278
+ * 1. Removes `location` from the body (the framework passes it as a top-level
279
+ * attribute which the azapi provider normalizes properly)
280
+ * 2. Enables `ignoreCasing` to suppress diffs from Azure's value normalization
281
+ * (e.g., "auto" vs "Auto" for transport, "enabled" vs "Enabled")
282
+ */
283
+ customizeResourceConfig(config) {
284
+ const updatedConfig = { ...config, ignoreCasing: true };
285
+ // Remove location from body to prevent "East US" vs "eastus" drift
286
+ // The framework handles location as a top-level azapi attribute instead
287
+ if (updatedConfig.body && updatedConfig.body.location) {
288
+ const { location: _location, ...bodyWithoutLocation } = updatedConfig.body;
289
+ return {
290
+ ...updatedConfig,
291
+ body: bodyWithoutLocation,
292
+ // Ensure location is set as top-level attribute
293
+ location: updatedConfig.location || _location,
294
+ };
295
+ }
296
+ return updatedConfig;
297
+ }
298
+ // =============================================================================
299
+ // PRIVATE HELPER METHODS
300
+ // =============================================================================
301
+ /**
302
+ * Applies ignore changes lifecycle rules if specified in props
303
+ */
304
+ _applyIgnoreChanges() {
305
+ if (this.props.ignoreChanges && this.props.ignoreChanges.length > 0) {
306
+ this.terraformResource.addOverride("lifecycle", [
307
+ {
308
+ ignore_changes: this.props.ignoreChanges,
309
+ },
310
+ ]);
311
+ }
312
+ }
313
+ }
314
+ exports.ContainerApp = ContainerApp;
315
+ _a = JSII_RTTI_SYMBOL_1;
316
+ ContainerApp[_a] = { fqn: "@microsoft/terraform-cdk-constructs.ContainerApp", version: "2.2.0" };
317
+ (() => {
318
+ azapi_resource_1.AzapiResource.registerSchemas(container_app_schemas_1.CONTAINER_APP_TYPE, container_app_schemas_1.ALL_CONTAINER_APP_VERSIONS);
319
+ })();
320
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"container-app.js","sourceRoot":"","sources":["../../../src/azure-containerapps/lib/container-app.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Unified Azure Container App implementation using AzapiResource framework\n *\n * This class provides a single, version-aware implementation that automatically handles\n * version management, schema validation, and property transformation across all\n * supported API versions.\n *\n * Supported API Versions:\n * - 2024-03-01 (Active)\n * - 2025-07-01 (Active, Latest)\n *\n * Features:\n * - Automatic latest version resolution when no version is specified\n * - Explicit version pinning for stability requirements\n * - Schema-driven validation and transformation\n * - Full backward compatibility with existing interface\n * - JSII compliance for multi-language support\n * - Ingress with external/internal access, TLS, CORS, sticky sessions\n * - Dapr sidecar integration for microservice patterns\n * - Horizontal scaling with HTTP, TCP, custom (KEDA), and Azure Queue rules\n * - Secrets management with Key Vault references\n * - Managed identity (SystemAssigned and UserAssigned)\n * - Init containers, volume mounts, and health probes\n * - Multiple revision mode for blue-green deployments\n */\n\nimport * as cdktn from \"cdktn\";\nimport { Construct } from \"constructs\";\nimport {\n  ALL_CONTAINER_APP_VERSIONS,\n  CONTAINER_APP_TYPE,\n} from \"./container-app-schemas\";\nimport {\n  AzapiResource,\n  AzapiResourceProps,\n} from \"../../core-azure/lib/azapi/azapi-resource\";\nimport { ApiSchema } from \"../../core-azure/lib/version-manager/interfaces/version-interfaces\";\n\n// =============================================================================\n// SUPPORTING INTERFACES\n// =============================================================================\n\n/**\n * Environment variable for a container\n */\nexport interface ContainerAppEnvVar {\n  /**\n   * Environment variable name\n   */\n  readonly name: string;\n\n  /**\n   * Non-secret environment variable value\n   */\n  readonly value?: string;\n\n  /**\n   * Name of the secret from which to pull the value\n   */\n  readonly secretRef?: string;\n}\n\n/**\n * Container resource requirements\n */\nexport interface ContainerAppContainerResources {\n  /**\n   * Required CPU in cores (e.g., 0.25, 0.5, 1.0, 2.0)\n   */\n  readonly cpu?: number;\n\n  /**\n   * Required memory (e.g., \"0.5Gi\", \"1Gi\", \"2Gi\")\n   */\n  readonly memory?: string;\n}\n\n/**\n * Volume mount for a container\n */\nexport interface ContainerAppVolumeMount {\n  /**\n   * Name of the volume to mount (must match a volume name)\n   */\n  readonly volumeName: string;\n\n  /**\n   * Path within the container to mount the volume\n   */\n  readonly mountPath: string;\n\n  /**\n   * Path within the volume from which the container's volume should be mounted\n   */\n  readonly subPath?: string;\n}\n\n/**\n * HTTP header for a probe\n */\nexport interface ContainerAppProbeHttpHeader {\n  /**\n   * The header field name\n   */\n  readonly name: string;\n\n  /**\n   * The header field value\n   */\n  readonly value: string;\n}\n\n/**\n * HTTP GET action for a probe\n */\nexport interface ContainerAppProbeHttpGet {\n  /**\n   * Path to access on the HTTP server\n   */\n  readonly path?: string;\n\n  /**\n   * Port number to access on the container\n   */\n  readonly port: number;\n\n  /**\n   * Scheme to use for connecting to the host (HTTP or HTTPS)\n   */\n  readonly scheme?: string;\n\n  /**\n   * Custom headers to set in the request\n   */\n  readonly httpHeaders?: ContainerAppProbeHttpHeader[];\n}\n\n/**\n * TCP socket action for a probe\n */\nexport interface ContainerAppProbeTcpSocket {\n  /**\n   * TCP port to connect to\n   */\n  readonly port: number;\n}\n\n/**\n * Container probe (liveness, readiness, or startup)\n */\nexport interface ContainerAppProbe {\n  /**\n   * The type of probe: Liveness, Readiness, or Startup\n   */\n  readonly type?: string;\n\n  /**\n   * HTTPGet specifies the HTTP request to perform\n   */\n  readonly httpGet?: ContainerAppProbeHttpGet;\n\n  /**\n   * TCPSocket specifies an action involving a TCP port\n   */\n  readonly tcpSocket?: ContainerAppProbeTcpSocket;\n\n  /**\n   * Number of seconds after the container has started before probes are initiated\n   */\n  readonly initialDelaySeconds?: number;\n\n  /**\n   * How often (in seconds) to perform the probe\n   */\n  readonly periodSeconds?: number;\n\n  /**\n   * Number of seconds after which the probe times out\n   */\n  readonly timeoutSeconds?: number;\n\n  /**\n   * Minimum consecutive successes for the probe to be considered successful\n   */\n  readonly successThreshold?: number;\n\n  /**\n   * Minimum consecutive failures for the probe to be considered failed\n   */\n  readonly failureThreshold?: number;\n}\n\n/**\n * Container definition\n */\nexport interface ContainerAppContainer {\n  /**\n   * Container image tag\n   */\n  readonly image: string;\n\n  /**\n   * Custom container name\n   */\n  readonly name: string;\n\n  /**\n   * Container resource requirements\n   */\n  readonly resources?: ContainerAppContainerResources;\n\n  /**\n   * Container environment variables\n   */\n  readonly env?: ContainerAppEnvVar[];\n\n  /**\n   * Container start command\n   */\n  readonly command?: string[];\n\n  /**\n   * Container start command arguments\n   */\n  readonly args?: string[];\n\n  /**\n   * List of probes for the container\n   */\n  readonly probes?: ContainerAppProbe[];\n\n  /**\n   * Container volume mounts\n   */\n  readonly volumeMounts?: ContainerAppVolumeMount[];\n}\n\n/**\n * Azure Queue scale rule configuration\n */\nexport interface ContainerAppAzureQueueScaleRule {\n  /**\n   * Azure Storage account name\n   */\n  readonly accountName?: string;\n\n  /**\n   * Azure Storage queue name\n   */\n  readonly queueName?: string;\n\n  /**\n   * Queue length threshold to trigger scaling\n   */\n  readonly queueLength?: number;\n\n  /**\n   * Managed identity to authenticate with the storage account\n   */\n  readonly identity?: string;\n}\n\n/**\n * Custom (KEDA) scale rule configuration\n */\nexport interface ContainerAppCustomScaleRule {\n  /**\n   * KEDA trigger type\n   */\n  readonly type: string;\n\n  /**\n   * Trigger metadata as key-value pairs\n   */\n  readonly metadata?: { [key: string]: string };\n\n  /**\n   * Managed identity for the custom scaler\n   */\n  readonly identity?: string;\n}\n\n/**\n * HTTP scale rule configuration\n */\nexport interface ContainerAppHttpScaleRule {\n  /**\n   * HTTP trigger metadata as key-value pairs\n   */\n  readonly metadata?: { [key: string]: string };\n}\n\n/**\n * TCP scale rule configuration\n */\nexport interface ContainerAppTcpScaleRule {\n  /**\n   * TCP trigger metadata as key-value pairs\n   */\n  readonly metadata?: { [key: string]: string };\n}\n\n/**\n * Scale rule definition\n */\nexport interface ContainerAppScaleRule {\n  /**\n   * Scale rule name\n   */\n  readonly name: string;\n\n  /**\n   * Azure Queue scale rule\n   */\n  readonly azureQueue?: ContainerAppAzureQueueScaleRule;\n\n  /**\n   * Custom scale rule (KEDA)\n   */\n  readonly custom?: ContainerAppCustomScaleRule;\n\n  /**\n   * HTTP scale rule\n   */\n  readonly http?: ContainerAppHttpScaleRule;\n\n  /**\n   * TCP scale rule\n   */\n  readonly tcp?: ContainerAppTcpScaleRule;\n}\n\n/**\n * Scale configuration for the Container App\n */\nexport interface ContainerAppScale {\n  /**\n   * Minimum number of replicas\n   * @default 0\n   */\n  readonly minReplicas?: number;\n\n  /**\n   * Maximum number of replicas\n   * @default 10\n   */\n  readonly maxReplicas?: number;\n\n  /**\n   * Cooldown period in seconds before scaling down (2025-07-01+)\n   */\n  readonly cooldownPeriod?: number;\n\n  /**\n   * Polling interval in seconds for checking scale rules (2025-07-01+)\n   */\n  readonly pollingInterval?: number;\n\n  /**\n   * Scaling rules\n   */\n  readonly rules?: ContainerAppScaleRule[];\n}\n\n/**\n * Volume definition for the Container App\n */\nexport interface ContainerAppVolume {\n  /**\n   * Volume name\n   */\n  readonly name: string;\n\n  /**\n   * Storage type: AzureFile, EmptyDir, NfsAzureFile, or Secret\n   */\n  readonly storageType?: string;\n\n  /**\n   * Name of the storage resource (for AzureFile and NfsAzureFile)\n   */\n  readonly storageName?: string;\n}\n\n/**\n * Service bind configuration (2025-07-01+)\n */\nexport interface ContainerAppServiceBind {\n  /**\n   * Resource ID of the service to bind to\n   */\n  readonly serviceId: string;\n\n  /**\n   * Name of the service binding\n   */\n  readonly name: string;\n}\n\n/**\n * Container App template\n */\nexport interface ContainerAppTemplate {\n  /**\n   * List of container definitions for the Container App\n   */\n  readonly containers: ContainerAppContainer[];\n\n  /**\n   * List of specialized containers that run before app containers\n   */\n  readonly initContainers?: ContainerAppContainer[];\n\n  /**\n   * Scaling properties for the Container App\n   */\n  readonly scale?: ContainerAppScale;\n\n  /**\n   * List of volume definitions for the Container App\n   */\n  readonly volumes?: ContainerAppVolume[];\n\n  /**\n   * User friendly suffix appended to the revision name\n   */\n  readonly revisionSuffix?: string;\n\n  /**\n   * Optional duration in seconds for graceful termination\n   */\n  readonly terminationGracePeriodSeconds?: number;\n\n  /**\n   * List of container app services bound to the app (2025-07-01+)\n   */\n  readonly serviceBinds?: ContainerAppServiceBind[];\n}\n\n/**\n * Traffic weight configuration for ingress\n */\nexport interface ContainerAppIngressTraffic {\n  /**\n   * Name of a revision\n   */\n  readonly revisionName?: string;\n\n  /**\n   * Indicates if this is the latest revision\n   */\n  readonly latestRevision?: boolean;\n\n  /**\n   * Traffic weight assigned to the revision\n   */\n  readonly weight?: number;\n\n  /**\n   * Associates a traffic label with a revision\n   */\n  readonly label?: string;\n}\n\n/**\n * IP security restriction for ingress\n */\nexport interface ContainerAppIngressIpSecurityRestriction {\n  /**\n   * Name for the restriction\n   */\n  readonly name: string;\n\n  /**\n   * Description of the restriction\n   */\n  readonly description?: string;\n\n  /**\n   * CIDR notation to match incoming IP address\n   */\n  readonly ipAddressRange: string;\n\n  /**\n   * Allow or Deny\n   */\n  readonly action: string;\n}\n\n/**\n * CORS policy for ingress\n */\nexport interface ContainerAppIngressCorsPolicy {\n  /**\n   * Allowed origins\n   */\n  readonly allowedOrigins?: string[];\n\n  /**\n   * Allowed methods\n   */\n  readonly allowedMethods?: string[];\n\n  /**\n   * Allowed headers\n   */\n  readonly allowedHeaders?: string[];\n\n  /**\n   * Expose headers\n   */\n  readonly exposeHeaders?: string[];\n\n  /**\n   * Max age in seconds\n   */\n  readonly maxAge?: number;\n\n  /**\n   * Allow credentials\n   */\n  readonly allowCredentials?: boolean;\n}\n\n/**\n * Sticky session configuration\n */\nexport interface ContainerAppIngressStickySessions {\n  /**\n   * Sticky session affinity: 'none' or 'sticky'\n   */\n  readonly affinity?: string;\n}\n\n/**\n * Additional port mapping for ingress (2025-07-01+)\n */\nexport interface ContainerAppIngressAdditionalPortMapping {\n  /**\n   * Whether the port is externally accessible\n   */\n  readonly external?: boolean;\n\n  /**\n   * Target port in the container\n   */\n  readonly targetPort: number;\n\n  /**\n   * Specifies the exposed port for the target port\n   */\n  readonly exposedPort?: number;\n}\n\n/**\n * Ingress configuration for the Container App\n */\nexport interface ContainerAppIngress {\n  /**\n   * Whether the ingress is externally accessible\n   */\n  readonly external?: boolean;\n\n  /**\n   * Target port in the container for incoming traffic\n   */\n  readonly targetPort?: number;\n\n  /**\n   * Ingress transport protocol: 'auto', 'http', 'http2', or 'tcp'\n   * @default \"auto\"\n   */\n  readonly transport?: string;\n\n  /**\n   * Traffic weights for revisions\n   */\n  readonly traffic?: ContainerAppIngressTraffic[];\n\n  /**\n   * Custom domain bindings for the Container App's hostname\n   */\n  readonly customDomains?: any[];\n\n  /**\n   * IP security restrictions\n   */\n  readonly ipSecurityRestrictions?: ContainerAppIngressIpSecurityRestriction[];\n\n  /**\n   * CORS policy for the Container App\n   */\n  readonly corsPolicy?: ContainerAppIngressCorsPolicy;\n\n  /**\n   * Sticky sessions configuration\n   */\n  readonly stickySessions?: ContainerAppIngressStickySessions;\n\n  /**\n   * Client certificate mode\n   */\n  readonly clientCertificateMode?: string;\n\n  /**\n   * Additional port mappings for the Container App (2025-07-01+)\n   */\n  readonly additionalPortMappings?: ContainerAppIngressAdditionalPortMapping[];\n}\n\n/**\n * Dapr configuration for the Container App\n */\nexport interface ContainerAppDapr {\n  /**\n   * Whether Dapr is enabled\n   */\n  readonly enabled?: boolean;\n\n  /**\n   * Dapr application identifier\n   */\n  readonly appId?: string;\n\n  /**\n   * Port on which the Dapr side car will listen\n   */\n  readonly appPort?: number;\n\n  /**\n   * Protocol used by Dapr to communicate with the app: 'http' or 'grpc'\n   * @default \"http\"\n   */\n  readonly appProtocol?: string;\n\n  /**\n   * Max size of HTTP request body in MB to handle by Dapr HTTP server\n   */\n  readonly httpMaxRequestSize?: number;\n\n  /**\n   * Max size of HTTP header read buffer in KB to handle by Dapr HTTP server\n   */\n  readonly httpReadBufferSize?: number;\n\n  /**\n   * Dapr log level: 'debug', 'info', 'warn', 'error'\n   */\n  readonly logLevel?: string;\n\n  /**\n   * Whether to enable API logging for the Dapr sidecar\n   */\n  readonly enableApiLogging?: boolean;\n\n  /**\n   * Maximum number of concurrent requests\n   */\n  readonly maxConcurrency?: number;\n\n  /**\n   * App health check configuration\n   */\n  readonly appHealth?: ContainerAppDaprHealthConfig;\n}\n\n/**\n * Dapr app health check configuration\n */\nexport interface ContainerAppDaprHealthConfig {\n  /**\n   * Whether app health check is enabled\n   */\n  readonly enabled?: boolean;\n\n  /**\n   * Health check endpoint path\n   */\n  readonly path?: string;\n\n  /**\n   * Probe interval in seconds\n   */\n  readonly probeIntervalSeconds?: number;\n\n  /**\n   * Probe timeout in milliseconds\n   */\n  readonly probeTimeoutMilliseconds?: number;\n\n  /**\n   * Number of consecutive failures before marking unhealthy\n   */\n  readonly threshold?: number;\n}\n\n/**\n * Container registry configuration\n */\nexport interface ContainerAppRegistry {\n  /**\n   * Container registry server URL\n   */\n  readonly server: string;\n\n  /**\n   * Managed identity to authenticate with the registry\n   */\n  readonly identity?: string;\n\n  /**\n   * Registry username\n   */\n  readonly username?: string;\n\n  /**\n   * Name of the secret containing the registry password\n   */\n  readonly passwordSecretRef?: string;\n}\n\n/**\n * Secret configuration for the Container App\n */\nexport interface ContainerAppSecret {\n  /**\n   * Secret name\n   */\n  readonly name: string;\n\n  /**\n   * Secret value (mutually exclusive with keyVaultUrl)\n   */\n  readonly value?: string;\n\n  /**\n   * Key Vault secret URI (mutually exclusive with value)\n   */\n  readonly keyVaultUrl?: string;\n\n  /**\n   * Resource ID of a managed identity to authenticate with Key Vault\n   */\n  readonly identity?: string;\n}\n\n/**\n * Identity settings for lifecycle phases (2025-07-01+)\n */\nexport interface ContainerAppIdentitySetting {\n  /**\n   * Resource ID of a managed identity or 'system' for system-assigned\n   */\n  readonly identity: string;\n\n  /**\n   * The lifecycle stage: 'All', 'Init', or 'Main'\n   */\n  readonly lifecycle: string;\n}\n\n/**\n * Java runtime configuration (2025-07-01+)\n */\nexport interface ContainerAppJavaRuntimeConfig {\n  /**\n   * Whether to enable Java metrics\n   */\n  readonly enableMetrics?: boolean;\n}\n\n/**\n * Runtime configuration (2025-07-01+)\n */\nexport interface ContainerAppRuntime {\n  /**\n   * Java runtime configuration\n   */\n  readonly java?: ContainerAppJavaRuntimeConfig;\n}\n\n/**\n * Service configuration for dev services\n */\nexport interface ContainerAppService {\n  /**\n   * Dev Service type (e.g., 'redis', 'postgres', 'kafka')\n   */\n  readonly type: string;\n}\n\n/**\n * Configuration properties for the Container App\n */\nexport interface ContainerAppConfiguration {\n  /**\n   * Active revisions mode: 'Single' or 'Multiple'\n   * @default \"Single\"\n   */\n  readonly activeRevisionsMode?: string;\n\n  /**\n   * Ingress configuration\n   */\n  readonly ingress?: ContainerAppIngress;\n\n  /**\n   * Dapr configuration\n   */\n  readonly dapr?: ContainerAppDapr;\n\n  /**\n   * Collection of private container registry credentials\n   */\n  readonly registries?: ContainerAppRegistry[];\n\n  /**\n   * Collection of secrets used by a Container App\n   */\n  readonly secrets?: ContainerAppSecret[];\n\n  /**\n   * Maximum number of inactive revisions to keep\n   */\n  readonly maxInactiveRevisions?: number;\n\n  /**\n   * Container App to be a dev Container App Service (2025-07-01+)\n   */\n  readonly service?: ContainerAppService;\n\n  /**\n   * Runtime configuration (2025-07-01+)\n   */\n  readonly runtime?: ContainerAppRuntime;\n\n  /**\n   * Identity settings for lifecycle phases (2025-07-01+)\n   */\n  readonly identitySettings?: ContainerAppIdentitySetting[];\n}\n\n/**\n * Managed service identity configuration\n */\nexport interface ContainerAppIdentity {\n  /**\n   * Type of managed service identity:\n   * 'None', 'SystemAssigned', 'UserAssigned', or 'SystemAssigned,UserAssigned'\n   */\n  readonly type: string;\n\n  /**\n   * User-assigned identity resource IDs\n   */\n  readonly userAssignedIdentities?: { [key: string]: any };\n}\n\n// =============================================================================\n// MAIN PROPS INTERFACE\n// =============================================================================\n\n/**\n * Properties for the unified Azure Container App\n *\n * Extends AzapiResourceProps with Container App specific properties\n */\nexport interface ContainerAppProps extends AzapiResourceProps {\n  /**\n   * Resource ID of the Container App Environment where this app will be hosted\n   */\n  readonly environmentId: string;\n\n  /**\n   * Container App versioned application definition\n   */\n  readonly template: ContainerAppTemplate;\n\n  /**\n   * Non-versioned Container App configuration properties\n   */\n  readonly configuration?: ContainerAppConfiguration;\n\n  /**\n   * Managed identity configuration\n   */\n  readonly identity?: ContainerAppIdentity;\n\n  /**\n   * Workload profile name to pin for container app execution\n   */\n  readonly workloadProfileName?: string;\n\n  /**\n   * The lifecycle rules to ignore changes\n   * @example [\"tags\"]\n   */\n  readonly ignoreChanges?: string[];\n\n  /**\n   * Resource group ID where the Container App will be created\n   */\n  readonly resourceGroupId?: string;\n}\n\n/**\n * The resource body interface for Azure Container App API calls\n */\nexport interface ContainerAppBody {\n  readonly location: string;\n  readonly tags?: { [key: string]: string };\n  readonly identity?: any;\n  readonly properties?: any;\n}\n\n// =============================================================================\n// MAIN CLASS\n// =============================================================================\n\n/**\n * Unified Azure Container App implementation\n *\n * Azure Container Apps enable you to run microservices and containerized applications\n * on a serverless platform. They provide built-in support for ingress, scaling,\n * Dapr integration, and more.\n *\n * Key features:\n * - External and internal ingress with TLS, CORS, and sticky sessions\n * - Dapr sidecar integration for microservice patterns\n * - Horizontal auto-scaling with HTTP, TCP, custom (KEDA), and Azure Queue rules\n * - Secrets management with Azure Key Vault references\n * - Managed identity (SystemAssigned, UserAssigned, or both)\n * - Init containers for setup tasks\n * - Volume mounts (Azure Files, NFS, Ephemeral, Secret)\n * - Health probes (Liveness, Readiness, Startup)\n * - Multiple revision mode for blue-green and canary deployments\n * - Service binds for dev services (Redis, Postgres, Kafka)\n *\n * @example\n * // Basic Container App with a single container:\n * const app = new ContainerApp(this, \"app\", {\n *   name: \"my-container-app\",\n *   location: \"eastus\",\n *   resourceGroupId: resourceGroup.id,\n *   environmentId: environment.id,\n *   template: {\n *     containers: [{\n *       name: \"my-app\",\n *       image: \"mcr.microsoft.com/azuredocs/containerapps-helloworld:latest\",\n *       resources: { cpu: 0.25, memory: \"0.5Gi\" },\n *     }],\n *   },\n * });\n *\n * @example\n * // Container App with external ingress and scaling:\n * const app = new ContainerApp(this, \"app\", {\n *   name: \"my-web-app\",\n *   location: \"eastus\",\n *   resourceGroupId: resourceGroup.id,\n *   environmentId: environment.id,\n *   template: {\n *     containers: [{\n *       name: \"web\",\n *       image: \"myregistry.azurecr.io/myapp:latest\",\n *       resources: { cpu: 0.5, memory: \"1Gi\" },\n *       env: [{ name: \"PORT\", value: \"3000\" }],\n *     }],\n *     scale: {\n *       minReplicas: 1,\n *       maxReplicas: 10,\n *       rules: [{\n *         name: \"http-rule\",\n *         http: { metadata: { concurrentRequests: \"50\" } },\n *       }],\n *     },\n *   },\n *   configuration: {\n *     ingress: {\n *       external: true,\n *       targetPort: 3000,\n *       transport: \"http\",\n *     },\n *   },\n * });\n *\n * @stability stable\n */\nexport class ContainerApp extends AzapiResource {\n  static {\n    AzapiResource.registerSchemas(\n      CONTAINER_APP_TYPE,\n      ALL_CONTAINER_APP_VERSIONS,\n    );\n  }\n\n  public readonly props: ContainerAppProps;\n\n  // Output properties for easy access and referencing\n  public readonly idOutput: cdktn.TerraformOutput;\n  public readonly locationOutput: cdktn.TerraformOutput;\n  public readonly nameOutput: cdktn.TerraformOutput;\n  public readonly tagsOutput: cdktn.TerraformOutput;\n  public readonly fqdnOutput: cdktn.TerraformOutput;\n  public readonly latestRevisionFqdnOutput: cdktn.TerraformOutput;\n  public readonly latestReadyRevisionNameOutput: cdktn.TerraformOutput;\n  public readonly provisioningStateOutput: cdktn.TerraformOutput;\n\n  /**\n   * Creates a new Azure Container App\n   *\n   * @param scope - The scope in which to define this construct\n   * @param id - The unique identifier for this instance\n   * @param props - Configuration properties for the Container App\n   */\n  constructor(scope: Construct, id: string, props: ContainerAppProps) {\n    super(scope, id, props);\n\n    this.props = props;\n\n    // Create Terraform outputs\n    this.idOutput = new cdktn.TerraformOutput(this, \"id\", {\n      value: this.id,\n      description: \"The ID of the Container App\",\n    });\n\n    this.locationOutput = new cdktn.TerraformOutput(this, \"location\", {\n      value: `\\${${this.terraformResource.fqn}.location}`,\n      description: \"The location of the Container App\",\n    });\n\n    this.nameOutput = new cdktn.TerraformOutput(this, \"name\", {\n      value: `\\${${this.terraformResource.fqn}.name}`,\n      description: \"The name of the Container App\",\n    });\n\n    this.tagsOutput = new cdktn.TerraformOutput(this, \"tags\", {\n      value: `\\${${this.terraformResource.fqn}.tags}`,\n      description: \"The tags assigned to the Container App\",\n    });\n\n    this.fqdnOutput = new cdktn.TerraformOutput(this, \"fqdn\", {\n      value: `\\${${this.terraformResource.fqn}.output.properties.configuration.ingress.fqdn}`,\n      description: \"The FQDN of the Container App (from ingress)\",\n    });\n\n    this.latestRevisionFqdnOutput = new cdktn.TerraformOutput(\n      this,\n      \"latest_revision_fqdn\",\n      {\n        value: `\\${${this.terraformResource.fqn}.output.properties.latestRevisionFqdn}`,\n        description: \"The FQDN of the latest revision of the Container App\",\n      },\n    );\n\n    this.latestReadyRevisionNameOutput = new cdktn.TerraformOutput(\n      this,\n      \"latest_ready_revision_name\",\n      {\n        value: `\\${${this.terraformResource.fqn}.output.properties.latestReadyRevisionName}`,\n        description:\n          \"The name of the latest ready revision of the Container App\",\n      },\n    );\n\n    this.provisioningStateOutput = new cdktn.TerraformOutput(\n      this,\n      \"provisioning_state\",\n      {\n        value: `\\${${this.terraformResource.fqn}.output.properties.provisioningState}`,\n        description: \"The provisioning state of the Container App\",\n      },\n    );\n\n    // Override logical IDs\n    this.idOutput.overrideLogicalId(\"id\");\n    this.locationOutput.overrideLogicalId(\"location\");\n    this.nameOutput.overrideLogicalId(\"name\");\n    this.tagsOutput.overrideLogicalId(\"tags\");\n    this.fqdnOutput.overrideLogicalId(\"fqdn\");\n    this.latestRevisionFqdnOutput.overrideLogicalId(\"latest_revision_fqdn\");\n    this.latestReadyRevisionNameOutput.overrideLogicalId(\n      \"latest_ready_revision_name\",\n    );\n    this.provisioningStateOutput.overrideLogicalId(\"provisioning_state\");\n\n    // Apply ignore changes if specified\n    this._applyIgnoreChanges();\n  }\n\n  // =============================================================================\n  // REQUIRED ABSTRACT METHODS FROM AzapiResource\n  // =============================================================================\n\n  /**\n   * Gets the default API version to use when no explicit version is specified\n   */\n  protected defaultVersion(): string {\n    return \"2025-02-02-preview\";\n  }\n\n  /**\n   * Gets the Azure resource type for Container Apps\n   */\n  protected resourceType(): string {\n    return CONTAINER_APP_TYPE;\n  }\n\n  /**\n   * Gets the API schema for the resolved version\n   */\n  protected apiSchema(): ApiSchema {\n    return this.resolveSchema();\n  }\n\n  /**\n   * Indicates that location is required for Container Apps\n   */\n  protected requiresLocation(): boolean {\n    return true;\n  }\n\n  /**\n   * Creates the resource body for the Azure API call\n   */\n  protected createResourceBody(props: any): any {\n    const typedProps = props as ContainerAppProps;\n\n    const properties: any = {\n      environmentId: typedProps.environmentId,\n      template: typedProps.template,\n    };\n\n    // Configuration\n    if (typedProps.configuration) {\n      properties.configuration = typedProps.configuration;\n    }\n\n    // Workload profile name\n    if (typedProps.workloadProfileName) {\n      properties.workloadProfileName = typedProps.workloadProfileName;\n    }\n\n    const body: ContainerAppBody = {\n      location: this.location!,\n      tags: this.allTags(),\n      properties,\n    };\n\n    // Identity (top-level, not inside properties)\n    if (typedProps.identity) {\n      (body as any).identity = typedProps.identity;\n    }\n\n    return body;\n  }\n\n  // =============================================================================\n  // PUBLIC METHODS - Read-only Properties\n  // =============================================================================\n\n  /**\n   * Get the FQDN of the Container App (from ingress configuration)\n   */\n  public get fqdn(): string {\n    return `\\${${this.terraformResource.fqn}.output.properties.configuration.ingress.fqdn}`;\n  }\n\n  /**\n   * Get the FQDN of the latest revision\n   */\n  public get latestRevisionFqdn(): string {\n    return `\\${${this.terraformResource.fqn}.output.properties.latestRevisionFqdn}`;\n  }\n\n  /**\n   * Get the name of the latest ready revision\n   */\n  public get latestReadyRevisionName(): string {\n    return `\\${${this.terraformResource.fqn}.output.properties.latestReadyRevisionName}`;\n  }\n\n  /**\n   * Get the provisioning state of the Container App\n   */\n  public get provisioningState(): string {\n    return `\\${${this.terraformResource.fqn}.output.properties.provisioningState}`;\n  }\n\n  /**\n   * Get the running status of the Container App\n   */\n  public get runningStatus(): string {\n    return `\\${${this.terraformResource.fqn}.output.properties.runningStatus}`;\n  }\n\n  /**\n   * Add a tag to the Container App\n   */\n  public addTag(key: string, value: string): void {\n    if (!this.props.tags) {\n      (this.props as any).tags = {};\n    }\n    this.props.tags![key] = value;\n  }\n\n  /**\n   * Remove a tag from the Container App\n   */\n  public removeTag(key: string): void {\n    if (this.props.tags && this.props.tags[key]) {\n      delete this.props.tags[key];\n    }\n  }\n\n  // =============================================================================\n  // RESOURCE CONFIG CUSTOMIZATION\n  // =============================================================================\n\n  /**\n   * Customizes the resource configuration to handle Azure value normalization.\n   *\n   * Azure normalizes values in API responses:\n   * - Location: \"eastus\" → \"East US\"\n   * - Enum values: \"auto\" → \"Auto\", \"enabled\" → \"Enabled\"\n   *\n   * This override:\n   * 1. Removes `location` from the body (the framework passes it as a top-level\n   *    attribute which the azapi provider normalizes properly)\n   * 2. Enables `ignoreCasing` to suppress diffs from Azure's value normalization\n   *    (e.g., \"auto\" vs \"Auto\" for transport, \"enabled\" vs \"Enabled\")\n   */\n  protected customizeResourceConfig(config: any): any {\n    const updatedConfig = { ...config, ignoreCasing: true };\n\n    // Remove location from body to prevent \"East US\" vs \"eastus\" drift\n    // The framework handles location as a top-level azapi attribute instead\n    if (updatedConfig.body && updatedConfig.body.location) {\n      const { location: _location, ...bodyWithoutLocation } =\n        updatedConfig.body;\n      return {\n        ...updatedConfig,\n        body: bodyWithoutLocation,\n        // Ensure location is set as top-level attribute\n        location: updatedConfig.location || _location,\n      };\n    }\n    return updatedConfig;\n  }\n\n  // =============================================================================\n  // PRIVATE HELPER METHODS\n  // =============================================================================\n\n  /**\n   * Applies ignore changes lifecycle rules if specified in props\n   */\n  private _applyIgnoreChanges(): void {\n    if (this.props.ignoreChanges && this.props.ignoreChanges.length > 0) {\n      this.terraformResource.addOverride(\"lifecycle\", [\n        {\n          ignore_changes: this.props.ignoreChanges,\n        },\n      ]);\n    }\n  }\n}\n"]}
@@ -0,0 +1,4 @@
1
+ export * from "./container-app-environment-schemas";
2
+ export * from "./container-app-environment";
3
+ export * from "./container-app-schemas";
4
+ export * from "./container-app";
@@ -0,0 +1,21 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./container-app-environment-schemas"), exports);
18
+ __exportStar(require("./container-app-environment"), exports);
19
+ __exportStar(require("./container-app-schemas"), exports);
20
+ __exportStar(require("./container-app"), exports);
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXp1cmUtY29udGFpbmVyYXBwcy9saWIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHNFQUFvRDtBQUNwRCw4REFBNEM7QUFDNUMsMERBQXdDO0FBQ3hDLGtEQUFnQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gXCIuL2NvbnRhaW5lci1hcHAtZW52aXJvbm1lbnQtc2NoZW1hc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vY29udGFpbmVyLWFwcC1lbnZpcm9ubWVudFwiO1xuZXhwb3J0ICogZnJvbSBcIi4vY29udGFpbmVyLWFwcC1zY2hlbWFzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9jb250YWluZXItYXBwXCI7XG4iXX0=
@@ -0,0 +1,10 @@
1
+ /**
2
+ * Integration test for Azure Container App Environment
3
+ *
4
+ * This test demonstrates usage of the ContainerAppEnvironment construct
5
+ * with Log Analytics integration and validates deployment, idempotency,
6
+ * and cleanup.
7
+ *
8
+ * Run with: npm run integration:nostream
9
+ */
10
+ import "cdktn/lib/testing/adapters/jest";
@@ -0,0 +1,82 @@
1
+ "use strict";
2
+ /**
3
+ * Integration test for Azure Container App Environment
4
+ *
5
+ * This test demonstrates usage of the ContainerAppEnvironment construct
6
+ * with Log Analytics integration and validates deployment, idempotency,
7
+ * and cleanup.
8
+ *
9
+ * Run with: npm run integration:nostream
10
+ */
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ const cdktn_1 = require("cdktn");
13
+ require("cdktn/lib/testing/adapters/jest");
14
+ const resource_group_1 = require("../../azure-resourcegroup/lib/resource-group");
15
+ const provider_1 = require("../../core-azure/lib/azapi/providers-azapi/provider");
16
+ const testing_1 = require("../../testing");
17
+ const metadata_1 = require("../../testing/lib/metadata");
18
+ const container_app_environment_1 = require("../lib/container-app-environment");
19
+ // Generate unique test run metadata for this test suite
20
+ const testMetadata = new metadata_1.TestRunMetadata("containerenv-integration", {
21
+ maxAgeHours: 4,
22
+ });
23
+ /**
24
+ * Example stack demonstrating Container App Environment usage
25
+ */
26
+ class ContainerAppEnvironmentExampleStack extends testing_1.BaseTestStack {
27
+ constructor(scope, id) {
28
+ super(scope, id, {
29
+ testRunOptions: {
30
+ maxAgeHours: testMetadata.maxAgeHours,
31
+ autoCleanup: testMetadata.autoCleanup,
32
+ cleanupPolicy: testMetadata.cleanupPolicy,
33
+ },
34
+ });
35
+ // Configure AZAPI provider
36
+ new provider_1.AzapiProvider(this, "azapi", {});
37
+ // Generate unique names
38
+ const resourceGroupName = this.generateResourceName("Microsoft.Resources/resourceGroups", "containerenv");
39
+ const envName = this.generateResourceName("Microsoft.App/managedEnvironments", "containerenv");
40
+ // Create resource group
41
+ const resourceGroup = new resource_group_1.ResourceGroup(this, "test-rg", {
42
+ name: resourceGroupName,
43
+ location: "eastus",
44
+ tags: {
45
+ ...this.systemTags(),
46
+ purpose: "container-app-env-testing",
47
+ },
48
+ });
49
+ // Create Container App Environment with basic configuration
50
+ new container_app_environment_1.ContainerAppEnvironment(this, "container-env", {
51
+ name: envName,
52
+ location: resourceGroup.location,
53
+ resourceGroupId: resourceGroup.id,
54
+ apiVersion: "2025-02-02-preview",
55
+ workloadProfiles: [
56
+ {
57
+ name: "Consumption",
58
+ workloadProfileType: "Consumption",
59
+ },
60
+ ],
61
+ tags: {
62
+ ...this.systemTags(),
63
+ example: "complete",
64
+ environment: "test",
65
+ purpose: "container-app-hosting",
66
+ },
67
+ });
68
+ }
69
+ }
70
+ describe("ContainerAppEnvironment Integration Test", () => {
71
+ it("should deploy Container App Environment, validate idempotency, and cleanup", () => {
72
+ const app = cdktn_1.Testing.app();
73
+ const stack = new ContainerAppEnvironmentExampleStack(app, "test-containerenv");
74
+ const synthesized = cdktn_1.Testing.fullSynth(stack);
75
+ // This will:
76
+ // 1. Run terraform apply to deploy resources
77
+ // 2. Run terraform plan to check idempotency (no changes expected)
78
+ // 3. Run terraform destroy to cleanup resources
79
+ (0, testing_1.TerraformApplyCheckAndDestroy)(synthesized, { verifyCleanup: true });
80
+ }, 600000); // 10 minute timeout for deployment and cleanup
81
+ });
82
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udGFpbmVyLWFwcC1lbnZpcm9ubWVudC5pbnRlZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9henVyZS1jb250YWluZXJhcHBzL3Rlc3QvY29udGFpbmVyLWFwcC1lbnZpcm9ubWVudC5pbnRlZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7Ozs7O0dBUUc7O0FBRUgsaUNBQWdDO0FBRWhDLDJDQUF5QztBQUN6QyxpRkFBNkU7QUFDN0Usa0ZBQW9GO0FBQ3BGLDJDQUE2RTtBQUM3RSx5REFBNkQ7QUFDN0QsZ0ZBQTJFO0FBRTNFLHdEQUF3RDtBQUN4RCxNQUFNLFlBQVksR0FBRyxJQUFJLDBCQUFlLENBQUMsMEJBQTBCLEVBQUU7SUFDbkUsV0FBVyxFQUFFLENBQUM7Q0FDZixDQUFDLENBQUM7QUFFSDs7R0FFRztBQUNILE1BQU0sbUNBQW9DLFNBQVEsdUJBQWE7SUFDN0QsWUFBWSxLQUFnQixFQUFFLEVBQVU7UUFDdEMsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUU7WUFDZixjQUFjLEVBQUU7Z0JBQ2QsV0FBVyxFQUFFLFlBQVksQ0FBQyxXQUFXO2dCQUNyQyxXQUFXLEVBQUUsWUFBWSxDQUFDLFdBQVc7Z0JBQ3JDLGFBQWEsRUFBRSxZQUFZLENBQUMsYUFBYTthQUMxQztTQUNGLENBQUMsQ0FBQztRQUVILDJCQUEyQjtRQUMzQixJQUFJLHdCQUFhLENBQUMsSUFBSSxFQUFFLE9BQU8sRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVyQyx3QkFBd0I7UUFDeEIsTUFBTSxpQkFBaUIsR0FBRyxJQUFJLENBQUMsb0JBQW9CLENBQ2pELG9DQUFvQyxFQUNwQyxjQUFjLENBQ2YsQ0FBQztRQUNGLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxvQkFBb0IsQ0FDdkMsbUNBQW1DLEVBQ25DLGNBQWMsQ0FDZixDQUFDO1FBRUYsd0JBQXdCO1FBQ3hCLE1BQU0sYUFBYSxHQUFHLElBQUksOEJBQWEsQ0FBQyxJQUFJLEVBQUUsU0FBUyxFQUFFO1lBQ3ZELElBQUksRUFBRSxpQkFBaUI7WUFDdkIsUUFBUSxFQUFFLFFBQVE7WUFDbEIsSUFBSSxFQUFFO2dCQUNKLEdBQUcsSUFBSSxDQUFDLFVBQVUsRUFBRTtnQkFDcEIsT0FBTyxFQUFFLDJCQUEyQjthQUNyQztTQUNGLENBQUMsQ0FBQztRQUVILDREQUE0RDtRQUM1RCxJQUFJLG1EQUF1QixDQUFDLElBQUksRUFBRSxlQUFlLEVBQUU7WUFDakQsSUFBSSxFQUFFLE9BQU87WUFDYixRQUFRLEVBQUUsYUFBYSxDQUFDLFFBQVE7WUFDaEMsZUFBZSxFQUFFLGFBQWEsQ0FBQyxFQUFFO1lBQ2pDLFVBQVUsRUFBRSxvQkFBb0I7WUFDaEMsZ0JBQWdCLEVBQUU7Z0JBQ2hCO29CQUNFLElBQUksRUFBRSxhQUFhO29CQUNuQixtQkFBbUIsRUFBRSxhQUFhO2lCQUNuQzthQUNGO1lBQ0QsSUFBSSxFQUFFO2dCQUNKLEdBQUcsSUFBSSxDQUFDLFVBQVUsRUFBRTtnQkFDcEIsT0FBTyxFQUFFLFVBQVU7Z0JBQ25CLFdBQVcsRUFBRSxNQUFNO2dCQUNuQixPQUFPLEVBQUUsdUJBQXVCO2FBQ2pDO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBRUQsUUFBUSxDQUFDLDBDQUEwQyxFQUFFLEdBQUcsRUFBRTtJQUN4RCxFQUFFLENBQUMsNEVBQTRFLEVBQUUsR0FBRyxFQUFFO1FBQ3BGLE1BQU0sR0FBRyxHQUFHLGVBQU8sQ0FBQyxHQUFHLEVBQUUsQ0FBQztRQUMxQixNQUFNLEtBQUssR0FBRyxJQUFJLG1DQUFtQyxDQUNuRCxHQUFHLEVBQ0gsbUJBQW1CLENBQ3BCLENBQUM7UUFDRixNQUFNLFdBQVcsR0FBRyxlQUFPLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBRTdDLGFBQWE7UUFDYiw2Q0FBNkM7UUFDN0MsbUVBQW1FO1FBQ25FLGdEQUFnRDtRQUNoRCxJQUFBLHVDQUE2QixFQUFDLFdBQVcsRUFBRSxFQUFFLGFBQWEsRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDO0lBQ3RFLENBQUMsRUFBRSxNQUFNLENBQUMsQ0FBQyxDQUFDLCtDQUErQztBQUM3RCxDQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogSW50ZWdyYXRpb24gdGVzdCBmb3IgQXp1cmUgQ29udGFpbmVyIEFwcCBFbnZpcm9ubWVudFxuICpcbiAqIFRoaXMgdGVzdCBkZW1vbnN0cmF0ZXMgdXNhZ2Ugb2YgdGhlIENvbnRhaW5lckFwcEVudmlyb25tZW50IGNvbnN0cnVjdFxuICogd2l0aCBMb2cgQW5hbHl0aWNzIGludGVncmF0aW9uIGFuZCB2YWxpZGF0ZXMgZGVwbG95bWVudCwgaWRlbXBvdGVuY3ksXG4gKiBhbmQgY2xlYW51cC5cbiAqXG4gKiBSdW4gd2l0aDogbnBtIHJ1biBpbnRlZ3JhdGlvbjpub3N0cmVhbVxuICovXG5cbmltcG9ydCB7IFRlc3RpbmcgfSBmcm9tIFwiY2RrdG5cIjtcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5pbXBvcnQgXCJjZGt0bi9saWIvdGVzdGluZy9hZGFwdGVycy9qZXN0XCI7XG5pbXBvcnQgeyBSZXNvdXJjZUdyb3VwIH0gZnJvbSBcIi4uLy4uL2F6dXJlLXJlc291cmNlZ3JvdXAvbGliL3Jlc291cmNlLWdyb3VwXCI7XG5pbXBvcnQgeyBBemFwaVByb3ZpZGVyIH0gZnJvbSBcIi4uLy4uL2NvcmUtYXp1cmUvbGliL2F6YXBpL3Byb3ZpZGVycy1hemFwaS9wcm92aWRlclwiO1xuaW1wb3J0IHsgQmFzZVRlc3RTdGFjaywgVGVycmFmb3JtQXBwbHlDaGVja0FuZERlc3Ryb3kgfSBmcm9tIFwiLi4vLi4vdGVzdGluZ1wiO1xuaW1wb3J0IHsgVGVzdFJ1bk1ldGFkYXRhIH0gZnJvbSBcIi4uLy4uL3Rlc3RpbmcvbGliL21ldGFkYXRhXCI7XG5pbXBvcnQgeyBDb250YWluZXJBcHBFbnZpcm9ubWVudCB9IGZyb20gXCIuLi9saWIvY29udGFpbmVyLWFwcC1lbnZpcm9ubWVudFwiO1xuXG4vLyBHZW5lcmF0ZSB1bmlxdWUgdGVzdCBydW4gbWV0YWRhdGEgZm9yIHRoaXMgdGVzdCBzdWl0ZVxuY29uc3QgdGVzdE1ldGFkYXRhID0gbmV3IFRlc3RSdW5NZXRhZGF0YShcImNvbnRhaW5lcmVudi1pbnRlZ3JhdGlvblwiLCB7XG4gIG1heEFnZUhvdXJzOiA0LFxufSk7XG5cbi8qKlxuICogRXhhbXBsZSBzdGFjayBkZW1vbnN0cmF0aW5nIENvbnRhaW5lciBBcHAgRW52aXJvbm1lbnQgdXNhZ2VcbiAqL1xuY2xhc3MgQ29udGFpbmVyQXBwRW52aXJvbm1lbnRFeGFtcGxlU3RhY2sgZXh0ZW5kcyBCYXNlVGVzdFN0YWNrIHtcbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCwge1xuICAgICAgdGVzdFJ1bk9wdGlvbnM6IHtcbiAgICAgICAgbWF4QWdlSG91cnM6IHRlc3RNZXRhZGF0YS5tYXhBZ2VIb3VycyxcbiAgICAgICAgYXV0b0NsZWFudXA6IHRlc3RNZXRhZGF0YS5hdXRvQ2xlYW51cCxcbiAgICAgICAgY2xlYW51cFBvbGljeTogdGVzdE1ldGFkYXRhLmNsZWFudXBQb2xpY3ksXG4gICAgICB9LFxuICAgIH0pO1xuXG4gICAgLy8gQ29uZmlndXJlIEFaQVBJIHByb3ZpZGVyXG4gICAgbmV3IEF6YXBpUHJvdmlkZXIodGhpcywgXCJhemFwaVwiLCB7fSk7XG5cbiAgICAvLyBHZW5lcmF0ZSB1bmlxdWUgbmFtZXNcbiAgICBjb25zdCByZXNvdXJjZUdyb3VwTmFtZSA9IHRoaXMuZ2VuZXJhdGVSZXNvdXJjZU5hbWUoXG4gICAgICBcIk1pY3Jvc29mdC5SZXNvdXJjZXMvcmVzb3VyY2VHcm91cHNcIixcbiAgICAgIFwiY29udGFpbmVyZW52XCIsXG4gICAgKTtcbiAgICBjb25zdCBlbnZOYW1lID0gdGhpcy5nZW5lcmF0ZVJlc291cmNlTmFtZShcbiAgICAgIFwiTWljcm9zb2Z0LkFwcC9tYW5hZ2VkRW52aXJvbm1lbnRzXCIsXG4gICAgICBcImNvbnRhaW5lcmVudlwiLFxuICAgICk7XG5cbiAgICAvLyBDcmVhdGUgcmVzb3VyY2UgZ3JvdXBcbiAgICBjb25zdCByZXNvdXJjZUdyb3VwID0gbmV3IFJlc291cmNlR3JvdXAodGhpcywgXCJ0ZXN0LXJnXCIsIHtcbiAgICAgIG5hbWU6IHJlc291cmNlR3JvdXBOYW1lLFxuICAgICAgbG9jYXRpb246IFwiZWFzdHVzXCIsXG4gICAgICB0YWdzOiB7XG4gICAgICAgIC4uLnRoaXMuc3lzdGVtVGFncygpLFxuICAgICAgICBwdXJwb3NlOiBcImNvbnRhaW5lci1hcHAtZW52LXRlc3RpbmdcIixcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICAvLyBDcmVhdGUgQ29udGFpbmVyIEFwcCBFbnZpcm9ubWVudCB3aXRoIGJhc2ljIGNvbmZpZ3VyYXRpb25cbiAgICBuZXcgQ29udGFpbmVyQXBwRW52aXJvbm1lbnQodGhpcywgXCJjb250YWluZXItZW52XCIsIHtcbiAgICAgIG5hbWU6IGVudk5hbWUsXG4gICAgICBsb2NhdGlvbjogcmVzb3VyY2VHcm91cC5sb2NhdGlvbixcbiAgICAgIHJlc291cmNlR3JvdXBJZDogcmVzb3VyY2VHcm91cC5pZCxcbiAgICAgIGFwaVZlcnNpb246IFwiMjAyNS0wMi0wMi1wcmV2aWV3XCIsXG4gICAgICB3b3JrbG9hZFByb2ZpbGVzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICBuYW1lOiBcIkNvbnN1bXB0aW9uXCIsXG4gICAgICAgICAgd29ya2xvYWRQcm9maWxlVHlwZTogXCJDb25zdW1wdGlvblwiLFxuICAgICAgICB9LFxuICAgICAgXSxcbiAgICAgIHRhZ3M6IHtcbiAgICAgICAgLi4udGhpcy5zeXN0ZW1UYWdzKCksXG4gICAgICAgIGV4YW1wbGU6IFwiY29tcGxldGVcIixcbiAgICAgICAgZW52aXJvbm1lbnQ6IFwidGVzdFwiLFxuICAgICAgICBwdXJwb3NlOiBcImNvbnRhaW5lci1hcHAtaG9zdGluZ1wiLFxuICAgICAgfSxcbiAgICB9KTtcbiAgfVxufVxuXG5kZXNjcmliZShcIkNvbnRhaW5lckFwcEVudmlyb25tZW50IEludGVncmF0aW9uIFRlc3RcIiwgKCkgPT4ge1xuICBpdChcInNob3VsZCBkZXBsb3kgQ29udGFpbmVyIEFwcCBFbnZpcm9ubWVudCwgdmFsaWRhdGUgaWRlbXBvdGVuY3ksIGFuZCBjbGVhbnVwXCIsICgpID0+IHtcbiAgICBjb25zdCBhcHAgPSBUZXN0aW5nLmFwcCgpO1xuICAgIGNvbnN0IHN0YWNrID0gbmV3IENvbnRhaW5lckFwcEVudmlyb25tZW50RXhhbXBsZVN0YWNrKFxuICAgICAgYXBwLFxuICAgICAgXCJ0ZXN0LWNvbnRhaW5lcmVudlwiLFxuICAgICk7XG4gICAgY29uc3Qgc3ludGhlc2l6ZWQgPSBUZXN0aW5nLmZ1bGxTeW50aChzdGFjayk7XG5cbiAgICAvLyBUaGlzIHdpbGw6XG4gICAgLy8gMS4gUnVuIHRlcnJhZm9ybSBhcHBseSB0byBkZXBsb3kgcmVzb3VyY2VzXG4gICAgLy8gMi4gUnVuIHRlcnJhZm9ybSBwbGFuIHRvIGNoZWNrIGlkZW1wb3RlbmN5IChubyBjaGFuZ2VzIGV4cGVjdGVkKVxuICAgIC8vIDMuIFJ1biB0ZXJyYWZvcm0gZGVzdHJveSB0byBjbGVhbnVwIHJlc291cmNlc1xuICAgIFRlcnJhZm9ybUFwcGx5Q2hlY2tBbmREZXN0cm95KHN5bnRoZXNpemVkLCB7IHZlcmlmeUNsZWFudXA6IHRydWUgfSk7XG4gIH0sIDYwMDAwMCk7IC8vIDEwIG1pbnV0ZSB0aW1lb3V0IGZvciBkZXBsb3ltZW50IGFuZCBjbGVhbnVwXG59KTtcbiJdfQ==
@@ -0,0 +1,9 @@
1
+ /**
2
+ * Comprehensive tests for the unified ContainerAppEnvironment implementation
3
+ *
4
+ * This test suite validates the unified ContainerAppEnvironment class using the
5
+ * AzapiResource framework. Tests cover automatic version resolution,
6
+ * explicit version pinning, schema validation, property transformation, and
7
+ * full backward compatibility.
8
+ */
9
+ export {};