@liflig/cdk 3.18.0 → 3.19.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +13 -12
- package/assets/pipeline-slack-notification-lambda/index.py +9 -0
- package/lib/cdk-pipelines/index.d.ts +1 -1
- package/lib/cdk-pipelines/index.js +2 -2
- package/lib/cdk-pipelines/slack-notification.d.ts +42 -0
- package/lib/cdk-pipelines/slack-notification.js +73 -1
- package/package.json +23 -23
package/README.md
CHANGED
|
@@ -13,22 +13,23 @@ are not yet resolved. Some relevant information:
|
|
|
13
13
|
|
|
14
14
|
- <https://github.com/aws/aws-cdk-rfcs/blob/master/text/0006-monolothic-packaging.md>
|
|
15
15
|
|
|
16
|
-
##
|
|
16
|
+
## Development
|
|
17
17
|
|
|
18
|
-
|
|
18
|
+
Project commands are defined using `Make`. Examples:
|
|
19
19
|
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
20
|
+
```sh
|
|
21
|
+
# Primary commands
|
|
22
|
+
$ make # runs '$ make build'
|
|
23
|
+
$ make build # build project, apply lint and formatting fixes, update snapshots
|
|
24
|
+
$ make verify # verify project, ensure lint, formatting and snapshots are up-to-date
|
|
25
25
|
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
26
|
+
# Misc commands
|
|
27
|
+
$ make lint # lint code
|
|
28
|
+
$ make fmt # reformat code
|
|
29
|
+
$ make snapshots # regenerate snapshots
|
|
30
|
+
```
|
|
30
31
|
|
|
31
|
-
|
|
32
|
+
For a complete list of commands, refer to the `Makefile`.
|
|
32
33
|
|
|
33
34
|
## Testing library changes before releasing
|
|
34
35
|
|
|
@@ -14,6 +14,7 @@ secrets_manager = boto3.client("secretsmanager")
|
|
|
14
14
|
ACCOUNT_FRIENDLY_NAME = os.getenv("ACCOUNT_FRIENDLY_NAME", None)
|
|
15
15
|
SLACK_URL_SECRET_NAME = os.getenv("SLACK_URL_SECRET_NAME", None)
|
|
16
16
|
NOTIFICATION_LEVEL = os.getenv("NOTIFICATION_LEVEL", "WARN")
|
|
17
|
+
SLACK_MENTIONS = os.getenv("SLACK_MENTIONS", None)
|
|
17
18
|
|
|
18
19
|
# Example event:
|
|
19
20
|
#
|
|
@@ -260,6 +261,9 @@ def handler(event, context):
|
|
|
260
261
|
for s in [f"*Execution:* <{execution_url}|{execution_id}>", text_for_failed]
|
|
261
262
|
if s
|
|
262
263
|
)
|
|
264
|
+
|
|
265
|
+
mentions_str = SLACK_MENTIONS if state == "FAILED" and not previous_failed else ""
|
|
266
|
+
|
|
263
267
|
pretext = " ".join(
|
|
264
268
|
s
|
|
265
269
|
for s in [
|
|
@@ -269,6 +273,11 @@ def handler(event, context):
|
|
|
269
273
|
]
|
|
270
274
|
if s
|
|
271
275
|
)
|
|
276
|
+
|
|
277
|
+
# Add mentions to pretext
|
|
278
|
+
if mentions_str:
|
|
279
|
+
pretext = f"{pretext} {mentions_str}"
|
|
280
|
+
|
|
272
281
|
fallback = f"Pipeline {pipeline_name} {state}"
|
|
273
282
|
attachments = [
|
|
274
283
|
{
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
export type { LifligCdkPipelineProps } from "./liflig-cdk-pipeline";
|
|
2
2
|
export { LifligCdkPipeline } from "./liflig-cdk-pipeline";
|
|
3
3
|
export type { SlackNotificationProps } from "./slack-notification";
|
|
4
|
-
export { SlackNotification } from "./slack-notification";
|
|
4
|
+
export { SlackMention, SlackNotification } from "./slack-notification";
|
|
5
5
|
export { getVariable } from "./variables";
|
|
@@ -1,4 +1,4 @@
|
|
|
1
1
|
export { LifligCdkPipeline } from "./liflig-cdk-pipeline";
|
|
2
|
-
export { SlackNotification } from "./slack-notification";
|
|
2
|
+
export { SlackMention, SlackNotification } from "./slack-notification";
|
|
3
3
|
export { getVariable } from "./variables";
|
|
4
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY2RrLXBpcGVsaW5lcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQTtBQUV6RCxPQUFPLEVBQUUsWUFBWSxFQUFFLGlCQUFpQixFQUFFLE1BQU0sc0JBQXNCLENBQUE7QUFDdEUsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGFBQWEsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIHsgTGlmbGlnQ2RrUGlwZWxpbmVQcm9wcyB9IGZyb20gXCIuL2xpZmxpZy1jZGstcGlwZWxpbmVcIlxuZXhwb3J0IHsgTGlmbGlnQ2RrUGlwZWxpbmUgfSBmcm9tIFwiLi9saWZsaWctY2RrLXBpcGVsaW5lXCJcbmV4cG9ydCB0eXBlIHsgU2xhY2tOb3RpZmljYXRpb25Qcm9wcyB9IGZyb20gXCIuL3NsYWNrLW5vdGlmaWNhdGlvblwiXG5leHBvcnQgeyBTbGFja01lbnRpb24sIFNsYWNrTm90aWZpY2F0aW9uIH0gZnJvbSBcIi4vc2xhY2stbm90aWZpY2F0aW9uXCJcbmV4cG9ydCB7IGdldFZhcmlhYmxlIH0gZnJvbSBcIi4vdmFyaWFibGVzXCJcbiJdfQ==
|
|
@@ -41,6 +41,12 @@ export interface SlackNotificationProps {
|
|
|
41
41
|
* @default - the Lambda function can read all objects in the artifacts bucket.
|
|
42
42
|
*/
|
|
43
43
|
triggerObjectKey?: string;
|
|
44
|
+
/**
|
|
45
|
+
* Slack mentions to include in failure notifications (only on new failures, not repeated ones).
|
|
46
|
+
* Use special mentions (@here, @channel, @everyone) or user/group IDs (e.g., 'U1234567890', 'S9876543210').
|
|
47
|
+
* @default - none
|
|
48
|
+
*/
|
|
49
|
+
mentions?: string[];
|
|
44
50
|
}
|
|
45
51
|
/**
|
|
46
52
|
* Monitor a CodePipeline and send message to Slack on failure
|
|
@@ -49,3 +55,39 @@ export interface SlackNotificationProps {
|
|
|
49
55
|
export declare class SlackNotification extends constructs.Construct {
|
|
50
56
|
constructor(scope: constructs.Construct, id: string, props: SlackNotificationProps);
|
|
51
57
|
}
|
|
58
|
+
/**
|
|
59
|
+
* Slack mention formatter with validation per Slack API format:
|
|
60
|
+
* https://docs.slack.dev/messaging/formatting-message-text/
|
|
61
|
+
*
|
|
62
|
+
* Supported mention types:
|
|
63
|
+
* - Special mentions: @here, @channel, @everyone
|
|
64
|
+
* - User IDs: U or W prefix + alphanumeric (e.g., U024BE7LH, W024BE7LH)
|
|
65
|
+
* - User group IDs: S prefix + alphanumeric (e.g., SAZ94GDB8)
|
|
66
|
+
*
|
|
67
|
+
* Usage:
|
|
68
|
+
* SlackMention.format(['@here', 'U024BE7LH', 'SAZ94GDB8'])
|
|
69
|
+
*/
|
|
70
|
+
export declare class SlackMention {
|
|
71
|
+
private static readonly SPECIAL_MENTIONS;
|
|
72
|
+
private static readonly USER_PATTERN;
|
|
73
|
+
private static readonly USER_GROUP_PATTERN;
|
|
74
|
+
/**
|
|
75
|
+
* Format an array of mentions into a single Slack-formatted string.
|
|
76
|
+
* @param mentions Array of mention strings
|
|
77
|
+
*/
|
|
78
|
+
static format(mentions: string[]): string;
|
|
79
|
+
/**
|
|
80
|
+
* Format a mention string for Slack API message format.
|
|
81
|
+
* Validates format and converts to proper Slack markup:
|
|
82
|
+
* '@here' -> '<!here>'
|
|
83
|
+
* 'U1234567890' -> '<@U1234567890>'
|
|
84
|
+
* 'S1234567890' -> '<!subteam^S1234567890>'
|
|
85
|
+
*
|
|
86
|
+
* @param mention Mention string (e.g., '@here', 'U1234567890', 'S1234567890')
|
|
87
|
+
* @throws if mention format is invalid
|
|
88
|
+
*/
|
|
89
|
+
static formatMention(mention: string): string;
|
|
90
|
+
private static formatSpecialMention;
|
|
91
|
+
private static formatUser;
|
|
92
|
+
private static formatUserGroup;
|
|
93
|
+
}
|
|
@@ -21,6 +21,9 @@ export class SlackNotification extends constructs.Construct {
|
|
|
21
21
|
if (props.accountFriendlyName != null) {
|
|
22
22
|
environment.ACCOUNT_FRIENDLY_NAME = props.accountFriendlyName;
|
|
23
23
|
}
|
|
24
|
+
if (props.mentions != null && props.mentions.length > 0) {
|
|
25
|
+
environment.SLACK_MENTIONS = SlackMention.format(props.mentions);
|
|
26
|
+
}
|
|
24
27
|
const reportFunction = new lambda.Function(this, "Function", {
|
|
25
28
|
code: lambda.Code.fromAsset(path.join(__dirname, "../../assets/pipeline-slack-notification-lambda")),
|
|
26
29
|
handler: "index.handler",
|
|
@@ -49,4 +52,73 @@ export class SlackNotification extends constructs.Construct {
|
|
|
49
52
|
});
|
|
50
53
|
}
|
|
51
54
|
}
|
|
52
|
-
|
|
55
|
+
/**
|
|
56
|
+
* Slack mention formatter with validation per Slack API format:
|
|
57
|
+
* https://docs.slack.dev/messaging/formatting-message-text/
|
|
58
|
+
*
|
|
59
|
+
* Supported mention types:
|
|
60
|
+
* - Special mentions: @here, @channel, @everyone
|
|
61
|
+
* - User IDs: U or W prefix + alphanumeric (e.g., U024BE7LH, W024BE7LH)
|
|
62
|
+
* - User group IDs: S prefix + alphanumeric (e.g., SAZ94GDB8)
|
|
63
|
+
*
|
|
64
|
+
* Usage:
|
|
65
|
+
* SlackMention.format(['@here', 'U024BE7LH', 'SAZ94GDB8'])
|
|
66
|
+
*/
|
|
67
|
+
export class SlackMention {
|
|
68
|
+
static SPECIAL_MENTIONS = [
|
|
69
|
+
"@here",
|
|
70
|
+
"@channel",
|
|
71
|
+
"@everyone",
|
|
72
|
+
];
|
|
73
|
+
// Note: Slack doesn't specify length constraints for these identifiers, leaving them unbounded
|
|
74
|
+
static USER_PATTERN = /^[UW][A-Z0-9]+$/;
|
|
75
|
+
static USER_GROUP_PATTERN = /^S[A-Z0-9]+$/;
|
|
76
|
+
/**
|
|
77
|
+
* Format an array of mentions into a single Slack-formatted string.
|
|
78
|
+
* @param mentions Array of mention strings
|
|
79
|
+
*/
|
|
80
|
+
static format(mentions) {
|
|
81
|
+
return mentions.map((m) => SlackMention.formatMention(m)).join(" ");
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Format a mention string for Slack API message format.
|
|
85
|
+
* Validates format and converts to proper Slack markup:
|
|
86
|
+
* '@here' -> '<!here>'
|
|
87
|
+
* 'U1234567890' -> '<@U1234567890>'
|
|
88
|
+
* 'S1234567890' -> '<!subteam^S1234567890>'
|
|
89
|
+
*
|
|
90
|
+
* @param mention Mention string (e.g., '@here', 'U1234567890', 'S1234567890')
|
|
91
|
+
* @throws if mention format is invalid
|
|
92
|
+
*/
|
|
93
|
+
static formatMention(mention) {
|
|
94
|
+
if (mention.startsWith("@")) {
|
|
95
|
+
return SlackMention.formatSpecialMention(mention);
|
|
96
|
+
}
|
|
97
|
+
if (mention.startsWith("U") || mention.startsWith("W")) {
|
|
98
|
+
return SlackMention.formatUser(mention);
|
|
99
|
+
}
|
|
100
|
+
if (mention.startsWith("S")) {
|
|
101
|
+
return SlackMention.formatUserGroup(mention);
|
|
102
|
+
}
|
|
103
|
+
throw new Error(`Unknown Slack mention format: ${mention}`);
|
|
104
|
+
}
|
|
105
|
+
static formatSpecialMention(mention) {
|
|
106
|
+
if (!SlackMention.SPECIAL_MENTIONS.includes(mention)) {
|
|
107
|
+
throw new Error(`Invalid special mention: ${mention}`);
|
|
108
|
+
}
|
|
109
|
+
return `<!${mention.substring(1)}>`;
|
|
110
|
+
}
|
|
111
|
+
static formatUser(mention) {
|
|
112
|
+
if (!SlackMention.USER_PATTERN.test(mention)) {
|
|
113
|
+
throw new Error(`Invalid user ID: ${mention}`);
|
|
114
|
+
}
|
|
115
|
+
return `<@${mention}>`;
|
|
116
|
+
}
|
|
117
|
+
static formatUserGroup(mention) {
|
|
118
|
+
if (!SlackMention.USER_GROUP_PATTERN.test(mention)) {
|
|
119
|
+
throw new Error(`Invalid user group ID: ${mention}`);
|
|
120
|
+
}
|
|
121
|
+
return `<!subteam^${mention}>`;
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slack-notification.js","sourceRoot":"","sources":["../../src/cdk-pipelines/slack-notification.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from \"node:path\"\nimport { fileURLToPath } from \"node:url\"\nimport * as cdk from \"aws-cdk-lib\"\nimport type * as codepipeline from \"aws-cdk-lib/aws-codepipeline\"\nimport * as eventsTargets from \"aws-cdk-lib/aws-events-targets\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport type * as s3 from \"aws-cdk-lib/aws-s3\"\nimport type * as secretsmanager from \"aws-cdk-lib/aws-secretsmanager\"\nimport * as constructs from \"constructs\"\n\nconst __filename = fileURLToPath(import.meta.url)\nconst __dirname = path.dirname(__filename)\n\nexport interface SlackNotificationProps {\n  /**\n   * CodePipeline to monitor.\n   */\n  pipeline: codepipeline.IPipeline\n  /**\n   * Artifacts bucket used by pipeline\n   */\n  artifactsBucket: s3.IBucket\n  /**\n   * A plaintext secret containing the URL of a Slack incoming webhook.\n   * The webhook should be created through a Slack app, and only allows posting to one specific Slack channel.\n   * See Slack's official documentation (e.g., https://api.slack.com/messaging/webhooks) for more details.\n   *\n   * NOTE: Incoming webhooks created through legacy custom integrations in Slack are not supported.\n   */\n  slackWebhookUrlSecret: secretsmanager.ISecret\n  /**\n   * An optional friendly name that will be used in the Slack notifications instead of the AWS account ID\n   */\n  accountFriendlyName?: string\n  /**\n   * Control the amount and types of notifications being sent to Slack.\n   * \"WARN\" is the least verbose, while \"DEBUG\" is the most verbose.\n   *\n   * \"WARN\" - Includes notifications related to the failure of a pipeline execution.\n   * \"INFO\" - Adds notifications for the success of a pipeline execution.\n   * \"DEBUG\" - Adds notifications for the start and superseding of a pipeline execution.\n   *\n   * @default \"WARN\"\n   */\n  notificationLevel?: \"WARN\" | \"INFO\" | \"DEBUG\"\n  /**\n   * The key of the object (e.g., `my-prefix/my-file.json`) that triggers the S3 Source Action associated with the pipeline.\n   * By configuring this parameter you can specify which objects the Lambda function that sends messages to Slack can access in the artifacts bucket.\n   *\n   * @default - the Lambda function can read all objects in the artifacts bucket.\n   */\n  triggerObjectKey?: string\n  /**\n   * Slack mentions to include in failure notifications (only on new failures, not repeated ones).\n   * Use special mentions (@here, @channel, @everyone) or user/group IDs (e.g., 'U1234567890', 'S9876543210').\n   * @default - none\n   */\n  mentions?: string[]\n}\n\n/**\n * Monitor a CodePipeline and send message to Slack on failure\n * and some succeeded events.\n */\nexport class SlackNotification extends constructs.Construct {\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: SlackNotificationProps,\n  ) {\n    super(scope, id)\n\n    const environment: Record<string, string> = {\n      SLACK_URL_SECRET_NAME: props.slackWebhookUrlSecret.secretName,\n      NOTIFICATION_LEVEL: props.notificationLevel ?? \"WARN\",\n    }\n\n    if (props.accountFriendlyName != null) {\n      environment.ACCOUNT_FRIENDLY_NAME = props.accountFriendlyName\n    }\n\n    if (props.mentions != null && props.mentions.length > 0) {\n      environment.SLACK_MENTIONS = SlackMention.format(props.mentions)\n    }\n\n    const reportFunction = new lambda.Function(this, \"Function\", {\n      code: lambda.Code.fromAsset(\n        path.join(__dirname, \"../../assets/pipeline-slack-notification-lambda\"),\n      ),\n      handler: \"index.handler\",\n      runtime: lambda.Runtime.PYTHON_3_13,\n      timeout: cdk.Duration.seconds(10),\n      environment,\n      description:\n        \"Handle CodePipeline pipeline state change and report to Slack\",\n    })\n\n    reportFunction.grantPrincipal.addToPrincipalPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"codepipeline:ListActionExecutions\",\n          \"codepipeline:ListPipelineExecutions\",\n        ],\n        resources: [props.pipeline.pipelineArn],\n      }),\n    )\n\n    props.slackWebhookUrlSecret.grantRead(reportFunction)\n\n    props.artifactsBucket.grantRead(reportFunction, props.triggerObjectKey)\n\n    props.pipeline.onStateChange(`Event${id}`, {\n      eventPattern: {\n        detail: {\n          // Available states: https://docs.aws.amazon.com/codepipeline/latest/userguide/detect-state-changes-cloudwatch-events.html\n          state: [\"SUCCEEDED\", \"FAILED\", \"STARTED\", \"SUPERSEDED\"],\n        },\n      },\n      target: new eventsTargets.LambdaFunction(reportFunction),\n    })\n  }\n}\n\n/**\n * Slack mention formatter with validation per Slack API format:\n * https://docs.slack.dev/messaging/formatting-message-text/\n *\n * Supported mention types:\n * - Special mentions: @here, @channel, @everyone\n * - User IDs: U or W prefix + alphanumeric (e.g., U024BE7LH, W024BE7LH)\n * - User group IDs: S prefix + alphanumeric (e.g., SAZ94GDB8)\n *\n * Usage:\n *   SlackMention.format(['@here', 'U024BE7LH', 'SAZ94GDB8'])\n */\nexport class SlackMention {\n  private static readonly SPECIAL_MENTIONS = [\n    \"@here\",\n    \"@channel\",\n    \"@everyone\",\n  ] as const\n\n  // Note: Slack doesn't specify length constraints for these identifiers, leaving them unbounded\n  private static readonly USER_PATTERN = /^[UW][A-Z0-9]+$/\n  private static readonly USER_GROUP_PATTERN = /^S[A-Z0-9]+$/\n\n  /**\n   * Format an array of mentions into a single Slack-formatted string.\n   * @param mentions Array of mention strings\n   */\n  static format(mentions: string[]): string {\n    return mentions.map((m) => SlackMention.formatMention(m)).join(\" \")\n  }\n\n  /**\n   * Format a mention string for Slack API message format.\n   * Validates format and converts to proper Slack markup:\n   *   '@here' -> '<!here>'\n   *   'U1234567890' -> '<@U1234567890>'\n   *   'S1234567890' -> '<!subteam^S1234567890>'\n   *\n   * @param mention Mention string (e.g., '@here', 'U1234567890', 'S1234567890')\n   * @throws if mention format is invalid\n   */\n  static formatMention(mention: string): string {\n    if (mention.startsWith(\"@\")) {\n      return SlackMention.formatSpecialMention(mention)\n    }\n    if (mention.startsWith(\"U\") || mention.startsWith(\"W\")) {\n      return SlackMention.formatUser(mention)\n    }\n    if (mention.startsWith(\"S\")) {\n      return SlackMention.formatUserGroup(mention)\n    }\n    throw new Error(`Unknown Slack mention format: ${mention}`)\n  }\n\n  private static formatSpecialMention(mention: string): string {\n    if (\n      !SlackMention.SPECIAL_MENTIONS.includes(\n        mention as \"@here\" | \"@channel\" | \"@everyone\",\n      )\n    ) {\n      throw new Error(`Invalid special mention: ${mention}`)\n    }\n    return `<!${mention.substring(1)}>`\n  }\n\n  private static formatUser(mention: string): string {\n    if (!SlackMention.USER_PATTERN.test(mention)) {\n      throw new Error(`Invalid user ID: ${mention}`)\n    }\n    return `<@${mention}>`\n  }\n\n  private static formatUserGroup(mention: string): string {\n    if (!SlackMention.USER_GROUP_PATTERN.test(mention)) {\n      throw new Error(`Invalid user group ID: ${mention}`)\n    }\n    return `<!subteam^${mention}>`\n  }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@liflig/cdk",
|
|
3
|
-
"version": "3.
|
|
3
|
+
"version": "3.19.0",
|
|
4
4
|
"description": "CDK library for Liflig",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"repository": {
|
|
@@ -40,31 +40,31 @@
|
|
|
40
40
|
},
|
|
41
41
|
"devDependencies": {
|
|
42
42
|
"@aws-cdk/assert": "2.68.0",
|
|
43
|
-
"@aws-sdk/client-cloudwatch-logs": "3.
|
|
44
|
-
"@aws-sdk/client-codebuild": "3.
|
|
45
|
-
"@aws-sdk/client-codepipeline": "3.
|
|
46
|
-
"@aws-sdk/client-ecs": "3.
|
|
47
|
-
"@aws-sdk/client-s3": "3.
|
|
48
|
-
"@aws-sdk/client-secrets-manager": "3.
|
|
49
|
-
"@aws-sdk/client-ses": "3.
|
|
50
|
-
"@aws-sdk/client-sesv2": "3.
|
|
51
|
-
"@aws-sdk/client-sfn": "3.
|
|
52
|
-
"@aws-sdk/client-ssm": "3.
|
|
53
|
-
"@aws-sdk/lib-storage": "3.
|
|
54
|
-
"@biomejs/biome": "2.3.
|
|
55
|
-
"@commitlint/cli": "20.
|
|
56
|
-
"@commitlint/config-conventional": "20.
|
|
43
|
+
"@aws-sdk/client-cloudwatch-logs": "3.958.0",
|
|
44
|
+
"@aws-sdk/client-codebuild": "3.958.0",
|
|
45
|
+
"@aws-sdk/client-codepipeline": "3.958.0",
|
|
46
|
+
"@aws-sdk/client-ecs": "3.958.0",
|
|
47
|
+
"@aws-sdk/client-s3": "3.958.0",
|
|
48
|
+
"@aws-sdk/client-secrets-manager": "3.958.0",
|
|
49
|
+
"@aws-sdk/client-ses": "3.958.0",
|
|
50
|
+
"@aws-sdk/client-sesv2": "3.958.0",
|
|
51
|
+
"@aws-sdk/client-sfn": "3.958.0",
|
|
52
|
+
"@aws-sdk/client-ssm": "3.958.0",
|
|
53
|
+
"@aws-sdk/lib-storage": "3.958.0",
|
|
54
|
+
"@biomejs/biome": "2.3.10",
|
|
55
|
+
"@commitlint/cli": "20.3.0",
|
|
56
|
+
"@commitlint/config-conventional": "20.3.0",
|
|
57
57
|
"@types/aws-lambda": "8.10.159",
|
|
58
58
|
"@types/jest": "30.0.0",
|
|
59
|
-
"@types/node": "24.10.
|
|
60
|
-
"aws-cdk": "2.
|
|
61
|
-
"aws-cdk-lib": "2.
|
|
62
|
-
"constructs": "10.4.
|
|
63
|
-
"esbuild": "0.27.
|
|
59
|
+
"@types/node": "24.10.4",
|
|
60
|
+
"aws-cdk": "2.1100.1",
|
|
61
|
+
"aws-cdk-lib": "2.232.2",
|
|
62
|
+
"constructs": "10.4.4",
|
|
63
|
+
"esbuild": "0.27.2",
|
|
64
64
|
"jest": "30.2.0",
|
|
65
65
|
"jest-cdk-snapshot": "2.3.6",
|
|
66
|
-
"lefthook": "2.0.
|
|
67
|
-
"npm-check-updates": "19.
|
|
66
|
+
"lefthook": "2.0.13",
|
|
67
|
+
"npm-check-updates": "19.2.0",
|
|
68
68
|
"semantic-release": "25.0.2",
|
|
69
69
|
"ts-jest": "29.4.6",
|
|
70
70
|
"tsx": "4.21.0",
|
|
@@ -72,7 +72,7 @@
|
|
|
72
72
|
"typescript": "5.9.3"
|
|
73
73
|
},
|
|
74
74
|
"dependencies": {
|
|
75
|
-
"@capraconsulting/webapp-deploy-lambda": "2.5.
|
|
75
|
+
"@capraconsulting/webapp-deploy-lambda": "2.5.3",
|
|
76
76
|
"aws-jwt-verify": "5.1.1"
|
|
77
77
|
},
|
|
78
78
|
"peerDependencies": {
|