@fjall/components-infrastructure 0.88.3 → 0.89.2

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 (152) hide show
  1. package/LICENSE +21 -0
  2. package/dist/lib/app.d.ts +33 -10
  3. package/dist/lib/app.js +79 -36
  4. package/dist/lib/aspects/index.d.ts +1 -0
  5. package/dist/lib/aspects/index.js +6 -0
  6. package/dist/lib/config/aws/accountAuditRole.d.ts +20 -0
  7. package/dist/lib/config/aws/accountAuditRole.js +38 -0
  8. package/dist/lib/config/aws/accountMonitoringRole.d.ts +22 -0
  9. package/dist/lib/config/aws/accountMonitoringRole.js +133 -0
  10. package/dist/lib/config/aws/cloudTrail.d.ts +0 -3
  11. package/dist/lib/config/aws/cloudTrail.js +2 -2
  12. package/dist/lib/config/aws/disasterRecovery.js +53 -20
  13. package/dist/lib/config/aws/ecrDefaultImage.js +4 -3
  14. package/dist/lib/config/aws/index.d.ts +4 -0
  15. package/dist/lib/config/aws/index.js +5 -1
  16. package/dist/lib/config/aws/oidcConnector.d.ts +8 -0
  17. package/dist/lib/config/aws/oidcConnector.js +46 -0
  18. package/dist/lib/config/aws/platform.d.ts +2 -0
  19. package/dist/lib/config/aws/platform.js +6 -0
  20. package/dist/lib/config/index.d.ts +2 -0
  21. package/dist/lib/config/index.js +21 -0
  22. package/dist/lib/patterns/aws/account.js +22 -10
  23. package/dist/lib/patterns/aws/cdn.d.ts +19 -40
  24. package/dist/lib/patterns/aws/cdn.js +21 -17
  25. package/dist/lib/patterns/aws/compute.d.ts +9 -720
  26. package/dist/lib/patterns/aws/compute.js +27 -432
  27. package/dist/lib/patterns/aws/computeEc2.d.ts +67 -0
  28. package/dist/lib/patterns/aws/computeEc2.js +46 -0
  29. package/dist/lib/patterns/aws/computeEcs.d.ts +446 -0
  30. package/dist/lib/patterns/aws/computeEcs.js +246 -0
  31. package/dist/lib/patterns/aws/computeLambda.d.ts +220 -0
  32. package/dist/lib/patterns/aws/computeLambda.js +147 -0
  33. package/dist/lib/patterns/aws/database.d.ts +7 -87
  34. package/dist/lib/patterns/aws/database.js +15 -38
  35. package/dist/lib/patterns/aws/domainDelegation.d.ts +8 -0
  36. package/dist/lib/patterns/aws/domainDelegation.js +54 -0
  37. package/dist/lib/patterns/aws/domainFactory.d.ts +8 -0
  38. package/dist/lib/patterns/aws/domainFactory.js +23 -0
  39. package/dist/lib/patterns/aws/index.d.ts +4 -1
  40. package/dist/lib/patterns/aws/index.js +6 -2
  41. package/dist/lib/patterns/aws/interfaces/cdn.d.ts +26 -0
  42. package/dist/lib/patterns/aws/interfaces/cdn.js +14 -0
  43. package/dist/lib/patterns/aws/interfaces/connector.d.ts +4 -181
  44. package/dist/lib/patterns/aws/interfaces/connector.js +16 -113
  45. package/dist/lib/patterns/aws/interfaces/domain.d.ts +2 -0
  46. package/dist/lib/patterns/aws/interfaces/domain.js +6 -0
  47. package/dist/lib/patterns/aws/interfaces/index.d.ts +2 -0
  48. package/dist/lib/patterns/aws/interfaces/index.js +5 -2
  49. package/dist/lib/patterns/aws/interfaces/pattern.d.ts +9 -6
  50. package/dist/lib/patterns/aws/interfaces/pattern.js +1 -1
  51. package/dist/lib/patterns/aws/network.js +6 -9
  52. package/dist/lib/patterns/aws/organisation.d.ts +4 -2
  53. package/dist/lib/patterns/aws/organisation.js +21 -8
  54. package/dist/lib/patterns/aws/payload.js +21 -12
  55. package/dist/lib/patterns/aws/storage.d.ts +3 -2
  56. package/dist/lib/patterns/aws/storage.js +1 -1
  57. package/dist/lib/resources/aws/audit/auditRole.js +4 -4
  58. package/dist/lib/resources/aws/audit/index.d.ts +1 -0
  59. package/dist/lib/resources/aws/audit/index.js +6 -0
  60. package/dist/lib/resources/aws/backup/backupPlan.js +3 -2
  61. package/dist/lib/resources/aws/backup/backupVault.js +5 -3
  62. package/dist/lib/resources/aws/base/awsStack.d.ts +4 -2
  63. package/dist/lib/resources/aws/base/awsStack.js +8 -2
  64. package/dist/lib/resources/aws/cdn/cloudFront.d.ts +14 -0
  65. package/dist/lib/resources/aws/cdn/cloudFront.js +52 -18
  66. package/dist/lib/resources/aws/compute/ec2.js +18 -22
  67. package/dist/lib/resources/aws/compute/ecs.d.ts +23 -10
  68. package/dist/lib/resources/aws/compute/ecs.js +121 -64
  69. package/dist/lib/resources/aws/compute/index.d.ts +1 -0
  70. package/dist/lib/resources/aws/compute/index.js +2 -1
  71. package/dist/lib/resources/aws/compute/lambda.d.ts +0 -2
  72. package/dist/lib/resources/aws/compute/lambda.js +12 -27
  73. package/dist/lib/resources/aws/database/dynamodb.js +3 -13
  74. package/dist/lib/resources/aws/database/index.d.ts +8 -2
  75. package/dist/lib/resources/aws/database/index.js +19 -3
  76. package/dist/lib/resources/aws/database/rdsAurora.d.ts +2 -3
  77. package/dist/lib/resources/aws/database/rdsAurora.js +32 -68
  78. package/dist/lib/resources/aws/database/rdsAuroraGlobal.d.ts +6 -6
  79. package/dist/lib/resources/aws/database/rdsAuroraGlobal.js +25 -29
  80. package/dist/lib/resources/aws/database/rdsDefaults.d.ts +11 -0
  81. package/dist/lib/resources/aws/database/rdsDefaults.js +15 -0
  82. package/dist/lib/resources/aws/database/rdsHelpers.d.ts +39 -0
  83. package/dist/lib/resources/aws/database/rdsHelpers.js +75 -0
  84. package/dist/lib/resources/aws/database/rdsInstance.d.ts +7 -8
  85. package/dist/lib/resources/aws/database/rdsInstance.js +40 -84
  86. package/dist/lib/resources/aws/database/rdsProxyOutput.d.ts +7 -0
  87. package/dist/lib/resources/aws/database/rdsProxyOutput.js +18 -0
  88. package/dist/lib/resources/aws/iam/index.d.ts +0 -1
  89. package/dist/lib/resources/aws/iam/index.js +1 -2
  90. package/dist/lib/resources/aws/index.d.ts +0 -1
  91. package/dist/lib/resources/aws/index.js +1 -2
  92. package/dist/lib/resources/aws/logging/cloudTrail.js +13 -3
  93. package/dist/lib/resources/aws/logging/index.d.ts +2 -0
  94. package/dist/lib/resources/aws/logging/index.js +19 -0
  95. package/dist/lib/resources/aws/messaging/index.d.ts +3 -2
  96. package/dist/lib/resources/aws/messaging/index.js +4 -3
  97. package/dist/lib/resources/aws/messaging/sqs.js +14 -11
  98. package/dist/lib/resources/aws/messaging/utils.d.ts +1 -2
  99. package/dist/lib/resources/aws/messaging/utils.js +3 -4
  100. package/dist/lib/resources/aws/monitoring/index.d.ts +0 -1
  101. package/dist/lib/resources/aws/monitoring/index.js +4 -17
  102. package/dist/lib/resources/aws/networking/domain.d.ts +13 -0
  103. package/dist/lib/resources/aws/networking/domain.js +102 -0
  104. package/dist/lib/resources/aws/networking/domainCertificate.d.ts +13 -0
  105. package/dist/lib/resources/aws/networking/domainCertificate.js +28 -0
  106. package/dist/lib/resources/aws/networking/hostedZone.d.ts +28 -0
  107. package/dist/lib/resources/aws/networking/hostedZone.js +150 -0
  108. package/dist/lib/resources/aws/networking/index.d.ts +4 -0
  109. package/dist/lib/resources/aws/networking/index.js +5 -1
  110. package/dist/lib/resources/aws/networking/ipamPool.js +57 -31
  111. package/dist/lib/resources/aws/networking/securityGroup.d.ts +5 -0
  112. package/dist/lib/resources/aws/networking/securityGroup.js +14 -0
  113. package/dist/lib/resources/aws/networking/vpc.js +9 -4
  114. package/dist/lib/resources/aws/organisation/costAllocationTagActivator.d.ts +17 -0
  115. package/dist/lib/resources/aws/organisation/costAllocationTagActivator.js +66 -0
  116. package/dist/lib/resources/aws/organisation/index.d.ts +1 -0
  117. package/dist/lib/resources/aws/organisation/index.js +4 -2
  118. package/dist/lib/resources/aws/secrets/index.d.ts +0 -1
  119. package/dist/lib/resources/aws/secrets/index.js +1 -2
  120. package/dist/lib/resources/aws/storage/ecr.d.ts +0 -1
  121. package/dist/lib/resources/aws/storage/ecr.js +5 -5
  122. package/dist/lib/resources/aws/storage/s3.d.ts +3 -3
  123. package/dist/lib/resources/aws/storage/s3.js +1 -1
  124. package/dist/lib/resources/aws/utilities/index.d.ts +5 -0
  125. package/dist/lib/resources/aws/utilities/index.js +22 -0
  126. package/dist/lib/utils/backupTierMapping.d.ts +11 -0
  127. package/dist/lib/utils/backupTierMapping.js +17 -0
  128. package/dist/lib/utils/capitaliseString.d.ts +1 -12
  129. package/dist/lib/utils/capitaliseString.js +8 -28
  130. package/dist/lib/utils/connections.d.ts +46 -0
  131. package/dist/lib/utils/connections.js +159 -0
  132. package/dist/lib/utils/connector.d.ts +183 -0
  133. package/dist/lib/utils/connector.js +117 -0
  134. package/dist/lib/utils/databaseTypes.d.ts +85 -0
  135. package/dist/lib/utils/databaseTypes.js +34 -0
  136. package/dist/lib/utils/dnsRecords.d.ts +4 -0
  137. package/dist/lib/utils/dnsRecords.js +108 -0
  138. package/dist/lib/utils/domainTypes.d.ts +37 -0
  139. package/dist/lib/utils/domainTypes.js +10 -0
  140. package/dist/lib/utils/env.d.ts +42 -0
  141. package/dist/lib/utils/env.js +122 -0
  142. package/dist/lib/utils/getConfig.d.ts +0 -5
  143. package/dist/lib/utils/getConfig.js +42 -19
  144. package/dist/lib/utils/index.d.ts +7 -0
  145. package/dist/lib/utils/index.js +8 -1
  146. package/dist/lib/utils/removalPolicy.d.ts +2 -0
  147. package/dist/lib/utils/removalPolicy.js +16 -0
  148. package/dist/lib/utils/standardTagsAspect.d.ts +4 -0
  149. package/dist/lib/utils/standardTagsAspect.js +8 -8
  150. package/dist/lib/utils/vpcUtils.d.ts +14 -0
  151. package/dist/lib/utils/vpcUtils.js +28 -0
  152. package/package.json +7 -6
@@ -1,69 +1,55 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.isEc2Compute = exports.isLambdaCompute = exports.isEcsCompute = exports.isCompute = exports.Ec2Compute = exports.LambdaCompute = exports.EcsCompute = exports.ComputeFactory = exports.InvokeMode = exports.HttpMethod = exports.Architecture = exports.COMPUTE_DEFAULTS = exports.ECS_CAPACITY_PROVIDER_CONFIG = exports.COMPUTE_TYPE_CONFIG = exports.ScalingType = void 0;
3
+ exports.isEc2Compute = exports.isLambdaCompute = exports.isEcsCompute = exports.isCompute = exports.ComputeFactory = exports.COMPUTE_DEFAULTS = exports.COMPUTE_TYPE_CONFIG = exports.Ec2Compute = exports.InvokeMode = exports.HttpMethod = exports.Architecture = exports.resolveLambdaDeployment = exports.LambdaCompute = exports.resolveScalingConfig = exports.buildContainerConfigs = exports.validateEcsProps = exports.ScalingType = exports.getEcsCapacityProviderConfig = exports.ECS_CAPACITY_PROVIDER_CONFIG = exports.EcsCompute = void 0;
4
4
  exports.getComputeTypeConfig = getComputeTypeConfig;
