Cost Monitoring Construct is a typescript CDK construct that helps track applications' costs separately and receive alerts in case of unpredicted resource usage to prevent surprise billings and have a better overview of resource usage by an application.
To install the Cost Monitoring Construct package, run the following command to download it from npm:
npm i cost-monitoring-construct
The easiest way to apply cost monitoring is by using predefined default budget strategies. The ApplicationCostMonitoring
has default strategy implementation for monitoring budgets per CDK applications, while the AccountCostMonitoring
has default strategy implementation for monitoring budgets per account. You can also create your own strategy by creating a class that applies to the IBudgetStrategy
or directly inherits from the ApplicationCostMonitoring
or AccountCostMonitoring
class to customize them to your needs.
This is the default strategy for monitoring CDK applications. You have to provide at least one stack by the first parameter of the initializer. If you wish to monitor more stacks, you can pass them as a list to the otherStacksIncludedInBudget
prop.
The ApplicationCostMonitoring
will always use the first stack (the first parameter of the initializer) to inject its resources into it. You can create a separate stack and pass it as the first stack if you wish to keep cost monitoring separate from your stacks. The cost monitoring class will monitor both the first stack and stacks from otherStacksIncludedInBudget
.
[!WARNING] ApplicationCostMonitoring uses AWS Tags to track resources' usages. You must activate your chosen tag key (
cm:application
by default) under Cost Allocation Tags. The tag key will appear in the AWS console up to 24 hours after at least one AWS resource has been created with that tag.
The example below shows how to use ApplicationCostMonitoring
to track your application in CDK code:
const app = new cdk.App();
// Holding resources for monitoring the application. Plus to be used by costMonitoring to inject its resources into it.
const monitoringStack = new MonitoringStack(app, "MyAppMonitoringStack", {});
const firstStack = new FirstStack(app, "FirstStack", {});
const secondStack = new SecondStack(app, "SecondStack", {});
const costMonitoring = new ApplicationCostMonitoring(monitoringStack, {
// the first stack ~~~~~~~👆~~~~~~
applicationName: "my-application",
monthlyLimitInDollars: 200,
otherStacksIncludedInBudget: [secondStack, firstStack], // Optional (you can add as many stacks as you want)
subscribers: ["[email protected]"], // Optional
costAllocationTag: "MyTag" // Optional, defaults to `cm:application`
});
costMonitoring.monitor();
This is the abstract base class that all budget strategies must apply to. This Abstract class enforces the implementation for createDailyBudgets
and createMonthlyBudgets
. Plus, it provides the createQuarterlyBudgets
and createYearlyBudgets
optional to implement.
The methods above abstract the implementations of budgeting strategy from boilerplate codes.
To implement those methods, you can instantiate from class Budget
to define your budgets according to their period. The class Budget
provides the clone
method to deep copy objects with small changes instead of providing duplicate configs for multiple Budget
instantiations.
The example below implements the createDailyBudgets
method with that pattern for a custom budget strategy
export class MyApplicationCostMonitoring extends IBudgetStrategy {
protected createDailyBudgets(
dailyLimit: number,
subscribers: Array<budgets.CfnBudget.SubscriberProperty>
): void {
new Budget(
this.stack,
`application_${this.applicationName}_daily_${dailyBudget}_%80`,
{
tags: [{ key: "application", value: "my-application" }],
limit: dailyLimit,
subscribers: subscribers,
alertContdition: {
threshold: 80,
period: TimeUnit.DAILY,
},
}
)
.clone(`application_${this.applicationName}_daily_${dailyLimit}_%90`, {
threshold: 90,
})
.clone(`application_${this.applicationName}_daily_${dailyLimit}_%100`, {
threshold: 100,
});
}
protected createMonthlyBudgets(
monthlyLimit: number,
subscribers: Array<budgets.CfnBudget.SubscriberProperty>
): void {
// defining monthly strategy here ...
} // optionally you can define Yearly and Quarterly Budgets as well.
}