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.
- package/aws-ecs/lib/index.d.ts +1 -0
- package/aws-ecs/lib/index.js +2 -1
- package/aws-ecs/lib/priority-allocator-handler.d.ts +38 -0
- package/aws-ecs/lib/priority-allocator-handler.js +347 -0
- package/aws-ecs/lib/priority-allocator.d.ts +84 -0
- package/aws-ecs/lib/priority-allocator.js +225 -0
- package/aws-ecs/lib/standard-application-fargate-service.d.ts +5 -1
- package/aws-ecs/lib/standard-application-fargate-service.js +17 -3
- package/package.json +6 -2
package/aws-ecs/lib/index.d.ts
CHANGED
|
@@ -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';
|
package/aws-ecs/lib/index.js
CHANGED
|
@@ -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,
|
|
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":";;;AAAA,2CAAqC;AACrC,qEAA6D;AAC7D,6CAOqB;AACrB,mEAAsD;AACtD,uDAA+C;AAC/C,kCAAkC;AAClC,iDAK6B;AAC7B,2DAA2E;AAC3E,sCAAsC;AAiBtC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8BG;AACH,MAAa,iBAAkB,SAAQ,sBAAS;IAqB9C;;;OAGG;IACK,MAAM,CAAC,gBAAgB,CAAC,KAAgB;QAC9C,MAAM,KAAK,GAAG,mBAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC;QAC9B,MAAM,OAAO,GAAG,wBAAwB,CAAC;QAEzC,0CAA0C;QAC1C,MAAM,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,OAAO,CAAsB,CAAC;QACvE,IAAI,QAAQ,EAAE,CAAC;YACb,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,8CAA8C;QAC9C,MAAM,KAAK,GAAG,IAAI,oBAAK,CAAC,KAAK,EAAE,OAAO,EAAE;YACtC,SAAS,EAAE,iBAAiB,CAAC,UAAU;YACvC,YAAY,EAAE;gBACZ,IAAI,EAAE,aAAa;gBACnB,IAAI,EAAE,4BAAa,CAAC,MAAM;aAC3B;YACD,OAAO,EAAE;gBACP,IAAI,EAAE,UAAU;gBAChB,IAAI,EAAE,4BAAa,CAAC,MAAM;aAC3B;YACD,WAAW,EAAE,0BAAW,CAAC,eAAe;YACxC,aAAa,EAAE,2BAAa,CAAC,MAAM,EAAE,uCAAuC;YAC5E,gCAAgC,EAAE;gBAChC,0BAA0B,EAAE,IAAI;aACjC;SACF,CAAC,CAAC;QAEH,6CAA6C;QAC7C,KAAK,CAAC,uBAAuB,CAAC;YAC5B,SAAS,EAAE,wBAAwB;YACnC,YAAY,EAAE;gBACZ,IAAI,EAAE,mBAAmB;gBACzB,IAAI,EAAE,4BAAa,CAAC,MAAM;aAC3B;YACD,OAAO,EAAE;gBACP,IAAI,EAAE,aAAa;gBACnB,IAAI,EAAE,4BAAa,CAAC,MAAM;aAC3B;SACF,CAAC,CAAC;QAEH,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;OAGG;IACK,MAAM,CAAC,iBAAiB,CAC9B,KAAgB,EAChB,KAAY;QAEZ,MAAM,KAAK,GAAG,mBAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC;QAC9B,MAAM,QAAQ,GAAG,yBAAyB,CAAC;QAE3C,2CAA2C;QAC3C,MAAM,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAEpC,CAAC;QACd,IAAI,QAAQ,EAAE,CAAC;YACb,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,6BAA6B;QAC7B,MAAM,IAAI,GAAG,IAAI,cAAI,CAAC,KAAK,EAAE,6BAA6B,EAAE;YAC1D,SAAS,EAAE,IAAI,0BAAgB,CAAC,sBAAsB,CAAC;YACvD,WAAW,EAAE,iDAAiD;SAC/D,CAAC,CAAC;QAEH,8BAA8B;QAC9B,IAAI,CAAC,WAAW,CACd,IAAI,yBAAe,CAAC;YAClB,MAAM,EAAE,gBAAM,CAAC,KAAK;YACpB,OAAO,EAAE;gBACP,qBAAqB;gBACrB,sBAAsB;gBACtB,mBAAmB;aACpB;YACD,SAAS,EAAE,CAAC,GAAG,CAAC;SACjB,CAAC,CACH,CAAC;QAEF,uBAAuB;QACvB,IAAI,CAAC,WAAW,CACd,IAAI,yBAAe,CAAC;YAClB,MAAM,EAAE,gBAAM,CAAC,KAAK;YACpB,OAAO,EAAE;gBACP,wCAAwC;gBACxC,oCAAoC;aACrC;YACD,SAAS,EAAE,CAAC,GAAG,CAAC;SACjB,CAAC,CACH,CAAC;QAEF,uBAAuB;QACvB,IAAI,CAAC,WAAW,CACd,IAAI,yBAAe,CAAC;YAClB,MAAM,EAAE,gBAAM,CAAC,KAAK;YACpB,OAAO,EAAE;gBACP,gBAAgB;gBAChB,kBAAkB;gBAClB,kBAAkB;gBAClB,qBAAqB;aACtB;YACD,SAAS,EAAE,CAAC,KAAK,CAAC,QAAQ,EAAE,GAAG,KAAK,CAAC,QAAQ,UAAU,CAAC;SACzD,CAAC,CACH,CAAC;QAEF,yBAAyB;QACzB,OAAO,IAAI,kCAAc,CAAC,KAAK,EAAE,QAAQ,EAAE;YACzC,IAAI;YACJ,OAAO,EAAE,oBAAO,CAAC,WAAW;YAC5B,OAAO,EAAE,SAAS;YAClB,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,+BAA+B,CAAC;YAC5D,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YAC7B,UAAU,EAAE,GAAG;YACf,WAAW,EAAE,oDAAoD;YACjE,YAAY,EAAE,8BAA8B;SAC7C,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACK,MAAM,CAAC,mBAAmB,CAChC,KAAgB,EAChB,MAAsB;QAEtB,MAAM,KAAK,GAAG,mBAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC;QAC9B,MAAM,UAAU,GAAG,2BAA2B,CAAC;QAE/C,6CAA6C;QAC7C,MAAM,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,UAAU,CAEtC,CAAC;QACd,IAAI,QAAQ,EAAE,CAAC;YACb,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,iDAAiD;QACjD,OAAO,IAAI,2BAAQ,CAAC,KAAK,EAAE,UAAU,EAAE;YACrC,cAAc,EAAE,MAAM;SACvB,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,yBAAyB,CAAC,WAAmB;QACnD,MAAM,KAAK,GAAG,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;QAC7B,MAAM,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC;QAC5B,MAAM,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC;QAC9B,MAAM,SAAS,GAAG,KAAK,CAAC,SAAS,CAAC;QAClC,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC;QAErC,4BAA4B;QAC5B,MAAM,KAAK,GAAG,GAAG,OAAO,IAAI,MAAM,IAAI,SAAS,IAAI,aAAa,IAAI,WAAW,EAAE,CAAC;QAClF,MAAM,IAAI,GAAG,MAAM;aAChB,UAAU,CAAC,QAAQ,CAAC;aACpB,MAAM,CAAC,KAAK,CAAC;aACb,MAAM,CAAC,KAAK,CAAC;aACb,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAEpB,4DAA4D;QAC5D,MAAM,kBAAkB,GAAG,SAAS;aACjC,UAAU,CAAC,gBAAgB,EAAE,GAAG,CAAC;aACjC,WAAW,EAAE,CAAC;QACjB,OAAO,GAAG,kBAAkB,IAAI,IAAI,EAAE,CAAC;IACzC,CAAC;IAED,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA6B;;QACrE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,oCAAoC;QACpC,MAAM,KAAK,GAAG,iBAAiB,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC;QACvD,MAAM,MAAM,GAAG,iBAAiB,CAAC,iBAAiB,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;QAChE,MAAM,QAAQ,GAAG,iBAAiB,CAAC,mBAAmB,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;QAErE,8BAA8B;QAC9B,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,yBAAyB,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QAE3E,mDAAmD;QACnD,IAAI,CAAC,QAAQ,GAAG,IAAI,4BAAc,CAAC,IAAI,EAAE,UAAU,EAAE;YACnD,YAAY,EAAE,QAAQ,CAAC,YAAY;YACnC,UAAU,EAAE;gBACV,WAAW,EAAE,KAAK,CAAC,WAAW;gBAC9B,iBAAiB,EAAE,IAAI,CAAC,iBAAiB;gBACzC,SAAS,EAAE,iBAAiB,CAAC,UAAU;gBACvC,iBAAiB,EAAE,MAAA,KAAK,CAAC,iBAAiB,0CAAE,QAAQ,EAAE;gBACtD,qDAAqD;gBACrD,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE;aACjC;SACF,CAAC,CAAC;QAEH,wCAAwC;QACxC,IAAI,CAAC,QAAQ,GAAG,mBAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC;QAEjE,2CAA2C;QAC3C,IAAI,uBAAS,CAAC,IAAI,EAAE,mBAAmB,EAAE;YACvC,KAAK,EAAE,IAAI,CAAC,iBAAiB;YAC7B,WAAW,EAAE,qDAAqD;SACnE,CAAC,CAAC;QAEH,IAAI,uBAAS,CAAC,IAAI,EAAE,mBAAmB,EAAE;YACvC,KAAK,EAAE,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE;YAC/B,WAAW,EAAE,+CAA+C;SAC7D,CAAC,CAAC;IACL,CAAC;;AAzOH,8CA0OC;AA1NC;;GAEG;AACqB,4BAAU,GAAG,yBAAyB,CAAC","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 -
|
|
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
|
|
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
|
|
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":";;;AAAA,yEAGoC;AAEpC,6CAAqC;AACrC,uFAUgD;AAChD,yDAA2E;AAC3E,mDAA6C;AAC7C,yEAAmE;AAEnE,6DAAuD;AA4KvD;;GAEG;AACH,MAAa,iCAAkC,SAAQ,iDAAsB;IAO3E,YACE,KAAgB,EAChB,EAAU,EACV,KAA6C;;QAE7C,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE;YACf,GAAG,KAAK;YACR,sBAAsB,EACpB,MAAA,KAAK,CAAC,sBAAsB,mCAAI,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;SACvD,CAAC,CAAC;QAEH,IAAI,wBAAwB,GAC1B,MAAA,KAAK,CAAC,wBAAwB,mCAAI,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACrD,IAAI,wBAAwB,CAAC,SAAS,EAAE,KAAK,CAAC,EAAE,CAAC;YAC/C,wBAAwB,GAAG,SAAS,CAAC;QACvC,CAAC;QAED,IAAI,oBAAwC,CAAC;QAC7C,IAAI,wBAAwB,KAAK,SAAS,EAAE,CAAC;YAC3C,oBAAoB,GAAG,MAAA,KAAK,CAAC,oBAAoB,mCAAI,aAAa,CAAC;QACrE,CAAC;QAED,IAAI,SAAS,GACX,MAAA,KAAK,CAAC,SAAS,mCAAI,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QAC1C,IAAI,SAAS,CAAC,SAAS,EAAE,KAAK,CAAC,EAAE,CAAC;YAChC,SAAS,GAAG,SAAS,CAAC;QACxB,CAAC;QAED,MAAM,WAAW,GAAG,IAAI,mDAAsB,CAAC,IAAI,EAAE,aAAa,EAAE;YAClE,OAAO,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC;YACvB,GAAG,EAAE,KAAK,CAAC,OAAO,CAAC,GAAG;YACtB,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,QAAQ,EAAE,MAAA,KAAK,CAAC,mBAAmB,mCAAI,gDAAmB,CAAC,IAAI;YAC/D,mBAAmB,EAAE,MAAA,KAAK,CAAC,mBAAmB,mCAAI,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACtE,SAAS;YACT,WAAW,EAAE;gBACX,OAAO,EAAE,IAAI;gBACb,QAAQ,EAAE,MAAA,KAAK,CAAC,mBAAmB,mCAAI,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;gBAC3D,IAAI,EAAE,MAAA,KAAK,CAAC,eAAe,mCAAI,SAAS;gBACxC,OAAO,EAAE,MAAA,KAAK,CAAC,kBAAkB,mCAAI,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;gBACxD,qBAAqB,EAAE,MAAA,KAAK,CAAC,qBAAqB,mCAAI,CAAC;gBACvD,uBAAuB,EAAE,MAAA,KAAK,CAAC,uBAAuB,mCAAI,CAAC;gBAC3D,gBAAgB,EAAE,MAAA,KAAK,CAAC,gBAAgB,mCAAI,SAAS;aACtD;YACD,oBAAoB;YACpB,wBAAwB;YACxB,0BAA0B,EACxB,MAAA,KAAK,CAAC,0BAA0B,mCAChC,kEAAqC,CAAC,WAAW;SACpD,CAAC,CAAC;QAEH,IAAI,KAAK,CAAC,qBAAqB,KAAK,SAAS,EAAE,CAAC;YAC9C,IAAI,CAAC,OAAO,CAAC,mBAAmB,CAAC,qBAAqB,EAAE;gBACtD,eAAe,EAAE,IAAI,CAAC,eAAe;gBACrC,gBAAgB,EAAE,IAAI,CAAC,gBAAgB;gBACvC,WAAW;gBACX,iBAAiB,EAAE,KAAK,CAAC,qBAAqB;aAC/C,CAAC,CAAC;QACL,CAAC;QAED,MAAM,qBAAqB,GAAwB,EAAE,CAAC;QACtD,qBAAqB,CAAC,IAAI,CACxB,8CAAiB,CAAC,YAAY,CAAC,MAAA,KAAK,CAAC,WAAW,mCAAI,CAAC,IAAI,CAAC,CAAC,CAC5D,CAAC;QACF,IAAI,KAAK,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;YACnC,qBAAqB,CAAC,IAAI,CACxB,8CAAiB,CAAC,WAAW,CAAC;gBAC5B,KAAK,CAAC,UAAU;gBAChB,GAAG,CAAC,MAAA,KAAK,CAAC,WAAW,mCAAI,EAAE,CAAC;aAC7B,CAAC,CACH,CAAC;QACJ,CAAC;QAED,IAAI,YAAsC,CAAC;QAC3C,IAAI,OAAO,KAAK,CAAC,YAAY,KAAK,QAAQ,EAAE,CAAC;YAC3C,IAAI,KAAK,CAAC,YAAY,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;gBAC1C,YAAY,GAAG,oDAAuB,CAAC,UAAU,CAC/C,IAAI,EACJ,cAAc,EACd;oBACE,eAAe,EAAE,KAAK,CAAC,YAAY;iBACpC,CACF,CAAC;YACJ,CAAC;iBAAM,CAAC;gBACN,YAAY,GAAG,oDAAuB,CAAC,UAAU,CAC/C,IAAI,EACJ,cAAc,EACd;oBACE,gBAAgB,EAAE;wBAChB,IAAI,EAAE,KAAK,CAAC,YAAY;qBACzB;iBACF,CACF,CAAC;YACJ,CAAC;QACH,CAAC;aAAM,CAAC;YACN,YAAY,GAAG,KAAK,CAAC,YAAY,CAAC;QACpC,CAAC;QAED,MAAM,QAAQ,GACZ,MAAA,KAAK,CAAC,QAAQ,mCACd,gDAAmB,CAAC,UAAU,CAAC,IAAI,EAAE,UAAU,EAAE;YAC/C,eAAe,EAAE,YAAY,CAAC,eAAe;YAC7C,gBAAgB,EAAE,gDAAmB,CAAC,KAAK;SAC5C,CAAC,CAAC;QAEL,oEAAoE;QACpE,IAAI,QAAgB,CAAC;QACrB,IAAI,KAAK,CAAC,mBAAmB,KAAK,SAAS,EAAE,CAAC;YAC5C,8CAA8C;YAC9C,QAAQ,GAAG,KAAK,CAAC,mBAAmB,CAAC;QACvC,CAAC;aAAM,CAAC;YACN,mDAAmD;YACnD,MAAM,SAAS,GAAG,IAAI,sCAAiB,CAAC,IAAI,EAAE,mBAAmB,EAAE;gBACjE,WAAW,EAAE,QAAQ,CAAC,WAAW;aAClC,CAAC,CAAC;YACH,QAAQ,GAAG,SAAS,CAAC,QAAQ,CAAC;QAChC,CAAC;QAED,QAAQ,CAAC,eAAe,CAAC,GAAG,EAAE,cAAc,EAAE;YAC5C,YAAY,EAAE,CAAC,WAAW,CAAC;YAC3B,UAAU,EAAE,qBAAqB;YACjC,QAAQ;SACT,CAAC,CAAC;QAEH,IACE,KAAK,CAAC,UAAU,KAAK,SAAS;YAC9B,KAAK,CAAC,UAAU,KAAK,SAAS;YAC9B,CAAC,KAAK,CAAC,wBAAwB,EAC/B,CAAC;YACD,IAAI,CAAC,UAAU,GAAG,wBAAU,CAAC,QAAQ,CAAC,KAAK,CAAC,UAAU,EAAE,KAAK,CAAC,UAAU,CAAC,CAAC;YAC1E,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAChD,IAAI,EACJ,0BAAY,CAAC,SAAS,CAAC,IAAI,wCAAkB,CAAC,YAAY,CAAC,CAAC,CAC7D,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC;QACjC,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC;QAE/B,IAAI,KAAK,CAAC,yBAAyB,EAAE,CAAC;YACpC,uCAAuC;YACvC,IAAI,CAAC,wBAAwB,CAC3B,2BAA2B,EAC3B,WAAW,CAAC,OAAO,CAAC,kBAAkB,CACpC,KAAK,CAAC,yBAAyB,CAAC,aAAa,CAC9C,EACD,KAAK,CAAC,yBAAyB,CAAC,SAAS,CAC1C,CAAC;QACJ,CAAC;IACH,CAAC;CACF;AA9JD,8EA8JC","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.
|
|
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",
|