5
- exports.getEcsCapacityProviderConfig = getEcsCapacityProviderConfig;
6
- exports.buildContainerConfigs = buildContainerConfigs;
7
- exports.resolveScalingConfig = resolveScalingConfig;
8
- exports.resolveLambdaDeployment = resolveLambdaDeployment;
9
- const aws_ecr_1 = require("aws-cdk-lib/aws-ecr");
10
- const aws_ec2_1 = require("aws-cdk-lib/aws-ec2");
11
5
  const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
12
- const aws_iam_1 = require("aws-cdk-lib/aws-iam");
13
- const constructs_1 = require("constructs");
14
- // Import type-safe interfaces
15
6
  const compute_js_1 = require("./interfaces/compute.js");
16
7
  Object.defineProperty(exports, "isCompute", { enumerable: true, get: function () { return compute_js_1.isCompute; } });
17
8
  Object.defineProperty(exports, "isEcsCompute", { enumerable: true, get: function () { return compute_js_1.isEcsCompute; } });
18
9
  Object.defineProperty(exports, "isLambdaCompute", { enumerable: true, get: function () { return compute_js_1.isLambdaCompute; } });
19
10
  Object.defineProperty(exports, "isEc2Compute", { enumerable: true, get: function () { return compute_js_1.isEc2Compute; } });
20
- const connections_js_1 = require("./connections.js");
21
11
  const validationLogger_js_1 = require("../../utils/validationLogger.js");
22
- const ecs_1 = require("../../resources/aws/compute/ecs");
23
- Object.defineProperty(exports, "ScalingType", { enumerable: true, get: function () { return ecs_1.ScalingType; } });
24
- const ec2_1 = require("../../resources/aws/compute/ec2");
25
- const lambda_1 = require("../../resources/aws/compute/lambda");
12
+ // Import and re-export from per-pattern files
13
+ const computeEcs_js_1 = require("./computeEcs.js");
14
+ Object.defineProperty(exports, "EcsCompute", { enumerable: true, get: function () { return computeEcs_js_1.EcsCompute; } });
15
+ Object.defineProperty(exports, "ECS_CAPACITY_PROVIDER_CONFIG", { enumerable: true, get: function () { return computeEcs_js_1.ECS_CAPACITY_PROVIDER_CONFIG; } });
16
+ Object.defineProperty(exports, "getEcsCapacityProviderConfig", { enumerable: true, get: function () { return computeEcs_js_1.getEcsCapacityProviderConfig; } });
17
+ Object.defineProperty(exports, "ScalingType", { enumerable: true, get: function () { return computeEcs_js_1.ScalingType; } });
18
+ Object.defineProperty(exports, "validateEcsProps", { enumerable: true, get: function () { return computeEcs_js_1.validateEcsProps; } });
19
+ Object.defineProperty(exports, "buildContainerConfigs", { enumerable: true, get: function () { return computeEcs_js_1.buildContainerConfigs; } });
20
+ Object.defineProperty(exports, "resolveScalingConfig", { enumerable: true, get: function () { return computeEcs_js_1.resolveScalingConfig; } });
21
+ const computeLambda_js_1 = require("./computeLambda.js");
22
+ Object.defineProperty(exports, "LambdaCompute", { enumerable: true, get: function () { return computeLambda_js_1.LambdaCompute; } });
23
+ Object.defineProperty(exports, "resolveLambdaDeployment", { enumerable: true, get: function () { return computeLambda_js_1.resolveLambdaDeployment; } });
24
+ Object.defineProperty(exports, "Architecture", { enumerable: true, get: function () { return computeLambda_js_1.Architecture; } });
25
+ Object.defineProperty(exports, "HttpMethod", { enumerable: true, get: function () { return computeLambda_js_1.HttpMethod; } });
26
+ Object.defineProperty(exports, "InvokeMode", { enumerable: true, get: function () { return computeLambda_js_1.InvokeMode; } });
27
+ const computeEc2_js_1 = require("./computeEc2.js");
28
+ Object.defineProperty(exports, "Ec2Compute", { enumerable: true, get: function () { return computeEc2_js_1.Ec2Compute; } });
26
29
  exports.COMPUTE_TYPE_CONFIG = {
27
30
  ecs: {
28
31
  defaultPort: 80,
29
32
  defaultScaling: { minCapacity: 2, maxCapacity: 5 },
30
- supportsConnections: true,
31
33
  requiresVpc: true
32
34
  },
33
35
  ec2: {
34
36
  defaultPort: 22,
35
37
  defaultScaling: { minCapacity: 2, maxCapacity: 3 },
36
- supportsConnections: true,
37
38
  requiresVpc: true
38
39
  },
39
40
  lambda: {
40
41
  defaultPort: 0,
41
42
  defaultScaling: { minCapacity: 0, maxCapacity: 100 },
42
- supportsConnections: false,
43
43
  requiresVpc: false
44
44
  }
45
45
  };
46
- exports.ECS_CAPACITY_PROVIDER_CONFIG = {
47
- FARGATE: {
48
- usesSpot: false,
49
- usesEc2Instances: false
50
- },
51
- FARGATE_SPOT: {
52
- usesSpot: true,
53
- usesEc2Instances: false
54
- },
55
- EC2: {
56
- usesSpot: false,
57
- usesEc2Instances: true
58
- }
59
- };
60
46
  /**
61
47
  * Default values for compute resource configuration.
62
48
  * Centralised constants to ensure consistency across the codebase.
63
49
  */
64
50
  exports.COMPUTE_DEFAULTS = {
65
51
  EC2: {
66
- INSTANCE_TYPE: "t3.micro",
52
+ INSTANCE_TYPE: "t4g.micro",
67
53
  MIN_CAPACITY: 1,
68
54
  MAX_CAPACITY: 1
69
55
  },
@@ -75,20 +61,13 @@ exports.COMPUTE_DEFAULTS = {
75
61
  },
76
62
  LAMBDA: {
77
63
  HANDLER: "index.handler",
78
- RUNTIME: aws_lambda_1.Runtime.NODEJS_22_X
64
+ RUNTIME: aws_lambda_1.Runtime.NODEJS_22_X,
65
+ ARCHITECTURE: computeLambda_js_1.Architecture.ARM_64
79
66
  }
80
67
  };
81
68
  function getComputeTypeConfig(type) {
82
69
  return exports.COMPUTE_TYPE_CONFIG[type];
83
70
  }
84
- function getEcsCapacityProviderConfig(provider) {
85
- return exports.ECS_CAPACITY_PROVIDER_CONFIG[provider];
86
- }
87
- // Re-export Lambda types from CDK for user convenience
88
- var aws_lambda_2 = require("aws-cdk-lib/aws-lambda");
89
- Object.defineProperty(exports, "Architecture", { enumerable: true, get: function () { return aws_lambda_2.Architecture; } });
90
- Object.defineProperty(exports, "HttpMethod", { enumerable: true, get: function () { return aws_lambda_2.HttpMethod; } });
91
- Object.defineProperty(exports, "InvokeMode", { enumerable: true, get: function () { return aws_lambda_2.InvokeMode; } });
92
71
  /**
93
72
  * Validates compute props and logs warnings for ignored or misconfigured options.
94
73
  * These checks help catch issues when props come from dynamic sources where
@@ -103,142 +82,8 @@ function validateComputeProps(props) {
103
82
  (0, validationLogger_js_1.warnIfPropertiesIgnored)(props, ["ssh", "userData", "machineImage"], "ec2", "EC2 compute");
104
83
  // ECS-specific validation - type narrowing handled by discriminated union
105
84
  if (props.type === "ecs") {
106
- validateEcsProps(props);
107
- }
108
- }
109
- /**
110
- * Validates ECS-specific props.
111
- * Extracted for clarity and detail parity with database/network patterns.
112
- */
113
- function validateEcsProps(props) {
114
- // Validate services array exists and is not empty
115
- if (!props.services || props.services.length === 0) {
116
- throw new Error("At least one service must be specified in 'services' array.");
117
- }
118
- // Validate unique service names
119
- const serviceNames = props.services.map((s) => s.name);
120
- const duplicateNames = serviceNames.filter((name, index) => serviceNames.indexOf(name) !== index);
121
- if (duplicateNames.length > 0) {
122
- throw new Error(`Duplicate service names: ${[...new Set(duplicateNames)].join(", ")}`);
123
- }
124
- // Validate service name format
125
- const invalidNames = props.services.filter((s) => !/^[a-zA-Z][a-zA-Z0-9-]*$/.test(s.name));
126
- if (invalidNames.length > 0) {
127
- throw new Error(`Invalid service names: ${invalidNames.map((s) => s.name).join(", ")}. ` +
128
- "Service names must start with a letter and contain only letters, numbers, and hyphens.");
129
- }
130
- // Validate routing when multiple services have ports
131
- const servicesWithPorts = props.services.filter((s) => s.containers?.some((c) => c.port));
132
- if (servicesWithPorts.length > 1) {
133
- const missingRouting = servicesWithPorts.filter((s) => {
134
- const rules = Array.isArray(s.routing)
135
- ? s.routing
136
- : s.routing
137
- ? [s.routing]
138
- : [];
139
- return !rules.some((r) => r.path || r.host);
140
- });
141
- if (missingRouting.length > 0) {
142
- throw new Error(`Services with ports require routing config when cluster has multiple services: ` +
143
- `${missingRouting.map((s) => s.name).join(", ")}. ` +
144
- "Add routing: { path: '/...' } or routing: { host: '...' } to each service.");
145
- }
146
- }
147
- // Validate each service's containers
148
- for (const service of props.services) {
149
- if (service.containers && service.containers.length > 0) {
150
- // Check for duplicate container names within service (only named containers)
151
- const containerNames = service.containers
152
- .filter((c) => c.name)
153
- .map((c) => c.name);
154
- const duplicateContainerNames = containerNames.filter((name, index) => containerNames.indexOf(name) !== index);
155
- if (duplicateContainerNames.length > 0) {
156
- throw new Error(`Service '${service.name}': Duplicate container names: ` +
157
- `${[...new Set(duplicateContainerNames)].join(", ")}`);
158
- }
159
- }
160
- if (service.capacityProvider === "EC2" && !service.ec2Config) {
161
- throw new Error(`Service '${service.name}' uses EC2 capacity provider but no ec2Config is defined. ` +
162
- "Provide ec2Config on the service.");
163
- }
164
- // Warn if service ec2Config is defined but capacityProvider is not EC2
165
- if (service.ec2Config && service.capacityProvider !== "EC2") {
166
- validationLogger_js_1.FjallLogger.warn(`Service '${service.name}' has ec2Config but capacityProvider is not 'EC2'. ` +
167
- "The ec2Config will be ignored unless capacityProvider is set to 'EC2'.");
168
- }
169
- }
170
- }
171
- /**
172
- * Build container configurations for an ECS service.
173
- * Converts user-facing EcsContainerConfig to internal EcsClusterProps format.
174
- * @internal Exported for testing only
175
- */
176
- function buildContainerConfigs(service) {
177
- if (service.containers && service.containers.length > 0) {
178
- return service.containers.map((c, index) => ({
179
- name: c.name || `${service.name}Container${index > 0 ? index : ""}`,
180
- image: c.image,
181
- port: c.port,
182
- environment: c.environment,
183
- secrets: c.secrets,
184
- secretsImport: c.secretsImport,
185
- command: c.command,
186
- entryPoint: c.entryPoint,
187
- essential: c.essential,
188
- healthCheck: c.healthCheck
189
- }));
190
- }
191
- // Default container (no port = worker)
192
- return [{ name: `${service.name}Container` }];
193
- }
194
- /**
195
- * Resolve scaling configuration from service props.
196
- * Handles the three cases: explicit config, disabled (false), or default (undefined).
197
- * @internal Exported for testing only
198
- */
199
- function resolveScalingConfig(scaling) {
200
- if (scaling === false) {
201
- return {
202
- scalingType: undefined,
203
- minCapacity: undefined,
204
- maxCapacity: undefined
205
- };
206
- }
207
- if (scaling === undefined) {
208
- return {
209
- scalingType: ecs_1.ScalingType.CPU,
210
- minCapacity: undefined,
211
- maxCapacity: undefined
212
- };
85
+ (0, computeEcs_js_1.validateEcsProps)(props);
213
86
  }
214
- return {
215
- scalingType: scaling.scalingType ?? ecs_1.ScalingType.CPU,
216
- minCapacity: scaling.minCapacity,
217
- maxCapacity: scaling.maxCapacity
218
- };
219
- }
220
- /**
221
- * Resolve Lambda deployment configuration from props.
222
- * Handles container vs code deployment types.
223
- */
224
- function resolveLambdaDeployment(props) {
225
- if (props.deployment === "container") {
226
- if (!(props.ecrRepository instanceof aws_ecr_1.Repository)) {
227
- throw new Error("ecrRepository must be a Repository instance for Lambda compute type");
228
- }
229
- return {
230
- code: aws_lambda_1.Code.fromEcrImage(props.ecrRepository, {
231
- tagOrDigest: exports.COMPUTE_DEFAULTS.ECS.IMAGE_TAG
232
- }),
233
- handler: aws_lambda_1.Handler.FROM_IMAGE,
234
- runtime: aws_lambda_1.Runtime.FROM_IMAGE
235
- };
236
- }
237
- return {
238
- code: props.code,
239
- handler: props.handler || exports.COMPUTE_DEFAULTS.LAMBDA.HANDLER,
240
- runtime: props.runtime || exports.COMPUTE_DEFAULTS.LAMBDA.RUNTIME
241
- };
242
87
  }
