@liflig/cdk 3.18.1 → 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 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
- ## Pre-commit checklist
16
+ ## Development
17
17
 
18
- 1. Lint code
18
+ Project commands are defined using `Make`. Examples:
19
19
 
20
- ```bash
21
- npm run lint
22
- ```
23
-
24
- 1. Run tests and update snapshots
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
- ```bash
27
- npm run snapshots
28
- npm run test -- -u
29
- ```
26
+ # Misc commands
27
+ $ make lint # lint code
28
+ $ make fmt # reformat code
29
+ $ make snapshots # regenerate snapshots
30
+ ```
30
31
 
31
- Investigate any changes before committing.
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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY2RrLXBpcGVsaW5lcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQTtBQUV6RCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQTtBQUN4RCxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sYUFBYSxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgeyBMaWZsaWdDZGtQaXBlbGluZVByb3BzIH0gZnJvbSBcIi4vbGlmbGlnLWNkay1waXBlbGluZVwiXG5leHBvcnQgeyBMaWZsaWdDZGtQaXBlbGluZSB9IGZyb20gXCIuL2xpZmxpZy1jZGstcGlwZWxpbmVcIlxuZXhwb3J0IHR5cGUgeyBTbGFja05vdGlmaWNhdGlvblByb3BzIH0gZnJvbSBcIi4vc2xhY2stbm90aWZpY2F0aW9uXCJcbmV4cG9ydCB7IFNsYWNrTm90aWZpY2F0aW9uIH0gZnJvbSBcIi4vc2xhY2stbm90aWZpY2F0aW9uXCJcbmV4cG9ydCB7IGdldFZhcmlhYmxlIH0gZnJvbSBcIi4vdmFyaWFibGVzXCJcbiJdfQ==
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
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slack-notification.js","sourceRoot":"","sources":["../../src/cdk-pipelines/slack-notification.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,IAAI,MAAM,WAAW,CAAA;AACjC,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAA;AACxC,OAAO,KAAK,GAAG,MAAM,aAAa,CAAA;AAElC,OAAO,KAAK,aAAa,MAAM,gCAAgC,CAAA;AAC/D,OAAO,KAAK,GAAG,MAAM,qBAAqB,CAAA;AAC1C,OAAO,KAAK,MAAM,MAAM,wBAAwB,CAAA;AAGhD,OAAO,KAAK,UAAU,MAAM,YAAY,CAAA;AAExC,MAAM,UAAU,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;AACjD,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAA;AA2C1C;;;GAGG;AACH,MAAM,OAAO,iBAAkB,SAAQ,UAAU,CAAC,SAAS;IACzD,YACE,KAA2B,EAC3B,EAAU,EACV,KAA6B;QAE7B,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAA;QAEhB,MAAM,WAAW,GAA2B;YAC1C,qBAAqB,EAAE,KAAK,CAAC,qBAAqB,CAAC,UAAU;YAC7D,kBAAkB,EAAE,KAAK,CAAC,iBAAiB,IAAI,MAAM;SACtD,CAAA;QAED,IAAI,KAAK,CAAC,mBAAmB,IAAI,IAAI,EAAE,CAAC;YACtC,WAAW,CAAC,qBAAqB,GAAG,KAAK,CAAC,mBAAmB,CAAA;QAC/D,CAAC;QAED,MAAM,cAAc,GAAG,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,UAAU,EAAE;YAC3D,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CACzB,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,iDAAiD,CAAC,CACxE;YACD,OAAO,EAAE,eAAe;YACxB,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,WAAW;YACnC,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,WAAW;YACX,WAAW,EACT,+DAA+D;SAClE,CAAC,CAAA;QAEF,cAAc,CAAC,cAAc,CAAC,oBAAoB,CAChD,IAAI,GAAG,CAAC,eAAe,CAAC;YACtB,OAAO,EAAE;gBACP,mCAAmC;gBACnC,qCAAqC;aACtC;YACD,SAAS,EAAE,CAAC,KAAK,CAAC,QAAQ,CAAC,WAAW,CAAC;SACxC,CAAC,CACH,CAAA;QAED,KAAK,CAAC,qBAAqB,CAAC,SAAS,CAAC,cAAc,CAAC,CAAA;QAErD,KAAK,CAAC,eAAe,CAAC,SAAS,CAAC,cAAc,EAAE,KAAK,CAAC,gBAAgB,CAAC,CAAA;QAEvE,KAAK,CAAC,QAAQ,CAAC,aAAa,CAAC,QAAQ,EAAE,EAAE,EAAE;YACzC,YAAY,EAAE;gBACZ,MAAM,EAAE;oBACN,0HAA0H;oBAC1H,KAAK,EAAE,CAAC,WAAW,EAAE,QAAQ,EAAE,SAAS,EAAE,YAAY,CAAC;iBACxD;aACF;YACD,MAAM,EAAE,IAAI,aAAa,CAAC,cAAc,CAAC,cAAc,CAAC;SACzD,CAAC,CAAA;IACJ,CAAC;CACF","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\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    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"]}
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.18.1",
3
+ "version": "3.19.0",
4
4
  "description": "CDK library for Liflig",
5
5
  "type": "module",
6
6
  "repository": {
@@ -40,30 +40,30 @@
40
40
  },
41
41
  "devDependencies": {
42
42
  "@aws-cdk/assert": "2.68.0",
43
- "@aws-sdk/client-cloudwatch-logs": "3.948.0",
44
- "@aws-sdk/client-codebuild": "3.948.0",
45
- "@aws-sdk/client-codepipeline": "3.948.0",
46
- "@aws-sdk/client-ecs": "3.948.0",
47
- "@aws-sdk/client-s3": "3.948.0",
48
- "@aws-sdk/client-secrets-manager": "3.950.0",
49
- "@aws-sdk/client-ses": "3.948.0",
50
- "@aws-sdk/client-sesv2": "3.950.0",
51
- "@aws-sdk/client-sfn": "3.948.0",
52
- "@aws-sdk/client-ssm": "3.948.0",
53
- "@aws-sdk/lib-storage": "3.948.0",
54
- "@biomejs/biome": "2.3.8",
55
- "@commitlint/cli": "20.2.0",
56
- "@commitlint/config-conventional": "20.2.0",
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.2",
60
- "aws-cdk": "2.1034.0",
61
- "aws-cdk-lib": "2.232.1",
59
+ "@types/node": "24.10.4",
60
+ "aws-cdk": "2.1100.1",
61
+ "aws-cdk-lib": "2.232.2",
62
62
  "constructs": "10.4.4",
63
- "esbuild": "0.27.1",
63
+ "esbuild": "0.27.2",
64
64
  "jest": "30.2.0",
65
65
  "jest-cdk-snapshot": "2.3.6",
66
- "lefthook": "2.0.11",
66
+ "lefthook": "2.0.13",
67
67
  "npm-check-updates": "19.2.0",
68
68
  "semantic-release": "25.0.2",
69
69
  "ts-jest": "29.4.6",