@gammarers/aws-daily-cost-usage-report-stack 2.0.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/.jsii +4144 -0
- package/API.md +964 -0
- package/LICENSE +202 -0
- package/README.md +63 -0
- package/assets/funcs/cost-reporter.lambda/index.js +18835 -0
- package/assets/funcs/cost-reporter.lambda/index.js.map +7 -0
- package/lib/funcs/cost-reporter-function.d.ts +13 -0
- package/lib/funcs/cost-reporter-function.js +23 -0
- package/lib/funcs/cost-reporter.lambda.d.ts +13 -0
- package/lib/funcs/cost-reporter.lambda.js +103 -0
- package/lib/funcs/lib/get-billing-command.d.ts +31 -0
- package/lib/funcs/lib/get-billing-command.js +158 -0
- package/lib/funcs/lib/get-date-range.d.ts +6 -0
- package/lib/funcs/lib/get-date-range.js +22 -0
- package/lib/index.d.ts +15 -0
- package/lib/index.js +102 -0
- package/package.json +158 -0
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import * as lambda from 'aws-cdk-lib/aws-lambda';
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
/**
|
|
4
|
+
* Props for CostReporterFunction
|
|
5
|
+
*/
|
|
6
|
+
export interface CostReporterFunctionProps extends lambda.FunctionOptions {
|
|
7
|
+
}
|
|
8
|
+
/**
|
|
9
|
+
* An AWS Lambda function which executes src/funcs/cost-reporter.
|
|
10
|
+
*/
|
|
11
|
+
export declare class CostReporterFunction extends lambda.Function {
|
|
12
|
+
constructor(scope: Construct, id: string, props?: CostReporterFunctionProps);
|
|
13
|
+
}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.CostReporterFunction = void 0;
|
|
4
|
+
// ~~ Generated by projen. To modify, edit .projenrc.ts and run "npx projen".
|
|
5
|
+
const path = require("path");
|
|
6
|
+
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
7
|
+
/**
|
|
8
|
+
* An AWS Lambda function which executes src/funcs/cost-reporter.
|
|
9
|
+
*/
|
|
10
|
+
class CostReporterFunction extends lambda.Function {
|
|
11
|
+
constructor(scope, id, props) {
|
|
12
|
+
super(scope, id, {
|
|
13
|
+
description: 'src/funcs/cost-reporter.lambda.ts',
|
|
14
|
+
...props,
|
|
15
|
+
runtime: new lambda.Runtime('nodejs20.x', lambda.RuntimeFamily.NODEJS),
|
|
16
|
+
handler: 'index.handler',
|
|
17
|
+
code: lambda.Code.fromAsset(path.join(__dirname, '../../assets/funcs/cost-reporter.lambda')),
|
|
18
|
+
});
|
|
19
|
+
this.addEnvironment('AWS_NODEJS_CONNECTION_REUSE_ENABLED', '1', { removeInEdge: true });
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
exports.CostReporterFunction = CostReporterFunction;
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29zdC1yZXBvcnRlci1mdW5jdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9mdW5jcy9jb3N0LXJlcG9ydGVyLWZ1bmN0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDZFQUE2RTtBQUM3RSw2QkFBNkI7QUFDN0IsaURBQWlEO0FBU2pEOztHQUVHO0FBQ0gsTUFBYSxvQkFBcUIsU0FBUSxNQUFNLENBQUMsUUFBUTtJQUN2RCxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQWlDO1FBQ3pFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFO1lBQ2YsV0FBVyxFQUFFLG1DQUFtQztZQUNoRCxHQUFHLEtBQUs7WUFDUixPQUFPLEVBQUUsSUFBSSxNQUFNLENBQUMsT0FBTyxDQUFDLFlBQVksRUFBRSxNQUFNLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQztZQUN0RSxPQUFPLEVBQUUsZUFBZTtZQUN4QixJQUFJLEVBQUUsTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLEVBQUUseUNBQXlDLENBQUMsQ0FBQztTQUM3RixDQUFDLENBQUM7UUFDSCxJQUFJLENBQUMsY0FBYyxDQUFDLHFDQUFxQyxFQUFFLEdBQUcsRUFBRSxFQUFFLFlBQVksRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDO0lBQzFGLENBQUM7Q0FDRjtBQVhELG9EQVdDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gfn4gR2VuZXJhdGVkIGJ5IHByb2plbi4gVG8gbW9kaWZ5LCBlZGl0IC5wcm9qZW5yYy50cyBhbmQgcnVuIFwibnB4IHByb2plblwiLlxuaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcbmltcG9ydCAqIGFzIGxhbWJkYSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuXG4vKipcbiAqIFByb3BzIGZvciBDb3N0UmVwb3J0ZXJGdW5jdGlvblxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvc3RSZXBvcnRlckZ1bmN0aW9uUHJvcHMgZXh0ZW5kcyBsYW1iZGEuRnVuY3Rpb25PcHRpb25zIHtcbn1cblxuLyoqXG4gKiBBbiBBV1MgTGFtYmRhIGZ1bmN0aW9uIHdoaWNoIGV4ZWN1dGVzIHNyYy9mdW5jcy9jb3N0LXJlcG9ydGVyLlxuICovXG5leHBvcnQgY2xhc3MgQ29zdFJlcG9ydGVyRnVuY3Rpb24gZXh0ZW5kcyBsYW1iZGEuRnVuY3Rpb24ge1xuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wcz86IENvc3RSZXBvcnRlckZ1bmN0aW9uUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQsIHtcbiAgICAgIGRlc2NyaXB0aW9uOiAnc3JjL2Z1bmNzL2Nvc3QtcmVwb3J0ZXIubGFtYmRhLnRzJyxcbiAgICAgIC4uLnByb3BzLFxuICAgICAgcnVudGltZTogbmV3IGxhbWJkYS5SdW50aW1lKCdub2RlanMyMC54JywgbGFtYmRhLlJ1bnRpbWVGYW1pbHkuTk9ERUpTKSxcbiAgICAgIGhhbmRsZXI6ICdpbmRleC5oYW5kbGVyJyxcbiAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21Bc3NldChwYXRoLmpvaW4oX19kaXJuYW1lLCAnLi4vLi4vYXNzZXRzL2Z1bmNzL2Nvc3QtcmVwb3J0ZXIubGFtYmRhJykpLFxuICAgIH0pO1xuICAgIHRoaXMuYWRkRW52aXJvbm1lbnQoJ0FXU19OT0RFSlNfQ09OTkVDVElPTl9SRVVTRV9FTkFCTEVEJywgJzEnLCB7IHJlbW92ZUluRWRnZTogdHJ1ZSB9KTtcbiAgfVxufSJdfQ==
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { Context } from 'aws-lambda';
|
|
2
|
+
export interface EventInput {
|
|
3
|
+
readonly Type: EventInputType;
|
|
4
|
+
}
|
|
5
|
+
export declare enum EventInputType {
|
|
6
|
+
ACCOUNTS = "Accounts",
|
|
7
|
+
SERVICES = "Services"
|
|
8
|
+
}
|
|
9
|
+
export interface MessageAttachmentField {
|
|
10
|
+
readonly title: string;
|
|
11
|
+
readonly value: string;
|
|
12
|
+
}
|
|
13
|
+
export declare const handler: (event: EventInput, context: Context) => Promise<string | Error>;
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.handler = exports.EventInputType = void 0;
|
|
4
|
+
const client_cost_explorer_1 = require("@aws-sdk/client-cost-explorer");
|
|
5
|
+
const web_api_1 = require("@slack/web-api");
|
|
6
|
+
const aws_lambda_errors_1 = require("@yicr/aws-lambda-errors");
|
|
7
|
+
const get_billing_command_1 = require("./lib/get-billing-command");
|
|
8
|
+
const get_date_range_1 = require("./lib/get-date-range");
|
|
9
|
+
var EventInputType;
|
|
10
|
+
(function (EventInputType) {
|
|
11
|
+
EventInputType["ACCOUNTS"] = "Accounts";
|
|
12
|
+
EventInputType["SERVICES"] = "Services";
|
|
13
|
+
})(EventInputType || (exports.EventInputType = EventInputType = {}));
|
|
14
|
+
const ceClient = new client_cost_explorer_1.CostExplorerClient({
|
|
15
|
+
region: 'us-east-1',
|
|
16
|
+
});
|
|
17
|
+
const handler = async (event, context) => {
|
|
18
|
+
console.log(`Event: ${JSON.stringify(event, null, 2)}`);
|
|
19
|
+
console.log(`Context: ${JSON.stringify(context, null, 2)}`);
|
|
20
|
+
// do validation
|
|
21
|
+
if (!process.env.SLACK_TOKEN) {
|
|
22
|
+
throw new aws_lambda_errors_1.MissingEnvironmentVariableError('missing environment variable SLACK_TOKEN.');
|
|
23
|
+
}
|
|
24
|
+
if (!process.env.SLACK_CHANNEL) {
|
|
25
|
+
throw new aws_lambda_errors_1.MissingEnvironmentVariableError('missing environment variable SLACK_CHANNEL.');
|
|
26
|
+
}
|
|
27
|
+
if (!event.Type) {
|
|
28
|
+
throw new aws_lambda_errors_1.MissingInputVariableError('missing input variable Type');
|
|
29
|
+
}
|
|
30
|
+
else {
|
|
31
|
+
if (!Object.values(EventInputType).includes(event.Type)) {
|
|
32
|
+
throw new aws_lambda_errors_1.InvalidInputVariableFormatError('invalid input variable format is Accounts or Services.');
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
// 👇Calculate Date Range
|
|
36
|
+
const dateRange = new get_date_range_1.GetDateRange();
|
|
37
|
+
console.log(`DateRange::${JSON.stringify(dateRange, null, 2)}`);
|
|
38
|
+
// 👇Get Total Billing
|
|
39
|
+
const totalBilling = await (new get_billing_command_1.GetTotalBilling(ceClient)).execute(dateRange);
|
|
40
|
+
console.log(`TotalBilling: ${JSON.stringify(totalBilling, null, 2)}`);
|
|
41
|
+
const fields = await (async () => {
|
|
42
|
+
switch (event.Type) {
|
|
43
|
+
case EventInputType.ACCOUNTS:
|
|
44
|
+
// 👇Get Accounts Billings
|
|
45
|
+
const accountBillings = await (new get_billing_command_1.GetAccountBillings(ceClient).execute(dateRange));
|
|
46
|
+
console.log(`AccountBillings: ${JSON.stringify(accountBillings, null, 2)}`);
|
|
47
|
+
return accountBillings?.map((value) => {
|
|
48
|
+
return {
|
|
49
|
+
title: value.account,
|
|
50
|
+
value: `${value.amount} ${value.unit}`,
|
|
51
|
+
};
|
|
52
|
+
});
|
|
53
|
+
case EventInputType.SERVICES:
|
|
54
|
+
// 👇Get Service Billings
|
|
55
|
+
const serviceBillings = await (new get_billing_command_1.GetServiceBilling(ceClient)).execute(dateRange);
|
|
56
|
+
console.log(`ServiceBilling: ${JSON.stringify(serviceBillings, null, 2)}`);
|
|
57
|
+
return serviceBillings?.map((value) => {
|
|
58
|
+
return {
|
|
59
|
+
title: value.service,
|
|
60
|
+
value: `${value.amount} ${value.unit}`,
|
|
61
|
+
};
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
})();
|
|
65
|
+
const client = new web_api_1.WebClient(process.env.SLACK_TOKEN);
|
|
66
|
+
const channel = process.env.SLACK_CHANNEL;
|
|
67
|
+
// Send the notification
|
|
68
|
+
await (async () => {
|
|
69
|
+
const result = await client.chat.postMessage({
|
|
70
|
+
channel,
|
|
71
|
+
icon_emoji: ':money-with-wings:',
|
|
72
|
+
text: `AWS Cost Reports (${dateRange.start} - ${dateRange.end})`,
|
|
73
|
+
attachments: [
|
|
74
|
+
{
|
|
75
|
+
title: ':moneybag: Total',
|
|
76
|
+
text: `${totalBilling?.amount} ${totalBilling?.unit}`,
|
|
77
|
+
color: '#ff8c00',
|
|
78
|
+
},
|
|
79
|
+
],
|
|
80
|
+
});
|
|
81
|
+
if (result.ok) {
|
|
82
|
+
await client.chat.postMessage({
|
|
83
|
+
channel,
|
|
84
|
+
thread_ts: result.ts,
|
|
85
|
+
attachments: [
|
|
86
|
+
{
|
|
87
|
+
color: '#ffd700',
|
|
88
|
+
fields: fields?.map((filed) => {
|
|
89
|
+
return {
|
|
90
|
+
title: `:aws: ${filed.title}`,
|
|
91
|
+
value: filed.value,
|
|
92
|
+
short: false,
|
|
93
|
+
};
|
|
94
|
+
}),
|
|
95
|
+
},
|
|
96
|
+
],
|
|
97
|
+
});
|
|
98
|
+
}
|
|
99
|
+
})();
|
|
100
|
+
return 'OK';
|
|
101
|
+
};
|
|
102
|
+
exports.handler = handler;
|
|
103
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cost-reporter.lambda.js","sourceRoot":"","sources":["../../src/funcs/cost-reporter.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { CostExplorerClient } from '@aws-sdk/client-cost-explorer';\nimport { WebClient } from '@slack/web-api';\nimport { MissingEnvironmentVariableError, MissingInputVariableError, InvalidInputVariableFormatError } from '@yicr/aws-lambda-errors';\nimport { Context } from 'aws-lambda';\nimport { GetAccountBillings, GetServiceBilling, GetTotalBilling } from './lib/get-billing-command';\nimport { GetDateRange } from './lib/get-date-range';\n\nexport interface EventInput {\n  readonly Type: EventInputType;\n}\n\nexport enum EventInputType {\n  ACCOUNTS = 'Accounts',\n  SERVICES = 'Services',\n}\n\nexport interface MessageAttachmentField {\n  readonly title: string;\n  readonly value: string;\n}\n\nconst ceClient = new CostExplorerClient({\n  region: 'us-east-1',\n});\n\nexport const handler = async (event: EventInput, context: Context): Promise<string | Error> => {\n  console.log(`Event: ${JSON.stringify(event, null, 2)}`);\n  console.log(`Context: ${JSON.stringify(context, null, 2)}`);\n\n  // do validation\n  if (!process.env.SLACK_TOKEN) {\n    throw new MissingEnvironmentVariableError('missing environment variable SLACK_TOKEN.');\n  }\n  if (!process.env.SLACK_CHANNEL) {\n    throw new MissingEnvironmentVariableError('missing environment variable SLACK_CHANNEL.');\n  }\n  if (!event.Type) {\n    throw new MissingInputVariableError('missing input variable Type');\n  } else {\n    if (!Object.values(EventInputType).includes(event.Type)) {\n      throw new InvalidInputVariableFormatError('invalid input variable format is Accounts or Services.');\n    }\n  }\n\n  // 👇Calculate Date Range\n  const dateRange = new GetDateRange();\n  console.log(`DateRange::${JSON.stringify(dateRange, null, 2)}`);\n\n  // 👇Get Total Billing\n  const totalBilling = await (new GetTotalBilling(ceClient)).execute(dateRange);\n  console.log(`TotalBilling: ${JSON.stringify(totalBilling, null, 2)}`);\n\n  const fields: MessageAttachmentField[] | undefined = await (async () => {\n    switch (event.Type) {\n      case EventInputType.ACCOUNTS:\n        // 👇Get Accounts Billings\n        const accountBillings = await (new GetAccountBillings(ceClient).execute(dateRange));\n        console.log(`AccountBillings: ${JSON.stringify(accountBillings, null, 2)}`);\n        return accountBillings?.map((value) => {\n          return {\n            title: value.account,\n            value: `${value.amount} ${value.unit}`,\n          };\n        });\n      case EventInputType.SERVICES:\n        // 👇Get Service Billings\n        const serviceBillings = await (new GetServiceBilling(ceClient)).execute(dateRange);\n        console.log(`ServiceBilling: ${JSON.stringify(serviceBillings, null, 2)}`);\n        return serviceBillings?.map((value) => {\n          return {\n            title: value.service,\n            value: `${value.amount} ${value.unit}`,\n          };\n        });\n    }\n  })();\n\n  const client = new WebClient(process.env.SLACK_TOKEN);\n\n  const channel = process.env.SLACK_CHANNEL;\n\n  // Send the notification\n  await (async () => {\n    const result = await client.chat.postMessage({\n      channel,\n      icon_emoji: ':money-with-wings:',\n      text: `AWS Cost Reports (${dateRange.start} - ${dateRange.end})`,\n      attachments: [\n        {\n          title: ':moneybag: Total',\n          text: `${totalBilling?.amount} ${totalBilling?.unit}`,\n          color: '#ff8c00',\n        },\n      ],\n    });\n    if (result.ok) {\n      await client.chat.postMessage({\n        channel,\n        thread_ts: result.ts,\n        attachments: [\n          {\n            color: '#ffd700',\n            fields: fields?.map((filed) => {\n              return {\n                title: `:aws: ${filed.title}`,\n                value: filed.value,\n                short: false,\n              };\n            }),\n          },\n        ],\n      });\n    }\n  })();\n\n  return 'OK';\n};\n\n"]}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { CostExplorerClient } from '@aws-sdk/client-cost-explorer';
|
|
2
|
+
import { GetDateRange } from './get-date-range';
|
|
3
|
+
export interface TotalBilling {
|
|
4
|
+
readonly unit: string;
|
|
5
|
+
readonly amount: number;
|
|
6
|
+
}
|
|
7
|
+
export declare class GetTotalBilling {
|
|
8
|
+
private client;
|
|
9
|
+
constructor(client: CostExplorerClient);
|
|
10
|
+
execute: (dateRange: GetDateRange) => Promise<TotalBilling | undefined>;
|
|
11
|
+
}
|
|
12
|
+
export interface ServiceBilling {
|
|
13
|
+
readonly service: string;
|
|
14
|
+
readonly unit: string;
|
|
15
|
+
readonly amount: number;
|
|
16
|
+
}
|
|
17
|
+
export declare class GetServiceBilling {
|
|
18
|
+
private client;
|
|
19
|
+
constructor(client: CostExplorerClient);
|
|
20
|
+
execute: (dateRange: GetDateRange, nextPageToken?: string) => Promise<ServiceBilling[] | undefined>;
|
|
21
|
+
}
|
|
22
|
+
export interface AccountBilling {
|
|
23
|
+
readonly account: string;
|
|
24
|
+
readonly amount: number;
|
|
25
|
+
readonly unit: string;
|
|
26
|
+
}
|
|
27
|
+
export declare class GetAccountBillings {
|
|
28
|
+
private client;
|
|
29
|
+
constructor(client: CostExplorerClient);
|
|
30
|
+
execute: (dateRange: GetDateRange, nextPageToken?: string) => Promise<AccountBilling[] | undefined>;
|
|
31
|
+
}
|
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.GetAccountBillings = exports.GetServiceBilling = exports.GetTotalBilling = void 0;
|
|
4
|
+
// eslint-disable-next-line import/no-extraneous-dependencies
|
|
5
|
+
const client_cost_explorer_1 = require("@aws-sdk/client-cost-explorer");
|
|
6
|
+
class GetTotalBilling {
|
|
7
|
+
constructor(client) {
|
|
8
|
+
this.client = client;
|
|
9
|
+
this.execute = async (dateRange) => {
|
|
10
|
+
const input = {
|
|
11
|
+
TimePeriod: {
|
|
12
|
+
Start: dateRange.start,
|
|
13
|
+
End: dateRange.end,
|
|
14
|
+
},
|
|
15
|
+
Granularity: 'MONTHLY',
|
|
16
|
+
Metrics: [
|
|
17
|
+
'AMORTIZED_COST',
|
|
18
|
+
],
|
|
19
|
+
};
|
|
20
|
+
console.log(`TotalBilling:Command:Input:${JSON.stringify(input)}`);
|
|
21
|
+
return this.client.send(new client_cost_explorer_1.GetCostAndUsageCommand(input))
|
|
22
|
+
.then((data) => {
|
|
23
|
+
if (data && data.ResultsByTime && data.ResultsByTime.length === 1) {
|
|
24
|
+
const cost = Object(data.ResultsByTime[0]).Total.AmortizedCost;
|
|
25
|
+
const result = {
|
|
26
|
+
unit: cost.Unit,
|
|
27
|
+
amount: cost.Amount,
|
|
28
|
+
};
|
|
29
|
+
console.log(`TotalBilling:Command:Output(Shaped):${JSON.stringify(result)}`);
|
|
30
|
+
return result;
|
|
31
|
+
}
|
|
32
|
+
return undefined;
|
|
33
|
+
})
|
|
34
|
+
.catch((error) => {
|
|
35
|
+
console.log('Error caught...');
|
|
36
|
+
console.log(`Error:${JSON.stringify(error)}`);
|
|
37
|
+
return undefined;
|
|
38
|
+
});
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
;
|
|
42
|
+
}
|
|
43
|
+
exports.GetTotalBilling = GetTotalBilling;
|
|
44
|
+
class GetServiceBilling {
|
|
45
|
+
constructor(client) {
|
|
46
|
+
this.client = client;
|
|
47
|
+
this.execute = async (dateRange, nextPageToken) => {
|
|
48
|
+
const input = {
|
|
49
|
+
NextPageToken: nextPageToken,
|
|
50
|
+
TimePeriod: {
|
|
51
|
+
Start: dateRange.start,
|
|
52
|
+
End: dateRange.end,
|
|
53
|
+
},
|
|
54
|
+
Granularity: 'MONTHLY',
|
|
55
|
+
Metrics: [
|
|
56
|
+
'AMORTIZED_COST',
|
|
57
|
+
],
|
|
58
|
+
GroupBy: [
|
|
59
|
+
{
|
|
60
|
+
Type: 'DIMENSION',
|
|
61
|
+
Key: 'SERVICE',
|
|
62
|
+
},
|
|
63
|
+
],
|
|
64
|
+
};
|
|
65
|
+
console.log(`ServiceBillings:Command:Input:${JSON.stringify(input)}`);
|
|
66
|
+
return this.client.send(new client_cost_explorer_1.GetCostAndUsageCommand(input))
|
|
67
|
+
.then(async (data) => {
|
|
68
|
+
const billings = [];
|
|
69
|
+
if (data.ResultsByTime && data.ResultsByTime.length === 1) {
|
|
70
|
+
for (const item of Object(data.ResultsByTime[0]).Groups) {
|
|
71
|
+
billings.push({
|
|
72
|
+
service: item.Keys[0],
|
|
73
|
+
unit: item.Metrics.AmortizedCost.Unit,
|
|
74
|
+
amount: item.Metrics.AmortizedCost.Amount,
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
console.log(`ServiceBillings:Command:Output(Shaped):${JSON.stringify(billings)}`);
|
|
78
|
+
if (data.NextPageToken) {
|
|
79
|
+
const nextBillings = await this.execute(dateRange, data.NextPageToken);
|
|
80
|
+
if (nextBillings) {
|
|
81
|
+
return billings.concat(nextBillings);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
return billings;
|
|
85
|
+
}
|
|
86
|
+
return undefined;
|
|
87
|
+
})
|
|
88
|
+
.catch(async (error) => {
|
|
89
|
+
console.log('Error caught...');
|
|
90
|
+
console.log(`Error:${JSON.stringify(error)}`);
|
|
91
|
+
return undefined;
|
|
92
|
+
});
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
;
|
|
96
|
+
}
|
|
97
|
+
exports.GetServiceBilling = GetServiceBilling;
|
|
98
|
+
class GetAccountBillings {
|
|
99
|
+
constructor(client) {
|
|
100
|
+
this.client = client;
|
|
101
|
+
this.execute = async (dateRange, nextPageToken) => {
|
|
102
|
+
const input = {
|
|
103
|
+
NextPageToken: nextPageToken,
|
|
104
|
+
TimePeriod: {
|
|
105
|
+
Start: dateRange.start,
|
|
106
|
+
End: dateRange.end,
|
|
107
|
+
},
|
|
108
|
+
Granularity: 'MONTHLY',
|
|
109
|
+
Metrics: [
|
|
110
|
+
'AMORTIZED_COST',
|
|
111
|
+
],
|
|
112
|
+
GroupBy: [
|
|
113
|
+
{
|
|
114
|
+
Type: 'DIMENSION',
|
|
115
|
+
Key: 'LINKED_ACCOUNT',
|
|
116
|
+
},
|
|
117
|
+
],
|
|
118
|
+
};
|
|
119
|
+
console.log(`AccountBillings:Command:Input:${JSON.stringify(input)}`);
|
|
120
|
+
return this.client.send(new client_cost_explorer_1.GetCostAndUsageCommand(input))
|
|
121
|
+
.then(async (data) => {
|
|
122
|
+
const billings = [];
|
|
123
|
+
if (data.ResultsByTime && data.ResultsByTime.length === 1) {
|
|
124
|
+
const groups = Object(data.ResultsByTime[0]).Groups;
|
|
125
|
+
const dimensionValueAttributes = data.DimensionValueAttributes;
|
|
126
|
+
for (const item of groups) {
|
|
127
|
+
for (const attr of dimensionValueAttributes) {
|
|
128
|
+
if (item.Keys[0] === attr.Value) {
|
|
129
|
+
billings.push({
|
|
130
|
+
account: `${attr.Value} (${attr.Attributes.description})`,
|
|
131
|
+
unit: item.Metrics.AmortizedCost.Unit,
|
|
132
|
+
amount: item.Metrics.AmortizedCost.Amount,
|
|
133
|
+
});
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
console.log(`AccountBillings:Command:Output(Shaped):${JSON.stringify(billings)}`);
|
|
138
|
+
if (data.NextPageToken) {
|
|
139
|
+
const nextBillings = await this.execute(dateRange, data.NextPageToken);
|
|
140
|
+
if (nextBillings) {
|
|
141
|
+
return billings.concat(nextBillings);
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
return billings;
|
|
145
|
+
}
|
|
146
|
+
return undefined;
|
|
147
|
+
})
|
|
148
|
+
.catch(async (error) => {
|
|
149
|
+
console.log('Error caught...');
|
|
150
|
+
console.log(`Error:${JSON.stringify(error)}`);
|
|
151
|
+
return undefined;
|
|
152
|
+
});
|
|
153
|
+
};
|
|
154
|
+
}
|
|
155
|
+
;
|
|
156
|
+
}
|
|
157
|
+
exports.GetAccountBillings = GetAccountBillings;
|
|
158
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"get-billing-command.js","sourceRoot":"","sources":["../../../src/funcs/lib/get-billing-command.ts"],"names":[],"mappings":"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","sourcesContent":["// eslint-disable-next-line import/no-extraneous-dependencies\nimport { CostExplorerClient, GetCostAndUsageCommand, GetCostAndUsageCommandInput, GetCostAndUsageCommandOutput } from '@aws-sdk/client-cost-explorer';\nimport { GetDateRange } from './get-date-range';\n\nexport interface TotalBilling {\n  readonly unit: string;\n  readonly amount: number;\n}\n\nexport class GetTotalBilling {\n\n  constructor(\n    private client: CostExplorerClient,\n  ) {};\n\n  public execute = async (dateRange: GetDateRange): Promise<TotalBilling | undefined> => {\n    const input: GetCostAndUsageCommandInput = {\n      TimePeriod: {\n        Start: dateRange.start,\n        End: dateRange.end,\n      },\n      Granularity: 'MONTHLY',\n      Metrics: [\n        'AMORTIZED_COST',\n      ],\n    };\n    console.log(`TotalBilling:Command:Input:${JSON.stringify(input)}`);\n    return this.client.send(new GetCostAndUsageCommand(input))\n      .then((data: GetCostAndUsageCommandOutput) => {\n        if (data && data.ResultsByTime && data.ResultsByTime.length === 1) {\n          const cost = Object(data.ResultsByTime[0]).Total.AmortizedCost;\n          const result: TotalBilling = {\n            unit: cost.Unit,\n            amount: cost.Amount,\n          };\n          console.log(`TotalBilling:Command:Output(Shaped):${JSON.stringify(result)}`);\n          return result;\n        }\n        return undefined;\n      })\n      .catch((error) => {\n        console.log('Error caught...');\n        console.log(`Error:${JSON.stringify(error)}`);\n        return undefined;\n      });\n  };\n}\n\nexport interface ServiceBilling {\n  readonly service: string;\n  readonly unit: string;\n  readonly amount: number;\n}\n\nexport class GetServiceBilling {\n  constructor(\n    private client: CostExplorerClient,\n  ) {};\n\n  public execute = async (dateRange: GetDateRange, nextPageToken?: string): Promise<ServiceBilling[] | undefined> => {\n    const input: GetCostAndUsageCommandInput = {\n      NextPageToken: nextPageToken,\n      TimePeriod: {\n        Start: dateRange.start,\n        End: dateRange.end,\n      },\n      Granularity: 'MONTHLY',\n      Metrics: [\n        'AMORTIZED_COST',\n      ],\n      GroupBy: [\n        {\n          Type: 'DIMENSION',\n          Key: 'SERVICE',\n        },\n      ],\n    };\n    console.log(`ServiceBillings:Command:Input:${JSON.stringify(input)}`);\n    return this.client.send(new GetCostAndUsageCommand(input))\n      .then(async (data) => {\n        const billings: ServiceBilling[] = [];\n        if (data.ResultsByTime && data.ResultsByTime.length === 1) {\n          for (const item of Object(data.ResultsByTime[0]).Groups) {\n            billings.push({\n              service: item.Keys[0],\n              unit: item.Metrics.AmortizedCost.Unit,\n              amount: item.Metrics.AmortizedCost.Amount,\n            });\n          }\n          console.log(`ServiceBillings:Command:Output(Shaped):${JSON.stringify(billings)}`);\n          if (data.NextPageToken) {\n            const nextBillings = await this.execute(dateRange, data.NextPageToken);\n            if (nextBillings) {\n              return billings.concat(nextBillings);\n            }\n          }\n          return billings;\n        }\n        return undefined;\n      })\n      .catch(async (error) => {\n        console.log('Error caught...');\n        console.log(`Error:${JSON.stringify(error)}`);\n        return undefined;\n      });\n  };\n}\n\nexport interface AccountBilling {\n  readonly account: string;\n  readonly amount: number;\n  readonly unit: string;\n}\n\nexport class GetAccountBillings {\n  constructor(\n    private client: CostExplorerClient,\n  ) {};\n\n  public execute = async (dateRange: GetDateRange, nextPageToken?: string): Promise<AccountBilling[] | undefined> => {\n    const input: GetCostAndUsageCommandInput = {\n      NextPageToken: nextPageToken,\n      TimePeriod: {\n        Start: dateRange.start,\n        End: dateRange.end,\n      },\n      Granularity: 'MONTHLY',\n      Metrics: [\n        'AMORTIZED_COST',\n      ],\n      GroupBy: [\n        {\n          Type: 'DIMENSION',\n          Key: 'LINKED_ACCOUNT',\n        },\n      ],\n    };\n    console.log(`AccountBillings:Command:Input:${JSON.stringify(input)}`);\n    return this.client.send(new GetCostAndUsageCommand(input))\n      .then(async (data) => {\n        const billings: AccountBilling[] = [];\n        if (data.ResultsByTime && data.ResultsByTime.length === 1) {\n          const groups = Object(data.ResultsByTime[0]).Groups;\n          const dimensionValueAttributes = data.DimensionValueAttributes!;\n          for (const item of groups) {\n            for (const attr of dimensionValueAttributes) {\n              if (item.Keys[0] === attr.Value) {\n                billings.push({\n                  account: `${attr.Value} (${attr.Attributes!.description})`,\n                  unit: item.Metrics.AmortizedCost.Unit,\n                  amount: item.Metrics.AmortizedCost.Amount,\n                });\n              }\n            }\n          }\n          console.log(`AccountBillings:Command:Output(Shaped):${JSON.stringify(billings)}`);\n          if (data.NextPageToken) {\n            const nextBillings = await this.execute(dateRange, data.NextPageToken);\n            if (nextBillings) {\n              return billings.concat(nextBillings);\n            }\n          }\n          return billings;\n        }\n        return undefined;\n      })\n      .catch(async (error) => {\n        console.log('Error caught...');\n        console.log(`Error:${JSON.stringify(error)}`);\n        return undefined;\n      });\n  };\n}"]}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.GetDateRange = void 0;
|
|
4
|
+
class GetDateRange {
|
|
5
|
+
constructor() {
|
|
6
|
+
this.dateFormatString = (date) => {
|
|
7
|
+
return (date.getFullYear()) + '-' + ('00' + (date.getMonth() + 1)).slice(-2) + '-' + ('00' + (date.getDate())).slice(-2);
|
|
8
|
+
};
|
|
9
|
+
const now = new Date(Date.now());
|
|
10
|
+
if (now.getDate() === 1) {
|
|
11
|
+
// Last month
|
|
12
|
+
this.start = this.dateFormatString(new Date(now.getFullYear(), now.getMonth() - 1, 1));
|
|
13
|
+
this.end = this.dateFormatString(new Date(now.getFullYear(), now.getMonth(), 0));
|
|
14
|
+
}
|
|
15
|
+
else {
|
|
16
|
+
this.start = this.dateFormatString(new Date(now.getFullYear(), now.getMonth(), 1));
|
|
17
|
+
this.end = this.dateFormatString(new Date(now.getFullYear(), now.getMonth(), now.getDate() - 1));
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
exports.GetDateRange = GetDateRange;
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2V0LWRhdGUtcmFuZ2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZnVuY3MvbGliL2dldC1kYXRlLXJhbmdlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVBLE1BQWEsWUFBWTtJQUt2QjtRQVlRLHFCQUFnQixHQUFHLENBQUMsSUFBVSxFQUFVLEVBQUU7WUFDaEQsT0FBTyxDQUFDLElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDLElBQUksR0FBRyxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDLElBQUksR0FBRyxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDM0gsQ0FBQyxDQUFDO1FBYkEsTUFBTSxHQUFHLEdBQUcsSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUM7UUFDakMsSUFBSSxHQUFHLENBQUMsT0FBTyxFQUFFLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDeEIsYUFBYTtZQUNiLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLGdCQUFnQixDQUFDLElBQUksSUFBSSxDQUFDLEdBQUcsQ0FBQyxXQUFXLEVBQUUsRUFBRSxHQUFHLENBQUMsUUFBUSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDdkYsSUFBSSxDQUFDLEdBQUcsR0FBRyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsSUFBSSxJQUFJLENBQUMsR0FBRyxDQUFDLFdBQVcsRUFBRSxFQUFFLEdBQUcsQ0FBQyxRQUFRLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ25GLENBQUM7YUFBTSxDQUFDO1lBQ04sSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsSUFBSSxJQUFJLENBQUMsR0FBRyxDQUFDLFdBQVcsRUFBRSxFQUFFLEdBQUcsQ0FBQyxRQUFRLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ25GLElBQUksQ0FBQyxHQUFHLEdBQUcsSUFBSSxDQUFDLGdCQUFnQixDQUFDLElBQUksSUFBSSxDQUFDLEdBQUcsQ0FBQyxXQUFXLEVBQUUsRUFBRSxHQUFHLENBQUMsUUFBUSxFQUFFLEVBQUUsR0FBRyxDQUFDLE9BQU8sRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDbkcsQ0FBQztJQUNILENBQUM7Q0FLRjtBQXBCRCxvQ0FvQkMiLCJzb3VyY2VzQ29udGVudCI6WyJcblxuZXhwb3J0IGNsYXNzIEdldERhdGVSYW5nZSB7XG5cbiAgcHVibGljIHN0YXJ0OiBzdHJpbmc7XG4gIHB1YmxpYyBlbmQ6IHN0cmluZztcblxuICBjb25zdHJ1Y3RvcigpIHtcbiAgICBjb25zdCBub3cgPSBuZXcgRGF0ZShEYXRlLm5vdygpKTtcbiAgICBpZiAobm93LmdldERhdGUoKSA9PT0gMSkge1xuICAgICAgLy8gTGFzdCBtb250aFxuICAgICAgdGhpcy5zdGFydCA9IHRoaXMuZGF0ZUZvcm1hdFN0cmluZyhuZXcgRGF0ZShub3cuZ2V0RnVsbFllYXIoKSwgbm93LmdldE1vbnRoKCkgLSAxLCAxKSk7XG4gICAgICB0aGlzLmVuZCA9IHRoaXMuZGF0ZUZvcm1hdFN0cmluZyhuZXcgRGF0ZShub3cuZ2V0RnVsbFllYXIoKSwgbm93LmdldE1vbnRoKCksIDApKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5zdGFydCA9IHRoaXMuZGF0ZUZvcm1hdFN0cmluZyhuZXcgRGF0ZShub3cuZ2V0RnVsbFllYXIoKSwgbm93LmdldE1vbnRoKCksIDEpKTtcbiAgICAgIHRoaXMuZW5kID0gdGhpcy5kYXRlRm9ybWF0U3RyaW5nKG5ldyBEYXRlKG5vdy5nZXRGdWxsWWVhcigpLCBub3cuZ2V0TW9udGgoKSwgbm93LmdldERhdGUoKSAtIDEpKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGRhdGVGb3JtYXRTdHJpbmcgPSAoZGF0ZTogRGF0ZSk6IHN0cmluZyA9PiB7XG4gICAgcmV0dXJuIChkYXRlLmdldEZ1bGxZZWFyKCkpICsgJy0nICsgKCcwMCcgKyAoZGF0ZS5nZXRNb250aCgpICsgMSkpLnNsaWNlKC0yKSArICctJyArICgnMDAnICsgKGRhdGUuZ2V0RGF0ZSgpKSkuc2xpY2UoLTIpO1xuICB9O1xufSJdfQ==
|
package/lib/index.d.ts
ADDED
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { Stack } from 'aws-cdk-lib';
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
export interface DailyCostUsageReportStackProps {
|
|
4
|
+
readonly slackToken: string;
|
|
5
|
+
readonly slackChannel: string;
|
|
6
|
+
readonly scheduleTimezone?: string;
|
|
7
|
+
readonly costGroupType: CostGroupType;
|
|
8
|
+
}
|
|
9
|
+
export declare enum CostGroupType {
|
|
10
|
+
ACCOUNTS = "Accounts",
|
|
11
|
+
SERVICES = "Services"
|
|
12
|
+
}
|
|
13
|
+
export declare class DailyCostUsageReportStack extends Stack {
|
|
14
|
+
constructor(scope: Construct, id: string, props: DailyCostUsageReportStackProps);
|
|
15
|
+
}
|
package/lib/index.js
ADDED
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.DailyCostUsageReportStack = exports.CostGroupType = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
7
|
+
const iam = require("aws-cdk-lib/aws-iam");
|
|
8
|
+
const scheduler = require("aws-cdk-lib/aws-scheduler");
|
|
9
|
+
const cost_reporter_function_1 = require("./funcs/cost-reporter-function");
|
|
10
|
+
var CostGroupType;
|
|
11
|
+
(function (CostGroupType) {
|
|
12
|
+
CostGroupType["ACCOUNTS"] = "Accounts";
|
|
13
|
+
CostGroupType["SERVICES"] = "Services";
|
|
14
|
+
})(CostGroupType || (exports.CostGroupType = CostGroupType = {}));
|
|
15
|
+
class DailyCostUsageReportStack extends aws_cdk_lib_1.Stack {
|
|
16
|
+
constructor(scope, id, props) {
|
|
17
|
+
super(scope, id);
|
|
18
|
+
// 👇Get current account & region
|
|
19
|
+
// const account = Stack.of(this).account;
|
|
20
|
+
// const region = cdk.Stack.of(this).region;
|
|
21
|
+
// 👇Lambda Exec Role
|
|
22
|
+
const lambdaExecutionRole = new iam.Role(this, 'LambdaExecutionRole', {
|
|
23
|
+
roleName: undefined,
|
|
24
|
+
description: '',
|
|
25
|
+
assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),
|
|
26
|
+
managedPolicies: [
|
|
27
|
+
iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole'),
|
|
28
|
+
],
|
|
29
|
+
inlinePolicies: {
|
|
30
|
+
['get-cost-usage']: new iam.PolicyDocument({
|
|
31
|
+
statements: [
|
|
32
|
+
new iam.PolicyStatement({
|
|
33
|
+
effect: iam.Effect.ALLOW,
|
|
34
|
+
actions: [
|
|
35
|
+
'ce:GetCostAndUsage',
|
|
36
|
+
],
|
|
37
|
+
resources: ['*'],
|
|
38
|
+
}),
|
|
39
|
+
],
|
|
40
|
+
}),
|
|
41
|
+
},
|
|
42
|
+
});
|
|
43
|
+
// 👇Lambda Function
|
|
44
|
+
const lambdaFunction = new cost_reporter_function_1.CostReporterFunction(this, 'CostReporterFunction', {
|
|
45
|
+
functionName: undefined,
|
|
46
|
+
description: 'A function to archive logs s3 bucket from CloudWatch Logs.',
|
|
47
|
+
environment: {
|
|
48
|
+
//BUCKET_NAME: logArchiveBucket.bucketName,
|
|
49
|
+
SLACK_TOKEN: props.slackToken,
|
|
50
|
+
SLACK_CHANNEL: props.slackChannel,
|
|
51
|
+
},
|
|
52
|
+
role: lambdaExecutionRole,
|
|
53
|
+
timeout: aws_cdk_lib_1.Duration.seconds(45),
|
|
54
|
+
});
|
|
55
|
+
// 👇EventBridge Scheduler IAM Role
|
|
56
|
+
const schedulerExecutionRole = new iam.Role(this, 'SchedulerExecutionRole', {
|
|
57
|
+
roleName: undefined,
|
|
58
|
+
assumedBy: new iam.ServicePrincipal('scheduler.amazonaws.com'),
|
|
59
|
+
inlinePolicies: {
|
|
60
|
+
['lambda-invoke-policy']: new iam.PolicyDocument({
|
|
61
|
+
statements: [
|
|
62
|
+
new iam.PolicyStatement({
|
|
63
|
+
effect: iam.Effect.ALLOW,
|
|
64
|
+
actions: [
|
|
65
|
+
'lambda:InvokeFunction',
|
|
66
|
+
],
|
|
67
|
+
resources: [
|
|
68
|
+
lambdaFunction.functionArn,
|
|
69
|
+
`${lambdaFunction.functionArn}:*`,
|
|
70
|
+
],
|
|
71
|
+
}),
|
|
72
|
+
],
|
|
73
|
+
}),
|
|
74
|
+
},
|
|
75
|
+
});
|
|
76
|
+
// 👇Schedule
|
|
77
|
+
new scheduler.CfnSchedule(this, 'Schedule', {
|
|
78
|
+
name: undefined,
|
|
79
|
+
description: 'aws account const reports.',
|
|
80
|
+
state: 'ENABLED',
|
|
81
|
+
//groupName: scheduleGroup.name, // default
|
|
82
|
+
flexibleTimeWindow: {
|
|
83
|
+
mode: 'OFF',
|
|
84
|
+
},
|
|
85
|
+
scheduleExpressionTimezone: props.scheduleTimezone ?? 'UTC',
|
|
86
|
+
scheduleExpression: 'cron(1 9 * * ? *)',
|
|
87
|
+
target: {
|
|
88
|
+
arn: lambdaFunction.functionArn,
|
|
89
|
+
roleArn: schedulerExecutionRole.roleArn,
|
|
90
|
+
input: JSON.stringify({ Type: props.costGroupType }),
|
|
91
|
+
retryPolicy: {
|
|
92
|
+
maximumEventAgeInSeconds: 60,
|
|
93
|
+
maximumRetryAttempts: 0,
|
|
94
|
+
},
|
|
95
|
+
},
|
|
96
|
+
});
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
exports.DailyCostUsageReportStack = DailyCostUsageReportStack;
|
|
100
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
101
|
+
DailyCostUsageReportStack[_a] = { fqn: "@gammarers/aws-daily-cost-usage-report-stack.DailyCostUsageReportStack", version: "2.0.0" };
|
|
102
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFBQSw2Q0FBOEM7QUFDOUMsMkNBQTJDO0FBQzNDLHVEQUF1RDtBQUV2RCwyRUFBc0U7QUFTdEUsSUFBWSxhQUdYO0FBSEQsV0FBWSxhQUFhO0lBQ3ZCLHNDQUFxQixDQUFBO0lBQ3JCLHNDQUFxQixDQUFBO0FBQ3ZCLENBQUMsRUFIVyxhQUFhLDZCQUFiLGFBQWEsUUFHeEI7QUFFRCxNQUFhLHlCQUEwQixTQUFRLG1CQUFLO0lBQ2xELFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBcUM7UUFDN0UsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixpQ0FBaUM7UUFDakMsMENBQTBDO1FBQzFDLDRDQUE0QztRQUU1QyxxQkFBcUI7UUFDckIsTUFBTSxtQkFBbUIsR0FBRyxJQUFJLEdBQUcsQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLHFCQUFxQixFQUFFO1lBQ3BFLFFBQVEsRUFBRSxTQUFTO1lBQ25CLFdBQVcsRUFBRSxFQUFFO1lBQ2YsU0FBUyxFQUFFLElBQUksR0FBRyxDQUFDLGdCQUFnQixDQUFDLHNCQUFzQixDQUFDO1lBQzNELGVBQWUsRUFBRTtnQkFDZixHQUFHLENBQUMsYUFBYSxDQUFDLHdCQUF3QixDQUFDLDBDQUEwQyxDQUFDO2FBQ3ZGO1lBQ0QsY0FBYyxFQUFFO2dCQUNkLENBQUMsZ0JBQWdCLENBQUMsRUFBRSxJQUFJLEdBQUcsQ0FBQyxjQUFjLENBQUM7b0JBQ3pDLFVBQVUsRUFBRTt3QkFDVixJQUFJLEdBQUcsQ0FBQyxlQUFlLENBQUM7NEJBQ3RCLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTSxDQUFDLEtBQUs7NEJBQ3hCLE9BQU8sRUFBRTtnQ0FDUCxvQkFBb0I7NkJBQ3JCOzRCQUNELFNBQVMsRUFBRSxDQUFDLEdBQUcsQ0FBQzt5QkFDakIsQ0FBQztxQkFDSDtpQkFDRixDQUFDO2FBQ0g7U0FDRixDQUFDLENBQUM7UUFFSCxvQkFBb0I7UUFDcEIsTUFBTSxjQUFjLEdBQUcsSUFBSSw2Q0FBb0IsQ0FBQyxJQUFJLEVBQUUsc0JBQXNCLEVBQUU7WUFDNUUsWUFBWSxFQUFFLFNBQVM7WUFDdkIsV0FBVyxFQUFFLDREQUE0RDtZQUN6RSxXQUFXLEVBQUU7Z0JBQ1gsMkNBQTJDO2dCQUMzQyxXQUFXLEVBQUUsS0FBSyxDQUFDLFVBQVU7Z0JBQzdCLGFBQWEsRUFBRSxLQUFLLENBQUMsWUFBWTthQUNsQztZQUNELElBQUksRUFBRSxtQkFBbUI7WUFDekIsT0FBTyxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztTQUM5QixDQUFDLENBQUM7UUFFSCxtQ0FBbUM7UUFDbkMsTUFBTSxzQkFBc0IsR0FBRyxJQUFJLEdBQUcsQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLHdCQUF3QixFQUFFO1lBQzFFLFFBQVEsRUFBRSxTQUFTO1lBQ25CLFNBQVMsRUFBRSxJQUFJLEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyx5QkFBeUIsQ0FBQztZQUM5RCxjQUFjLEVBQUU7Z0JBQ2QsQ0FBQyxzQkFBc0IsQ0FBQyxFQUFFLElBQUksR0FBRyxDQUFDLGNBQWMsQ0FBQztvQkFDL0MsVUFBVSxFQUFFO3dCQUNWLElBQUksR0FBRyxDQUFDLGVBQWUsQ0FBQzs0QkFDdEIsTUFBTSxFQUFFLEdBQUcsQ0FBQyxNQUFNLENBQUMsS0FBSzs0QkFDeEIsT0FBTyxFQUFFO2dDQUNQLHVCQUF1Qjs2QkFDeEI7NEJBQ0QsU0FBUyxFQUFFO2dDQUNULGNBQWMsQ0FBQyxXQUFXO2dDQUMxQixHQUFHLGNBQWMsQ0FBQyxXQUFXLElBQUk7NkJBQ2xDO3lCQUNGLENBQUM7cUJBQ0g7aUJBQ0YsQ0FBQzthQUNIO1NBQ0YsQ0FBQyxDQUFDO1FBRUgsYUFBYTtRQUNiLElBQUksU0FBUyxDQUFDLFdBQVcsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO1lBQzFDLElBQUksRUFBRSxTQUFTO1lBQ2YsV0FBVyxFQUFFLDRCQUE0QjtZQUN6QyxLQUFLLEVBQUUsU0FBUztZQUNoQiwyQ0FBMkM7WUFDM0Msa0JBQWtCLEVBQUU7Z0JBQ2xCLElBQUksRUFBRSxLQUFLO2FBQ1o7WUFDRCwwQkFBMEIsRUFBRSxLQUFLLENBQUMsZ0JBQWdCLElBQUksS0FBSztZQUMzRCxrQkFBa0IsRUFBRSxtQkFBbUI7WUFDdkMsTUFBTSxFQUFFO2dCQUNOLEdBQUcsRUFBRSxjQUFjLENBQUMsV0FBVztnQkFDL0IsT0FBTyxFQUFFLHNCQUFzQixDQUFDLE9BQU87Z0JBQ3ZDLEtBQUssRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDLEVBQUUsSUFBSSxFQUFFLEtBQUssQ0FBQyxhQUFhLEVBQUUsQ0FBQztnQkFDcEQsV0FBVyxFQUFFO29CQUNYLHdCQUF3QixFQUFFLEVBQUU7b0JBQzVCLG9CQUFvQixFQUFFLENBQUM7aUJBQ3hCO2FBQ0Y7U0FDRixDQUFDLENBQUM7SUFDTCxDQUFDOztBQXZGSCw4REF5RkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEdXJhdGlvbiwgU3RhY2sgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgKiBhcyBpYW0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWlhbSc7XG5pbXBvcnQgKiBhcyBzY2hlZHVsZXIgZnJvbSAnYXdzLWNkay1saWIvYXdzLXNjaGVkdWxlcic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IENvc3RSZXBvcnRlckZ1bmN0aW9uIH0gZnJvbSAnLi9mdW5jcy9jb3N0LXJlcG9ydGVyLWZ1bmN0aW9uJztcblxuZXhwb3J0IGludGVyZmFjZSBEYWlseUNvc3RVc2FnZVJlcG9ydFN0YWNrUHJvcHMge1xuICByZWFkb25seSBzbGFja1Rva2VuOiBzdHJpbmc7XG4gIHJlYWRvbmx5IHNsYWNrQ2hhbm5lbDogc3RyaW5nO1xuICByZWFkb25seSBzY2hlZHVsZVRpbWV6b25lPzogc3RyaW5nO1xuICByZWFkb25seSBjb3N0R3JvdXBUeXBlOiBDb3N0R3JvdXBUeXBlO1xufVxuXG5leHBvcnQgZW51bSBDb3N0R3JvdXBUeXBlIHtcbiAgQUNDT1VOVFMgPSAnQWNjb3VudHMnLFxuICBTRVJWSUNFUyA9ICdTZXJ2aWNlcycsXG59XG5cbmV4cG9ydCBjbGFzcyBEYWlseUNvc3RVc2FnZVJlcG9ydFN0YWNrIGV4dGVuZHMgU3RhY2sge1xuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogRGFpbHlDb3N0VXNhZ2VSZXBvcnRTdGFja1Byb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgIC8vIPCfkYdHZXQgY3VycmVudCBhY2NvdW50ICYgcmVnaW9uXG4gICAgLy8gY29uc3QgYWNjb3VudCA9IFN0YWNrLm9mKHRoaXMpLmFjY291bnQ7XG4gICAgLy8gY29uc3QgcmVnaW9uID0gY2RrLlN0YWNrLm9mKHRoaXMpLnJlZ2lvbjtcblxuICAgIC8vIPCfkYdMYW1iZGEgRXhlYyBSb2xlXG4gICAgY29uc3QgbGFtYmRhRXhlY3V0aW9uUm9sZSA9IG5ldyBpYW0uUm9sZSh0aGlzLCAnTGFtYmRhRXhlY3V0aW9uUm9sZScsIHtcbiAgICAgIHJvbGVOYW1lOiB1bmRlZmluZWQsXG4gICAgICBkZXNjcmlwdGlvbjogJycsXG4gICAgICBhc3N1bWVkQnk6IG5ldyBpYW0uU2VydmljZVByaW5jaXBhbCgnbGFtYmRhLmFtYXpvbmF3cy5jb20nKSxcbiAgICAgIG1hbmFnZWRQb2xpY2llczogW1xuICAgICAgICBpYW0uTWFuYWdlZFBvbGljeS5mcm9tQXdzTWFuYWdlZFBvbGljeU5hbWUoJ3NlcnZpY2Utcm9sZS9BV1NMYW1iZGFCYXNpY0V4ZWN1dGlvblJvbGUnKSxcbiAgICAgIF0sXG4gICAgICBpbmxpbmVQb2xpY2llczoge1xuICAgICAgICBbJ2dldC1jb3N0LXVzYWdlJ106IG5ldyBpYW0uUG9saWN5RG9jdW1lbnQoe1xuICAgICAgICAgIHN0YXRlbWVudHM6IFtcbiAgICAgICAgICAgIG5ldyBpYW0uUG9saWN5U3RhdGVtZW50KHtcbiAgICAgICAgICAgICAgZWZmZWN0OiBpYW0uRWZmZWN0LkFMTE9XLFxuICAgICAgICAgICAgICBhY3Rpb25zOiBbXG4gICAgICAgICAgICAgICAgJ2NlOkdldENvc3RBbmRVc2FnZScsXG4gICAgICAgICAgICAgIF0sXG4gICAgICAgICAgICAgIHJlc291cmNlczogWycqJ10sXG4gICAgICAgICAgICB9KSxcbiAgICAgICAgICBdLFxuICAgICAgICB9KSxcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICAvLyDwn5GHTGFtYmRhIEZ1bmN0aW9uXG4gICAgY29uc3QgbGFtYmRhRnVuY3Rpb24gPSBuZXcgQ29zdFJlcG9ydGVyRnVuY3Rpb24odGhpcywgJ0Nvc3RSZXBvcnRlckZ1bmN0aW9uJywge1xuICAgICAgZnVuY3Rpb25OYW1lOiB1bmRlZmluZWQsXG4gICAgICBkZXNjcmlwdGlvbjogJ0EgZnVuY3Rpb24gdG8gYXJjaGl2ZSBsb2dzIHMzIGJ1Y2tldCBmcm9tIENsb3VkV2F0Y2ggTG9ncy4nLFxuICAgICAgZW52aXJvbm1lbnQ6IHtcbiAgICAgICAgLy9CVUNLRVRfTkFNRTogbG9nQXJjaGl2ZUJ1Y2tldC5idWNrZXROYW1lLFxuICAgICAgICBTTEFDS19UT0tFTjogcHJvcHMuc2xhY2tUb2tlbixcbiAgICAgICAgU0xBQ0tfQ0hBTk5FTDogcHJvcHMuc2xhY2tDaGFubmVsLFxuICAgICAgfSxcbiAgICAgIHJvbGU6IGxhbWJkYUV4ZWN1dGlvblJvbGUsXG4gICAgICB0aW1lb3V0OiBEdXJhdGlvbi5zZWNvbmRzKDQ1KSxcbiAgICB9KTtcblxuICAgIC8vIPCfkYdFdmVudEJyaWRnZSBTY2hlZHVsZXIgSUFNIFJvbGVcbiAgICBjb25zdCBzY2hlZHVsZXJFeGVjdXRpb25Sb2xlID0gbmV3IGlhbS5Sb2xlKHRoaXMsICdTY2hlZHVsZXJFeGVjdXRpb25Sb2xlJywge1xuICAgICAgcm9sZU5hbWU6IHVuZGVmaW5lZCxcbiAgICAgIGFzc3VtZWRCeTogbmV3IGlhbS5TZXJ2aWNlUHJpbmNpcGFsKCdzY2hlZHVsZXIuYW1hem9uYXdzLmNvbScpLFxuICAgICAgaW5saW5lUG9saWNpZXM6IHtcbiAgICAgICAgWydsYW1iZGEtaW52b2tlLXBvbGljeSddOiBuZXcgaWFtLlBvbGljeURvY3VtZW50KHtcbiAgICAgICAgICBzdGF0ZW1lbnRzOiBbXG4gICAgICAgICAgICBuZXcgaWFtLlBvbGljeVN0YXRlbWVudCh7XG4gICAgICAgICAgICAgIGVmZmVjdDogaWFtLkVmZmVjdC5BTExPVyxcbiAgICAgICAgICAgICAgYWN0aW9uczogW1xuICAgICAgICAgICAgICAgICdsYW1iZGE6SW52b2tlRnVuY3Rpb24nLFxuICAgICAgICAgICAgICBdLFxuICAgICAgICAgICAgICByZXNvdXJjZXM6IFtcbiAgICAgICAgICAgICAgICBsYW1iZGFGdW5jdGlvbi5mdW5jdGlvbkFybixcbiAgICAgICAgICAgICAgICBgJHtsYW1iZGFGdW5jdGlvbi5mdW5jdGlvbkFybn06KmAsXG4gICAgICAgICAgICAgIF0sXG4gICAgICAgICAgICB9KSxcbiAgICAgICAgICBdLFxuICAgICAgICB9KSxcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICAvLyDwn5GHU2NoZWR1bGVcbiAgICBuZXcgc2NoZWR1bGVyLkNmblNjaGVkdWxlKHRoaXMsICdTY2hlZHVsZScsIHtcbiAgICAgIG5hbWU6IHVuZGVmaW5lZCxcbiAgICAgIGRlc2NyaXB0aW9uOiAnYXdzIGFjY291bnQgY29uc3QgcmVwb3J0cy4nLFxuICAgICAgc3RhdGU6ICdFTkFCTEVEJyxcbiAgICAgIC8vZ3JvdXBOYW1lOiBzY2hlZHVsZUdyb3VwLm5hbWUsIC8vIGRlZmF1bHRcbiAgICAgIGZsZXhpYmxlVGltZVdpbmRvdzoge1xuICAgICAgICBtb2RlOiAnT0ZGJyxcbiAgICAgIH0sXG4gICAgICBzY2hlZHVsZUV4cHJlc3Npb25UaW1lem9uZTogcHJvcHMuc2NoZWR1bGVUaW1lem9uZSA/PyAnVVRDJyxcbiAgICAgIHNjaGVkdWxlRXhwcmVzc2lvbjogJ2Nyb24oMSA5ICogKiA/ICopJyxcbiAgICAgIHRhcmdldDoge1xuICAgICAgICBhcm46IGxhbWJkYUZ1bmN0aW9uLmZ1bmN0aW9uQXJuLFxuICAgICAgICByb2xlQXJuOiBzY2hlZHVsZXJFeGVjdXRpb25Sb2xlLnJvbGVBcm4sXG4gICAgICAgIGlucHV0OiBKU09OLnN0cmluZ2lmeSh7IFR5cGU6IHByb3BzLmNvc3RHcm91cFR5cGUgfSksXG4gICAgICAgIHJldHJ5UG9saWN5OiB7XG4gICAgICAgICAgbWF4aW11bUV2ZW50QWdlSW5TZWNvbmRzOiA2MCxcbiAgICAgICAgICBtYXhpbXVtUmV0cnlBdHRlbXB0czogMCxcbiAgICAgICAgfSxcbiAgICAgIH0sXG4gICAgfSk7XG4gIH1cblxufSJdfQ==
|