243
88
  /**
244
89
  * Factory for creating compute resources with type-safe return types.
@@ -268,7 +113,7 @@ function resolveLambdaDeployment(props) {
268
113
  * // EC2 compute - returns Ec2Compute with EC2-specific methods
269
114
  * const ec2 = app.addCompute(ComputeFactory.build("Instance", {
270
115
  * type: "ec2",
271
- * instanceType: "t3.micro"
116
+ * instanceType: "t4g.micro"
272
117
  * }));
273
118
  * ec2.getAutoScalingGroup(); // Available on Ec2Compute
274
119
  */
@@ -339,7 +184,7 @@ class ComputeFactory {
339
184
  `/${appName}/${clusterName}/${service.name}`;
340
185
  collector.addService(manifestService);
341
186
  }
342
- return new EcsCompute(scope, id, computeProps);
187
+ return new computeEcs_js_1.EcsCompute(scope, id, computeProps);
343
188
  }
344
189
  case "lambda": {
345
190
  // Add appName to props for Lambda (needed for secrets path derivation)
@@ -361,10 +206,10 @@ class ComputeFactory {
361
206
  manifestLambda.secrets = lambdaComputeProps.secrets;
362
207
  }
363
208
  collector.addLambda(manifestLambda);
364
- return new LambdaCompute(scope, id, lambdaComputeProps);
209
+ return new computeLambda_js_1.LambdaCompute(scope, id, lambdaComputeProps);
365
210
  }
366
211
  case "ec2":
367
- return new Ec2Compute(scope, id, computeProps);
212
+ return new computeEc2_js_1.Ec2Compute(scope, id, computeProps);
368
213
  default: {
369
214
  const _exhaustive = computeProps;
370
215
  throw new Error(`Unsupported compute type: ${String(_exhaustive.type)}`);
@@ -374,254 +219,4 @@ class ComputeFactory {
374
219
  }
375
220
  }
376
221
  exports.ComputeFactory = ComputeFactory;
