truemark-cdk-lib 1.21.0 → 1.21.1-alpha.1768569999

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.
@@ -1,6 +1,7 @@
1
1
  export * from './ecs-service-update';
2
2
  export * from './log-configuration';
3
3
  export * from './otel-configuration';
4
+ export * from './priority-allocator';
4
5
  export * from './standard-fargate-cluster';
5
6
  export * from './standard-fargate-service';
6
7
  export * from './standard-application-fargate-service';
@@ -17,8 +17,9 @@ Object.defineProperty(exports, "__esModule", { value: true });
17
17
  __exportStar(require("./ecs-service-update"), exports);
18
18
  __exportStar(require("./log-configuration"), exports);
19
19
  __exportStar(require("./otel-configuration"), exports);
20
+ __exportStar(require("./priority-allocator"), exports);
20
21
  __exportStar(require("./standard-fargate-cluster"), exports);
21
22
  __exportStar(require("./standard-fargate-service"), exports);
22
23
  __exportStar(require("./standard-application-fargate-service"), exports);
23
24
  __exportStar(require("./standard-network-fargate-service"), exports);
24
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsdURBQXFDO0FBQ3JDLHNEQUFvQztBQUNwQyx1REFBcUM7QUFDckMsNkRBQTJDO0FBQzNDLDZEQUEyQztBQUMzQyx5RUFBdUQ7QUFDdkQscUVBQW1EIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9lY3Mtc2VydmljZS11cGRhdGUnO1xuZXhwb3J0ICogZnJvbSAnLi9sb2ctY29uZmlndXJhdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL290ZWwtY29uZmlndXJhdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL3N0YW5kYXJkLWZhcmdhdGUtY2x1c3Rlcic7XG5leHBvcnQgKiBmcm9tICcuL3N0YW5kYXJkLWZhcmdhdGUtc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3N0YW5kYXJkLWFwcGxpY2F0aW9uLWZhcmdhdGUtc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3N0YW5kYXJkLW5ldHdvcmstZmFyZ2F0ZS1zZXJ2aWNlJztcbiJdfQ==
25
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsdURBQXFDO0FBQ3JDLHNEQUFvQztBQUNwQyx1REFBcUM7QUFDckMsdURBQXFDO0FBQ3JDLDZEQUEyQztBQUMzQyw2REFBMkM7QUFDM0MseUVBQXVEO0FBQ3ZELHFFQUFtRCIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vZWNzLXNlcnZpY2UtdXBkYXRlJztcbmV4cG9ydCAqIGZyb20gJy4vbG9nLWNvbmZpZ3VyYXRpb24nO1xuZXhwb3J0ICogZnJvbSAnLi9vdGVsLWNvbmZpZ3VyYXRpb24nO1xuZXhwb3J0ICogZnJvbSAnLi9wcmlvcml0eS1hbGxvY2F0b3InO1xuZXhwb3J0ICogZnJvbSAnLi9zdGFuZGFyZC1mYXJnYXRlLWNsdXN0ZXInO1xuZXhwb3J0ICogZnJvbSAnLi9zdGFuZGFyZC1mYXJnYXRlLXNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9zdGFuZGFyZC1hcHBsaWNhdGlvbi1mYXJnYXRlLXNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9zdGFuZGFyZC1uZXR3b3JrLWZhcmdhdGUtc2VydmljZSc7XG4iXX0=
@@ -0,0 +1,38 @@
1
+ interface CustomResourceEvent {
2
+ readonly RequestType: 'Create' | 'Update' | 'Delete';
3
+ readonly ResponseURL: string;
4
+ readonly StackId: string;
5
+ readonly RequestId: string;
6
+ readonly ResourceType: string;
7
+ readonly LogicalResourceId: string;
8
+ readonly PhysicalResourceId?: string;
9
+ readonly ResourceProperties: {
10
+ readonly ListenerArn: string;
11
+ readonly ServiceIdentifier: string;
12
+ readonly TableName: string;
13
+ readonly PreferredPriority?: string;
14
+ };
15
+ readonly OldResourceProperties?: {
16
+ readonly ListenerArn: string;
17
+ readonly ServiceIdentifier: string;
18
+ readonly TableName: string;
19
+ readonly PreferredPriority?: string;
20
+ };
21
+ }
22
+ interface CustomResourceResponse {
23
+ Status: 'SUCCESS' | 'FAILED';
24
+ Reason?: string;
25
+ PhysicalResourceId: string;
26
+ StackId: string;
27
+ RequestId: string;
28
+ LogicalResourceId: string;
29
+ NoEcho?: boolean;
30
+ Data?: {
31
+ Priority: string;
32
+ };
33
+ }
34
+ /**
35
+ * Main Lambda handler for the Custom Resource
36
+ */
37
+ export declare function handler(event: CustomResourceEvent): Promise<CustomResourceResponse>;
38
+ export {};
@@ -0,0 +1,347 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.handler = handler;
4
+ const client_elastic_load_balancing_v2_1 = require("@aws-sdk/client-elastic-load-balancing-v2");
5
+ const client_dynamodb_1 = require("@aws-sdk/client-dynamodb");
6
+ const crypto = require("node:crypto");
7
+ const elbv2Client = new client_elastic_load_balancing_v2_1.ElasticLoadBalancingV2Client({});
8
+ const dynamoClient = new client_dynamodb_1.DynamoDBClient({});
9
+ const MAX_PRIORITY = 50000;
10
+ const MAX_RETRIES = 10;
11
+ /**
12
+ * Creates a deterministic hash of the service identifier
13
+ */
14
+ function hashServiceIdentifier(serviceIdentifier) {
15
+ return crypto.createHash('sha256').update(serviceIdentifier).digest('hex');
16
+ }
17
+ /**
18
+ * Creates a success response
19
+ */
20
+ function success(event, physicalResourceId, priority) {
21
+ return {
22
+ Status: 'SUCCESS',
23
+ PhysicalResourceId: physicalResourceId,
24
+ StackId: event.StackId,
25
+ RequestId: event.RequestId,
26
+ LogicalResourceId: event.LogicalResourceId,
27
+ NoEcho: false,
28
+ Data: {
29
+ Priority: priority.toString(),
30
+ },
31
+ };
32
+ }
33
+ /**
34
+ * Creates a failure response
35
+ */
36
+ function fail(event, physicalResourceId, reason) {
37
+ return {
38
+ Status: 'FAILED',
39
+ Reason: reason,
40
+ PhysicalResourceId: physicalResourceId,
41
+ StackId: event.StackId,
42
+ RequestId: event.RequestId,
43
+ LogicalResourceId: event.LogicalResourceId,
44
+ NoEcho: false,
45
+ };
46
+ }
47
+ /**
48
+ * Extracts valid priority from a rule
49
+ */
50
+ function extractPriorityFromRule(rule) {
51
+ if (!rule.Priority || rule.Priority === 'default') {
52
+ return null;
53
+ }
54
+ const priority = Number.parseInt(rule.Priority, 10);
55
+ return Number.isNaN(priority) ? null : priority;
56
+ }
57
+ /**
58
+ * Formats priority list for logging
59
+ */
60
+ function formatPrioritiesForLog(priorities) {
61
+ const sorted = Array.from(priorities).sort((a, b) => a - b);
62
+ const preview = sorted.slice(0, 10).join(', ');
63
+ return priorities.size > 10 ? `${preview}...` : preview;
64
+ }
65
+ /**
66
+ * Gets all priorities currently in use on the ALB listener
67
+ */
68
+ async function getAlbListenerPriorities(listenerArn) {
69
+ const priorities = new Set();
70
+ try {
71
+ let nextMarker = undefined;
72
+ do {
73
+ const command = new client_elastic_load_balancing_v2_1.DescribeRulesCommand({
74
+ ListenerArn: listenerArn,
75
+ Marker: nextMarker,
76
+ });
77
+ const response = await elbv2Client.send(command);
78
+ if (response.Rules) {
79
+ for (const rule of response.Rules) {
80
+ const priority = extractPriorityFromRule(rule);
81
+ if (priority !== null) {
82
+ priorities.add(priority);
83
+ }
84
+ }
85
+ }
86
+ nextMarker = response.NextMarker;
87
+ } while (nextMarker);
88
+ const formatted = formatPrioritiesForLog(priorities);
89
+ console.log(`Found ${priorities.size} priorities in use on ALB listener: ${formatted}`);
90
+ }
91
+ catch (error) {
92
+ console.error('Error fetching ALB listener priorities:', error);
93
+ throw error;
94
+ }
95
+ return priorities;
96
+ }
97
+ /**
98
+ * Extracts valid priority from a DynamoDB item
99
+ */
100
+ function extractPriorityFromDynamoItem(item) {
101
+ var _a;
102
+ if (!((_a = item.Priority) === null || _a === void 0 ? void 0 : _a.N)) {
103
+ return null;
104
+ }
105
+ const priority = Number.parseInt(item.Priority.N, 10);
106
+ return Number.isNaN(priority) ? null : priority;
107
+ }
108
+ /**
109
+ * Gets all priorities tracked in DynamoDB for this listener
110
+ */
111
+ async function getDynamoDbPriorities(tableName, listenerArn) {
112
+ const priorities = new Set();
113
+ try {
114
+ let lastEvaluatedKey = undefined;
115
+ do {
116
+ const command = new client_dynamodb_1.QueryCommand({
117
+ TableName: tableName,
118
+ KeyConditionExpression: 'ListenerArn = :arn',
119
+ ExpressionAttributeValues: {
120
+ ':arn': { S: listenerArn },
121
+ },
122
+ ExclusiveStartKey: lastEvaluatedKey,
123
+ });
124
+ const response = await dynamoClient.send(command);
125
+ if (response.Items) {
126
+ for (const item of response.Items) {
127
+ const priority = extractPriorityFromDynamoItem(item);
128
+ if (priority !== null) {
129
+ priorities.add(priority);
130
+ }
131
+ }
132
+ }
133
+ lastEvaluatedKey = response.LastEvaluatedKey;
134
+ } while (lastEvaluatedKey);
135
+ console.log(`Found ${priorities.size} priorities tracked in DynamoDB for listener`);
136
+ }
137
+ catch (error) {
138
+ console.error('Error fetching DynamoDB priorities:', error);
139
+ throw error;
140
+ }
141
+ return priorities;
142
+ }
143
+ /**
144
+ * Checks if the service already has an allocated priority (for idempotency)
145
+ */
146
+ async function getExistingAllocation(tableName, listenerArn, serviceIdentifier) {
147
+ var _a, _b;
148
+ try {
149
+ const command = new client_dynamodb_1.QueryCommand({
150
+ TableName: tableName,
151
+ IndexName: 'ServiceIdentifierIndex',
152
+ KeyConditionExpression: 'ServiceIdentifier = :sid AND ListenerArn = :arn',
153
+ ExpressionAttributeValues: {
154
+ ':sid': { S: serviceIdentifier },
155
+ ':arn': { S: listenerArn },
156
+ },
157
+ });
158
+ const response = await dynamoClient.send(command);
159
+ if ((_a = response.Items) === null || _a === void 0 ? void 0 : _a[0]) {
160
+ const item = response.Items[0];
161
+ if ((_b = item.Priority) === null || _b === void 0 ? void 0 : _b.N) {
162
+ const priority = Number.parseInt(item.Priority.N, 10);
163
+ console.log(`Found existing allocation: priority ${priority} for service ${serviceIdentifier}`);
164
+ return priority;
165
+ }
166
+ }
167
+ return null;
168
+ }
169
+ catch (error) {
170
+ console.error('Error checking existing allocation:', error);
171
+ throw error;
172
+ }
173
+ }
174
+ /**
175
+ * Attempts to allocate a specific priority atomically
176
+ */
177
+ async function tryAllocatePriority(tableName, listenerArn, serviceIdentifier, priority) {
178
+ try {
179
+ const now = new Date().toISOString();
180
+ const command = new client_dynamodb_1.PutItemCommand({
181
+ TableName: tableName,
182
+ Item: {
183
+ ListenerArn: { S: listenerArn },
184
+ Priority: { N: priority.toString() },
185
+ ServiceIdentifier: { S: serviceIdentifier },
186
+ AllocatedAt: { S: now },
187
+ Source: { S: 'CDK' },
188
+ },
189
+ ConditionExpression: 'attribute_not_exists(ListenerArn)',
190
+ });
191
+ await dynamoClient.send(command);
192
+ console.log(`Successfully allocated priority ${priority} for service ${serviceIdentifier}`);
193
+ return true;
194
+ }
195
+ catch (error) {
196
+ if (error instanceof client_dynamodb_1.ConditionalCheckFailedException) {
197
+ console.log(`Priority ${priority} already taken (race condition), will try next available`);
198
+ return false;
199
+ }
200
+ console.error('Error allocating priority:', error);
201
+ throw error;
202
+ }
203
+ }
204
+ /**
205
+ * Validates if a priority is within valid range
206
+ */
207
+ function isValidPriorityRange(priority) {
208
+ return priority >= 1 && priority <= MAX_PRIORITY;
209
+ }
210
+ /**
211
+ * Attempts to allocate preferred priority if available
212
+ */
213
+ async function tryPreferredPriority(tableName, listenerArn, serviceIdentifier, preferredPriority, usedPriorities) {
214
+ if (!isValidPriorityRange(preferredPriority)) {
215
+ return null;
216
+ }
217
+ if (usedPriorities.has(preferredPriority)) {
218
+ console.log(`Preferred priority ${preferredPriority} is already in use, finding next available`);
219
+ return null;
220
+ }
221
+ const allocated = await tryAllocatePriority(tableName, listenerArn, serviceIdentifier, preferredPriority);
222
+ return allocated ? preferredPriority : null;
223
+ }
224
+ /**
225
+ * Finds lowest available priority with gap filling
226
+ */
227
+ async function findLowestAvailablePriority(tableName, listenerArn, serviceIdentifier, usedPriorities) {
228
+ let retries = 0;
229
+ for (let priority = 1; priority <= MAX_PRIORITY; priority++) {
230
+ if (usedPriorities.has(priority)) {
231
+ continue;
232
+ }
233
+ const allocated = await tryAllocatePriority(tableName, listenerArn, serviceIdentifier, priority);
234
+ if (allocated) {
235
+ return priority;
236
+ }
237
+ // Race condition: someone else took this priority, try next
238
+ retries++;
239
+ if (retries >= MAX_RETRIES) {
240
+ throw new Error(`Failed to allocate priority after ${MAX_RETRIES} retries due to race conditions`);
241
+ }
242
+ }
243
+ throw new Error(`No available priorities found (all ${MAX_PRIORITY} in use)`);
244
+ }
245
+ /**
246
+ * Finds the lowest available priority and allocates it
247
+ */
248
+ async function allocatePriority(tableName, listenerArn, serviceIdentifier, preferredPriority) {
249
+ // Step 1: Check if service already has an allocation (idempotency)
250
+ const existingPriority = await getExistingAllocation(tableName, listenerArn, serviceIdentifier);
251
+ if (existingPriority !== null) {
252
+ return existingPriority;
253
+ }
254
+ // Step 2: Get all used priorities from ALB
255
+ const albPriorities = await getAlbListenerPriorities(listenerArn);
256
+ // Step 3: Get all tracked priorities from DynamoDB
257
+ const dynamoPriorities = await getDynamoDbPriorities(tableName, listenerArn);
258
+ // Step 4: Merge both sources to get complete picture
259
+ const allUsedPriorities = new Set([...albPriorities, ...dynamoPriorities]);
260
+ console.log(`Total priorities in use (ALB + DynamoDB): ${allUsedPriorities.size}`);
261
+ // Step 5: Try preferred priority first if provided
262
+ if (preferredPriority) {
263
+ const result = await tryPreferredPriority(tableName, listenerArn, serviceIdentifier, preferredPriority, allUsedPriorities);
264
+ if (result !== null) {
265
+ return result;
266
+ }
267
+ }
268
+ // Step 6: Find lowest available priority (gap filling)
269
+ return findLowestAvailablePriority(tableName, listenerArn, serviceIdentifier, allUsedPriorities);
270
+ }
271
+ /**
272
+ * Deletes a priority allocation from DynamoDB
273
+ */
274
+ async function deletePriorityAllocation(tableName, listenerArn, serviceIdentifier) {
275
+ var _a;
276
+ try {
277
+ // Find the priority allocated to this service
278
+ const command = new client_dynamodb_1.QueryCommand({
279
+ TableName: tableName,
280
+ IndexName: 'ServiceIdentifierIndex',
281
+ KeyConditionExpression: 'ServiceIdentifier = :sid AND ListenerArn = :arn',
282
+ ExpressionAttributeValues: {
283
+ ':sid': { S: serviceIdentifier },
284
+ ':arn': { S: listenerArn },
285
+ },
286
+ });
287
+ const response = await dynamoClient.send(command);
288
+ if (!response.Items || response.Items.length === 0) {
289
+ console.log(`No priority found for service ${serviceIdentifier}, nothing to delete`);
290
+ return;
291
+ }
292
+ const item = response.Items[0];
293
+ const priorityValue = (_a = item.Priority) === null || _a === void 0 ? void 0 : _a.N;
294
+ if (!priorityValue) {
295
+ console.error('Priority not found in item:', item);
296
+ return;
297
+ }
298
+ const priority = Number.parseInt(priorityValue, 10);
299
+ // Delete the allocation
300
+ const deleteCommand = new client_dynamodb_1.DeleteItemCommand({
301
+ TableName: tableName,
302
+ Key: {
303
+ ListenerArn: { S: listenerArn },
304
+ Priority: { N: priority.toString() },
305
+ },
306
+ });
307
+ await dynamoClient.send(deleteCommand);
308
+ console.log(`Successfully deleted priority ${priority} for service ${serviceIdentifier}`);
309
+ }
310
+ catch (error) {
311
+ console.error('Error deleting priority allocation:', error);
312
+ throw error;
313
+ }
314
+ }
315
+ /**
316
+ * Main Lambda handler for the Custom Resource
317
+ */
318
+ async function handler(event) {
319
+ console.log('Received event:', JSON.stringify(event, null, 2));
320
+ const listenerArn = event.ResourceProperties.ListenerArn;
321
+ const serviceIdentifier = event.ResourceProperties.ServiceIdentifier;
322
+ const tableName = event.ResourceProperties.TableName;
323
+ const preferredPriority = event.ResourceProperties.PreferredPriority
324
+ ? Number.parseInt(event.ResourceProperties.PreferredPriority, 10)
325
+ : undefined;
326
+ // Physical resource ID is based on listener ARN and service identifier
327
+ const physicalResourceId = hashServiceIdentifier(`${listenerArn}/${serviceIdentifier}`);
328
+ try {
329
+ // Handle Delete operation
330
+ if (event.RequestType === 'Delete') {
331
+ console.log('Handling DELETE request - removing priority allocation');
332
+ await deletePriorityAllocation(tableName, listenerArn, serviceIdentifier);
333
+ // Return success with priority 0 (doesn't matter for delete)
334
+ return success(event, physicalResourceId, 0);
335
+ }
336
+ // Handle Create and Update operations
337
+ console.log(`Handling ${event.RequestType} request - allocating priority`);
338
+ const priority = await allocatePriority(tableName, listenerArn, serviceIdentifier, preferredPriority);
339
+ return success(event, physicalResourceId, priority);
340
+ }
341
+ catch (error) {
342
+ const errorMessage = error instanceof Error ? error.message : String(error);
343
+ console.error('Error in handler:', errorMessage);
344
+ return fail(event, physicalResourceId, errorMessage);
345
+ }
346
+ }
347
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"priority-allocator-handler.js","sourceRoot":"","sources":["priority-allocator-handler.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ElasticLoadBalancingV2Client,\n  DescribeRulesCommand,\n  type DescribeRulesCommandOutput,\n} from '@aws-sdk/client-elastic-load-balancing-v2';\nimport {\n  DynamoDBClient,\n  PutItemCommand,\n  QueryCommand,\n  DeleteItemCommand,\n  ConditionalCheckFailedException,\n  type QueryCommandOutput,\n  type AttributeValue,\n} from '@aws-sdk/client-dynamodb';\nimport * as crypto from 'node:crypto';\n\n// CloudFormation Custom Resource event interface\ninterface CustomResourceEvent {\n  readonly RequestType: 'Create' | 'Update' | 'Delete';\n  readonly ResponseURL: string;\n  readonly StackId: string;\n  readonly RequestId: string;\n  readonly ResourceType: string;\n  readonly LogicalResourceId: string;\n  readonly PhysicalResourceId?: string;\n  readonly ResourceProperties: {\n    readonly ListenerArn: string;\n    readonly ServiceIdentifier: string;\n    readonly TableName: string;\n    readonly PreferredPriority?: string;\n  };\n  readonly OldResourceProperties?: {\n    readonly ListenerArn: string;\n    readonly ServiceIdentifier: string;\n    readonly TableName: string;\n    readonly PreferredPriority?: string;\n  };\n}\n\ninterface CustomResourceResponse {\n  Status: 'SUCCESS' | 'FAILED';\n  Reason?: string;\n  PhysicalResourceId: string;\n  StackId: string;\n  RequestId: string;\n  LogicalResourceId: string;\n  NoEcho?: boolean;\n  Data?: {\n    Priority: string;\n  };\n}\n\nconst elbv2Client = new ElasticLoadBalancingV2Client({});\nconst dynamoClient = new DynamoDBClient({});\n\nconst MAX_PRIORITY = 50000;\nconst MAX_RETRIES = 10;\n\n/**\n * Creates a deterministic hash of the service identifier\n */\nfunction hashServiceIdentifier(serviceIdentifier: string): string {\n  return crypto.createHash('sha256').update(serviceIdentifier).digest('hex');\n}\n\n/**\n * Creates a success response\n */\nfunction success(\n  event: CustomResourceEvent,\n  physicalResourceId: string,\n  priority: number,\n): CustomResourceResponse {\n  return {\n    Status: 'SUCCESS',\n    PhysicalResourceId: physicalResourceId,\n    StackId: event.StackId,\n    RequestId: event.RequestId,\n    LogicalResourceId: event.LogicalResourceId,\n    NoEcho: false,\n    Data: {\n      Priority: priority.toString(),\n    },\n  };\n}\n\n/**\n * Creates a failure response\n */\nfunction fail(\n  event: CustomResourceEvent,\n  physicalResourceId: string,\n  reason: string,\n): CustomResourceResponse {\n  return {\n    Status: 'FAILED',\n    Reason: reason,\n    PhysicalResourceId: physicalResourceId,\n    StackId: event.StackId,\n    RequestId: event.RequestId,\n    LogicalResourceId: event.LogicalResourceId,\n    NoEcho: false,\n  };\n}\n\n/**\n * Extracts valid priority from a rule\n */\nfunction extractPriorityFromRule(rule: {Priority?: string}): number | null {\n  if (!rule.Priority || rule.Priority === 'default') {\n    return null;\n  }\n  const priority = Number.parseInt(rule.Priority, 10);\n  return Number.isNaN(priority) ? null : priority;\n}\n\n/**\n * Formats priority list for logging\n */\nfunction formatPrioritiesForLog(priorities: Set<number>): string {\n  const sorted = Array.from(priorities).sort((a, b) => a - b);\n  const preview = sorted.slice(0, 10).join(', ');\n  return priorities.size > 10 ? `${preview}...` : preview;\n}\n\n/**\n * Gets all priorities currently in use on the ALB listener\n */\nasync function getAlbListenerPriorities(\n  listenerArn: string,\n): Promise<Set<number>> {\n  const priorities = new Set<number>();\n\n  try {\n    let nextMarker: string | undefined = undefined;\n\n    do {\n      const command = new DescribeRulesCommand({\n        ListenerArn: listenerArn,\n        Marker: nextMarker,\n      });\n\n      const response: DescribeRulesCommandOutput =\n        await elbv2Client.send(command);\n\n      if (response.Rules) {\n        for (const rule of response.Rules) {\n          const priority = extractPriorityFromRule(rule);\n          if (priority !== null) {\n            priorities.add(priority);\n          }\n        }\n      }\n\n      nextMarker = response.NextMarker;\n    } while (nextMarker);\n\n    const formatted = formatPrioritiesForLog(priorities);\n    console.log(\n      `Found ${priorities.size} priorities in use on ALB listener: ${formatted}`,\n    );\n  } catch (error) {\n    console.error('Error fetching ALB listener priorities:', error);\n    throw error;\n  }\n\n  return priorities;\n}\n\n/**\n * Extracts valid priority from a DynamoDB item\n */\nfunction extractPriorityFromDynamoItem(\n  item: Record<string, AttributeValue>,\n): number | null {\n  if (!item.Priority?.N) {\n    return null;\n  }\n  const priority = Number.parseInt(item.Priority.N, 10);\n  return Number.isNaN(priority) ? null : priority;\n}\n\n/**\n * Gets all priorities tracked in DynamoDB for this listener\n */\nasync function getDynamoDbPriorities(\n  tableName: string,\n  listenerArn: string,\n): Promise<Set<number>> {\n  const priorities = new Set<number>();\n\n  try {\n    let lastEvaluatedKey: Record<string, AttributeValue> | undefined =\n      undefined;\n\n    do {\n      const command = new QueryCommand({\n        TableName: tableName,\n        KeyConditionExpression: 'ListenerArn = :arn',\n        ExpressionAttributeValues: {\n          ':arn': {S: listenerArn},\n        },\n        ExclusiveStartKey: lastEvaluatedKey,\n      });\n\n      const response: QueryCommandOutput = await dynamoClient.send(command);\n\n      if (response.Items) {\n        for (const item of response.Items) {\n          const priority = extractPriorityFromDynamoItem(item);\n          if (priority !== null) {\n            priorities.add(priority);\n          }\n        }\n      }\n\n      lastEvaluatedKey = response.LastEvaluatedKey;\n    } while (lastEvaluatedKey);\n\n    console.log(\n      `Found ${priorities.size} priorities tracked in DynamoDB for listener`,\n    );\n  } catch (error) {\n    console.error('Error fetching DynamoDB priorities:', error);\n    throw error;\n  }\n\n  return priorities;\n}\n\n/**\n * Checks if the service already has an allocated priority (for idempotency)\n */\nasync function getExistingAllocation(\n  tableName: string,\n  listenerArn: string,\n  serviceIdentifier: string,\n): Promise<number | null> {\n  try {\n    const command = new QueryCommand({\n      TableName: tableName,\n      IndexName: 'ServiceIdentifierIndex',\n      KeyConditionExpression: 'ServiceIdentifier = :sid AND ListenerArn = :arn',\n      ExpressionAttributeValues: {\n        ':sid': {S: serviceIdentifier},\n        ':arn': {S: listenerArn},\n      },\n    });\n\n    const response: QueryCommandOutput = await dynamoClient.send(command);\n\n    if (response.Items?.[0]) {\n      const item = response.Items[0];\n      if (item.Priority?.N) {\n        const priority = Number.parseInt(item.Priority.N, 10);\n        console.log(\n          `Found existing allocation: priority ${priority} for service ${serviceIdentifier}`,\n        );\n        return priority;\n      }\n    }\n\n    return null;\n  } catch (error) {\n    console.error('Error checking existing allocation:', error);\n    throw error;\n  }\n}\n\n/**\n * Attempts to allocate a specific priority atomically\n */\nasync function tryAllocatePriority(\n  tableName: string,\n  listenerArn: string,\n  serviceIdentifier: string,\n  priority: number,\n): Promise<boolean> {\n  try {\n    const now = new Date().toISOString();\n\n    const command = new PutItemCommand({\n      TableName: tableName,\n      Item: {\n        ListenerArn: {S: listenerArn},\n        Priority: {N: priority.toString()},\n        ServiceIdentifier: {S: serviceIdentifier},\n        AllocatedAt: {S: now},\n        Source: {S: 'CDK'},\n      },\n      ConditionExpression: 'attribute_not_exists(ListenerArn)',\n    });\n\n    await dynamoClient.send(command);\n    console.log(\n      `Successfully allocated priority ${priority} for service ${serviceIdentifier}`,\n    );\n    return true;\n  } catch (error) {\n    if (error instanceof ConditionalCheckFailedException) {\n      console.log(\n        `Priority ${priority} already taken (race condition), will try next available`,\n      );\n      return false;\n    }\n    console.error('Error allocating priority:', error);\n    throw error;\n  }\n}\n\n/**\n * Validates if a priority is within valid range\n */\nfunction isValidPriorityRange(priority: number): boolean {\n  return priority >= 1 && priority <= MAX_PRIORITY;\n}\n\n/**\n * Attempts to allocate preferred priority if available\n */\nasync function tryPreferredPriority(\n  tableName: string,\n  listenerArn: string,\n  serviceIdentifier: string,\n  preferredPriority: number,\n  usedPriorities: Set<number>,\n): Promise<number | null> {\n  if (!isValidPriorityRange(preferredPriority)) {\n    return null;\n  }\n\n  if (usedPriorities.has(preferredPriority)) {\n    console.log(\n      `Preferred priority ${preferredPriority} is already in use, finding next available`,\n    );\n    return null;\n  }\n\n  const allocated = await tryAllocatePriority(\n    tableName,\n    listenerArn,\n    serviceIdentifier,\n    preferredPriority,\n  );\n  return allocated ? preferredPriority : null;\n}\n\n/**\n * Finds lowest available priority with gap filling\n */\nasync function findLowestAvailablePriority(\n  tableName: string,\n  listenerArn: string,\n  serviceIdentifier: string,\n  usedPriorities: Set<number>,\n): Promise<number> {\n  let retries = 0;\n\n  for (let priority = 1; priority <= MAX_PRIORITY; priority++) {\n    if (usedPriorities.has(priority)) {\n      continue;\n    }\n\n    const allocated = await tryAllocatePriority(\n      tableName,\n      listenerArn,\n      serviceIdentifier,\n      priority,\n    );\n\n    if (allocated) {\n      return priority;\n    }\n\n    // Race condition: someone else took this priority, try next\n    retries++;\n    if (retries >= MAX_RETRIES) {\n      throw new Error(\n        `Failed to allocate priority after ${MAX_RETRIES} retries due to race conditions`,\n      );\n    }\n  }\n\n  throw new Error(`No available priorities found (all ${MAX_PRIORITY} in use)`);\n}\n\n/**\n * Finds the lowest available priority and allocates it\n */\nasync function allocatePriority(\n  tableName: string,\n  listenerArn: string,\n  serviceIdentifier: string,\n  preferredPriority?: number,\n): Promise<number> {\n  // Step 1: Check if service already has an allocation (idempotency)\n  const existingPriority = await getExistingAllocation(\n    tableName,\n    listenerArn,\n    serviceIdentifier,\n  );\n  if (existingPriority !== null) {\n    return existingPriority;\n  }\n\n  // Step 2: Get all used priorities from ALB\n  const albPriorities = await getAlbListenerPriorities(listenerArn);\n\n  // Step 3: Get all tracked priorities from DynamoDB\n  const dynamoPriorities = await getDynamoDbPriorities(tableName, listenerArn);\n\n  // Step 4: Merge both sources to get complete picture\n  const allUsedPriorities = new Set([...albPriorities, ...dynamoPriorities]);\n\n  console.log(\n    `Total priorities in use (ALB + DynamoDB): ${allUsedPriorities.size}`,\n  );\n\n  // Step 5: Try preferred priority first if provided\n  if (preferredPriority) {\n    const result = await tryPreferredPriority(\n      tableName,\n      listenerArn,\n      serviceIdentifier,\n      preferredPriority,\n      allUsedPriorities,\n    );\n    if (result !== null) {\n      return result;\n    }\n  }\n\n  // Step 6: Find lowest available priority (gap filling)\n  return findLowestAvailablePriority(\n    tableName,\n    listenerArn,\n    serviceIdentifier,\n    allUsedPriorities,\n  );\n}\n\n/**\n * Deletes a priority allocation from DynamoDB\n */\nasync function deletePriorityAllocation(\n  tableName: string,\n  listenerArn: string,\n  serviceIdentifier: string,\n): Promise<void> {\n  try {\n    // Find the priority allocated to this service\n    const command = new QueryCommand({\n      TableName: tableName,\n      IndexName: 'ServiceIdentifierIndex',\n      KeyConditionExpression: 'ServiceIdentifier = :sid AND ListenerArn = :arn',\n      ExpressionAttributeValues: {\n        ':sid': {S: serviceIdentifier},\n        ':arn': {S: listenerArn},\n      },\n    });\n\n    const response = await dynamoClient.send(command);\n\n    if (!response.Items || response.Items.length === 0) {\n      console.log(\n        `No priority found for service ${serviceIdentifier}, nothing to delete`,\n      );\n      return;\n    }\n\n    const item = response.Items[0];\n    const priorityValue = item.Priority?.N;\n    if (!priorityValue) {\n      console.error('Priority not found in item:', item);\n      return;\n    }\n\n    const priority = Number.parseInt(priorityValue, 10);\n\n    // Delete the allocation\n    const deleteCommand = new DeleteItemCommand({\n      TableName: tableName,\n      Key: {\n        ListenerArn: {S: listenerArn},\n        Priority: {N: priority.toString()},\n      },\n    });\n\n    await dynamoClient.send(deleteCommand);\n    console.log(\n      `Successfully deleted priority ${priority} for service ${serviceIdentifier}`,\n    );\n  } catch (error) {\n    console.error('Error deleting priority allocation:', error);\n    throw error;\n  }\n}\n\n/**\n * Main Lambda handler for the Custom Resource\n */\nexport async function handler(\n  event: CustomResourceEvent,\n): Promise<CustomResourceResponse> {\n  console.log('Received event:', JSON.stringify(event, null, 2));\n\n  const listenerArn = event.ResourceProperties.ListenerArn;\n  const serviceIdentifier = event.ResourceProperties.ServiceIdentifier;\n  const tableName = event.ResourceProperties.TableName;\n  const preferredPriority = event.ResourceProperties.PreferredPriority\n    ? Number.parseInt(event.ResourceProperties.PreferredPriority, 10)\n    : undefined;\n\n  // Physical resource ID is based on listener ARN and service identifier\n  const physicalResourceId = hashServiceIdentifier(\n    `${listenerArn}/${serviceIdentifier}`,\n  );\n\n  try {\n    // Handle Delete operation\n    if (event.RequestType === 'Delete') {\n      console.log('Handling DELETE request - removing priority allocation');\n      await deletePriorityAllocation(tableName, listenerArn, serviceIdentifier);\n      // Return success with priority 0 (doesn't matter for delete)\n      return success(event, physicalResourceId, 0);\n    }\n\n    // Handle Create and Update operations\n    console.log(`Handling ${event.RequestType} request - allocating priority`);\n    const priority = await allocatePriority(\n      tableName,\n      listenerArn,\n      serviceIdentifier,\n      preferredPriority,\n    );\n\n    return success(event, physicalResourceId, priority);\n  } catch (error) {\n    const errorMessage = error instanceof Error ? error.message : String(error);\n    console.error('Error in handler:', errorMessage);\n    return fail(event, physicalResourceId, errorMessage);\n  }\n}\n"]}
@@ -0,0 +1,84 @@
1
+ import { Construct } from 'constructs';
2
+ import { CustomResource } from 'aws-cdk-lib';
3
+ export interface PriorityAllocatorProps {
4
+ /**
5
+ * The ARN of the ALB listener for which to allocate a priority.
6
+ */
7
+ readonly listenerArn: string;
8
+ /**
9
+ * Optional preferred priority. If available, this priority will be allocated.
10
+ * If not available, the next available priority will be allocated.
11
+ *
12
+ * @default - Next available priority is allocated
13
+ */
14
+ readonly preferredPriority?: number;
15
+ }
16
+ /**
17
+ * Allocates a unique priority for an ALB listener rule using a Lambda-backed Custom Resource.
18
+ *
19
+ * This construct implements a singleton pattern for the Lambda function and DynamoDB table,
20
+ * ensuring that only one instance of each exists per AWS account/region regardless of how
21
+ * many services use priority allocation.
22
+ *
23
+ * The allocation algorithm:
24
+ * 1. Checks if this service already has an allocated priority (idempotent)
25
+ * 2. Queries all priorities currently on the ALB listener (source of truth)
26
+ * 3. Queries priorities tracked in DynamoDB
27
+ * 4. Merges both sources to get complete picture
28
+ * 5. Finds lowest available priority (gap filling)
29
+ * 6. Allocates atomically with DynamoDB conditional write
30
+ * 7. Returns allocated priority to CloudFormation
31
+ *
32
+ * On stack deletion, the priority is released back to the pool for reuse.
33
+ *
34
+ * @example
35
+ * ```typescript
36
+ * const allocator = new PriorityAllocator(this, 'PriorityAllocator', {
37
+ * listenerArn: listener.listenerArn,
38
+ * });
39
+ *
40
+ * // Use the allocated priority
41
+ * listener.addTargetGroups('TargetGroup', {
42
+ * targetGroups: [targetGroup],
43
+ * priority: allocator.priority,
44
+ * });
45
+ * ```
46
+ */
47
+ export declare class PriorityAllocator extends Construct {
48
+ /**
49
+ * The allocated priority for the ALB listener rule.
50
+ */
51
+ readonly priority: number;
52
+ /**
53
+ * The service identifier used for tracking this allocation.
54
+ */
55
+ readonly serviceIdentifier: string;
56
+ /**
57
+ * The Custom Resource that manages the priority allocation.
58
+ */
59
+ readonly resource: CustomResource;
60
+ /**
61
+ * Singleton table name - shared across all priority allocators in the account/region.
62
+ */
63
+ private static readonly TABLE_NAME;
64
+ /**
65
+ * Gets or creates the singleton DynamoDB table for priority tracking.
66
+ * Only one table exists per AWS account/region.
67
+ */
68
+ private static getOrCreateTable;
69
+ /**
70
+ * Gets or creates the singleton Lambda function for priority allocation.
71
+ * Only one Lambda function exists per AWS account/region.
72
+ */
73
+ private static getOrCreateLambda;
74
+ /**
75
+ * Gets or creates the singleton Custom Resource Provider.
76
+ * Only one provider exists per AWS account/region.
77
+ */
78
+ private static getOrCreateProvider;
79
+ /**
80
+ * Generates a deterministic service identifier based on the construct path and listener ARN.
81
+ */
82
+ private generateServiceIdentifier;
83
+ constructor(scope: Construct, id: string, props: PriorityAllocatorProps);
84
+ }
@@ -0,0 +1,225 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.PriorityAllocator = void 0;
4
+ const constructs_1 = require("constructs");
5
+ const aws_lambda_nodejs_1 = require("aws-cdk-lib/aws-lambda-nodejs");
6
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
7
+ const custom_resources_1 = require("aws-cdk-lib/custom-resources");
8
+ const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
9
+ const path = require("node:path");
10
+ const aws_iam_1 = require("aws-cdk-lib/aws-iam");
11
+ const aws_dynamodb_1 = require("aws-cdk-lib/aws-dynamodb");
12
+ const crypto = require("node:crypto");
13
+ /**
14
+ * Allocates a unique priority for an ALB listener rule using a Lambda-backed Custom Resource.
15
+ *
16
+ * This construct implements a singleton pattern for the Lambda function and DynamoDB table,
17
+ * ensuring that only one instance of each exists per AWS account/region regardless of how
18
+ * many services use priority allocation.
19
+ *
20
+ * The allocation algorithm:
21
+ * 1. Checks if this service already has an allocated priority (idempotent)
22
+ * 2. Queries all priorities currently on the ALB listener (source of truth)
23
+ * 3. Queries priorities tracked in DynamoDB
24
+ * 4. Merges both sources to get complete picture
25
+ * 5. Finds lowest available priority (gap filling)
26
+ * 6. Allocates atomically with DynamoDB conditional write
27
+ * 7. Returns allocated priority to CloudFormation
28
+ *
29
+ * On stack deletion, the priority is released back to the pool for reuse.
30
+ *
31
+ * @example
32
+ * ```typescript
33
+ * const allocator = new PriorityAllocator(this, 'PriorityAllocator', {
34
+ * listenerArn: listener.listenerArn,
35
+ * });
36
+ *
37
+ * // Use the allocated priority
38
+ * listener.addTargetGroups('TargetGroup', {
39
+ * targetGroups: [targetGroup],
40
+ * priority: allocator.priority,
41
+ * });
42
+ * ```
43
+ */
44
+ class PriorityAllocator extends constructs_1.Construct {
45
+ /**
46
+ * Gets or creates the singleton DynamoDB table for priority tracking.
47
+ * Only one table exists per AWS account/region.
48
+ */
49
+ static getOrCreateTable(scope) {
50
+ const stack = aws_cdk_lib_1.Stack.of(scope);
51
+ const tableId = 'PriorityAllocatorTable';
52
+ // Try to find existing table in the stack
53
+ const existing = stack.node.tryFindChild(tableId);
54
+ if (existing) {
55
+ return existing;
56
+ }
57
+ // Create new table at stack level (singleton)
58
+ const table = new aws_dynamodb_1.Table(stack, tableId, {
59
+ tableName: PriorityAllocator.TABLE_NAME,
60
+ partitionKey: {
61
+ name: 'ListenerArn',
62
+ type: aws_dynamodb_1.AttributeType.STRING,
63
+ },
64
+ sortKey: {
65
+ name: 'Priority',
66
+ type: aws_dynamodb_1.AttributeType.NUMBER,
67
+ },
68
+ billingMode: aws_dynamodb_1.BillingMode.PAY_PER_REQUEST,
69
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.RETAIN, // Never delete - shared infrastructure
70
+ pointInTimeRecoverySpecification: {
71
+ pointInTimeRecoveryEnabled: true,
72
+ },
73
+ });
74
+ // Add GSI for querying by service identifier
75
+ table.addGlobalSecondaryIndex({
76
+ indexName: 'ServiceIdentifierIndex',
77
+ partitionKey: {
78
+ name: 'ServiceIdentifier',
79
+ type: aws_dynamodb_1.AttributeType.STRING,
80
+ },
81
+ sortKey: {
82
+ name: 'ListenerArn',
83
+ type: aws_dynamodb_1.AttributeType.STRING,
84
+ },
85
+ });
86
+ return table;
87
+ }
88
+ /**
89
+ * Gets or creates the singleton Lambda function for priority allocation.
90
+ * Only one Lambda function exists per AWS account/region.
91
+ */
92
+ static getOrCreateLambda(scope, table) {
93
+ const stack = aws_cdk_lib_1.Stack.of(scope);
94
+ const lambdaId = 'PriorityAllocatorLambda';
95
+ // Try to find existing Lambda in the stack
96
+ const existing = stack.node.tryFindChild(lambdaId);
97
+ if (existing) {
98
+ return existing;
99
+ }
100
+ // Create IAM role for Lambda
101
+ const role = new aws_iam_1.Role(stack, 'PriorityAllocatorLambdaRole', {
102
+ assumedBy: new aws_iam_1.ServicePrincipal('lambda.amazonaws.com'),
103
+ description: 'Role for ALB Priority Allocator Lambda function',
104
+ });
105
+ // CloudWatch Logs permissions
106
+ role.addToPolicy(new aws_iam_1.PolicyStatement({
107
+ effect: aws_iam_1.Effect.ALLOW,
108
+ actions: [
109
+ 'logs:CreateLogGroup',
110
+ 'logs:CreateLogStream',
111
+ 'logs:PutLogEvents',
112
+ ],
113
+ resources: ['*'],
114
+ }));
115
+ // ALB read permissions
116
+ role.addToPolicy(new aws_iam_1.PolicyStatement({
117
+ effect: aws_iam_1.Effect.ALLOW,
118
+ actions: [
119
+ 'elasticloadbalancing:DescribeListeners',
120
+ 'elasticloadbalancing:DescribeRules',
121
+ ],
122
+ resources: ['*'],
123
+ }));
124
+ // DynamoDB permissions
125
+ role.addToPolicy(new aws_iam_1.PolicyStatement({
126
+ effect: aws_iam_1.Effect.ALLOW,
127
+ actions: [
128
+ 'dynamodb:Query',
129
+ 'dynamodb:GetItem',
130
+ 'dynamodb:PutItem',
131
+ 'dynamodb:DeleteItem',
132
+ ],
133
+ resources: [table.tableArn, `${table.tableArn}/index/*`],
134
+ }));
135
+ // Create Lambda function
136
+ return new aws_lambda_nodejs_1.NodejsFunction(stack, lambdaId, {
137
+ role,
138
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
139
+ handler: 'handler',
140
+ entry: path.join(__dirname, 'priority-allocator-handler.js'),
141
+ timeout: aws_cdk_lib_1.Duration.seconds(30),
142
+ memorySize: 256,
143
+ description: 'Allocates unique priorities for ALB listener rules',
144
+ functionName: 'priority-allocator-singleton',
145
+ });
146
+ }
147
+ /**
148
+ * Gets or creates the singleton Custom Resource Provider.
149
+ * Only one provider exists per AWS account/region.
150
+ */
151
+ static getOrCreateProvider(scope, lambda) {
152
+ const stack = aws_cdk_lib_1.Stack.of(scope);
153
+ const providerId = 'PriorityAllocatorProvider';
154
+ // Try to find existing provider in the stack
155
+ const existing = stack.node.tryFindChild(providerId);
156
+ if (existing) {
157
+ return existing;
158
+ }
159
+ // Create new provider at stack level (singleton)
160
+ return new custom_resources_1.Provider(stack, providerId, {
161
+ onEventHandler: lambda,
162
+ });
163
+ }
164
+ /**
165
+ * Generates a deterministic service identifier based on the construct path and listener ARN.
166
+ */
167
+ generateServiceIdentifier(listenerArn) {
168
+ const stack = aws_cdk_lib_1.Stack.of(this);
169
+ const region = stack.region;
170
+ const account = stack.account;
171
+ const stackName = stack.stackName;
172
+ const constructPath = this.node.path;
173
+ // Create deterministic hash
174
+ const input = `${account}/${region}/${stackName}/${constructPath}/${listenerArn}`;
175
+ const hash = crypto
176
+ .createHash('sha256')
177
+ .update(input)
178
+ .digest('hex')
179
+ .substring(0, 12);
180
+ // Create human-readable identifier with stack name and hash
181
+ const sanitizedStackName = stackName
182
+ .replaceAll(/[^a-zA-Z0-9-]/g, '-')
183
+ .toLowerCase();
184
+ return `${sanitizedStackName}-${hash}`;
185
+ }
186
+ constructor(scope, id, props) {
187
+ var _a;
188
+ super(scope, id);
189
+ // Get or create singleton resources
190
+ const table = PriorityAllocator.getOrCreateTable(this);
191
+ const lambda = PriorityAllocator.getOrCreateLambda(this, table);
192
+ const provider = PriorityAllocator.getOrCreateProvider(this, lambda);
193
+ // Generate service identifier
194
+ this.serviceIdentifier = this.generateServiceIdentifier(props.listenerArn);
195
+ // Create Custom Resource for this specific service
196
+ this.resource = new aws_cdk_lib_1.CustomResource(this, 'Resource', {
197
+ serviceToken: provider.serviceToken,
198
+ properties: {
199
+ ListenerArn: props.listenerArn,
200
+ ServiceIdentifier: this.serviceIdentifier,
201
+ TableName: PriorityAllocator.TABLE_NAME,
202
+ PreferredPriority: (_a = props.preferredPriority) === null || _a === void 0 ? void 0 : _a.toString(),
203
+ // Add timestamp to ensure update on property changes
204
+ Timestamp: Date.now().toString(),
205
+ },
206
+ });
207
+ // Extract priority from custom resource
208
+ this.priority = aws_cdk_lib_1.Token.asNumber(this.resource.getAtt('Priority'));
209
+ // Add CloudFormation outputs for debugging
210
+ new aws_cdk_lib_1.CfnOutput(this, 'ServiceIdentifier', {
211
+ value: this.serviceIdentifier,
212
+ description: 'Service identifier for priority allocation tracking',
213
+ });
214
+ new aws_cdk_lib_1.CfnOutput(this, 'AllocatedPriority', {
215
+ value: this.priority.toString(),
216
+ description: 'Auto-allocated priority for ALB listener rule',
217
+ });
218
+ }
219
+ }
220
+ exports.PriorityAllocator = PriorityAllocator;
221
+ /**
222
+ * Singleton table name - shared across all priority allocators in the account/region.
223
+ */
224
+ PriorityAllocator.TABLE_NAME = 'alb-listener-priorities';
225
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"priority-allocator.js","sourceRoot":"","sources":["priority-allocator.ts"],"names":[],"mappings":"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","sourcesContent":["import {Construct} from 'constructs';\nimport {NodejsFunction} from 'aws-cdk-lib/aws-lambda-nodejs';\nimport {\n  CustomResource,\n  Duration,\n  RemovalPolicy,\n  Stack,\n  CfnOutput,\n  Token,\n} from 'aws-cdk-lib';\nimport {Provider} from 'aws-cdk-lib/custom-resources';\nimport {Runtime} from 'aws-cdk-lib/aws-lambda';\nimport * as path from 'node:path';\nimport {\n  Effect,\n  PolicyStatement,\n  Role,\n  ServicePrincipal,\n} from 'aws-cdk-lib/aws-iam';\nimport {AttributeType, BillingMode, Table} from 'aws-cdk-lib/aws-dynamodb';\nimport * as crypto from 'node:crypto';\n\nexport interface PriorityAllocatorProps {\n  /**\n   * The ARN of the ALB listener for which to allocate a priority.\n   */\n  readonly listenerArn: string;\n\n  /**\n   * Optional preferred priority. If available, this priority will be allocated.\n   * If not available, the next available priority will be allocated.\n   *\n   * @default - Next available priority is allocated\n   */\n  readonly preferredPriority?: number;\n}\n\n/**\n * Allocates a unique priority for an ALB listener rule using a Lambda-backed Custom Resource.\n *\n * This construct implements a singleton pattern for the Lambda function and DynamoDB table,\n * ensuring that only one instance of each exists per AWS account/region regardless of how\n * many services use priority allocation.\n *\n * The allocation algorithm:\n * 1. Checks if this service already has an allocated priority (idempotent)\n * 2. Queries all priorities currently on the ALB listener (source of truth)\n * 3. Queries priorities tracked in DynamoDB\n * 4. Merges both sources to get complete picture\n * 5. Finds lowest available priority (gap filling)\n * 6. Allocates atomically with DynamoDB conditional write\n * 7. Returns allocated priority to CloudFormation\n *\n * On stack deletion, the priority is released back to the pool for reuse.\n *\n * @example\n * ```typescript\n * const allocator = new PriorityAllocator(this, 'PriorityAllocator', {\n *   listenerArn: listener.listenerArn,\n * });\n *\n * // Use the allocated priority\n * listener.addTargetGroups('TargetGroup', {\n *   targetGroups: [targetGroup],\n *   priority: allocator.priority,\n * });\n * ```\n */\nexport class PriorityAllocator extends Construct {\n  /**\n   * The allocated priority for the ALB listener rule.\n   */\n  readonly priority: number;\n\n  /**\n   * The service identifier used for tracking this allocation.\n   */\n  readonly serviceIdentifier: string;\n\n  /**\n   * The Custom Resource that manages the priority allocation.\n   */\n  readonly resource: CustomResource;\n\n  /**\n   * Singleton table name - shared across all priority allocators in the account/region.\n   */\n  private static readonly TABLE_NAME = 'alb-listener-priorities';\n\n  /**\n   * Gets or creates the singleton DynamoDB table for priority tracking.\n   * Only one table exists per AWS account/region.\n   */\n  private static getOrCreateTable(scope: Construct): Table {\n    const stack = Stack.of(scope);\n    const tableId = 'PriorityAllocatorTable';\n\n    // Try to find existing table in the stack\n    const existing = stack.node.tryFindChild(tableId) as Table | undefined;\n    if (existing) {\n      return existing;\n    }\n\n    // Create new table at stack level (singleton)\n    const table = new Table(stack, tableId, {\n      tableName: PriorityAllocator.TABLE_NAME,\n      partitionKey: {\n        name: 'ListenerArn',\n        type: AttributeType.STRING,\n      },\n      sortKey: {\n        name: 'Priority',\n        type: AttributeType.NUMBER,\n      },\n      billingMode: BillingMode.PAY_PER_REQUEST,\n      removalPolicy: RemovalPolicy.RETAIN, // Never delete - shared infrastructure\n      pointInTimeRecoverySpecification: {\n        pointInTimeRecoveryEnabled: true,\n      },\n    });\n\n    // Add GSI for querying by service identifier\n    table.addGlobalSecondaryIndex({\n      indexName: 'ServiceIdentifierIndex',\n      partitionKey: {\n        name: 'ServiceIdentifier',\n        type: AttributeType.STRING,\n      },\n      sortKey: {\n        name: 'ListenerArn',\n        type: AttributeType.STRING,\n      },\n    });\n\n    return table;\n  }\n\n  /**\n   * Gets or creates the singleton Lambda function for priority allocation.\n   * Only one Lambda function exists per AWS account/region.\n   */\n  private static getOrCreateLambda(\n    scope: Construct,\n    table: Table,\n  ): NodejsFunction {\n    const stack = Stack.of(scope);\n    const lambdaId = 'PriorityAllocatorLambda';\n\n    // Try to find existing Lambda in the stack\n    const existing = stack.node.tryFindChild(lambdaId) as\n      | NodejsFunction\n      | undefined;\n    if (existing) {\n      return existing;\n    }\n\n    // Create IAM role for Lambda\n    const role = new Role(stack, 'PriorityAllocatorLambdaRole', {\n      assumedBy: new ServicePrincipal('lambda.amazonaws.com'),\n      description: 'Role for ALB Priority Allocator Lambda function',\n    });\n\n    // CloudWatch Logs permissions\n    role.addToPolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: [\n          'logs:CreateLogGroup',\n          'logs:CreateLogStream',\n          'logs:PutLogEvents',\n        ],\n        resources: ['*'],\n      }),\n    );\n\n    // ALB read permissions\n    role.addToPolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: [\n          'elasticloadbalancing:DescribeListeners',\n          'elasticloadbalancing:DescribeRules',\n        ],\n        resources: ['*'],\n      }),\n    );\n\n    // DynamoDB permissions\n    role.addToPolicy(\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: [\n          'dynamodb:Query',\n          'dynamodb:GetItem',\n          'dynamodb:PutItem',\n          'dynamodb:DeleteItem',\n        ],\n        resources: [table.tableArn, `${table.tableArn}/index/*`],\n      }),\n    );\n\n    // Create Lambda function\n    return new NodejsFunction(stack, lambdaId, {\n      role,\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'handler',\n      entry: path.join(__dirname, 'priority-allocator-handler.js'),\n      timeout: Duration.seconds(30),\n      memorySize: 256,\n      description: 'Allocates unique priorities for ALB listener rules',\n      functionName: 'priority-allocator-singleton',\n    });\n  }\n\n  /**\n   * Gets or creates the singleton Custom Resource Provider.\n   * Only one provider exists per AWS account/region.\n   */\n  private static getOrCreateProvider(\n    scope: Construct,\n    lambda: NodejsFunction,\n  ): Provider {\n    const stack = Stack.of(scope);\n    const providerId = 'PriorityAllocatorProvider';\n\n    // Try to find existing provider in the stack\n    const existing = stack.node.tryFindChild(providerId) as\n      | Provider\n      | undefined;\n    if (existing) {\n      return existing;\n    }\n\n    // Create new provider at stack level (singleton)\n    return new Provider(stack, providerId, {\n      onEventHandler: lambda,\n    });\n  }\n\n  /**\n   * Generates a deterministic service identifier based on the construct path and listener ARN.\n   */\n  private generateServiceIdentifier(listenerArn: string): string {\n    const stack = Stack.of(this);\n    const region = stack.region;\n    const account = stack.account;\n    const stackName = stack.stackName;\n    const constructPath = this.node.path;\n\n    // Create deterministic hash\n    const input = `${account}/${region}/${stackName}/${constructPath}/${listenerArn}`;\n    const hash = crypto\n      .createHash('sha256')\n      .update(input)\n      .digest('hex')\n      .substring(0, 12);\n\n    // Create human-readable identifier with stack name and hash\n    const sanitizedStackName = stackName\n      .replaceAll(/[^a-zA-Z0-9-]/g, '-')\n      .toLowerCase();\n    return `${sanitizedStackName}-${hash}`;\n  }\n\n  constructor(scope: Construct, id: string, props: PriorityAllocatorProps) {\n    super(scope, id);\n\n    // Get or create singleton resources\n    const table = PriorityAllocator.getOrCreateTable(this);\n    const lambda = PriorityAllocator.getOrCreateLambda(this, table);\n    const provider = PriorityAllocator.getOrCreateProvider(this, lambda);\n\n    // Generate service identifier\n    this.serviceIdentifier = this.generateServiceIdentifier(props.listenerArn);\n\n    // Create Custom Resource for this specific service\n    this.resource = new CustomResource(this, 'Resource', {\n      serviceToken: provider.serviceToken,\n      properties: {\n        ListenerArn: props.listenerArn,\n        ServiceIdentifier: this.serviceIdentifier,\n        TableName: PriorityAllocator.TABLE_NAME,\n        PreferredPriority: props.preferredPriority?.toString(),\n        // Add timestamp to ensure update on property changes\n        Timestamp: Date.now().toString(),\n      },\n    });\n\n    // Extract priority from custom resource\n    this.priority = Token.asNumber(this.resource.getAtt('Priority'));\n\n    // Add CloudFormation outputs for debugging\n    new CfnOutput(this, 'ServiceIdentifier', {\n      value: this.serviceIdentifier,\n      description: 'Service identifier for priority allocation tracking',\n    });\n\n    new CfnOutput(this, 'AllocatedPriority', {\n      value: this.priority.toString(),\n      description: 'Auto-allocated priority for ALB listener rule',\n    });\n  }\n}\n"]}
@@ -138,8 +138,12 @@ export interface StandardApplicationFargateServiceProps extends StandardFargateS
138
138
  readonly listener?: IApplicationListener;
