UNPKG

@gammarers/aws-budgets-notification

Version:
186 lines • 20 kB
"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.47" }; //# 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}"]}