377
- /**
378
- * ECS compute wrapper implementing IEcsCompute.
379
- * Provides type-safe access to ECS-specific resources.
380
- */
381
- class EcsCompute extends constructs_1.Construct {
382
- constructor(scope, id, props) {
383
- super(scope, id);
384
- this.computeType = "ecs";
385
- // Transform EcsServiceConfig[] to EcsServiceProps[] for EcsCluster
386
- const services = props.services.map((service) => {
387
- const containers = buildContainerConfigs(service);
388
- const { scalingType, minCapacity, maxCapacity } = resolveScalingConfig(service.scaling);
389
- return {
390
- name: service.name,
391
- image: service.image,
392
- containers,
393
- cpu: service.cpu,
394
- memoryLimitMiB: service.memoryLimitMiB,
395
- desiredCount: service.desiredCount,
396
- scalingType,
397
- minCapacity,
398
- maxCapacity,
399
- routing: service.routing,
400
- taskRoleInlinePolicies: service.taskRoleInlinePolicies,
401
- taskRoleManagedPolicies: service.taskRoleManagedPolicies,
402
- connections: service.connections,
403
- capacityProvider: service.capacityProvider,
404
- ec2Config: service.ec2Config,
405
- ssmSecretsPath: service.ssmSecretsPath,
406
- dockerTarget: service.dockerTarget
407
- };
408
- });
409
- // Build cluster config
410
- const cluster = props.cluster
411
- ? {
412
- domain: props.cluster.domain,
413
- loadBalancer: props.cluster.loadBalancer,
414
- directAccess: props.cluster.directAccess,
415
- domainConfig: props.cluster.domainConfig
416
- }
417
- : undefined;
418
- const ecsProps = {
419
- clusterName: id,
420
- appName: props.appName,
421
- vpc: props.vpc,
422
- ecrRepository: props.ecrRepository || exports.COMPUTE_DEFAULTS.ECS.FALLBACK_IMAGE,
423
- cluster,
424
- services
425
- };
426
- this.ecsCluster = new ecs_1.default(scope, `${id}Ecs`, ecsProps);
427
- this.connections = this.ecsCluster.connections;
428
- }
429
- /** Get the ECS cluster. */
430
- getCluster() {
431
- return this.ecsCluster.getCluster();
432
- }
433
- /** Get the Application Load Balancer if one was created. */
434
- getLoadBalancer() {
435
- return this.ecsCluster.getLoadBalancer();
436
- }
437
- /** Get a specific service by name. */
438
- getService(name) {
439
- return this.ecsCluster.getService(name);
440
- }
441
- /** Get all services in the cluster. */
442
- getAllServices() {
443
- const servicesMap = this.ecsCluster.getServices();
444
- return Array.from(servicesMap.values());
445
- }
446
- /** Get the security group for the cluster. */
447
- getSecurityGroup() {
448
- // Return the first security group from connections
449
- const securityGroups = this.connections.securityGroups;
450
- if (securityGroups.length === 0) {
451
- throw new Error("No security groups found for ECS cluster");
452
- }
453
- return securityGroups[0];
454
- }
455
- /**
456
- * Get the ALB listener if this is an ECS compute with ALB.
457
- */
458
- getListener() {
459
- return this.ecsCluster.getListener();
460
- }
461
- /**
462
- * Get the underlying ECS cluster construct.
463
- */
464
- getEcsCluster() {
465
- return this.ecsCluster;
466
- }
467
- /**
468
- * Grants ecs:ExecuteCommand permission for ECS services.
469
- * Uses wildcard resource because ecs:ExecuteCommand targets task ARNs
470
- * which are not known until runtime (tasks are ephemeral).
471
- */
472
- grantExecuteCommand(grantee) {
473
- return aws_iam_1.Grant.addToPrincipal({
474
- grantee,
475
- actions: ["ecs:ExecuteCommand"],
476
- resourceArns: ["*"]
477
- });
478
- }
479
- }
480
- exports.EcsCompute = EcsCompute;
481
- /**
482
- * Lambda compute wrapper implementing ILambdaCompute.
483
- * Provides type-safe access to Lambda-specific resources.
484
- */
485
- class LambdaCompute extends constructs_1.Construct {
486
- constructor(scope, id, props) {
487
- super(scope, id);
488
- this.computeType = "lambda";
489
- const { code, handler, runtime } = resolveLambdaDeployment(props);
490
- // Convert presence-based functionUrl config to legacy boolean format
491
- const functionUrlConfig = props.functionUrl;
492
- const enableFunctionUrl = functionUrlConfig !== undefined && functionUrlConfig !== false;
493
- const functionUrlAuthType = enableFunctionUrl && typeof functionUrlConfig === "object"
494
- ? functionUrlConfig.authType
495
- : undefined;
496
- const functionUrlCors = enableFunctionUrl && typeof functionUrlConfig === "object"
497
- ? functionUrlConfig.cors
498
- : undefined;
499
- const functionUrlInvokeMode = enableFunctionUrl && typeof functionUrlConfig === "object"
500
- ? functionUrlConfig.invokeMode
501
- : undefined;
502
- const lambdaProps = {
503
- code,
504
- vpc: props.vpc,
505
- handler,
506
- runtime,
507
- architecture: props.architecture,
508
- timeout: props.timeout,
509
- memorySize: props.memorySize,
510
- lambdaDescription: props.description,
511
- roleDescription: props.roleDescription,
512
- inlinePolicy: props.inlinePolicy || [],
513
- enableFunctionUrl,
514
- functionUrlAuthType,
515
- functionUrlCors,
516
- functionUrlInvokeMode,
517
- scheduleExpression: props.scheduleExpression,
518
- ephemeralStorageSize: props.ephemeralStorageSize,
519
- secrets: props.secrets,
520
- ssmSecretsPath: props.ssmSecretsPath,
521
- secretsImport: props.secretsImport,
522
- environment: props.environment,
523
- appName: props.appName,
524
- functionName: id
525
- };
526
- this.lambdaFunction = new lambda_1.LambdaFunction(scope, `${id}Lambda`, lambdaProps);
527
- // CDK's Function.connections getter throws for non-VPC Lambdas, so only
528
- // access it when a VPC is configured. Non-VPC Lambdas get an empty
529
- // Connections object to satisfy the IConnectable interface.
530
- this.connections = props.vpc
531
- ? this.lambdaFunction.connections
532
- : new aws_ec2_1.Connections();
533
- // Process connections using the unified connector interface
534
- if (props.connections && props.connections.length > 0) {
535
- if (!props.vpc) {
536
- throw new Error(`Lambda '${id}' has connections configured but no VPC. ` +
537
- "A VPC is required for security group connections.");
538
- }
539
- try {
540
- (0, connections_js_1.processConnections)(props.connections, this.lambdaFunction, // IGrantable (execution role)
541
- this.lambdaFunction // IConnectable (security group)
542
- );
543
- }
544
- catch (error) {
545
- throw new Error(`Failed to process connections for Lambda '${id}': ${error instanceof Error ? error.message : String(error)}`);
546
- }
547
- }
548
- }
549
- /**
550
- * Get a Lambda function by name.
551
- * Since we only have one function, name is ignored.
552
- */
553
- getFunction(_name) {
554
- return this.lambdaFunction;
555
- }
556
- /** Get all Lambda functions. */
557
- getAllFunctions() {
558
- return [this.lambdaFunction];
559
- }
560
- /**
561
- * Get the function URL for a Lambda function.
562
- */
563
- getFunctionUrl(_name) {
564
- return this.lambdaFunction.getFunctionUrl();
565
- }
566
- /**
567
- * Grant invoke permissions to a grantee.
568
- */
569
- grantInvoke(grantee, _functionName) {
570
- return this.lambdaFunction.grantInvoke(grantee);
571
- }
572
- /**
573
- * Get the security group for VPC-enabled Lambda functions.
574
- * Returns undefined if the Lambda is not VPC-enabled.
575
- */
576
- getSecurityGroup() {
577
- const securityGroups = this.connections.securityGroups;
578
- return securityGroups.length > 0 ? securityGroups[0] : undefined;
579
- }
580
- /**
581
- * Get the underlying Lambda function construct.
582
- */
583
- getLambdaFunction() {
584
- return this.lambdaFunction;
585
- }
586
- }
587
- exports.LambdaCompute = LambdaCompute;
588
- /**
589
- * EC2 compute wrapper implementing IEc2Compute.
590
- * Provides type-safe access to EC2-specific resources.
591
- */
592
- class Ec2Compute extends constructs_1.Construct {
593
- constructor(scope, id, props) {
594
- super(scope, id);
595
- this.computeType = "ec2";
596
- const enableSSH = props.ssh !== undefined && props.ssh !== false;
597
- const ec2Props = {
598
- serviceName: `${id}Service`,
599
- instanceType: props.instanceType || exports.COMPUTE_DEFAULTS.EC2.INSTANCE_TYPE,
600
- enableSSH,
601
- userData: props.userData,
602
- machineImage: props.machineImage,
603
- minCapacity: props.minCapacity || exports.COMPUTE_DEFAULTS.EC2.MIN_CAPACITY,
604
- maxCapacity: props.maxCapacity || exports.COMPUTE_DEFAULTS.EC2.MAX_CAPACITY,
605
- vpc: props.vpc,
606
- spotCapacityPercentage: props.spotCapacityPercentage
607
- };
608
- this.ec2Instance = new ec2_1.Ec2Instance(scope, `${id}Ec2`, ec2Props);
609
- this.connections = this.ec2Instance.connections;
610
- }
611
- /** Get the Auto Scaling Group. */
612
- getAutoScalingGroup() {
613
- return this.ec2Instance.getAutoScalingGroup();
614
- }
615
- /** Get the security group. */
616
- getSecurityGroup() {
617
- return this.ec2Instance.asgSecurityGroup;
618
- }
619
- /**
620
- * Get the underlying EC2 instance construct.
621
- */
622
- getEc2Instance() {
623
- return this.ec2Instance;
624
- }
625
- }
626
- exports.Ec2Compute = Ec2Compute;
627
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"compute.js","sourceRoot":"","sources":["../../../../lib/patterns/aws/compute.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  type RepositoryImage,\n  type ICluster,\n  type IBaseService\n} from \"aws-cdk-lib/aws-ecs\";\nimport { Repository } from \"aws-cdk-lib/aws-ecr\";\nimport {\n  Connections,\n  type IVpc,\n  type UserData,\n  type IMachineImage,\n  type ISecurityGroup\n} from \"aws-cdk-lib/aws-ec2\";\nimport {\n  Code,\n  Runtime,\n  Handler,\n  type Architecture,\n  type FunctionUrlAuthType,\n  type FunctionUrlCorsOptions,\n  type IFunction,\n  type InvokeMode\n} from \"aws-cdk-lib/aws-lambda\";\nimport {\n  type PolicyStatement,\n  type PolicyDocument,\n  type IManagedPolicy,\n  type IGrantable,\n  Grant\n} from \"aws-cdk-lib/aws-iam\";\nimport {\n  type IApplicationLoadBalancer,\n  type ApplicationListener\n} from \"aws-cdk-lib/aws-elasticloadbalancingv2\";\nimport { type IAutoScalingGroup } from \"aws-cdk-lib/aws-autoscaling\";\n\nimport { Construct } from \"constructs\";\n\n// Import type-safe interfaces\nimport {\n  type IEcsCompute,\n  type ILambdaCompute,\n  type IEc2Compute,\n  type AnyCompute,\n  isCompute,\n  isEcsCompute,\n  isLambdaCompute,\n  isEc2Compute\n} from \"./interfaces/compute.js\";\nimport { type ConnectionSpec } from \"./interfaces/connector.js\";\nimport { processConnections } from \"./connections.js\";\n\nimport type App from \"../../app\";\nimport {\n  FjallLogger,\n  warnIfPropertiesIgnored\n} from \"../../utils/validationLogger.js\";\nimport {\n  type ManifestService,\n  type ManifestLambda\n} from \"../../utils/manifestWriter.js\";\n\nimport EcsCluster, {\n  type EcsClusterProps,\n  ScalingType,\n  type DomainConfig,\n  type EcsCapacityProvider,\n  type Ec2CapacityConfig\n} from \"../../resources/aws/compute/ecs\";\nimport { Ec2Instance } from \"../../resources/aws/compute/ec2\";\nimport { LambdaFunction } from \"../../resources/aws/compute/lambda\";\nimport { type SecretImport } from \"../../resources/aws/secrets\";\n\nexport type ComputeType = \"ecs\" | \"ec2\" | \"lambda\";\n\n// Re-export from ecs.ts to maintain single source of truth\nexport { ScalingType };\nexport type { EcsCapacityProvider, Ec2CapacityConfig };\n\n/**\n * Configuration defaults for each compute type.\n */\nexport interface ComputeTypeConfig {\n  /** Default port for the compute type. ECS: 80, EC2: 22, Lambda: 0 (no port) */\n  defaultPort: number;\n  /** Default scaling limits */\n  defaultScaling: {\n    minCapacity: number;\n    maxCapacity: number;\n  };\n  /** Whether this compute type supports security group connections */\n  supportsConnections: boolean;\n  /** Whether this compute type requires a VPC */\n  requiresVpc: boolean;\n}\n\nexport const COMPUTE_TYPE_CONFIG: Record<ComputeType, ComputeTypeConfig> = {\n  ecs: {\n    defaultPort: 80,\n    defaultScaling: { minCapacity: 2, maxCapacity: 5 },\n    supportsConnections: true,\n    requiresVpc: true\n  },\n  ec2: {\n    defaultPort: 22,\n    defaultScaling: { minCapacity: 2, maxCapacity: 3 },\n    supportsConnections: true,\n    requiresVpc: true\n  },\n  lambda: {\n    defaultPort: 0,\n    defaultScaling: { minCapacity: 0, maxCapacity: 100 },\n    supportsConnections: false,\n    requiresVpc: false\n  }\n};\n\n/**\n * Configuration for ECS capacity providers.\n */\nexport interface EcsCapacityProviderConfig {\n  /** Whether this uses Spot pricing */\n  usesSpot: boolean;\n  /** Whether this runs on EC2 instances (vs serverless Fargate) */\n  usesEc2Instances: boolean;\n}\n\nexport const ECS_CAPACITY_PROVIDER_CONFIG: Record<\n  EcsCapacityProvider,\n  EcsCapacityProviderConfig\n> = {\n  FARGATE: {\n    usesSpot: false,\n    usesEc2Instances: false\n  },\n  FARGATE_SPOT: {\n    usesSpot: true,\n    usesEc2Instances: false\n  },\n  EC2: {\n    usesSpot: false,\n    usesEc2Instances: true\n  }\n};\n\n/**\n * Default values for compute resource configuration.\n * Centralised constants to ensure consistency across the codebase.\n */\nexport const COMPUTE_DEFAULTS = {\n  EC2: {\n    INSTANCE_TYPE: \"t3.micro\",\n    MIN_CAPACITY: 1,\n    MAX_CAPACITY: 1\n  },\n  ECS: {\n    /** AWS sample image used when no ECR repository is provided */\n    FALLBACK_IMAGE: \"amazon/amazon-ecs-sample\",\n    /** Default tag for ECR images */\n    IMAGE_TAG: \"latest\"\n  },\n  LAMBDA: {\n    HANDLER: \"index.handler\",\n    RUNTIME: Runtime.NODEJS_22_X\n  }\n} as const;\n\nexport function getComputeTypeConfig(type: ComputeType): ComputeTypeConfig {\n  return COMPUTE_TYPE_CONFIG[type];\n}\n\nexport function getEcsCapacityProviderConfig(\n  provider: EcsCapacityProvider\n): EcsCapacityProviderConfig {\n  return ECS_CAPACITY_PROVIDER_CONFIG[provider];\n}\n\n// Re-export Lambda types from CDK for user convenience\nexport {\n  Architecture,\n  HttpMethod,\n  InvokeMode,\n  type FunctionUrlCorsOptions\n} from \"aws-cdk-lib/aws-lambda\";\n\n/**\n * Configuration for a container in an ECS task.\n *\n * For single-container services, `name` is optional and defaults to `${serviceName}Container`.\n * For multi-container tasks, the first container with a `port` is the **primary container**\n * that receives load balancer traffic.\n *\n * @example\n * // Single container (name auto-generated)\n * containers: [{ port: 3000 }]\n *\n * @example\n * // Multi-container with sidecars\n * containers: [\n *   { name: \"app\", port: 3000 },                    // Primary - receives ALB traffic\n *   { name: \"datadog\", image: \"datadog/agent\" }     // Sidecar - monitoring\n * ]\n */\nexport interface EcsContainerConfig {\n  /** Container name. Optional for single-container services. */\n  name?: string;\n  /**\n   * Container image. Options:\n   * - Omit: Uses app's default ECR repository (primary container only)\n   * - string: ECR repository name or public image URL\n   * - Repository: CDK ECR Repository construct\n   */\n  image?: string | Repository;\n  /**\n   * Port the container listens on.\n   * The first container with a port becomes the **primary container**\n   * and is registered with the load balancer.\n   */\n  port?: number;\n  /** Environment variables */\n  environment?: Record<string, string>;\n  /**\n   * Secrets from AWS SSM Parameter Store.\n   * Array of secret names that will be fetched from the service's SSM namespace.\n   * The namespace path is auto-determined from app/cluster/service names.\n   *\n   * @example\n   * // Secrets at /myapp/api-cluster/users/API_KEY and /myapp/api-cluster/users/DB_PASSWORD\n   * secrets: [\"API_KEY\", \"DB_PASSWORD\"]\n   */\n  secrets?: string[];\n  /** Secrets imported from other CDK resources (AWS Secrets Manager) */\n  secretsImport?: Record<string, SecretImport>;\n  /** Command to run in the container */\n  command?: string[];\n  /** Entry point for the container */\n  entryPoint?: string[];\n  /**\n   * Whether this container is essential.\n   * If an essential container stops, all containers in the task stop.\n   * Default: true\n   */\n  essential?: boolean;\n  /**\n   * Health check configuration.\n   * Default: For primary container with port, uses curl health check.\n   */\n  healthCheck?: {\n    command: string[];\n    interval?: number;\n    timeout?: number;\n    retries?: number;\n    startPeriod?: number;\n  };\n}\n\n/**\n * ECS scaling configuration.\n * - Omit: enabled with defaults\n * - `{}`: enabled with defaults\n * - `{ minCapacity: 2, maxCapacity: 10 }`: custom scaling\n * - `false`: explicitly disabled\n */\nexport interface EcsScalingConfig {\n  minCapacity?: number;\n  maxCapacity?: number;\n  scalingType?: ScalingType;\n}\n\n/**\n * Cluster-level configuration.\n * Controls the shared ALB for all services in this cluster.\n */\nexport interface EcsClusterConfig {\n  /**\n   * Domain for HTTPS access.\n   * - Omit: ALB created with default DNS (*.elb.amazonaws.com)\n   * - Specified: Creates ACM certificate + Route53 DNS A record\n   */\n  domain?: string;\n\n  /**\n   * Load balancer configuration.\n   * - Omit or \"public\": Internet-facing ALB (default)\n   * - \"internal\": VPC-only ALB\n   * - false: No ALB (for workers/background processors)\n   */\n  loadBalancer?: false | \"public\" | \"internal\";\n\n  /**\n   * Enable direct EC2 access without ALB.\n   * Uses host network mode for predictable ports.\n   * Access via EC2 public IP at container port.\n   */\n  directAccess?: boolean;\n\n  /**\n   * Advanced domain configuration for routing policies (latency, weighted, geo).\n   * Only used when domain is specified.\n   * Allows for multi-region deployments with advanced DNS routing.\n   */\n  domainConfig?: DomainConfig;\n}\n\n/**\n * Routing configuration for path/host-based routing on the ALB.\n * Required when cluster has multiple services with ports.\n * Optional for single service (gets all traffic automatically).\n */\nexport interface EcsRoutingConfig {\n  /**\n   * Path pattern for routing (e.g., \"/api/*\", \"/users/*\").\n   * Uses ALB path-based routing.\n   */\n  path?: string;\n\n  /**\n   * Host header for routing (e.g., \"api.example.com\").\n   * Uses ALB host-based routing.\n   */\n  host?: string;\n\n  /**\n   * Priority for this routing rule (1-50000).\n   * Lower number = higher priority.\n   * Auto-assigned if omitted.\n   */\n  priority?: number;\n\n  /**\n   * Health check path for this service's target group.\n   * Default: \"/\"\n   */\n  healthCheckPath?: string;\n}\n\n/**\n * Configuration for a service in an ECS cluster.\n * Each service gets its own task definition, scaling config, and target group.\n *\n * @example\n * // Simple service\n * { name: \"api\", containers: [{ port: 3000 }] }\n *\n * @example\n * // Service with routing (for multi-service clusters)\n * { name: \"users\", containers: [{ port: 3000 }], routing: { path: \"/users/*\", priority: 100 } }\n *\n * @example\n * // Service with multiple routing rules (same target group)\n * { name: \"web\", containers: [{ port: 3000 }], routing: [\n *   { path: \"/api/v2/*\", priority: 50 },\n *   { path: \"/*\", priority: 200 },\n * ]}\n *\n * @example\n * // Service with sidecars\n * {\n *   name: \"api\",\n *   containers: [\n *     { name: \"app\", port: 3000 },\n *     { name: \"datadog\", image: \"datadog/agent\" }\n *   ]\n * }\n */\nexport interface EcsServiceConfig {\n  /** Service name (unique within cluster) */\n  name: string;\n\n  /**\n   * Container image for this service (applies to first container without explicit image).\n   * - Omit: Uses app's default ECR repository\n   * - string: ECR repository name or public image URL\n   * - Repository: CDK ECR Repository construct\n   */\n  image?: string | Repository;\n\n  /**\n   * Container configuration(s) for this service.\n   * For single-container services, container name is optional and auto-generated.\n   * For multi-container services, the first container with a port is the primary container.\n   */\n  containers?: EcsContainerConfig[];\n\n  /**\n   * Routing rules for this service on the cluster's ALB.\n   * Required when cluster has multiple services with ports.\n   * Optional for single service (gets /* automatically).\n   * Can be a single rule or an array of rules pointing to the same target group.\n   *\n   * @example\n   * // Multiple routes for the same service\n   * routing: [\n   *   { path: \"/api/v2/*\", priority: 50 },\n   *   { path: \"/*\", priority: 200 },\n   * ]\n   */\n  routing?: EcsRoutingConfig | EcsRoutingConfig[];\n\n  /** CPU units for this service's tasks (256-4096) */\n  cpu?: number;\n\n  /** Memory in MiB for this service's tasks (512-30720) */\n  memoryLimitMiB?: number;\n\n  /** Desired number of tasks. Default: 2 */\n  desiredCount?: number;\n\n  /**\n   * Scaling configuration.\n   * - Omit: enabled with defaults\n   * - false: disabled\n   */\n  scaling?: EcsScalingConfig | false;\n\n  /**\n   * Path to Dockerfile for building this service's image.\n   * Metadata for CLI build process, not used during CDK synthesis.\n   */\n  dockerfilePath?: string;\n\n  /**\n   * Docker build target stage for multi-stage Dockerfiles.\n   * When specified, the CLI builds with `--target <dockerTarget>`.\n   * The image tag suffix is also updated: `<service>-<target>-latest`.\n   *\n   * @example\n   * // Dockerfile: FROM node AS base ... FROM base AS api ... FROM base AS worker\n   * { name: \"api\", dockerTarget: \"api\" }   // builds: myapp-api-api-latest\n   * { name: \"worker\", dockerTarget: \"worker\" }  // builds: myapp-worker-worker-latest\n   */\n  dockerTarget?: string;\n\n  /**\n   * Additional inline policies for this service's task role.\n   * Added on top of the default ECS Exec permissions.\n   * Use for service-specific AWS permissions (S3, DynamoDB, SQS, etc.).\n   */\n  taskRoleInlinePolicies?: Record<string, PolicyDocument>;\n\n  /**\n   * Additional managed policies for this service's task role.\n   * Added on top of the default ECS Exec permissions.\n   */\n  taskRoleManagedPolicies?: IManagedPolicy[];\n\n  /**\n   * Resources this service needs to connect to (e.g., databases, S3 buckets, SQS queues).\n   * Creates security group rules for IConnectable resources and IAM grants for IAM resources.\n   * Follows least-privilege - only this service gets access, not all services in the cluster.\n   *\n   * Supports:\n   * - IConnectable: Security group resources (RDS, ECS, etc.)\n   * - IStorageConnector: S3 buckets (IAM grants)\n   * - IDynamoDBConnector: DynamoDB tables (IAM grants)\n   * - IQueueConnector: SQS queues (IAM grants)\n   * - ConnectionConfig: Explicit access level configuration\n   *\n   * @example\n   * // Simple connections (default permissions)\n   * connections: [database, bucket, cache, queue]\n   *\n   * @example\n   * // Explicit access levels\n   * connections: [\n   *   database,                              // Security group (RDS)\n   *   { resource: cache, access: \"read\" },   // Read-only DynamoDB\n   *   { resource: bucket, access: \"write\" }, // Write-only S3\n   *   { resource: queue, access: \"consume\" } // Consume-only SQS\n   * ]\n   */\n  connections?: ConnectionSpec[];\n\n  /**\n   * Capacity provider for this service. REQUIRED.\n   * Each service specifies its own capacity provider.\n   *\n   * @example\n   * // Mixed FARGATE and EC2 services in same cluster\n   * {\n   *   services: [\n   *     { name: \"api\", capacityProvider: \"FARGATE\" },\n   *     { name: \"worker\", capacityProvider: \"EC2\", ec2Config: { instanceType: \"t3.micro\" } }\n   *   ]\n   * }\n   */\n  capacityProvider: EcsCapacityProvider;\n\n  /**\n   * EC2 capacity configuration for this service.\n   * Only used when service capacityProvider is \"EC2\".\n   * Services with matching ec2Config share an ASG for efficiency.\n   */\n  ec2Config?: Ec2CapacityConfig;\n\n  /**\n   * SSM Parameter Store path for secrets.\n   * If not specified, secrets are fetched from /<app>/<cluster>/<service>.\n   * Use this to override the default convention.\n   *\n   * @example\n   * // Override default path\n   * ssmSecretsPath: \"/custom/path/to/secrets\"\n   */\n  ssmSecretsPath?: string;\n}\n\n/**\n * SSH access configuration for EC2 instances.\n * - Omit: disabled (default)\n * - `{}`: enabled with auto-generated key\n * - `{ keyName: \"my-key\" }`: enabled with existing key\n * - `false`: explicitly disabled\n */\nexport interface SshConfig {\n  /** SSH key pair name */\n  keyName?: string;\n  /** Allowed CIDR blocks for SSH access */\n  allowedCidrs?: string[];\n}\n\n/**\n * Lambda function URL configuration.\n * - Omit: disabled (default)\n * - `{}`: enabled with IAM auth\n * - `{ authType: \"NONE\", cors: {...} }`: public with CORS\n * - `false`: explicitly disabled\n */\nexport interface FunctionUrlConfig {\n  /** Authentication type. Default: AWS_IAM */\n  authType?: FunctionUrlAuthType;\n  /** CORS configuration */\n  cors?: FunctionUrlCorsOptions;\n  /** Invoke mode. Use RESPONSE_STREAM for Lambda streaming. Default: BUFFERED */\n  invokeMode?: InvokeMode;\n}\n\ninterface BaseComputeProps {\n  vpc?: IVpc;\n}\n\n/**\n * ECS compute configuration.\n * Creates an ECS cluster with one or more services sharing a load balancer.\n *\n * @example\n * // Single service\n * app.addCompute(ComputeFactory.build(\"WebApp\", {\n *   type: \"ecs\",\n *   cluster: { domain: \"app.example.com\" },\n *   services: [{ name: \"web\", containers: [{ port: 3000 }] }]\n * }));\n *\n * @example\n * // Multi-service cluster with routing\n * app.addCompute(ComputeFactory.build(\"ApiCluster\", {\n *   type: \"ecs\",\n *   cluster: { domain: \"api.example.com\" },\n *   services: [\n *     { name: \"users\", containers: [{ port: 3000 }], routing: { path: \"/users/*\" } },\n *     { name: \"orders\", containers: [{ port: 3001 }], routing: { path: \"/orders/*\" } }\n *   ]\n * }));\n *\n * @example\n * // Internal workers (no ALB)\n * app.addCompute(ComputeFactory.build(\"Workers\", {\n *   type: \"ecs\",\n *   cluster: { loadBalancer: false },\n *   services: [{ name: \"processor\" }, { name: \"emailer\" }]\n * }));\n */\nexport interface EcsComputeProps extends BaseComputeProps {\n  type: \"ecs\";\n\n  /**\n   * Application name for SSM secrets namespace.\n   * When containers use secrets, the path is derived as: /<appName>/<clusterName>/<serviceName>\n   * Auto-derived from App.getName() if not specified.\n   */\n  appName?: string;\n\n  /**\n   * Cluster configuration.\n   * Controls the shared ALB for all services in this cluster.\n   * - Omit: ALB created with default settings\n   * - `{ domain: \"...\" }`: ALB with HTTPS + DNS\n   * - `{ loadBalancer: false }`: No ALB (internal workers)\n   */\n  cluster?: EcsClusterConfig;\n\n  /**\n   * Services in this cluster.\n   * Each service gets its own task definition, scaling, and target group.\n   * Each service MUST specify its own capacityProvider.\n   * All services share the cluster's ALB (unless disabled).\n   */\n  services: EcsServiceConfig[];\n\n  /**\n   * ECR repository for all services (default image).\n   * Individual services can override with their own `image` property.\n   */\n  ecrRepository?: Repository | RepositoryImage;\n\n  /**\n   * Path to Dockerfile for building custom image.\n   * Note: This is metadata for the CLI build process,\n   * not used during CDK synthesis.\n   */\n  dockerfilePath?: string;\n}\n\nexport interface Ec2ComputeProps extends BaseComputeProps {\n  type: \"ec2\";\n\n  /** EC2 instance type. Default: \"t3.micro\" */\n  instanceType?: string;\n\n  /**\n   * SSH access configuration.\n   * - Omit: disabled (default)\n   * - `{}`: enabled with defaults\n   * - `false`: explicitly disabled\n   */\n  ssh?: SshConfig | false;\n  /** User data script */\n  userData?: UserData;\n  /** Machine image (AMI) */\n  machineImage?: IMachineImage;\n  /** Minimum number of instances. Default: 1 */\n  minCapacity?: number;\n  /** Maximum number of instances. Default: 1 */\n  maxCapacity?: number;\n  /**\n   * Percentage of capacity to run on Spot instances (0-100).\n   * - Omit or 0: All On-Demand instances (default)\n   * - 100: All Spot instances\n   * - 50: Half Spot, half On-Demand\n   *\n   * Spot instances can reduce costs by up to 90% but may be interrupted.\n   * Use for fault-tolerant workloads.\n   */\n  spotCapacityPercentage?: number;\n}\n\n/**\n * Base Lambda configuration shared by both container and code deployments.\n */\ninterface BaseLambdaProps extends BaseComputeProps {\n  type: \"lambda\";\n  /** Timeout in seconds. Default: 3 */\n  timeout?: number;\n  /** Memory size in MB. Default: 128 */\n  memorySize?: number;\n  ephemeralStorageSize?: number;\n  /** CPU architecture. Default: x86_64. Use Architecture.ARM_64 for Graviton2. */\n  architecture?: Architecture;\n  /** Lambda function description */\n  description?: string;\n  /** IAM role description */\n  roleDescription?: string;\n\n  // Inline policies - routed to execution role for ECS, task role for Lambda\n  /** Inline IAM policy statements */\n  inlinePolicy?: PolicyStatement[];\n  /**\n   * Function URL configuration.\n   * - Omit: disabled (default)\n   * - `{}`: enabled with IAM auth\n   * - `{ authType: \"NONE\" }`: public access\n   * - `false`: explicitly disabled\n   */\n  functionUrl?: FunctionUrlConfig | false;\n  /** Environment variables */\n  environment?: Record<string, string>;\n\n  /**\n   * Secrets from AWS SSM Parameter Store.\n   * Array of secret names that will be fetched from the Lambda's SSM namespace.\n   * The namespace path is auto-determined as: /<appName>/lambda/<functionName>\n   *\n   * @example\n   * secrets: [\"API_KEY\", \"STRIPE_SECRET\"]\n   */\n  secrets?: string[];\n\n  /**\n   * SSM Parameter Store path for secrets.\n   * If secrets are defined, this path is used as the base path.\n   * If not specified, uses: /<appName>/lambda/<functionName>\n   *\n   * @example\n   * ssmSecretsPath: \"/myapp/custom/path\"\n   */\n  ssmSecretsPath?: string;\n\n  /**\n   * Secrets imported from other CDK resources (AWS Secrets Manager).\n   * Used for CDK-managed secrets like database credentials.\n   *\n   * @example\n   * secretsImport: {\n   *   DATABASE_USERNAME: database.getCredentials().getImport(\"username\"),\n   *   DATABASE_PASSWORD: database.getCredentials().getImport(\"password\")\n   * }\n   */\n  secretsImport?: Record<string, SecretImport>;\n\n  /**\n   * Application name for SSM secrets path derivation.\n   * Auto-derived from App instance when using ComputeFactory.\n   * Only specify for advanced use cases.\n   */\n  appName?: string;\n\n  /**\n   * Resources this Lambda needs to connect to (e.g., databases, S3 buckets, SQS queues).\n   * Creates security group rules for IConnectable resources and IAM grants for IAM resources.\n   *\n   * Supports:\n   * - IConnectable: Security group resources (RDS, ECS, etc.)\n   * - IStorageConnector: S3 buckets (IAM grants)\n   * - IDynamoDBConnector: DynamoDB tables (IAM grants)\n   * - IQueueConnector: SQS queues (IAM grants)\n   * - ConnectionConfig: Explicit access level configuration\n   *\n   * @example\n   * connections: [\n   *   database,\n   *   { resource: queue, access: \"send\" }\n   * ]\n   */\n  connections?: ConnectionSpec[];\n\n  /**\n   * EventBridge schedule expression for scheduled Lambda invocations.\n   * Uses cron or rate syntax: \"rate(1 hour)\" or \"cron(0 12 * * ? *)\".\n   */\n  scheduleExpression?: string;\n}\n\n/**\n * Container-based Lambda using ECR image.\n *\n * Uses Docker image from ECR repository. Handler and runtime are\n * automatically set to FROM_IMAGE.\n *\n * @example\n * app.addCompute(ComputeFactory.build(\"ImageLambda\", {\n *   type: \"lambda\",\n *   deployment: \"container\",\n *   ecrRepository: app.getDefaultContainerRegistry()\n * }));\n */\nexport interface ContainerLambdaProps extends BaseLambdaProps {\n  /** Container-based deployment using ECR image */\n  deployment: \"container\";\n  /** ECR repository containing the Lambda container image */\n  ecrRepository: Repository | RepositoryImage;\n}\n\n/**\n * Code-based Lambda using inline code or S3.\n *\n * Uses traditional Lambda deployment with code, handler, and runtime.\n *\n * @example\n * app.addCompute(ComputeFactory.build(\"CodeLambda\", {\n *   type: \"lambda\",\n *   deployment: \"code\",\n *   code: Code.fromAsset(\"./lambda\"),\n *   handler: \"index.handler\",\n *   runtime: Runtime.NODEJS_20_X\n * }));\n */\nexport interface CodeLambdaProps extends BaseLambdaProps {\n  /** Code-based deployment */\n  deployment: \"code\";\n  /** Lambda code (from asset, S3, or inline) */\n  code: Code;\n  /** Handler function. Default: \"index.handler\" */\n  handler?: string;\n  /** Lambda runtime. Default: NODEJS_22_X */\n  runtime?: Runtime;\n}\n\n/**\n * Lambda compute configuration.\n *\n * Discriminated union ensuring type-safe Lambda configuration:\n * - `deployment: \"container\"` requires `ecrRepository`\n * - `deployment: \"code\"` requires `code` and allows `handler`/`runtime`\n *\n * @example\n * // Container-based Lambda\n * { type: \"lambda\", deployment: \"container\", ecrRepository: ecr }\n *\n * @example\n * // Code-based Lambda\n * { type: \"lambda\", deployment: \"code\", code: Code.fromAsset(\"./lambda\") }\n */\nexport type LambdaComputeProps = ContainerLambdaProps | CodeLambdaProps;\n\n// Type-safe props with literal discriminators for factory overloads\nexport type IEcsComputeProps = EcsComputeProps;\nexport type ILambdaComputeProps = LambdaComputeProps;\nexport type IEc2ComputeProps = Ec2ComputeProps;\n\nexport type IComputeProps =\n  | IEcsComputeProps\n  | ILambdaComputeProps\n  | IEc2ComputeProps;\n\n/**\n * Validates compute props and logs warnings for ignored or misconfigured options.\n * These checks help catch issues when props come from dynamic sources where\n * TypeScript's compile-time checks may not apply.\n */\nfunction validateComputeProps(props: IComputeProps): void {\n  // ECS-specific options on non-ECS compute\n  warnIfPropertiesIgnored(\n    props,\n    [\"capacityProvider\", \"ec2Config\", \"services\", \"cluster\"],\n    \"ecs\",\n    \"ECS compute\"\n  );\n\n  // Lambda-specific options on non-Lambda compute\n  warnIfPropertiesIgnored(\n    props,\n    [\"functionUrl\", \"handler\", \"runtime\"],\n    \"lambda\",\n    \"Lambda compute\"\n  );\n\n  // EC2-specific options on non-EC2 compute\n  warnIfPropertiesIgnored(\n    props,\n    [\"ssh\", \"userData\", \"machineImage\"],\n    \"ec2\",\n    \"EC2 compute\"\n  );\n\n  // ECS-specific validation - type narrowing handled by discriminated union\n  if (props.type === \"ecs\") {\n    validateEcsProps(props);\n  }\n}\n\n/**\n * Validates ECS-specific props.\n * Extracted for clarity and detail parity with database/network patterns.\n */\nfunction validateEcsProps(props: EcsComputeProps): void {\n  // Validate services array exists and is not empty\n  if (!props.services || props.services.length === 0) {\n    throw new Error(\n      \"At least one service must be specified in 'services' array.\"\n    );\n  }\n\n  // Validate unique service names\n  const serviceNames = props.services.map((s) => s.name);\n  const duplicateNames = serviceNames.filter(\n    (name, index) => serviceNames.indexOf(name) !== index\n  );\n  if (duplicateNames.length > 0) {\n    throw new Error(\n      `Duplicate service names: ${[...new Set(duplicateNames)].join(\", \")}`\n    );\n  }\n\n  // Validate service name format\n  const invalidNames = props.services.filter(\n    (s) => !/^[a-zA-Z][a-zA-Z0-9-]*$/.test(s.name)\n  );\n  if (invalidNames.length > 0) {\n    throw new Error(\n      `Invalid service names: ${invalidNames.map((s) => s.name).join(\", \")}. ` +\n        \"Service names must start with a letter and contain only letters, numbers, and hyphens.\"\n    );\n  }\n\n  // Validate routing when multiple services have ports\n  const servicesWithPorts = props.services.filter((s) =>\n    s.containers?.some((c) => c.port)\n  );\n\n  if (servicesWithPorts.length > 1) {\n    const missingRouting = servicesWithPorts.filter((s) => {\n      const rules = Array.isArray(s.routing)\n        ? s.routing\n        : s.routing\n          ? [s.routing]\n          : [];\n      return !rules.some((r) => r.path || r.host);\n    });\n    if (missingRouting.length > 0) {\n      throw new Error(\n        `Services with ports require routing config when cluster has multiple services: ` +\n          `${missingRouting.map((s) => s.name).join(\", \")}. ` +\n          \"Add routing: { path: '/...' } or routing: { host: '...' } to each service.\"\n      );\n    }\n  }\n\n  // Validate each service's containers\n  for (const service of props.services) {\n    if (service.containers && service.containers.length > 0) {\n      // Check for duplicate container names within service (only named containers)\n      const containerNames = service.containers\n        .filter((c) => c.name)\n        .map((c) => c.name);\n      const duplicateContainerNames = containerNames.filter(\n        (name, index) => containerNames.indexOf(name) !== index\n      );\n      if (duplicateContainerNames.length > 0) {\n        throw new Error(\n          `Service '${service.name}': Duplicate container names: ` +\n            `${[...new Set(duplicateContainerNames)].join(\", \")}`\n        );\n      }\n    }\n\n    if (service.capacityProvider === \"EC2\" && !service.ec2Config) {\n      throw new Error(\n        `Service '${service.name}' uses EC2 capacity provider but no ec2Config is defined. ` +\n          \"Provide ec2Config on the service.\"\n      );\n    }\n\n    // Warn if service ec2Config is defined but capacityProvider is not EC2\n    if (service.ec2Config && service.capacityProvider !== \"EC2\") {\n      FjallLogger.warn(\n        `Service '${service.name}' has ec2Config but capacityProvider is not 'EC2'. ` +\n          \"The ec2Config will be ignored unless capacityProvider is set to 'EC2'.\"\n      );\n    }\n  }\n}\n\n/**\n * Build container configurations for an ECS service.\n * Converts user-facing EcsContainerConfig to internal EcsClusterProps format.\n * @internal Exported for testing only\n */\nexport function buildContainerConfigs(\n  service: EcsServiceConfig\n): EcsClusterProps[\"services\"][number][\"containers\"] {\n  if (service.containers && service.containers.length > 0) {\n    return service.containers.map((c, index) => ({\n      name: c.name || `${service.name}Container${index > 0 ? index : \"\"}`,\n      image: c.image,\n      port: c.port,\n      environment: c.environment,\n      secrets: c.secrets,\n      secretsImport: c.secretsImport,\n      command: c.command,\n      entryPoint: c.entryPoint,\n      essential: c.essential,\n      healthCheck: c.healthCheck\n    }));\n  }\n  // Default container (no port = worker)\n  return [{ name: `${service.name}Container` }];\n}\n\n/**\n * Resolved scaling configuration for an ECS service.\n * @internal Exported for testing only\n */\nexport interface ResolvedScalingConfig {\n  scalingType: ScalingType | undefined;\n  minCapacity: number | undefined;\n  maxCapacity: number | undefined;\n}\n\n/**\n * Resolve scaling configuration from service props.\n * Handles the three cases: explicit config, disabled (false), or default (undefined).\n * @internal Exported for testing only\n */\nexport function resolveScalingConfig(\n  scaling: EcsScalingConfig | false | undefined\n): ResolvedScalingConfig {\n  if (scaling === false) {\n    return {\n      scalingType: undefined,\n      minCapacity: undefined,\n      maxCapacity: undefined\n    };\n  }\n  if (scaling === undefined) {\n    return {\n      scalingType: ScalingType.CPU,\n      minCapacity: undefined,\n      maxCapacity: undefined\n    };\n  }\n  return {\n    scalingType: scaling.scalingType ?? ScalingType.CPU,\n    minCapacity: scaling.minCapacity,\n    maxCapacity: scaling.maxCapacity\n  };\n}\n\n/**\n * Resolved Lambda deployment configuration.\n */\nexport interface ResolvedLambdaDeployment {\n  code: Code;\n  handler: string;\n  runtime: Runtime;\n}\n\n/**\n * Resolve Lambda deployment configuration from props.\n * Handles container vs code deployment types.\n */\nexport function resolveLambdaDeployment(\n  props: ILambdaComputeProps\n): ResolvedLambdaDeployment {\n  if (props.deployment === \"container\") {\n    if (!(props.ecrRepository instanceof Repository)) {\n      throw new Error(\n        \"ecrRepository must be a Repository instance for Lambda compute type\"\n      );\n    }\n    return {\n      code: Code.fromEcrImage(props.ecrRepository, {\n        tagOrDigest: COMPUTE_DEFAULTS.ECS.IMAGE_TAG\n      }),\n      handler: Handler.FROM_IMAGE,\n      runtime: Runtime.FROM_IMAGE\n    };\n  }\n  return {\n    code: props.code,\n    handler: props.handler || COMPUTE_DEFAULTS.LAMBDA.HANDLER,\n    runtime: props.runtime || COMPUTE_DEFAULTS.LAMBDA.RUNTIME\n  };\n}\n\n/**\n * Factory for creating compute resources with type-safe return types.\n *\n * The factory uses overloads to infer the specific compute type from props,\n * ensuring TypeScript provides only the relevant methods for each compute type.\n *\n * @example\n * // ECS compute - returns EcsCompute with ECS-specific methods\n * const ecs = app.addCompute(ComputeFactory.build(\"WebApp\", {\n *   type: \"ecs\",\n *   cluster: { domain: \"app.example.com\" },\n *   services: [{ name: \"web\", containers: [{ port: 3000 }] }]\n * }));\n * ecs.getLoadBalancer(); // Available on EcsCompute\n *\n * @example\n * // Lambda compute - returns LambdaCompute with Lambda-specific methods\n * const lambda = app.addCompute(ComputeFactory.build(\"Fn\", {\n *   type: \"lambda\",\n *   deployment: \"code\",\n *   code: Code.fromAsset(\"./lambda\")\n * }));\n * lambda.getFunctionUrl(); // Available on LambdaCompute\n *\n * @example\n * // EC2 compute - returns Ec2Compute with EC2-specific methods\n * const ec2 = app.addCompute(ComputeFactory.build(\"Instance\", {\n *   type: \"ec2\",\n *   instanceType: \"t3.micro\"\n * }));\n * ec2.getAutoScalingGroup(); // Available on Ec2Compute\n */\nexport class ComputeFactory {\n  // Overload: ECS props returns EcsCompute\n  static build(\n    id: string,\n    props: IEcsComputeProps\n  ): (app: App, scope: Construct) => EcsCompute;\n  // Overload: Lambda props returns LambdaCompute\n  static build(\n    id: string,\n    props: ILambdaComputeProps\n  ): (app: App, scope: Construct) => LambdaCompute;\n  // Overload: EC2 props returns Ec2Compute\n  static build(\n    id: string,\n    props: IEc2ComputeProps\n  ): (app: App, scope: Construct) => Ec2Compute;\n  // Implementation signature\n  static build(\n    id: string,\n    props: IComputeProps\n  ): (app: App, scope: Construct) => AnyCompute {\n    return (app: App, scope: Construct): AnyCompute => {\n      validateComputeProps(props);\n\n      // Auto-derive appName from App instance for ECS compute\n      const typeConfig = COMPUTE_TYPE_CONFIG[props.type];\n      const needsVpc =\n        typeConfig.requiresVpc ||\n        (\"connections\" in props &&\n          Array.isArray(props.connections) &&\n          props.connections.length > 0);\n      let computeProps: IComputeProps = {\n        ...{\n          vpc: needsVpc ? app.getVpc() : undefined,\n          ecrRepository: app.getDefaultContainerRegistry()\n        },\n        ...props\n      };\n\n      if (computeProps.type === \"ecs\") {\n        computeProps = {\n          ...computeProps,\n          appName: computeProps.appName ?? app.getName()\n        };\n      }\n\n      // Create specific compute class based on type discriminator\n      switch (computeProps.type) {\n        case \"ecs\": {\n          // Register ECS services with manifest collector for CLI discovery\n          const collector = app.getManifestCollector();\n          const clusterName = id;\n          const appName = computeProps.appName ?? app.getName();\n\n          for (const service of computeProps.services) {\n            const manifestService: ManifestService = {\n              name: service.name,\n              clusterName\n            };\n\n            // Include dockerfilePath if specified (metadata for CLI)\n            if (service.dockerfilePath) {\n              manifestService.dockerfilePath = service.dockerfilePath;\n            }\n\n            // Include dockerTarget if specified (metadata for CLI)\n            if (service.dockerTarget) {\n              manifestService.dockerTarget = service.dockerTarget;\n            }\n\n            // Find container port from first container with a port\n            const containerWithPort = service.containers?.find(\n              (c) => c.port !== undefined\n            );\n            if (containerWithPort?.port !== undefined) {\n              manifestService.containerPort = containerWithPort.port;\n            }\n\n            // Aggregate secrets from all containers (deduplicated)\n            const allSecrets = new Set<string>();\n            for (const container of service.containers ?? []) {\n              if (container.secrets) {\n                for (const secret of container.secrets) {\n                  allSecrets.add(secret);\n                }\n              }\n            }\n            if (allSecrets.size > 0) {\n              manifestService.secrets = Array.from(allSecrets);\n            }\n\n            // Include ssmSecretsPath (explicit or derived)\n            manifestService.ssmSecretsPath =\n              service.ssmSecretsPath ??\n              `/${appName}/${clusterName}/${service.name}`;\n\n            collector.addService(manifestService);\n          }\n          return new EcsCompute(scope, id, computeProps);\n        }\n        case \"lambda\": {\n          // Add appName to props for Lambda (needed for secrets path derivation)\n          const lambdaComputeProps = {\n            ...computeProps,\n            appName: app.getName()\n          };\n\n          // Always register Lambda with manifest collector for CLI discovery\n          // This ensures all Lambdas are discoverable for secrets management,\n          // even those that don't currently have secrets configured\n          const collector = app.getManifestCollector();\n          const manifestLambda: ManifestLambda = {\n            name: id,\n            ssmSecretsPath:\n              lambdaComputeProps.ssmSecretsPath ||\n              `/${lambdaComputeProps.appName}/lambda/${id}`\n          };\n          if (\n            lambdaComputeProps.secrets &&\n            lambdaComputeProps.secrets.length > 0\n          ) {\n            manifestLambda.secrets = lambdaComputeProps.secrets;\n          }\n          collector.addLambda(manifestLambda);\n\n          return new LambdaCompute(scope, id, lambdaComputeProps);\n        }\n        case \"ec2\":\n          return new Ec2Compute(scope, id, computeProps);\n        default: {\n          const _exhaustive: never = computeProps;\n          throw new Error(\n            `Unsupported compute type: ${String((_exhaustive as IComputeProps).type)}`\n          );\n        }\n      }\n    };\n  }\n}\n\n/**\n * ECS compute wrapper implementing IEcsCompute.\n * Provides type-safe access to ECS-specific resources.\n */\nexport class EcsCompute extends Construct implements IEcsCompute {\n  public readonly computeType = \"ecs\" as const;\n  public readonly connections: Connections;\n\n  private readonly ecsCluster: EcsCluster;\n\n  constructor(scope: Construct, id: string, props: IEcsComputeProps) {\n    super(scope, id);\n\n    // Transform EcsServiceConfig[] to EcsServiceProps[] for EcsCluster\n    const services: EcsClusterProps[\"services\"] = props.services.map(\n      (service) => {\n        const containers = buildContainerConfigs(service);\n        const { scalingType, minCapacity, maxCapacity } = resolveScalingConfig(\n          service.scaling\n        );\n\n        return {\n          name: service.name,\n          image: service.image,\n          containers,\n          cpu: service.cpu,\n          memoryLimitMiB: service.memoryLimitMiB,\n          desiredCount: service.desiredCount,\n          scalingType,\n          minCapacity,\n          maxCapacity,\n          routing: service.routing,\n          taskRoleInlinePolicies: service.taskRoleInlinePolicies,\n          taskRoleManagedPolicies: service.taskRoleManagedPolicies,\n          connections: service.connections,\n          capacityProvider: service.capacityProvider,\n          ec2Config: service.ec2Config,\n          ssmSecretsPath: service.ssmSecretsPath,\n          dockerTarget: service.dockerTarget\n        };\n      }\n    );\n\n    // Build cluster config\n    const cluster: EcsClusterProps[\"cluster\"] = props.cluster\n      ? {\n          domain: props.cluster.domain,\n          loadBalancer: props.cluster.loadBalancer,\n          directAccess: props.cluster.directAccess,\n          domainConfig: props.cluster.domainConfig\n        }\n      : undefined;\n\n    const ecsProps: EcsClusterProps = {\n      clusterName: id,\n      appName: props.appName,\n      vpc: props.vpc,\n      ecrRepository: props.ecrRepository || COMPUTE_DEFAULTS.ECS.FALLBACK_IMAGE,\n      cluster,\n      services\n    };\n\n    this.ecsCluster = new EcsCluster(scope, `${id}Ecs`, ecsProps);\n    this.connections = this.ecsCluster.connections;\n  }\n\n  /** Get the ECS cluster. */\n  getCluster(): ICluster {\n    return this.ecsCluster.getCluster();\n  }\n\n  /** Get the Application Load Balancer if one was created. */\n  getLoadBalancer(): IApplicationLoadBalancer | undefined {\n    return this.ecsCluster.getLoadBalancer();\n  }\n\n  /** Get a specific service by name. */\n  getService(name: string): IBaseService | undefined {\n    return this.ecsCluster.getService(name);\n  }\n\n  /** Get all services in the cluster. */\n  getAllServices(): IBaseService[] {\n    const servicesMap = this.ecsCluster.getServices();\n    return Array.from(servicesMap.values());\n  }\n\n  /** Get the security group for the cluster. */\n  getSecurityGroup(): ISecurityGroup {\n    // Return the first security group from connections\n    const securityGroups = this.connections.securityGroups;\n    if (securityGroups.length === 0) {\n      throw new Error(\"No security groups found for ECS cluster\");\n    }\n    return securityGroups[0];\n  }\n\n  /**\n   * Get the ALB listener if this is an ECS compute with ALB.\n   */\n  getListener(): ApplicationListener | undefined {\n    return this.ecsCluster.getListener();\n  }\n\n  /**\n   * Get the underlying ECS cluster construct.\n   */\n  getEcsCluster(): EcsCluster {\n    return this.ecsCluster;\n  }\n\n  /**\n   * Grants ecs:ExecuteCommand permission for ECS services.\n   * Uses wildcard resource because ecs:ExecuteCommand targets task ARNs\n   * which are not known until runtime (tasks are ephemeral).\n   */\n  grantExecuteCommand(grantee: IGrantable): Grant {\n    return Grant.addToPrincipal({\n      grantee,\n      actions: [\"ecs:ExecuteCommand\"],\n      resourceArns: [\"*\"]\n    });\n  }\n}\n\n/**\n * Lambda compute wrapper implementing ILambdaCompute.\n * Provides type-safe access to Lambda-specific resources.\n */\nexport class LambdaCompute extends Construct implements ILambdaCompute {\n  public readonly computeType = \"lambda\" as const;\n  public readonly connections: Connections;\n\n  private readonly lambdaFunction: LambdaFunction;\n\n  constructor(scope: Construct, id: string, props: ILambdaComputeProps) {\n    super(scope, id);\n\n    const { code, handler, runtime } = resolveLambdaDeployment(props);\n\n    // Convert presence-based functionUrl config to legacy boolean format\n    const functionUrlConfig = props.functionUrl;\n    const enableFunctionUrl =\n      functionUrlConfig !== undefined && functionUrlConfig !== false;\n    const functionUrlAuthType =\n      enableFunctionUrl && typeof functionUrlConfig === \"object\"\n        ? functionUrlConfig.authType\n        : undefined;\n    const functionUrlCors =\n      enableFunctionUrl && typeof functionUrlConfig === \"object\"\n        ? functionUrlConfig.cors\n        : undefined;\n    const functionUrlInvokeMode =\n      enableFunctionUrl && typeof functionUrlConfig === \"object\"\n        ? functionUrlConfig.invokeMode\n        : undefined;\n\n    const lambdaProps = {\n      code,\n      vpc: props.vpc,\n      handler,\n      runtime,\n      architecture: props.architecture,\n      timeout: props.timeout,\n      memorySize: props.memorySize,\n      lambdaDescription: props.description,\n      roleDescription: props.roleDescription,\n      inlinePolicy: props.inlinePolicy || [],\n      enableFunctionUrl,\n      functionUrlAuthType,\n      functionUrlCors,\n      functionUrlInvokeMode,\n      scheduleExpression: props.scheduleExpression,\n      ephemeralStorageSize: props.ephemeralStorageSize,\n      secrets: props.secrets,\n      ssmSecretsPath: props.ssmSecretsPath,\n      secretsImport: props.secretsImport,\n      environment: props.environment,\n      appName: props.appName,\n      functionName: id\n    };\n\n    this.lambdaFunction = new LambdaFunction(scope, `${id}Lambda`, lambdaProps);\n\n    // CDK's Function.connections getter throws for non-VPC Lambdas, so only\n    // access it when a VPC is configured. Non-VPC Lambdas get an empty\n    // Connections object to satisfy the IConnectable interface.\n    this.connections = props.vpc\n      ? this.lambdaFunction.connections\n      : new Connections();\n\n    // Process connections using the unified connector interface\n    if (props.connections && props.connections.length > 0) {\n      if (!props.vpc) {\n        throw new Error(\n          `Lambda '${id}' has connections configured but no VPC. ` +\n            \"A VPC is required for security group connections.\"\n        );\n      }\n      try {\n        processConnections(\n          props.connections,\n          this.lambdaFunction, // IGrantable (execution role)\n          this.lambdaFunction // IConnectable (security group)\n        );\n      } catch (error) {\n        throw new Error(\n          `Failed to process connections for Lambda '${id}': ${\n            error instanceof Error ? error.message : String(error)\n          }`\n        );\n      }\n    }\n  }\n\n  /**\n   * Get a Lambda function by name.\n   * Since we only have one function, name is ignored.\n   */\n  getFunction(_name?: string): IFunction | undefined {\n    return this.lambdaFunction;\n  }\n\n  /** Get all Lambda functions. */\n  getAllFunctions(): IFunction[] {\n    return [this.lambdaFunction];\n  }\n\n  /**\n   * Get the function URL for a Lambda function.\n   */\n  getFunctionUrl(_name?: string): string | undefined {\n    return this.lambdaFunction.getFunctionUrl();\n  }\n\n  /**\n   * Grant invoke permissions to a grantee.\n   */\n  grantInvoke(grantee: IGrantable, _functionName?: string): Grant {\n    return this.lambdaFunction.grantInvoke(grantee);\n  }\n\n  /**\n   * Get the security group for VPC-enabled Lambda functions.\n   * Returns undefined if the Lambda is not VPC-enabled.\n   */\n  getSecurityGroup(): ISecurityGroup | undefined {\n    const securityGroups = this.connections.securityGroups;\n    return securityGroups.length > 0 ? securityGroups[0] : undefined;\n  }\n\n  /**\n   * Get the underlying Lambda function construct.\n   */\n  getLambdaFunction(): LambdaFunction {\n    return this.lambdaFunction;\n  }\n}\n\n/**\n * EC2 compute wrapper implementing IEc2Compute.\n * Provides type-safe access to EC2-specific resources.\n */\nexport class Ec2Compute extends Construct implements IEc2Compute {\n  public readonly computeType = \"ec2\" as const;\n  public readonly connections: Connections;\n\n  private readonly ec2Instance: Ec2Instance;\n\n  constructor(scope: Construct, id: string, props: IEc2ComputeProps) {\n    super(scope, id);\n\n    const enableSSH = props.ssh !== undefined && props.ssh !== false;\n\n    const ec2Props = {\n      serviceName: `${id}Service`,\n      instanceType: props.instanceType || COMPUTE_DEFAULTS.EC2.INSTANCE_TYPE,\n      enableSSH,\n      userData: props.userData,\n      machineImage: props.machineImage,\n      minCapacity: props.minCapacity || COMPUTE_DEFAULTS.EC2.MIN_CAPACITY,\n      maxCapacity: props.maxCapacity || COMPUTE_DEFAULTS.EC2.MAX_CAPACITY,\n      vpc: props.vpc,\n      spotCapacityPercentage: props.spotCapacityPercentage\n    };\n\n    this.ec2Instance = new Ec2Instance(scope, `${id}Ec2`, ec2Props);\n    this.connections = this.ec2Instance.connections;\n  }\n\n  /** Get the Auto Scaling Group. */\n  getAutoScalingGroup(): IAutoScalingGroup {\n    return this.ec2Instance.getAutoScalingGroup();\n  }\n\n  /** Get the security group. */\n  getSecurityGroup(): ISecurityGroup {\n    return this.ec2Instance.asgSecurityGroup;\n  }\n\n  /**\n   * Get the underlying EC2 instance construct.\n   */\n  getEc2Instance(): Ec2Instance {\n    return this.ec2Instance;\n  }\n}\n\n// Re-export type guards for convenience\nexport { isCompute, isEcsCompute, isLambdaCompute, isEc2Compute };\nexport type { IEcsCompute, ILambdaCompute, IEc2Compute, AnyCompute };\n"]}
222
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"compute.js","sourceRoot":"","sources":["../../../../lib/patterns/aws/compute.ts"],"names":[],"mappings":"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","sourcesContent":["import { Runtime } from \"aws-cdk-lib/aws-lambda\";\n\nimport { type Construct } from \"constructs\";\n\nimport {\n  type IEcsCompute,\n  type ILambdaCompute,\n  type IEc2Compute,\n  type AnyCompute,\n  isCompute,\n  isEcsCompute,\n  isLambdaCompute,\n  isEc2Compute\n} from \"./interfaces/compute.js\";\nimport { warnIfPropertiesIgnored } from \"../../utils/validationLogger.js\";\nimport {\n  type ManifestService,\n  type ManifestLambda\n} from \"../../utils/manifestWriter.js\";\n\nimport type App from \"../../app\";\n\n// Import and re-export from per-pattern files\nimport {\n  EcsCompute,\n  type EcsComputeProps,\n  type EcsServiceConfig,\n  type EcsContainerConfig,\n  type EcsScalingConfig,\n  type EcsClusterConfig,\n  type EcsRoutingConfig,\n  type EcsCapacityProviderConfig,\n  ECS_CAPACITY_PROVIDER_CONFIG,\n  getEcsCapacityProviderConfig,\n  ScalingType,\n  type EcsCapacityProvider,\n  type Ec2CapacityConfig,\n  validateEcsProps,\n  buildContainerConfigs,\n  type ResolvedScalingConfig,\n  resolveScalingConfig\n} from \"./computeEcs.js\";\n\nimport {\n  LambdaCompute,\n  type LambdaComputeProps,\n  type ContainerLambdaProps,\n  type CodeLambdaProps,\n  type FunctionUrlConfig,\n  type ResolvedLambdaDeployment,\n  resolveLambdaDeployment,\n  Architecture,\n  HttpMethod,\n  InvokeMode,\n  type FunctionUrlCorsOptions\n} from \"./computeLambda.js\";\n\nimport {\n  Ec2Compute,\n  type Ec2ComputeProps,\n  type SshConfig\n} from \"./computeEc2.js\";\n\n// Re-export everything from per-pattern files\nexport {\n  // ECS\n  EcsCompute,\n  type EcsComputeProps,\n  type EcsServiceConfig,\n  type EcsContainerConfig,\n  type EcsScalingConfig,\n  type EcsClusterConfig,\n  type EcsRoutingConfig,\n  type EcsCapacityProviderConfig,\n  ECS_CAPACITY_PROVIDER_CONFIG,\n  getEcsCapacityProviderConfig,\n  ScalingType,\n  type EcsCapacityProvider,\n  type Ec2CapacityConfig,\n  validateEcsProps,\n  buildContainerConfigs,\n  type ResolvedScalingConfig,\n  resolveScalingConfig,\n  // Lambda\n  LambdaCompute,\n  type LambdaComputeProps,\n  type ContainerLambdaProps,\n  type CodeLambdaProps,\n  type FunctionUrlConfig,\n  type ResolvedLambdaDeployment,\n  resolveLambdaDeployment,\n  Architecture,\n  HttpMethod,\n  InvokeMode,\n  type FunctionUrlCorsOptions,\n  // EC2\n  Ec2Compute,\n  type Ec2ComputeProps,\n  type SshConfig\n};\n\nexport type ComputeType = \"ecs\" | \"ec2\" | \"lambda\";\n\n/**\n * Configuration defaults for each compute type.\n */\nexport interface ComputeTypeConfig {\n  /** Default port for the compute type. ECS: 80, EC2: 22, Lambda: 0 (no port) */\n  defaultPort: number;\n  /** Default scaling limits */\n  defaultScaling: {\n    minCapacity: number;\n    maxCapacity: number;\n  };\n  /** Whether this compute type requires a VPC */\n  requiresVpc: boolean;\n}\n\nexport const COMPUTE_TYPE_CONFIG: Record<ComputeType, ComputeTypeConfig> = {\n  ecs: {\n    defaultPort: 80,\n    defaultScaling: { minCapacity: 2, maxCapacity: 5 },\n    requiresVpc: true\n  },\n  ec2: {\n    defaultPort: 22,\n    defaultScaling: { minCapacity: 2, maxCapacity: 3 },\n    requiresVpc: true\n  },\n  lambda: {\n    defaultPort: 0,\n    defaultScaling: { minCapacity: 0, maxCapacity: 100 },\n    requiresVpc: false\n  }\n};\n\n/**\n * Default values for compute resource configuration.\n * Centralised constants to ensure consistency across the codebase.\n */\nexport const COMPUTE_DEFAULTS = {\n  EC2: {\n    INSTANCE_TYPE: \"t4g.micro\",\n    MIN_CAPACITY: 1,\n    MAX_CAPACITY: 1\n  },\n  ECS: {\n    /** AWS sample image used when no ECR repository is provided */\n    FALLBACK_IMAGE: \"amazon/amazon-ecs-sample\",\n    /** Default tag for ECR images */\n    IMAGE_TAG: \"latest\"\n  },\n  LAMBDA: {\n    HANDLER: \"index.handler\",\n    RUNTIME: Runtime.NODEJS_22_X,\n    ARCHITECTURE: Architecture.ARM_64\n  }\n} as const;\n\nexport function getComputeTypeConfig(type: ComputeType): ComputeTypeConfig {\n  return COMPUTE_TYPE_CONFIG[type];\n}\n\n// Type-safe props with literal discriminators for factory overloads\nexport type IEcsComputeProps = EcsComputeProps;\nexport type ILambdaComputeProps = LambdaComputeProps;\nexport type IEc2ComputeProps = Ec2ComputeProps;\n\nexport type IComputeProps =\n  | IEcsComputeProps\n  | ILambdaComputeProps\n  | IEc2ComputeProps;\n\n/**\n * Validates compute props and logs warnings for ignored or misconfigured options.\n * These checks help catch issues when props come from dynamic sources where\n * TypeScript's compile-time checks may not apply.\n */\nfunction validateComputeProps(props: IComputeProps): void {\n  // ECS-specific options on non-ECS compute\n  warnIfPropertiesIgnored(\n    props,\n    [\"capacityProvider\", \"ec2Config\", \"services\", \"cluster\"],\n    \"ecs\",\n    \"ECS compute\"\n  );\n\n  // Lambda-specific options on non-Lambda compute\n  warnIfPropertiesIgnored(\n    props,\n    [\"functionUrl\", \"handler\", \"runtime\"],\n    \"lambda\",\n    \"Lambda compute\"\n  );\n\n  // EC2-specific options on non-EC2 compute\n  warnIfPropertiesIgnored(\n    props,\n    [\"ssh\", \"userData\", \"machineImage\"],\n    \"ec2\",\n    \"EC2 compute\"\n  );\n\n  // ECS-specific validation - type narrowing handled by discriminated union\n  if (props.type === \"ecs\") {\n    validateEcsProps(props);\n  }\n}\n\n/**\n * Factory for creating compute resources with type-safe return types.\n *\n * The factory uses overloads to infer the specific compute type from props,\n * ensuring TypeScript provides only the relevant methods for each compute type.\n *\n * @example\n * // ECS compute - returns EcsCompute with ECS-specific methods\n * const ecs = app.addCompute(ComputeFactory.build(\"WebApp\", {\n *   type: \"ecs\",\n *   cluster: { domain: \"app.example.com\" },\n *   services: [{ name: \"web\", containers: [{ port: 3000 }] }]\n * }));\n * ecs.getLoadBalancer(); // Available on EcsCompute\n *\n * @example\n * // Lambda compute - returns LambdaCompute with Lambda-specific methods\n * const lambda = app.addCompute(ComputeFactory.build(\"Fn\", {\n *   type: \"lambda\",\n *   deployment: \"code\",\n *   code: Code.fromAsset(\"./lambda\")\n * }));\n * lambda.getFunctionUrl(); // Available on LambdaCompute\n *\n * @example\n * // EC2 compute - returns Ec2Compute with EC2-specific methods\n * const ec2 = app.addCompute(ComputeFactory.build(\"Instance\", {\n *   type: \"ec2\",\n *   instanceType: \"t4g.micro\"\n * }));\n * ec2.getAutoScalingGroup(); // Available on Ec2Compute\n */\nexport class ComputeFactory {\n  // Overload: ECS props returns EcsCompute\n  static build(\n    id: string,\n    props: IEcsComputeProps\n  ): (app: App, scope: Construct) => EcsCompute;\n  // Overload: Lambda props returns LambdaCompute\n  static build(\n    id: string,\n    props: ILambdaComputeProps\n  ): (app: App, scope: Construct) => LambdaCompute;\n  // Overload: EC2 props returns Ec2Compute\n  static build(\n    id: string,\n    props: IEc2ComputeProps\n  ): (app: App, scope: Construct) => Ec2Compute;\n  // Implementation signature\n  static build(\n    id: string,\n    props: IComputeProps\n  ): (app: App, scope: Construct) => AnyCompute {\n    return (app: App, scope: Construct): AnyCompute => {\n      validateComputeProps(props);\n\n      // Auto-derive appName from App instance for ECS compute\n      const typeConfig = COMPUTE_TYPE_CONFIG[props.type];\n      const needsVpc =\n        typeConfig.requiresVpc ||\n        (\"connections\" in props &&\n          Array.isArray(props.connections) &&\n          props.connections.length > 0);\n      let computeProps: IComputeProps = {\n        ...{\n          vpc: needsVpc ? app.getVpc() : undefined,\n          ecrRepository: app.getDefaultContainerRegistry()\n        },\n        ...props\n      };\n\n      if (computeProps.type === \"ecs\") {\n        computeProps = {\n          ...computeProps,\n          appName: computeProps.appName ?? app.getName()\n        };\n      }\n\n      // Create specific compute class based on type discriminator\n      switch (computeProps.type) {\n        case \"ecs\": {\n          // Register ECS services with manifest collector for CLI discovery\n          const collector = app.getManifestCollector();\n          const clusterName = id;\n          const appName = computeProps.appName ?? app.getName();\n\n          for (const service of computeProps.services) {\n            const manifestService: ManifestService = {\n              name: service.name,\n              clusterName\n            };\n\n            // Include dockerfilePath if specified (metadata for CLI)\n            if (service.dockerfilePath) {\n              manifestService.dockerfilePath = service.dockerfilePath;\n            }\n\n            // Include dockerTarget if specified (metadata for CLI)\n            if (service.dockerTarget) {\n              manifestService.dockerTarget = service.dockerTarget;\n            }\n\n            // Find container port from first container with a port\n            const containerWithPort = service.containers?.find(\n              (c) => c.port !== undefined\n            );\n            if (containerWithPort?.port !== undefined) {\n              manifestService.containerPort = containerWithPort.port;\n            }\n\n            // Aggregate secrets from all containers (deduplicated)\n            const allSecrets = new Set<string>();\n            for (const container of service.containers ?? []) {\n              if (container.secrets) {\n                for (const secret of container.secrets) {\n                  allSecrets.add(secret);\n                }\n              }\n            }\n            if (allSecrets.size > 0) {\n              manifestService.secrets = Array.from(allSecrets);\n            }\n\n            // Include ssmSecretsPath (explicit or derived)\n            manifestService.ssmSecretsPath =\n              service.ssmSecretsPath ??\n              `/${appName}/${clusterName}/${service.name}`;\n\n            collector.addService(manifestService);\n          }\n          return new EcsCompute(scope, id, computeProps);\n        }\n        case \"lambda\": {\n          // Add appName to props for Lambda (needed for secrets path derivation)\n          const lambdaComputeProps = {\n            ...computeProps,\n            appName: app.getName()\n          };\n\n          // Always register Lambda with manifest collector for CLI discovery\n          // This ensures all Lambdas are discoverable for secrets management,\n          // even those that don't currently have secrets configured\n          const collector = app.getManifestCollector();\n          const manifestLambda: ManifestLambda = {\n            name: id,\n            ssmSecretsPath:\n              lambdaComputeProps.ssmSecretsPath ||\n              `/${lambdaComputeProps.appName}/lambda/${id}`\n          };\n          if (\n            lambdaComputeProps.secrets &&\n            lambdaComputeProps.secrets.length > 0\n          ) {\n            manifestLambda.secrets = lambdaComputeProps.secrets;\n          }\n          collector.addLambda(manifestLambda);\n\n          return new LambdaCompute(scope, id, lambdaComputeProps);\n        }\n        case \"ec2\":\n          return new Ec2Compute(scope, id, computeProps);\n        default: {\n          const _exhaustive: never = computeProps;\n          throw new Error(\n            `Unsupported compute type: ${String((_exhaustive as IComputeProps).type)}`\n          );\n        }\n      }\n    };\n  }\n}\n\n// Re-export type guards for convenience\nexport { isCompute, isEcsCompute, isLambdaCompute, isEc2Compute };\nexport type { IEcsCompute, ILambdaCompute, IEc2Compute, AnyCompute };\n"]}