139
139
  /**
140
140
  * The priority to give the target group on the ALB.
141
+ * If not specified, a unique priority is automatically allocated using
142
+ * the PriorityAllocator, which coordinates with other services across
143
+ * multiple teams and tools (CDK, Terraform, manual) to find the lowest
144
+ * available priority.
141
145
  *
142
- * @default - 1
146
+ * @default - Automatically allocated (recommended for most use cases)
143
147
  */
144
148
  readonly targetGroupPriority?: number;
145
149
  /**
@@ -7,12 +7,13 @@ const aws_elasticloadbalancingv2_1 = require("aws-cdk-lib/aws-elasticloadbalanci
7
7
  const aws_route53_1 = require("aws-cdk-lib/aws-route53");
8
8
  const aws_route53_2 = require("../../aws-route53");
9
9
  const aws_route53_targets_1 = require("aws-cdk-lib/aws-route53-targets");
10
+ const priority_allocator_1 = require("./priority-allocator");
10
11
  /**
11
12
  * Creates an ECS Fargate service and maps it to an Application Load Balancer (ALB).
12
13
  */
13
14
  class StandardApplicationFargateService extends standard_fargate_service_1.StandardFargateService {
14
15
  constructor(scope, id, props) {
15
- var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r, _s;
16
+ var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r;
16
17
  super(scope, id, {
17
18
  ...props,
18
19
  healthCheckGracePeriod: (_a = props.healthCheckGracePeriod) !== null && _a !== void 0 ? _a : aws_cdk_lib_1.Duration.seconds(60),
@@ -87,10 +88,23 @@ class StandardApplicationFargateService extends standard_fargate_service_1.Stand
87
88
  loadBalancerArn: loadBalancer.loadBalancerArn,
88
89
  listenerProtocol: aws_elasticloadbalancingv2_1.ApplicationProtocol.HTTPS,
89
90
  });
91
+ // Determine priority: use explicit value, or allocate automatically
92
+ let priority;
93
+ if (props.targetGroupPriority !== undefined) {
94
+ // Manual priority specified - use it directly
95
+ priority = props.targetGroupPriority;
96
+ }
97
+ else {
98
+ // No priority specified - use automatic allocation
99
+ const allocator = new priority_allocator_1.PriorityAllocator(this, 'PriorityAllocator', {
100
+ listenerArn: listener.listenerArn,
101
+ });
102
+ priority = allocator.priority;
103
+ }
90
104
  listener.addTargetGroups(`${id}TargetGroups`, {
91
105
  targetGroups: [targetGroup],
92
106
  conditions: targetGroupConditions,
93
- priority: (_s = props.targetGroupPriority) !== null && _s !== void 0 ? _s : 1,
107
+ priority,
94
108
  });
95
109
  if (props.domainName !== undefined &&
96
110
  props.domainZone !== undefined &&
@@ -108,4 +122,4 @@ class StandardApplicationFargateService extends standard_fargate_service_1.Stand
108
122
  }
109
123
  }
110
124
  exports.StandardApplicationFargateService = StandardApplicationFargateService;
111
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"standard-application-fargate-service.js","sourceRoot":"","sources":["standard-application-fargate-service.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  StandardFargateService,\n  StandardFargateServiceProps,\n} from './standard-fargate-service';\nimport {Construct} from 'constructs';\nimport {Duration} from 'aws-cdk-lib';\nimport {\n  ApplicationListener,\n  ApplicationLoadBalancer,\n  ApplicationProtocol,\n  ApplicationTargetGroup,\n  IApplicationListener,\n  IApplicationLoadBalancer,\n  IApplicationTargetGroup,\n  ListenerCondition,\n  TargetGroupLoadBalancingAlgorithmType,\n} from 'aws-cdk-lib/aws-elasticloadbalancingv2';\nimport {ARecord, IHostedZone, RecordTarget} from 'aws-cdk-lib/aws-route53';\nimport {DomainName} from '../../aws-route53';\nimport {LoadBalancerTarget} from 'aws-cdk-lib/aws-route53-targets';\nimport {MetricOptions} from 'aws-cdk-lib/aws-cloudwatch';\n\n/**\n * Properties for StandardApplicationFargateService\n */\nexport interface StandardApplicationFargateServiceProps extends StandardFargateServiceProps {\n  /**\n   * The name of an application-based stickiness cookie.\n   *\n   * @default - lb_affinity\n   */\n  readonly stickinessCookieName?: string;\n\n  /**\n   * The stickiness cookie expiration period. Set to 0 to disable.\n   *\n   * @default - Duration.days(1)\n   */\n  readonly stickinessCookieDuration?: Duration;\n\n  /**\n   * The time period during which the load balancer sends a newly registered target a\n   * linearly increasing share of the traffic to the target group. Set this to 0\n   * to disable.\n   *\n   * @default - Duration.seconds(30)\n   */\n  readonly slowStart?: Duration;\n\n  /**\n   * The protocol used by the application in the container.\n   *\n   * @default - ApplicationProtocol.HTTP\n   */\n  readonly applicationProtocol?: ApplicationProtocol;\n\n  /**\n   * The amount of time for Elastic Load Balancing to wait before deregistering a target.\n   *\n   * @default - Duration.seconds(10)\n   */\n  readonly deregistrationDelay?: Duration;\n\n  /**\n   * The approximate number of seconds between health checks for an individual target.\n   *\n   * @default - Duration.seconds(10)\n   */\n  readonly healthCheckInterval?: Duration;\n\n  /**\n   * The period of time, in seconds, that the Amazon ECS service scheduler ignores unhealthy\n   * Elastic Load Balancing target health checks after a task has first started.\n   *\n   * @default - defaults to 60 seconds\n   */\n  readonly healthCheckGracePeriod?: Duration;\n\n  /**\n   * The ping path destination where Elastic Load Balancing sends health check requests.\n   *\n   * @default - /health\n   */\n  readonly healthCheckPath?: string;\n\n  /**\n   * The amount of time, in seconds, during which no response from a target means a failed health check\n   *\n   * @default - Duration.seconds(3)\n   */\n  readonly healthCheckTimeout?: Duration;\n\n  /**\n   * The number of consecutive health checks successes required before considering an unhealthy target healthy.\n   *\n   * @default - 2\n   */\n  readonly healthyThresholdCount?: number;\n\n  /**\n   * The number of consecutive health check failures required before considering a target unhealthy.\n   *\n   * @default - 2\n   */\n  readonly unhealthyThresholdCount?: number;\n\n  /**\n   * HTTP code to use when checking for a successful response from a target\n   *\n   * @default - 200-299\n   */\n  readonly healthyHttpCodes?: string;\n\n  /**\n   * The load balancing algorithm to select targets for routing requests.\n   * To set this to LEAST_OUTSTANDING_REQUESTS, stickiness must be disabled.\n   *\n   * @default - ROUND_ROBIN\n   */\n  readonly loadBalancingAlgorithmType?: TargetGroupLoadBalancingAlgorithmType;\n\n  /**\n   * The number of ALB requests to target for scaling.\n   * Disabled by default.\n   */\n  readonly scaleRequestPerTarget?: number;\n\n  /**\n   * Target response time for scaling\n   * Disabled by default\n   */\n  readonly scaleOnTargetResponseTime?: {\n    /** Threshold in seconds */\n    threshold: number;\n    metricOptions?: MetricOptions;\n  };\n\n  /**\n   * Domain name associated with this service.\n   */\n  readonly domainName?: string;\n\n  /**\n   * Additional domain names to associate with this service.\n   */\n  readonly domainNames?: string[];\n\n  /**\n   * Set this to true to skip the creation of route53 records. By default records will be created in domainName and domainZone is provided.\n   *\n   * @default - false\n   */\n  readonly skipCreateRoute53Records?: boolean;\n\n  /**\n   * Path pattern to match on the load balancer.\n   *\n   * @default - [\"/*\"]\n   */\n  readonly pathPattern?: string[];\n\n  /**\n   * Load balancer to attach this service to. If passed an ARN or name a lookup will be\n   * performed to locate the load balancer.\n   */\n  readonly loadBalancer: IApplicationLoadBalancer | string;\n\n  /**\n   * The listener to attach this service to. If one is not provided an HTTPS listener is obtained from a lookup.\n   *\n   * @default - ApplicationProtocol.HTTPS\n   */\n  readonly listener?: IApplicationListener;\n\n  /**\n   * The priority to give the target group on the ALB.\n   *\n   * @default - 1\n   */\n  readonly targetGroupPriority?: number;\n\n  /**\n   * Zone of the domain name. If set, a route53 record is created for the service.\n   *\n   */\n  readonly domainZone?: IHostedZone;\n}\n\n/**\n * Creates an ECS Fargate service and maps it to an Application Load Balancer (ALB).\n */\nexport class StandardApplicationFargateService extends StandardFargateService {\n  readonly loadBalancer: IApplicationLoadBalancer;\n  readonly listener: IApplicationListener;\n  readonly domainName?: DomainName;\n  readonly route53Record?: ARecord;\n  readonly targetGroup: IApplicationTargetGroup;\n\n  constructor(\n    scope: Construct,\n    id: string,\n    props: StandardApplicationFargateServiceProps,\n  ) {\n    super(scope, id, {\n      ...props,\n      healthCheckGracePeriod:\n        props.healthCheckGracePeriod ?? Duration.seconds(60),\n    });\n\n    let stickinessCookieDuration: Duration | undefined =\n      props.stickinessCookieDuration ?? Duration.days(1);\n    if (stickinessCookieDuration.toSeconds() === 0) {\n      stickinessCookieDuration = undefined;\n    }\n\n    let stickinessCookieName: string | undefined;\n    if (stickinessCookieDuration !== undefined) {\n      stickinessCookieName = props.stickinessCookieName ?? 'lb_affinity';\n    }\n\n    let slowStart: Duration | undefined =\n      props.slowStart ?? Duration.seconds(30);\n    if (slowStart.toSeconds() === 0) {\n      slowStart = undefined;\n    }\n\n    const targetGroup = new ApplicationTargetGroup(this, 'TargetGroup', {\n      targets: [this.service],\n      vpc: props.cluster.vpc,\n      port: this.port,\n      protocol: props.applicationProtocol ?? ApplicationProtocol.HTTP,\n      deregistrationDelay: props.deregistrationDelay ?? Duration.seconds(10),\n      slowStart,\n      healthCheck: {\n        enabled: true,\n        interval: props.healthCheckInterval ?? Duration.seconds(10),\n        path: props.healthCheckPath ?? '/health',\n        timeout: props.healthCheckTimeout ?? Duration.seconds(3),\n        healthyThresholdCount: props.healthyThresholdCount ?? 2,\n        unhealthyThresholdCount: props.unhealthyThresholdCount ?? 2,\n        healthyHttpCodes: props.healthyHttpCodes ?? '200-299',\n      },\n      stickinessCookieName,\n      stickinessCookieDuration,\n      loadBalancingAlgorithmType:\n        props.loadBalancingAlgorithmType ??\n        TargetGroupLoadBalancingAlgorithmType.ROUND_ROBIN,\n    });\n\n    if (props.scaleRequestPerTarget !== undefined) {\n      this.scaling.scaleOnRequestCount('RequestCountScaling', {\n        scaleInCooldown: this.scaleInCooldown,\n        scaleOutCooldown: this.scaleOutCooldown,\n        targetGroup,\n        requestsPerTarget: props.scaleRequestPerTarget,\n      });\n    }\n\n    const targetGroupConditions: ListenerCondition[] = [];\n    targetGroupConditions.push(\n      ListenerCondition.pathPatterns(props.pathPattern ?? ['/*']),\n    );\n    if (props.domainName !== undefined) {\n      targetGroupConditions.push(\n        ListenerCondition.hostHeaders([\n          props.domainName,\n          ...(props.domainNames ?? []),\n        ]),\n      );\n    }\n\n    let loadBalancer: IApplicationLoadBalancer;\n    if (typeof props.loadBalancer === 'string') {\n      if (props.loadBalancer.startsWith('arn:')) {\n        loadBalancer = ApplicationLoadBalancer.fromLookup(\n          this,\n          'LoadBalancer',\n          {\n            loadBalancerArn: props.loadBalancer,\n          },\n        );\n      } else {\n        loadBalancer = ApplicationLoadBalancer.fromLookup(\n          this,\n          'LoadBalancer',\n          {\n            loadBalancerTags: {\n              Name: props.loadBalancer,\n            },\n          },\n        );\n      }\n    } else {\n      loadBalancer = props.loadBalancer;\n    }\n\n    const listener =\n      props.listener ??\n      ApplicationListener.fromLookup(this, 'Listener', {\n        loadBalancerArn: loadBalancer.loadBalancerArn,\n        listenerProtocol: ApplicationProtocol.HTTPS,\n      });\n\n    listener.addTargetGroups(`${id}TargetGroups`, {\n      targetGroups: [targetGroup],\n      conditions: targetGroupConditions,\n      priority: props.targetGroupPriority ?? 1,\n    });\n\n    if (\n      props.domainName !== undefined &&\n      props.domainZone !== undefined &&\n      !props.skipCreateRoute53Records\n    ) {\n      this.domainName = DomainName.fromFqdn(props.domainName, props.domainZone);\n      this.route53Record = this.domainName.createARecord(\n        this,\n        RecordTarget.fromAlias(new LoadBalancerTarget(loadBalancer)),\n      );\n    }\n\n    this.loadBalancer = loadBalancer;\n    this.listener = listener;\n    this.targetGroup = targetGroup;\n\n    if (props.scaleOnTargetResponseTime) {\n      // Attach scaling policy to the service\n      this.scaleToTrackCustomMetric(\n        'TargetResponseTimeScaling',\n        targetGroup.metrics.targetResponseTime(\n          props.scaleOnTargetResponseTime.metricOptions,\n        ),\n        props.scaleOnTargetResponseTime.threshold,\n      );\n    }\n  }\n}\n"]}
125
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"standard-application-fargate-service.js","sourceRoot":"","sources":["standard-application-fargate-service.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  StandardFargateService,\n  StandardFargateServiceProps,\n} from './standard-fargate-service';\nimport {Construct} from 'constructs';\nimport {Duration} from 'aws-cdk-lib';\nimport {\n  ApplicationListener,\n  ApplicationLoadBalancer,\n  ApplicationProtocol,\n  ApplicationTargetGroup,\n  IApplicationListener,\n  IApplicationLoadBalancer,\n  IApplicationTargetGroup,\n  ListenerCondition,\n  TargetGroupLoadBalancingAlgorithmType,\n} from 'aws-cdk-lib/aws-elasticloadbalancingv2';\nimport {ARecord, IHostedZone, RecordTarget} from 'aws-cdk-lib/aws-route53';\nimport {DomainName} from '../../aws-route53';\nimport {LoadBalancerTarget} from 'aws-cdk-lib/aws-route53-targets';\nimport {MetricOptions} from 'aws-cdk-lib/aws-cloudwatch';\nimport {PriorityAllocator} from './priority-allocator';\n\n/**\n * Properties for StandardApplicationFargateService\n */\nexport interface StandardApplicationFargateServiceProps extends StandardFargateServiceProps {\n  /**\n   * The name of an application-based stickiness cookie.\n   *\n   * @default - lb_affinity\n   */\n  readonly stickinessCookieName?: string;\n\n  /**\n   * The stickiness cookie expiration period. Set to 0 to disable.\n   *\n   * @default - Duration.days(1)\n   */\n  readonly stickinessCookieDuration?: Duration;\n\n  /**\n   * The time period during which the load balancer sends a newly registered target a\n   * linearly increasing share of the traffic to the target group. Set this to 0\n   * to disable.\n   *\n   * @default - Duration.seconds(30)\n   */\n  readonly slowStart?: Duration;\n\n  /**\n   * The protocol used by the application in the container.\n   *\n   * @default - ApplicationProtocol.HTTP\n   */\n  readonly applicationProtocol?: ApplicationProtocol;\n\n  /**\n   * The amount of time for Elastic Load Balancing to wait before deregistering a target.\n   *\n   * @default - Duration.seconds(10)\n   */\n  readonly deregistrationDelay?: Duration;\n\n  /**\n   * The approximate number of seconds between health checks for an individual target.\n   *\n   * @default - Duration.seconds(10)\n   */\n  readonly healthCheckInterval?: Duration;\n\n  /**\n   * The period of time, in seconds, that the Amazon ECS service scheduler ignores unhealthy\n   * Elastic Load Balancing target health checks after a task has first started.\n   *\n   * @default - defaults to 60 seconds\n   */\n  readonly healthCheckGracePeriod?: Duration;\n\n  /**\n   * The ping path destination where Elastic Load Balancing sends health check requests.\n   *\n   * @default - /health\n   */\n  readonly healthCheckPath?: string;\n\n  /**\n   * The amount of time, in seconds, during which no response from a target means a failed health check\n   *\n   * @default - Duration.seconds(3)\n   */\n  readonly healthCheckTimeout?: Duration;\n\n  /**\n   * The number of consecutive health checks successes required before considering an unhealthy target healthy.\n   *\n   * @default - 2\n   */\n  readonly healthyThresholdCount?: number;\n\n  /**\n   * The number of consecutive health check failures required before considering a target unhealthy.\n   *\n   * @default - 2\n   */\n  readonly unhealthyThresholdCount?: number;\n\n  /**\n   * HTTP code to use when checking for a successful response from a target\n   *\n   * @default - 200-299\n   */\n  readonly healthyHttpCodes?: string;\n\n  /**\n   * The load balancing algorithm to select targets for routing requests.\n   * To set this to LEAST_OUTSTANDING_REQUESTS, stickiness must be disabled.\n   *\n   * @default - ROUND_ROBIN\n   */\n  readonly loadBalancingAlgorithmType?: TargetGroupLoadBalancingAlgorithmType;\n\n  /**\n   * The number of ALB requests to target for scaling.\n   * Disabled by default.\n   */\n  readonly scaleRequestPerTarget?: number;\n\n  /**\n   * Target response time for scaling\n   * Disabled by default\n   */\n  readonly scaleOnTargetResponseTime?: {\n    /** Threshold in seconds */\n    threshold: number;\n    metricOptions?: MetricOptions;\n  };\n\n  /**\n   * Domain name associated with this service.\n   */\n  readonly domainName?: string;\n\n  /**\n   * Additional domain names to associate with this service.\n   */\n  readonly domainNames?: string[];\n\n  /**\n   * Set this to true to skip the creation of route53 records. By default records will be created in domainName and domainZone is provided.\n   *\n   * @default - false\n   */\n  readonly skipCreateRoute53Records?: boolean;\n\n  /**\n   * Path pattern to match on the load balancer.\n   *\n   * @default - [\"/*\"]\n   */\n  readonly pathPattern?: string[];\n\n  /**\n   * Load balancer to attach this service to. If passed an ARN or name a lookup will be\n   * performed to locate the load balancer.\n   */\n  readonly loadBalancer: IApplicationLoadBalancer | string;\n\n  /**\n   * The listener to attach this service to. If one is not provided an HTTPS listener is obtained from a lookup.\n   *\n   * @default - ApplicationProtocol.HTTPS\n   */\n  readonly listener?: IApplicationListener;\n\n  /**\n   * The priority to give the target group on the ALB.\n   * If not specified, a unique priority is automatically allocated using\n   * the PriorityAllocator, which coordinates with other services across\n   * multiple teams and tools (CDK, Terraform, manual) to find the lowest\n   * available priority.\n   *\n   * @default - Automatically allocated (recommended for most use cases)\n   */\n  readonly targetGroupPriority?: number;\n\n  /**\n   * Zone of the domain name. If set, a route53 record is created for the service.\n   *\n   */\n  readonly domainZone?: IHostedZone;\n}\n\n/**\n * Creates an ECS Fargate service and maps it to an Application Load Balancer (ALB).\n */\nexport class StandardApplicationFargateService extends StandardFargateService {\n  readonly loadBalancer: IApplicationLoadBalancer;\n  readonly listener: IApplicationListener;\n  readonly domainName?: DomainName;\n  readonly route53Record?: ARecord;\n  readonly targetGroup: IApplicationTargetGroup;\n\n  constructor(\n    scope: Construct,\n    id: string,\n    props: StandardApplicationFargateServiceProps,\n  ) {\n    super(scope, id, {\n      ...props,\n      healthCheckGracePeriod:\n        props.healthCheckGracePeriod ?? Duration.seconds(60),\n    });\n\n    let stickinessCookieDuration: Duration | undefined =\n      props.stickinessCookieDuration ?? Duration.days(1);\n    if (stickinessCookieDuration.toSeconds() === 0) {\n      stickinessCookieDuration = undefined;\n    }\n\n    let stickinessCookieName: string | undefined;\n    if (stickinessCookieDuration !== undefined) {\n      stickinessCookieName = props.stickinessCookieName ?? 'lb_affinity';\n    }\n\n    let slowStart: Duration | undefined =\n      props.slowStart ?? Duration.seconds(30);\n    if (slowStart.toSeconds() === 0) {\n      slowStart = undefined;\n    }\n\n    const targetGroup = new ApplicationTargetGroup(this, 'TargetGroup', {\n      targets: [this.service],\n      vpc: props.cluster.vpc,\n      port: this.port,\n      protocol: props.applicationProtocol ?? ApplicationProtocol.HTTP,\n      deregistrationDelay: props.deregistrationDelay ?? Duration.seconds(10),\n      slowStart,\n      healthCheck: {\n        enabled: true,\n        interval: props.healthCheckInterval ?? Duration.seconds(10),\n        path: props.healthCheckPath ?? '/health',\n        timeout: props.healthCheckTimeout ?? Duration.seconds(3),\n        healthyThresholdCount: props.healthyThresholdCount ?? 2,\n        unhealthyThresholdCount: props.unhealthyThresholdCount ?? 2,\n        healthyHttpCodes: props.healthyHttpCodes ?? '200-299',\n      },\n      stickinessCookieName,\n      stickinessCookieDuration,\n      loadBalancingAlgorithmType:\n        props.loadBalancingAlgorithmType ??\n        TargetGroupLoadBalancingAlgorithmType.ROUND_ROBIN,\n    });\n\n    if (props.scaleRequestPerTarget !== undefined) {\n      this.scaling.scaleOnRequestCount('RequestCountScaling', {\n        scaleInCooldown: this.scaleInCooldown,\n        scaleOutCooldown: this.scaleOutCooldown,\n        targetGroup,\n        requestsPerTarget: props.scaleRequestPerTarget,\n      });\n    }\n\n    const targetGroupConditions: ListenerCondition[] = [];\n    targetGroupConditions.push(\n      ListenerCondition.pathPatterns(props.pathPattern ?? ['/*']),\n    );\n    if (props.domainName !== undefined) {\n      targetGroupConditions.push(\n        ListenerCondition.hostHeaders([\n          props.domainName,\n          ...(props.domainNames ?? []),\n        ]),\n      );\n    }\n\n    let loadBalancer: IApplicationLoadBalancer;\n    if (typeof props.loadBalancer === 'string') {\n      if (props.loadBalancer.startsWith('arn:')) {\n        loadBalancer = ApplicationLoadBalancer.fromLookup(\n          this,\n          'LoadBalancer',\n          {\n            loadBalancerArn: props.loadBalancer,\n          },\n        );\n      } else {\n        loadBalancer = ApplicationLoadBalancer.fromLookup(\n          this,\n          'LoadBalancer',\n          {\n            loadBalancerTags: {\n              Name: props.loadBalancer,\n            },\n          },\n        );\n      }\n    } else {\n      loadBalancer = props.loadBalancer;\n    }\n\n    const listener =\n      props.listener ??\n      ApplicationListener.fromLookup(this, 'Listener', {\n        loadBalancerArn: loadBalancer.loadBalancerArn,\n        listenerProtocol: ApplicationProtocol.HTTPS,\n      });\n\n    // Determine priority: use explicit value, or allocate automatically\n    let priority: number;\n    if (props.targetGroupPriority !== undefined) {\n      // Manual priority specified - use it directly\n      priority = props.targetGroupPriority;\n    } else {\n      // No priority specified - use automatic allocation\n      const allocator = new PriorityAllocator(this, 'PriorityAllocator', {\n        listenerArn: listener.listenerArn,\n      });\n      priority = allocator.priority;\n    }\n\n    listener.addTargetGroups(`${id}TargetGroups`, {\n      targetGroups: [targetGroup],\n      conditions: targetGroupConditions,\n      priority,\n    });\n\n    if (\n      props.domainName !== undefined &&\n      props.domainZone !== undefined &&\n      !props.skipCreateRoute53Records\n    ) {\n      this.domainName = DomainName.fromFqdn(props.domainName, props.domainZone);\n      this.route53Record = this.domainName.createARecord(\n        this,\n        RecordTarget.fromAlias(new LoadBalancerTarget(loadBalancer)),\n      );\n    }\n\n    this.loadBalancer = loadBalancer;\n    this.listener = listener;\n    this.targetGroup = targetGroup;\n\n    if (props.scaleOnTargetResponseTime) {\n      // Attach scaling policy to the service\n      this.scaleToTrackCustomMetric(\n        'TargetResponseTimeScaling',\n        targetGroup.metrics.targetResponseTime(\n          props.scaleOnTargetResponseTime.metricOptions,\n        ),\n        props.scaleOnTargetResponseTime.threshold,\n      );\n    }\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "truemark-cdk-lib",
3
3
  "description": "AWS CDK constructs created by TrueMark",
4
- "version": "1.21.0",
4
+ "version": "1.21.1-alpha.1768569999",
5
5
  "main": "index.js",
6
6
  "types": "index.d.ts",
7
7
  "author": "TrueMark Technologies, Inc.",
@@ -22,8 +22,11 @@
22
22
  ],
23
23
  "devDependencies": {
24
24
  "@eslint/js": "^9.39.1",
25
+ "@smithy/types": "^4.9.0",
25
26
  "@types/jest": "^29.5.14",
26
27
  "@types/node": "^22.19.2",
28
+ "aws-sdk-client-mock": "^4.1.0",
29
+ "aws-sdk-client-mock-jest": "^4.1.0",
27
30
  "esbuild": "^0.27.1",
28
31
  "eslint": "^9.39.1",
29
32
  "jest": "^29.7.0",
@@ -33,7 +36,8 @@
33
36
  "typescript": "~5.9.3",
34
37
  "typescript-eslint": "^8.49.0",
35
38
  "@aws-sdk/util-dynamodb": "^3.947.0",
36
- "@aws-sdk/client-dynamodb": "^3.947.0"
39
+ "@aws-sdk/client-dynamodb": "^3.947.0",
40
+ "@aws-sdk/client-elastic-load-balancing-v2": "^3.947.0"
37
41
  },
38
42
  "dependencies": {
39
43
  "@aws-cdk/aws-lambda-go-alpha": "^2.232.1-alpha.0",