@gammarers/aws-budgets-notification
Version:
AWS Budgets Notification
186 lines • 20 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.BudgetsNotification = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const crypto = require("crypto");
const cdk = require("aws-cdk-lib");
const budgets = require("aws-cdk-lib/aws-budgets");
const bot = require("aws-cdk-lib/aws-chatbot");
const iam = require("aws-cdk-lib/aws-iam");
//import * as logs from 'aws-cdk-lib/aws-logs';
const sns = require("aws-cdk-lib/aws-sns");
const constructs_1 = require("constructs");
class BudgetsNotification extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
// 👇Get current account & region
const account = cdk.Stack.of(this).account;
// const region = cdk.Stack.of(this).region;
// 👇Create random key
const randomNameKey = crypto.createHash('shake256', { outputLength: 4 })
.update(`${cdk.Names.uniqueId(scope)}-${cdk.Names.uniqueId(this)}`)
.digest('hex');
// 👇Create SNS Topic
const topic = new sns.Topic(this, 'NotificationTopic', {
topicName: `budget-notification-${randomNameKey}-topic`,
displayName: `budget-notification-${randomNameKey}-topic`,
});
topic.addToResourcePolicy(new iam.PolicyStatement({
sid: 'AWSBudgetsSNSPublishingPermissions',
effect: iam.Effect.ALLOW,
principals: [
new iam.ServicePrincipal('budgets.amazonaws.com'),
],
actions: [
'sns:Publish',
],
resources: [
topic.topicArn,
],
}));
topic.addToResourcePolicy(new iam.PolicyStatement({
sid: 'OwnerSNSActionPermissions',
effect: iam.Effect.ALLOW,
principals: [
new iam.AnyPrincipal(),
],
actions: [
'sns:GetTopicAttributes',
'sns:SetTopicAttributes',
'sns:AddPermission',
'sns:RemovePermission',
'sns:DeleteTopic',
'sns:Subscribe',
'sns:ListSubscriptionsByTopic',
'sns:Publish',
'sns:Receive',
],
resources: [
topic.topicArn,
],
conditions: {
StringEquals: {
'AWS:SourceOwner': account,
},
},
}));
// 👇Create ChatBot
new bot.SlackChannelConfiguration(this, 'SlackChannelConfig', {
slackChannelConfigurationName: `slack-channel-budget-notification-${randomNameKey}-config`,
slackWorkspaceId: props.slackWorkspaceId,
slackChannelId: props.slackChannelId,
//logRetention: logs.RetentionDays.TWO_MONTHS,
//logRetentionRetryOptions: undefined,
//logRetentionRole: undefined,
loggingLevel: bot.LoggingLevel.ERROR,
notificationTopics: [
topic,
],
role: new iam.Role(this, 'SlackChannelConfigRole', {
roleName: `slack-channel-budget-notification-${randomNameKey}-config-role`,
description: 'slack channel budget notification config role.',
assumedBy: new iam.ServicePrincipal('chatbot.amazonaws.com'),
inlinePolicies: {
'chatbot-policy': new iam.PolicyDocument({
statements: [
new iam.PolicyStatement({
effect: iam.Effect.ALLOW,
actions: [
'cloudwatch:Describe*',
'cloudwatch:Get*',
'cloudwatch:List*',
],
resources: ['*'],
}),
],
}),
},
}),
});
// 👇Common notification with subscribers.
const notificationsWithSubscribers = [
{
notification: {
notificationType: 'ACTUAL',
comparisonOperator: 'GREATER_THAN',
threshold: 60,
thresholdType: 'PERCENTAGE',
},
subscribers: [
{
subscriptionType: 'SNS',
address: topic.topicArn,
},
],
},
{
notification: {
notificationType: 'ACTUAL',
comparisonOperator: 'GREATER_THAN',
threshold: 80,
thresholdType: 'PERCENTAGE',
},
subscribers: [
{
subscriptionType: 'SNS',
address: topic.topicArn,
},
],
},
{
notification: {
notificationType: 'ACTUAL',
comparisonOperator: 'GREATER_THAN',
threshold: 100,
thresholdType: 'PERCENTAGE',
},
subscribers: [
{
subscriptionType: 'SNS',
address: topic.topicArn,
},
],
},
];
if (props.linkedAccounts && props.linkedAccounts.length >= 1) {
for (const linkedAccount of props.linkedAccounts) {
new budgets.CfnBudget(this, `Budget${linkedAccount}`, {
budget: {
budgetType: 'COST',
budgetName: `Monthly usage for ${linkedAccount}.`,
timeUnit: 'MONTHLY',
costFilters: {
LinkedAccount: [
linkedAccount,
],
},
budgetLimit: {
amount: props.budgetLimitAmount,
unit: 'USD',
},
},
notificationsWithSubscribers,
});
}
}
else {
new budgets.CfnBudget(this, 'Budget', {
budget: {
budgetType: 'COST',
budgetName: 'Monthly usage',
timeUnit: 'MONTHLY',
budgetLimit: {
amount: props.budgetLimitAmount,
unit: 'USD',
},
},
notificationsWithSubscribers,
});
}
}
}
exports.BudgetsNotification = BudgetsNotification;
_a = JSII_RTTI_SYMBOL_1;
BudgetsNotification[_a] = { fqn: "@gammarers/aws-budgets-notification.BudgetsNotification", version: "1.2.77" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as cdk from 'aws-cdk-lib';\nimport * as budgets from 'aws-cdk-lib/aws-budgets';\nimport * as bot from 'aws-cdk-lib/aws-chatbot';\nimport * as iam from 'aws-cdk-lib/aws-iam';\n//import * as logs from 'aws-cdk-lib/aws-logs';\nimport * as sns from 'aws-cdk-lib/aws-sns';\nimport { Construct } from 'constructs';\n\n\nexport interface BudgetsNotificationProps {\n  readonly slackWorkspaceId: string;\n  readonly slackChannelId: string;\n  readonly budgetLimitAmount: number;\n  readonly linkedAccounts?: string[];\n}\n\nexport class BudgetsNotification extends Construct {\n\n  constructor(scope: Construct, id: string, props: BudgetsNotificationProps) {\n    super(scope, id);\n\n    // 👇Get current account & region\n    const account = cdk.Stack.of(this).account;\n    // const region = cdk.Stack.of(this).region;\n\n    // 👇Create random key\n    const randomNameKey = crypto.createHash('shake256', { outputLength: 4 })\n      .update(`${cdk.Names.uniqueId(scope)}-${cdk.Names.uniqueId(this)}`)\n      .digest('hex');\n\n    // 👇Create SNS Topic\n    const topic = new sns.Topic(this, 'NotificationTopic', {\n      topicName: `budget-notification-${randomNameKey}-topic`,\n      displayName: `budget-notification-${randomNameKey}-topic`,\n    });\n    topic.addToResourcePolicy(new iam.PolicyStatement({\n      sid: 'AWSBudgetsSNSPublishingPermissions',\n      effect: iam.Effect.ALLOW,\n      principals: [\n        new iam.ServicePrincipal('budgets.amazonaws.com'),\n      ],\n      actions: [\n        'sns:Publish',\n      ],\n      resources: [\n        topic.topicArn,\n      ],\n    }));\n    topic.addToResourcePolicy(new iam.PolicyStatement({\n      sid: 'OwnerSNSActionPermissions',\n      effect: iam.Effect.ALLOW,\n      principals: [\n        new iam.AnyPrincipal(),\n      ],\n      actions: [\n        'sns:GetTopicAttributes',\n        'sns:SetTopicAttributes',\n        'sns:AddPermission',\n        'sns:RemovePermission',\n        'sns:DeleteTopic',\n        'sns:Subscribe',\n        'sns:ListSubscriptionsByTopic',\n        'sns:Publish',\n        'sns:Receive',\n      ],\n      resources: [\n        topic.topicArn,\n      ],\n      conditions: {\n        StringEquals: {\n          'AWS:SourceOwner': account,\n        },\n      },\n    }));\n\n    // 👇Create ChatBot\n    new bot.SlackChannelConfiguration(this, 'SlackChannelConfig', {\n      slackChannelConfigurationName: `slack-channel-budget-notification-${randomNameKey}-config`,\n      slackWorkspaceId: props.slackWorkspaceId,\n      slackChannelId: props.slackChannelId,\n      //logRetention: logs.RetentionDays.TWO_MONTHS,\n      //logRetentionRetryOptions: undefined,\n      //logRetentionRole: undefined,\n      loggingLevel: bot.LoggingLevel.ERROR,\n      notificationTopics: [\n        topic,\n      ],\n      role: new iam.Role(this, 'SlackChannelConfigRole', {\n        roleName: `slack-channel-budget-notification-${randomNameKey}-config-role`,\n        description: 'slack channel budget notification config role.',\n        assumedBy: new iam.ServicePrincipal('chatbot.amazonaws.com'),\n        inlinePolicies: {\n          'chatbot-policy': new iam.PolicyDocument({\n            statements: [\n              new iam.PolicyStatement({\n                effect: iam.Effect.ALLOW,\n                actions: [\n                  'cloudwatch:Describe*',\n                  'cloudwatch:Get*',\n                  'cloudwatch:List*',\n                ],\n                resources: ['*'],\n              }),\n            ],\n          }),\n        },\n      }),\n    });\n\n    // 👇Common notification with subscribers.\n    const notificationsWithSubscribers: Array<budgets.CfnBudget.NotificationWithSubscribersProperty> = [\n      {\n        notification: {\n          notificationType: 'ACTUAL',\n          comparisonOperator: 'GREATER_THAN',\n          threshold: 60,\n          thresholdType: 'PERCENTAGE',\n        },\n        subscribers: [\n          {\n            subscriptionType: 'SNS',\n            address: topic.topicArn,\n          },\n        ],\n      },\n      {\n        notification: {\n          notificationType: 'ACTUAL',\n          comparisonOperator: 'GREATER_THAN',\n          threshold: 80,\n          thresholdType: 'PERCENTAGE',\n        },\n        subscribers: [\n          {\n            subscriptionType: 'SNS',\n            address: topic.topicArn,\n          },\n        ],\n      },\n      {\n        notification: {\n          notificationType: 'ACTUAL',\n          comparisonOperator: 'GREATER_THAN',\n          threshold: 100,\n          thresholdType: 'PERCENTAGE',\n        },\n        subscribers: [\n          {\n            subscriptionType: 'SNS',\n            address: topic.topicArn,\n          },\n        ],\n      },\n    ];\n\n    if (props.linkedAccounts && props.linkedAccounts.length >= 1) {\n      for (const linkedAccount of props.linkedAccounts) {\n        new budgets.CfnBudget(this, `Budget${linkedAccount}`, {\n          budget: {\n            budgetType: 'COST',\n            budgetName: `Monthly usage for ${linkedAccount}.`,\n            timeUnit: 'MONTHLY',\n            costFilters: {\n              LinkedAccount: [\n                linkedAccount,\n              ],\n            },\n            budgetLimit: {\n              amount: props.budgetLimitAmount,\n              unit: 'USD',\n            },\n          },\n          notificationsWithSubscribers,\n        });\n      }\n    } else {\n      new budgets.CfnBudget(this, 'Budget', {\n        budget: {\n          budgetType: 'COST',\n          budgetName: 'Monthly usage',\n          timeUnit: 'MONTHLY',\n          budgetLimit: {\n            amount: props.budgetLimitAmount,\n            unit: 'USD',\n          },\n        },\n        notificationsWithSubscribers,\n      });\n    }\n  }\n}"]}