@gammarers/aws-codepipeline-execution-state-change-notification-stack
Version:
This AWS CDK Construct Stack receives all state changes of CodePipeline and sends a message to the specified notification destination when the CodePipeline is tagged with a specified tag. Therefore, you can send messages simply by adding tags without need
83 lines • 14.5 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.CodePipelineExecutionStateChangeNotificationStack = exports.CodePipelineExecutionStateChangeNotificationStackResourceNamingType = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const aws_codesuite_state_change_detection_event_rules_1 = require("@gammarers/aws-codesuite-state-change-detection-event-rules");
const aws_resource_naming_1 = require("@gammarers/aws-resource-naming");
Object.defineProperty(exports, "CodePipelineExecutionStateChangeNotificationStackResourceNamingType", { enumerable: true, get: function () { return aws_resource_naming_1.ResourceNamingType; } });
const aws_sns_slack_message_lambda_subscription_1 = require("@gammarers/aws-sns-slack-message-lambda-subscription");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const events = require("aws-cdk-lib/aws-events");
const targets = require("aws-cdk-lib/aws-events-targets");
const sns = require("aws-cdk-lib/aws-sns");
const subscriptions = require("aws-cdk-lib/aws-sns-subscriptions");
const notification_state_machine_1 = require("./resources/notification-state-machine");
class CodePipelineExecutionStateChangeNotificationStack extends aws_cdk_lib_1.Stack {
constructor(scope, id, props) {
super(scope, id, props);
// 👇 Create random 8 length string
const random = aws_resource_naming_1.ResourceNaming.createRandomString(`${aws_cdk_lib_1.Names.uniqueId(scope)}.${aws_cdk_lib_1.Names.uniqueId(this)}`);
// 👇 Auto naeming
const autoNaming = {
stateMachineName: `codepipeline-exec-state-change-notification-${random}-machine`,
notificationTopicName: `codepipeline-execution-state-change-notification-${random}-topic`,
notificationTopicDisplayName: 'CodePipeline Execution state change Notification Topic',
pipelineEventDetectionRuleName: `codepipeline-exe-state-change-${random}-detection-event-rule`,
};
// 👇 Resource Names
const names = aws_resource_naming_1.ResourceNaming.naming(autoNaming, props.resourceNamingOption);
// 👇 SNS Topic for notifications
const topic = new sns.Topic(this, 'NotificationTopic', {
topicName: names.notificationTopicName,
displayName: names.notificationTopicDisplayName,
});
// const secret = cdk.SecretValue.secretsManager('my-email-array-secret');
// const emails = JSON.parse(secret.unsafeUnwrap()) as string[];
// Subscribe an email endpoint to the topic
const emails = props.notifications.emails ?? [];
for (const email of emails) {
topic.addSubscription(new subscriptions.EmailSubscription(email));
}
// 👇 Subscription slack webhook
if (props.notifications?.slack) {
new aws_sns_slack_message_lambda_subscription_1.SNSSlackMessageLambdaSubscription(this, 'SNSSlackMessageLambdaSubscription', {
topic,
slackWebhookSecretName: props.notifications.slack.webhookSecretName,
});
}
// Subscribe a HTTP endpoint (Slack Webhook) to the topic
// topic.addSubscription(new subs.UrlSubscription('https://hooks.slack.com/services/YOUR/SLACK/WEBHOOK'));
// 👇 Create State Machine
const stateMachine = new notification_state_machine_1.NotificationStateMachine(this, 'StateMachine', {
stateMachineName: names.stateMachineName,
timeout: aws_cdk_lib_1.Duration.minutes(5),
notificationTopic: topic,
});
// 👇 Rule state
//enabled
const enableRule = (() => {
return props?.enabled === undefined || props.enabled;
})();
// 👇 Create EventBridge Rule
new aws_codesuite_state_change_detection_event_rules_1.CodePipelinePipelineExecutionStateChangeDetectionEventRule(this, 'EventRule', {
ruleName: names.pipelineEventDetectionRuleName,
enabled: enableRule,
targets: [
new targets.SfnStateMachine(stateMachine, {
input: events.RuleTargetInput.fromObject({
event: events.EventField.fromPath('$'),
params: {
tagKey: props.targetResource.tagKey,
tagValues: props.targetResource.tagValues,
},
}),
}),
],
});
}
}
exports.CodePipelineExecutionStateChangeNotificationStack = CodePipelineExecutionStateChangeNotificationStack;
_a = JSII_RTTI_SYMBOL_1;
CodePipelineExecutionStateChangeNotificationStack[_a] = { fqn: "@gammarers/aws-codepipeline-execution-state-change-notification-stack.CodePipelineExecutionStateChangeNotificationStack", version: "2.4.2" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { CodePipelinePipelineExecutionStateChangeDetectionEventRule } from '@gammarers/aws-codesuite-state-change-detection-event-rules';\nimport { ResourceAutoNaming, ResourceDefaultNaming, ResourceNaming, ResourceNamingType as CodePipelineExecutionStateChangeNotificationStackResourceNamingType } from '@gammarers/aws-resource-naming';\nimport { SNSSlackMessageLambdaSubscription } from '@gammarers/aws-sns-slack-message-lambda-subscription';\nimport { Duration, Names, Stack, StackProps } from 'aws-cdk-lib';\nimport * as events from 'aws-cdk-lib/aws-events';\nimport * as targets from 'aws-cdk-lib/aws-events-targets';\nimport * as sns from 'aws-cdk-lib/aws-sns';\nimport * as subscriptions from 'aws-cdk-lib/aws-sns-subscriptions';\nimport { Construct } from 'constructs';\nimport { NotificationStateMachine } from './resources/notification-state-machine';\n\nexport { CodePipelineExecutionStateChangeNotificationStackResourceNamingType };\n\nexport interface TargetResourceProperty {\n  readonly tagKey: string;\n  readonly tagValues: string[];\n}\n\nexport interface Slack {\n  readonly webhookSecretName: string;\n}\n\nexport interface NotificationsProperty {\n  readonly emails?: string[];\n  readonly slack?: Slack;\n}\n\nexport interface CodePipelineExecutionStateChangeNotificationStackProps extends StackProps {\n  readonly targetResource: TargetResourceProperty;\n  readonly enabled?: boolean;\n  readonly notifications: NotificationsProperty;\n  readonly resourceNamingOption?: ResourceNamingOption;\n}\n\nexport interface CustomNaming {\n  readonly type: CodePipelineExecutionStateChangeNotificationStackResourceNamingType.CUSTOM;\n  readonly stateMachineName: string;\n  readonly notificationTopicName: string;\n  readonly notificationTopicDisplayName: string;\n  readonly pipelineEventDetectionRuleName: string;\n}\n\nexport type ResourceNamingOption = ResourceDefaultNaming | ResourceAutoNaming | CustomNaming;\n\nexport class CodePipelineExecutionStateChangeNotificationStack extends Stack {\n  constructor(scope: Construct, id: string, props: CodePipelineExecutionStateChangeNotificationStackProps) {\n    super(scope, id, props);\n\n    // 👇 Create random 8 length string\n    const random = ResourceNaming.createRandomString(`${Names.uniqueId(scope)}.${Names.uniqueId(this)}`);\n    // 👇 Auto naeming\n    const autoNaming = {\n      stateMachineName: `codepipeline-exec-state-change-notification-${random}-machine`,\n      notificationTopicName: `codepipeline-execution-state-change-notification-${random}-topic`,\n      notificationTopicDisplayName: 'CodePipeline Execution state change Notification Topic',\n      pipelineEventDetectionRuleName: `codepipeline-exe-state-change-${random}-detection-event-rule`,\n    };\n    // 👇 Resource Names\n    const names = ResourceNaming.naming(autoNaming, props.resourceNamingOption as ResourceNaming.ResourceNamingOption);\n\n    // 👇 SNS Topic for notifications\n    const topic: sns.Topic = new sns.Topic(this, 'NotificationTopic', {\n      topicName: names.notificationTopicName,\n      displayName: names.notificationTopicDisplayName,\n    });\n\n    //    const secret = cdk.SecretValue.secretsManager('my-email-array-secret');\n    //    const emails = JSON.parse(secret.unsafeUnwrap()) as string[];\n\n    // Subscribe an email endpoint to the topic\n    const emails = props.notifications.emails ?? [];\n    for (const email of emails) {\n      topic.addSubscription(new subscriptions.EmailSubscription(email));\n    }\n\n    // 👇 Subscription slack webhook\n    if (props.notifications?.slack) {\n      new SNSSlackMessageLambdaSubscription(this, 'SNSSlackMessageLambdaSubscription', {\n        topic,\n        slackWebhookSecretName: props.notifications.slack.webhookSecretName,\n      });\n    }\n\n    // Subscribe a HTTP endpoint (Slack Webhook) to the topic\n    // topic.addSubscription(new subs.UrlSubscription('https://hooks.slack.com/services/YOUR/SLACK/WEBHOOK'));\n\n    // 👇 Create State Machine\n    const stateMachine = new NotificationStateMachine(this, 'StateMachine', {\n      stateMachineName: names.stateMachineName,\n      timeout: Duration.minutes(5),\n      notificationTopic: topic,\n    });\n\n    // 👇 Rule state\n    //enabled\n    const enableRule: boolean = (() => {\n      return props?.enabled === undefined || props.enabled;\n    })();\n\n    // 👇 Create EventBridge Rule\n    new CodePipelinePipelineExecutionStateChangeDetectionEventRule(this, 'EventRule', {\n      ruleName: names.pipelineEventDetectionRuleName,\n      enabled: enableRule,\n      targets: [\n        new targets.SfnStateMachine(stateMachine, {\n          input: events.RuleTargetInput.fromObject({\n            event: events.EventField.fromPath('$'),\n            params: {\n              tagKey: props.targetResource.tagKey,\n              tagValues: props.targetResource.tagValues,\n            },\n          }),\n        }),\n      ],\n    });\n  }\n}\n"]}