@liflig/cdk 1.48.1 → 1.51.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/lib/index.d.ts CHANGED
@@ -7,6 +7,7 @@ import * as webapp from "./webapp";
7
7
  import * as configureParameters from "./configure-parameters";
8
8
  import * as ecs from "./ecs";
9
9
  import * as loadBalancer from "./load-balancer";
10
+ import * as cloudTrailSlackIntegration from "./cloudtrail-slack-integration";
10
11
  import * as rds from "./rds";
11
12
  import * as platform from "./platform";
12
13
  export { BastionHost } from "./bastion-host";
@@ -20,7 +21,7 @@ export { SsmParameterBackedResource } from "./ssm-parameter-backed-resource";
20
21
  export { SsmParameterReader } from "./ssm-parameter-reader";
21
22
  export { tagResources } from "./tags";
22
23
  export { WebappDeployViaRole } from "./webapp-deploy-via-role";
23
- export { alarms, cdkPipelines, griid, pipelines, ses, webapp, configureParameters, ecs, loadBalancer, rds, platform, };
24
+ export { alarms, cdkPipelines, griid, pipelines, ses, webapp, configureParameters, ecs, loadBalancer, rds, platform, cloudTrailSlackIntegration, };
24
25
  /**
25
26
  * Check if we are synthesizing a snapshot by setting IS_SNAPSHOT
26
27
  * environment variable to true.
package/lib/index.js CHANGED
@@ -10,7 +10,7 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
10
10
  for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
11
11
  };
12
12
  Object.defineProperty(exports, "__esModule", { value: true });
13
- exports.isSnapshot = exports.platform = exports.rds = exports.loadBalancer = exports.ecs = exports.configureParameters = exports.webapp = exports.ses = exports.pipelines = exports.griid = exports.cdkPipelines = exports.alarms = exports.WebappDeployViaRole = exports.tagResources = exports.SsmParameterReader = exports.SsmParameterBackedResource = exports.createCloudAssemblySnapshot = exports.HostedZoneWithParam = exports.CrossRegionSsmParameter = exports.BastionHost = void 0;
13
+ exports.isSnapshot = exports.cloudTrailSlackIntegration = exports.platform = exports.rds = exports.loadBalancer = exports.ecs = exports.configureParameters = exports.webapp = exports.ses = exports.pipelines = exports.griid = exports.cdkPipelines = exports.alarms = exports.WebappDeployViaRole = exports.tagResources = exports.SsmParameterReader = exports.SsmParameterBackedResource = exports.createCloudAssemblySnapshot = exports.HostedZoneWithParam = exports.CrossRegionSsmParameter = exports.BastionHost = void 0;
14
14
  const alarms = require("./alarms");
15
15
  exports.alarms = alarms;
16
16
  const cdkPipelines = require("./cdk-pipelines");
@@ -29,6 +29,8 @@ const ecs = require("./ecs");
29
29
  exports.ecs = ecs;
30
30
  const loadBalancer = require("./load-balancer");
31
31
  exports.loadBalancer = loadBalancer;
32
+ const cloudTrailSlackIntegration = require("./cloudtrail-slack-integration");
33
+ exports.cloudTrailSlackIntegration = cloudTrailSlackIntegration;
32
34
  const rds = require("./rds");
33
35
  exports.rds = rds;
34
36
  const platform = require("./platform");
@@ -62,4 +64,4 @@ Object.defineProperty(exports, "WebappDeployViaRole", { enumerable: true, get: f
62
64
  * happen during snapshot creation.
63
65
  */
64
66
  exports.isSnapshot = process.env.IS_SNAPSHOT === "true";
65
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7OztBQUFBLG1DQUFrQztBQTRCaEMsd0JBQU07QUEzQlIsZ0RBQStDO0FBNEI3QyxvQ0FBWTtBQTNCZCxpQ0FBZ0M7QUE0QjlCLHNCQUFLO0FBM0JQLHlDQUF3QztBQTRCdEMsOEJBQVM7QUEzQlgsNkJBQTRCO0FBNEIxQixrQkFBRztBQTNCTCxtQ0FBa0M7QUE0QmhDLHdCQUFNO0FBM0JSLDhEQUE2RDtBQTRCM0Qsa0RBQW1CO0FBM0JyQiw2QkFBNEI7QUE0QjFCLGtCQUFHO0FBM0JMLGdEQUErQztBQTRCN0Msb0NBQVk7QUEzQmQsNkJBQTRCO0FBNEIxQixrQkFBRztBQTNCTCx1Q0FBc0M7QUE0QnBDLDRCQUFRO0FBMUJWLGdFQUFnRTtBQUNoRSx1Q0FBdUM7QUFFdkMsK0NBQTRDO0FBQW5DLDJHQUFBLFdBQVcsT0FBQTtBQUNwQixvREFBaUM7QUFDakMsK0NBQTRCO0FBQzVCLDJFQUFzRTtBQUE3RCxxSUFBQSx1QkFBdUIsT0FBQTtBQUNoQyxxREFBa0M7QUFDbEMsbUVBQThEO0FBQXJELDZIQUFBLG1CQUFtQixPQUFBO0FBQzVCLHlDQUF5RDtBQUFoRCx3SEFBQSwyQkFBMkIsT0FBQTtBQUNwQyxpRkFBNEU7QUFBbkUsMklBQUEsMEJBQTBCLE9BQUE7QUFDbkMsK0RBQTJEO0FBQWxELDBIQUFBLGtCQUFrQixPQUFBO0FBQzNCLCtCQUFxQztBQUE1QixvR0FBQSxZQUFZLE9BQUE7QUFDckIsbUVBQThEO0FBQXJELDZIQUFBLG1CQUFtQixPQUFBO0FBZ0I1Qjs7Ozs7O0dBTUc7QUFDVSxRQUFBLFVBQVUsR0FBRyxPQUFPLENBQUMsR0FBRyxDQUFDLFdBQVcsS0FBSyxNQUFNLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBhbGFybXMgZnJvbSBcIi4vYWxhcm1zXCJcbmltcG9ydCAqIGFzIGNka1BpcGVsaW5lcyBmcm9tIFwiLi9jZGstcGlwZWxpbmVzXCJcbmltcG9ydCAqIGFzIGdyaWlkIGZyb20gXCIuL2dyaWlkXCJcbmltcG9ydCAqIGFzIHBpcGVsaW5lcyBmcm9tIFwiLi9waXBlbGluZXNcIlxuaW1wb3J0ICogYXMgc2VzIGZyb20gXCIuL3Nlc1wiXG5pbXBvcnQgKiBhcyB3ZWJhcHAgZnJvbSBcIi4vd2ViYXBwXCJcbmltcG9ydCAqIGFzIGNvbmZpZ3VyZVBhcmFtZXRlcnMgZnJvbSBcIi4vY29uZmlndXJlLXBhcmFtZXRlcnNcIlxuaW1wb3J0ICogYXMgZWNzIGZyb20gXCIuL2Vjc1wiXG5pbXBvcnQgKiBhcyBsb2FkQmFsYW5jZXIgZnJvbSBcIi4vbG9hZC1iYWxhbmNlclwiXG5pbXBvcnQgKiBhcyByZHMgZnJvbSBcIi4vcmRzXCJcbmltcG9ydCAqIGFzIHBsYXRmb3JtIGZyb20gXCIuL3BsYXRmb3JtXCJcblxuLy8gVE9ETzogV2Ugd2FudCB0byBzd2l0Y2ggZXhwb3J0cyBzbyB0aGV5IGV2ZXJ5IGNvbnN0cnVjdCB1bmRlclxuLy8gIGEgbmFtZXNwYWNlIHN1Y2ggYXMgdGhlIHNucyBleHBvcnQuXG5cbmV4cG9ydCB7IEJhc3Rpb25Ib3N0IH0gZnJvbSBcIi4vYmFzdGlvbi1ob3N0XCJcbmV4cG9ydCAqIGZyb20gXCIuL2J1aWxkLWFydGlmYWN0c1wiXG5leHBvcnQgKiBmcm9tIFwiLi9jZGstZGVwbG95XCJcbmV4cG9ydCB7IENyb3NzUmVnaW9uU3NtUGFyYW1ldGVyIH0gZnJvbSBcIi4vY3Jvc3MtcmVnaW9uLXNzbS1wYXJhbWV0ZXJcIlxuZXhwb3J0ICogZnJvbSBcIi4vZWNzLXVwZGF0ZS1pbWFnZVwiXG5leHBvcnQgeyBIb3N0ZWRab25lV2l0aFBhcmFtIH0gZnJvbSBcIi4vaG9zdGVkLXpvbmUtd2l0aC1wYXJhbVwiXG5leHBvcnQgeyBjcmVhdGVDbG91ZEFzc2VtYmx5U25hcHNob3QgfSBmcm9tIFwiLi9zbmFwc2hvdHNcIlxuZXhwb3J0IHsgU3NtUGFyYW1ldGVyQmFja2VkUmVzb3VyY2UgfSBmcm9tIFwiLi9zc20tcGFyYW1ldGVyLWJhY2tlZC1yZXNvdXJjZVwiXG5leHBvcnQgeyBTc21QYXJhbWV0ZXJSZWFkZXIgfSBmcm9tIFwiLi9zc20tcGFyYW1ldGVyLXJlYWRlclwiXG5leHBvcnQgeyB0YWdSZXNvdXJjZXMgfSBmcm9tIFwiLi90YWdzXCJcbmV4cG9ydCB7IFdlYmFwcERlcGxveVZpYVJvbGUgfSBmcm9tIFwiLi93ZWJhcHAtZGVwbG95LXZpYS1yb2xlXCJcblxuZXhwb3J0IHtcbiAgYWxhcm1zLFxuICBjZGtQaXBlbGluZXMsXG4gIGdyaWlkLFxuICBwaXBlbGluZXMsXG4gIHNlcyxcbiAgd2ViYXBwLFxuICBjb25maWd1cmVQYXJhbWV0ZXJzLFxuICBlY3MsXG4gIGxvYWRCYWxhbmNlcixcbiAgcmRzLFxuICBwbGF0Zm9ybSxcbn1cblxuLyoqXG4gKiBDaGVjayBpZiB3ZSBhcmUgc3ludGhlc2l6aW5nIGEgc25hcHNob3QgYnkgc2V0dGluZyBJU19TTkFQU0hPVFxuICogZW52aXJvbm1lbnQgdmFyaWFibGUgdG8gdHJ1ZS5cbiAqXG4gKiBUaGlzIGFsbG93cyBmb3Igc3BlY2lhbCBjb25kaXRpb25hbCBsb2dpYyB0aGF0IHNob3VsZCBvbmx5XG4gKiBoYXBwZW4gZHVyaW5nIHNuYXBzaG90IGNyZWF0aW9uLlxuICovXG5leHBvcnQgY29uc3QgaXNTbmFwc2hvdCA9IHByb2Nlc3MuZW52LklTX1NOQVBTSE9UID09PSBcInRydWVcIlxuIl19
67
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7OztBQUFBLG1DQUFrQztBQTZCaEMsd0JBQU07QUE1QlIsZ0RBQStDO0FBNkI3QyxvQ0FBWTtBQTVCZCxpQ0FBZ0M7QUE2QjlCLHNCQUFLO0FBNUJQLHlDQUF3QztBQTZCdEMsOEJBQVM7QUE1QlgsNkJBQTRCO0FBNkIxQixrQkFBRztBQTVCTCxtQ0FBa0M7QUE2QmhDLHdCQUFNO0FBNUJSLDhEQUE2RDtBQTZCM0Qsa0RBQW1CO0FBNUJyQiw2QkFBNEI7QUE2QjFCLGtCQUFHO0FBNUJMLGdEQUErQztBQTZCN0Msb0NBQVk7QUE1QmQsNkVBQTRFO0FBK0IxRSxnRUFBMEI7QUE5QjVCLDZCQUE0QjtBQTRCMUIsa0JBQUc7QUEzQkwsdUNBQXNDO0FBNEJwQyw0QkFBUTtBQTFCVixnRUFBZ0U7QUFDaEUsdUNBQXVDO0FBRXZDLCtDQUE0QztBQUFuQywyR0FBQSxXQUFXLE9BQUE7QUFDcEIsb0RBQWlDO0FBQ2pDLCtDQUE0QjtBQUM1QiwyRUFBc0U7QUFBN0QscUlBQUEsdUJBQXVCLE9BQUE7QUFDaEMscURBQWtDO0FBQ2xDLG1FQUE4RDtBQUFyRCw2SEFBQSxtQkFBbUIsT0FBQTtBQUM1Qix5Q0FBeUQ7QUFBaEQsd0hBQUEsMkJBQTJCLE9BQUE7QUFDcEMsaUZBQTRFO0FBQW5FLDJJQUFBLDBCQUEwQixPQUFBO0FBQ25DLCtEQUEyRDtBQUFsRCwwSEFBQSxrQkFBa0IsT0FBQTtBQUMzQiwrQkFBcUM7QUFBNUIsb0dBQUEsWUFBWSxPQUFBO0FBQ3JCLG1FQUE4RDtBQUFyRCw2SEFBQSxtQkFBbUIsT0FBQTtBQWlCNUI7Ozs7OztHQU1HO0FBQ1UsUUFBQSxVQUFVLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxXQUFXLEtBQUssTUFBTSxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgYWxhcm1zIGZyb20gXCIuL2FsYXJtc1wiXG5pbXBvcnQgKiBhcyBjZGtQaXBlbGluZXMgZnJvbSBcIi4vY2RrLXBpcGVsaW5lc1wiXG5pbXBvcnQgKiBhcyBncmlpZCBmcm9tIFwiLi9ncmlpZFwiXG5pbXBvcnQgKiBhcyBwaXBlbGluZXMgZnJvbSBcIi4vcGlwZWxpbmVzXCJcbmltcG9ydCAqIGFzIHNlcyBmcm9tIFwiLi9zZXNcIlxuaW1wb3J0ICogYXMgd2ViYXBwIGZyb20gXCIuL3dlYmFwcFwiXG5pbXBvcnQgKiBhcyBjb25maWd1cmVQYXJhbWV0ZXJzIGZyb20gXCIuL2NvbmZpZ3VyZS1wYXJhbWV0ZXJzXCJcbmltcG9ydCAqIGFzIGVjcyBmcm9tIFwiLi9lY3NcIlxuaW1wb3J0ICogYXMgbG9hZEJhbGFuY2VyIGZyb20gXCIuL2xvYWQtYmFsYW5jZXJcIlxuaW1wb3J0ICogYXMgY2xvdWRUcmFpbFNsYWNrSW50ZWdyYXRpb24gZnJvbSBcIi4vY2xvdWR0cmFpbC1zbGFjay1pbnRlZ3JhdGlvblwiXG5pbXBvcnQgKiBhcyByZHMgZnJvbSBcIi4vcmRzXCJcbmltcG9ydCAqIGFzIHBsYXRmb3JtIGZyb20gXCIuL3BsYXRmb3JtXCJcblxuLy8gVE9ETzogV2Ugd2FudCB0byBzd2l0Y2ggZXhwb3J0cyBzbyB0aGV5IGV2ZXJ5IGNvbnN0cnVjdCB1bmRlclxuLy8gIGEgbmFtZXNwYWNlIHN1Y2ggYXMgdGhlIHNucyBleHBvcnQuXG5cbmV4cG9ydCB7IEJhc3Rpb25Ib3N0IH0gZnJvbSBcIi4vYmFzdGlvbi1ob3N0XCJcbmV4cG9ydCAqIGZyb20gXCIuL2J1aWxkLWFydGlmYWN0c1wiXG5leHBvcnQgKiBmcm9tIFwiLi9jZGstZGVwbG95XCJcbmV4cG9ydCB7IENyb3NzUmVnaW9uU3NtUGFyYW1ldGVyIH0gZnJvbSBcIi4vY3Jvc3MtcmVnaW9uLXNzbS1wYXJhbWV0ZXJcIlxuZXhwb3J0ICogZnJvbSBcIi4vZWNzLXVwZGF0ZS1pbWFnZVwiXG5leHBvcnQgeyBIb3N0ZWRab25lV2l0aFBhcmFtIH0gZnJvbSBcIi4vaG9zdGVkLXpvbmUtd2l0aC1wYXJhbVwiXG5leHBvcnQgeyBjcmVhdGVDbG91ZEFzc2VtYmx5U25hcHNob3QgfSBmcm9tIFwiLi9zbmFwc2hvdHNcIlxuZXhwb3J0IHsgU3NtUGFyYW1ldGVyQmFja2VkUmVzb3VyY2UgfSBmcm9tIFwiLi9zc20tcGFyYW1ldGVyLWJhY2tlZC1yZXNvdXJjZVwiXG5leHBvcnQgeyBTc21QYXJhbWV0ZXJSZWFkZXIgfSBmcm9tIFwiLi9zc20tcGFyYW1ldGVyLXJlYWRlclwiXG5leHBvcnQgeyB0YWdSZXNvdXJjZXMgfSBmcm9tIFwiLi90YWdzXCJcbmV4cG9ydCB7IFdlYmFwcERlcGxveVZpYVJvbGUgfSBmcm9tIFwiLi93ZWJhcHAtZGVwbG95LXZpYS1yb2xlXCJcblxuZXhwb3J0IHtcbiAgYWxhcm1zLFxuICBjZGtQaXBlbGluZXMsXG4gIGdyaWlkLFxuICBwaXBlbGluZXMsXG4gIHNlcyxcbiAgd2ViYXBwLFxuICBjb25maWd1cmVQYXJhbWV0ZXJzLFxuICBlY3MsXG4gIGxvYWRCYWxhbmNlcixcbiAgcmRzLFxuICBwbGF0Zm9ybSxcbiAgY2xvdWRUcmFpbFNsYWNrSW50ZWdyYXRpb24sXG59XG5cbi8qKlxuICogQ2hlY2sgaWYgd2UgYXJlIHN5bnRoZXNpemluZyBhIHNuYXBzaG90IGJ5IHNldHRpbmcgSVNfU05BUFNIT1RcbiAqIGVudmlyb25tZW50IHZhcmlhYmxlIHRvIHRydWUuXG4gKlxuICogVGhpcyBhbGxvd3MgZm9yIHNwZWNpYWwgY29uZGl0aW9uYWwgbG9naWMgdGhhdCBzaG91bGQgb25seVxuICogaGFwcGVuIGR1cmluZyBzbmFwc2hvdCBjcmVhdGlvbi5cbiAqL1xuZXhwb3J0IGNvbnN0IGlzU25hcHNob3QgPSBwcm9jZXNzLmVudi5JU19TTkFQU0hPVCA9PT0gXCJ0cnVlXCJcbiJdfQ==
@@ -0,0 +1 @@
1
+ export { KinesisToDatadogStream, KinesisToDatadogStreamProps, } from "./kinesis-to-datadog-stream";
@@ -0,0 +1,6 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.KinesisToDatadogStream = void 0;
4
+ var kinesis_to_datadog_stream_1 = require("./kinesis-to-datadog-stream");
5
+ Object.defineProperty(exports, "KinesisToDatadogStream", { enumerable: true, get: function () { return kinesis_to_datadog_stream_1.KinesisToDatadogStream; } });
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMva2luZXNpcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5RUFHb0M7QUFGbEMsbUlBQUEsc0JBQXNCLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQge1xuICBLaW5lc2lzVG9EYXRhZG9nU3RyZWFtLFxuICBLaW5lc2lzVG9EYXRhZG9nU3RyZWFtUHJvcHMsXG59IGZyb20gXCIuL2tpbmVzaXMtdG8tZGF0YWRvZy1zdHJlYW1cIlxuIl19
@@ -0,0 +1,27 @@
1
+ import * as logs from "@aws-cdk/aws-logs";
2
+ import * as cdk from "@aws-cdk/core";
3
+ export interface KinesisToDatadogStreamProps {
4
+ /**
5
+ *
6
+ * The name of the SecretsManager secret where your Datadog API key is saved.
7
+ *
8
+ * The secret must be a JSON object on the format { "value": "SECRET" }
9
+ *
10
+ */
11
+ datadogApiKeySecretName: string;
12
+ /**
13
+ *
14
+ * The CloudWatch log groups from you are streaming to Datadog
15
+ *
16
+ */
17
+ logGroups: logs.LogGroup[];
18
+ }
19
+ /**
20
+ *
21
+ * Forwards logs from log-groups in CloudWatch to a Datadog account.
22
+ * The logs are delivered through a Firehose delivery stream, which is being subscribed to the log-groups in CloudWatch.
23
+ *
24
+ */
25
+ export declare class KinesisToDatadogStream extends cdk.Construct {
26
+ constructor(scope: cdk.Construct, id: string, props: KinesisToDatadogStreamProps);
27
+ }
@@ -0,0 +1,124 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.KinesisToDatadogStream = void 0;
4
+ const iam = require("@aws-cdk/aws-iam");
5
+ const firehose = require("@aws-cdk/aws-kinesisfirehose");
6
+ const logs = require("@aws-cdk/aws-logs");
7
+ const s3 = require("@aws-cdk/aws-s3");
8
+ const aws_s3_1 = require("@aws-cdk/aws-s3");
9
+ const secretsmanager = require("@aws-cdk/aws-secretsmanager");
10
+ const cdk = require("@aws-cdk/core");
11
+ /**
12
+ *
13
+ * Forwards logs from log-groups in CloudWatch to a Datadog account.
14
+ * The logs are delivered through a Firehose delivery stream, which is being subscribed to the log-groups in CloudWatch.
15
+ *
16
+ */
17
+ class KinesisToDatadogStream extends cdk.Construct {
18
+ constructor(scope, id, props) {
19
+ super(scope, id);
20
+ const deliveryStreamLogGroup = new logs.LogGroup(this, "DeliveryStreamLogGroup");
21
+ const deliveryStreamLogStream = new logs.LogStream(this, "DeliveryStreamLogStream", {
22
+ logGroup: deliveryStreamLogGroup,
23
+ });
24
+ const failedDataBucket = new s3.Bucket(this, "FailedDataBucket", {
25
+ blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL,
26
+ });
27
+ const cloudWatchLogsRole = new iam.Role(this, "CloudWatchLogsRole", {
28
+ assumedBy: new iam.ServicePrincipal(`logs.${cdk.Stack.of(this).region}.amazonaws.com`),
29
+ });
30
+ const firehoseLogsRole = new iam.Role(this, "FirehoseLogsRole", {
31
+ assumedBy: new iam.ServicePrincipal("firehose.amazonaws.com"),
32
+ });
33
+ const datadogDeliveryStream = new firehose.CfnDeliveryStream(this, "DeliveryStream", {
34
+ deliveryStreamType: "DirectPut",
35
+ httpEndpointDestinationConfiguration: {
36
+ roleArn: firehoseLogsRole.roleArn,
37
+ endpointConfiguration: {
38
+ url: "https://aws-kinesis-http-intake.logs.datadoghq.eu/v1/input",
39
+ accessKey: secretsmanager.Secret.fromSecretNameV2(scope, "DatadogApiKey", props.datadogApiKeySecretName)
40
+ .secretValueFromJson("value")
41
+ .toString(),
42
+ name: "datadog-logs-endpoint",
43
+ },
44
+ requestConfiguration: {
45
+ contentEncoding: "GZIP",
46
+ },
47
+ cloudWatchLoggingOptions: {
48
+ enabled: true,
49
+ logGroupName: deliveryStreamLogGroup.logGroupName,
50
+ logStreamName: deliveryStreamLogStream.logStreamName,
51
+ },
52
+ bufferingHints: {
53
+ intervalInSeconds: 60,
54
+ sizeInMBs: 4,
55
+ },
56
+ retryOptions: {
57
+ durationInSeconds: 60,
58
+ },
59
+ s3BackupMode: "FailedDataOnly",
60
+ s3Configuration: {
61
+ bucketArn: failedDataBucket.bucketArn,
62
+ compressionFormat: "UNCOMPRESSED",
63
+ roleArn: firehoseLogsRole.roleArn,
64
+ },
65
+ },
66
+ });
67
+ new iam.Policy(this, "CloudWatchLogsPolicy", {
68
+ document: new iam.PolicyDocument({
69
+ statements: [
70
+ new iam.PolicyStatement({
71
+ actions: ["firehose:PutRecord", "firehose:PutRecordBatch"],
72
+ resources: [datadogDeliveryStream.attrArn],
73
+ }),
74
+ ],
75
+ }),
76
+ roles: [cloudWatchLogsRole],
77
+ });
78
+ new iam.Policy(this, "FirehoseLogsPolicy", {
79
+ document: new iam.PolicyDocument({
80
+ statements: [
81
+ new iam.PolicyStatement({
82
+ actions: [
83
+ "s3:AbortMultipartUpload",
84
+ "s3:GetBucketLocation",
85
+ "s3:GetObject",
86
+ "s3:ListBucket",
87
+ "s3:ListBucketMultipartUploads",
88
+ "s3:PutObject",
89
+ ],
90
+ resources: [
91
+ failedDataBucket.bucketArn,
92
+ `${failedDataBucket.bucketArn}/*`,
93
+ ],
94
+ }),
95
+ new iam.PolicyStatement({
96
+ actions: ["logs:PutLogEvents"],
97
+ resources: [
98
+ `arn:aws:logs:${cdk.Stack.of(this).region}:${cdk.Stack.of(this).account}:log-group:${deliveryStreamLogGroup.logGroupName}:log-stream:${deliveryStreamLogStream.logStreamName}`,
99
+ ],
100
+ }),
101
+ new iam.PolicyStatement({
102
+ actions: [
103
+ "kinesis:DescribeStream",
104
+ "kinesis:GetShardIterator",
105
+ "kinesis:GetRecords",
106
+ ],
107
+ resources: [datadogDeliveryStream.attrArn],
108
+ }),
109
+ ],
110
+ }),
111
+ roles: [firehoseLogsRole],
112
+ });
113
+ props.logGroups.forEach((logGroup, index) => {
114
+ new logs.CfnSubscriptionFilter(this, `SubscriptionFilter${index}`, {
115
+ logGroupName: logGroup.logGroupName,
116
+ destinationArn: datadogDeliveryStream.attrArn,
117
+ filterPattern: logs.FilterPattern.allEvents().logPatternString,
118
+ roleArn: cloudWatchLogsRole.roleArn,
119
+ });
120
+ });
121
+ }
122
+ }
123
+ exports.KinesisToDatadogStream = KinesisToDatadogStream;
124
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kinesis-to-datadog-stream.js","sourceRoot":"","sources":["../../src/kinesis/kinesis-to-datadog-stream.ts"],"names":[],"mappings":"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","sourcesContent":["import * as iam from \"@aws-cdk/aws-iam\"\nimport * as firehose from \"@aws-cdk/aws-kinesisfirehose\"\nimport * as logs from \"@aws-cdk/aws-logs\"\nimport * as s3 from \"@aws-cdk/aws-s3\"\nimport { BlockPublicAccess } from \"@aws-cdk/aws-s3\"\nimport * as secretsmanager from \"@aws-cdk/aws-secretsmanager\"\nimport * as cdk from \"@aws-cdk/core\"\n\nexport interface KinesisToDatadogStreamProps {\n  /**\n   *\n   * The name of the SecretsManager secret where your Datadog API key is saved.\n   *\n   * The secret must be a JSON object on the format { \"value\": \"SECRET\" }\n   *\n   */\n  datadogApiKeySecretName: string\n  /**\n   *\n   * The CloudWatch log groups from you are streaming to Datadog\n   *\n   */\n  logGroups: logs.LogGroup[]\n}\n\n/**\n *\n * Forwards logs from log-groups in CloudWatch to a Datadog account.\n * The logs are delivered through a Firehose delivery stream, which is being subscribed to the log-groups in CloudWatch.\n *\n */\nexport class KinesisToDatadogStream extends cdk.Construct {\n  constructor(\n    scope: cdk.Construct,\n    id: string,\n    props: KinesisToDatadogStreamProps,\n  ) {\n    super(scope, id)\n\n    const deliveryStreamLogGroup = new logs.LogGroup(\n      this,\n      \"DeliveryStreamLogGroup\",\n    )\n\n    const deliveryStreamLogStream = new logs.LogStream(\n      this,\n      \"DeliveryStreamLogStream\",\n      {\n        logGroup: deliveryStreamLogGroup,\n      },\n    )\n\n    const failedDataBucket = new s3.Bucket(this, \"FailedDataBucket\", {\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n    })\n\n    const cloudWatchLogsRole = new iam.Role(this, \"CloudWatchLogsRole\", {\n      assumedBy: new iam.ServicePrincipal(\n        `logs.${cdk.Stack.of(this).region}.amazonaws.com`,\n      ),\n    })\n\n    const firehoseLogsRole = new iam.Role(this, \"FirehoseLogsRole\", {\n      assumedBy: new iam.ServicePrincipal(\"firehose.amazonaws.com\"),\n    })\n\n    const datadogDeliveryStream = new firehose.CfnDeliveryStream(\n      this,\n      \"DeliveryStream\",\n      {\n        deliveryStreamType: \"DirectPut\",\n        httpEndpointDestinationConfiguration: {\n          roleArn: firehoseLogsRole.roleArn,\n          endpointConfiguration: {\n            url: \"https://aws-kinesis-http-intake.logs.datadoghq.eu/v1/input\",\n            accessKey: secretsmanager.Secret.fromSecretNameV2(\n              scope,\n              \"DatadogApiKey\",\n              props.datadogApiKeySecretName,\n            )\n              .secretValueFromJson(\"value\")\n              .toString(),\n            name: \"datadog-logs-endpoint\",\n          },\n          requestConfiguration: {\n            contentEncoding: \"GZIP\",\n          },\n          cloudWatchLoggingOptions: {\n            enabled: true,\n            logGroupName: deliveryStreamLogGroup.logGroupName,\n            logStreamName: deliveryStreamLogStream.logStreamName,\n          },\n          bufferingHints: {\n            intervalInSeconds: 60,\n            sizeInMBs: 4,\n          },\n          retryOptions: {\n            durationInSeconds: 60,\n          },\n          s3BackupMode: \"FailedDataOnly\",\n          s3Configuration: {\n            bucketArn: failedDataBucket.bucketArn,\n            compressionFormat: \"UNCOMPRESSED\",\n            roleArn: firehoseLogsRole.roleArn,\n          },\n        },\n      },\n    )\n\n    new iam.Policy(this, \"CloudWatchLogsPolicy\", {\n      document: new iam.PolicyDocument({\n        statements: [\n          new iam.PolicyStatement({\n            actions: [\"firehose:PutRecord\", \"firehose:PutRecordBatch\"],\n            resources: [datadogDeliveryStream.attrArn],\n          }),\n        ],\n      }),\n      roles: [cloudWatchLogsRole],\n    })\n\n    new iam.Policy(this, \"FirehoseLogsPolicy\", {\n      document: new iam.PolicyDocument({\n        statements: [\n          new iam.PolicyStatement({\n            actions: [\n              \"s3:AbortMultipartUpload\",\n              \"s3:GetBucketLocation\",\n              \"s3:GetObject\",\n              \"s3:ListBucket\",\n              \"s3:ListBucketMultipartUploads\",\n              \"s3:PutObject\",\n            ],\n            resources: [\n              failedDataBucket.bucketArn,\n              `${failedDataBucket.bucketArn}/*`,\n            ],\n          }),\n          new iam.PolicyStatement({\n            actions: [\"logs:PutLogEvents\"],\n            resources: [\n              `arn:aws:logs:${cdk.Stack.of(this).region}:${\n                cdk.Stack.of(this).account\n              }:log-group:${deliveryStreamLogGroup.logGroupName}:log-stream:${\n                deliveryStreamLogStream.logStreamName\n              }`,\n            ],\n          }),\n          new iam.PolicyStatement({\n            actions: [\n              \"kinesis:DescribeStream\",\n              \"kinesis:GetShardIterator\",\n              \"kinesis:GetRecords\",\n            ],\n            resources: [datadogDeliveryStream.attrArn],\n          }),\n        ],\n      }),\n      roles: [firehoseLogsRole],\n    })\n\n    props.logGroups.forEach((logGroup, index) => {\n      new logs.CfnSubscriptionFilter(this, `SubscriptionFilter${index}`, {\n        logGroupName: logGroup.logGroupName,\n        destinationArn: datadogDeliveryStream.attrArn,\n        filterPattern: logs.FilterPattern.allEvents().logPatternString,\n        roleArn: cloudWatchLogsRole.roleArn,\n      })\n    })\n  }\n}\n"]}
package/lib/snapshots.js CHANGED
@@ -209,4 +209,4 @@ async function createCloudAssemblySnapshot(src, dst) {
209
209
  }
210
210
  }
211
211
  exports.createCloudAssemblySnapshot = createCloudAssemblySnapshot;
212
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshots.js","sourceRoot":"","sources":["../src/snapshots.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-assignment */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-unsafe-return */\nimport * as cpy from \"cpy\"\nimport * as del from \"del\"\nimport * as fs from \"fs\"\nimport * as glob from \"glob\"\nimport * as path from \"path\"\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeVersion(data: any): any {\n  const cp = {\n    ...data,\n  }\n\n  delete cp.version\n  return cp\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeTrace(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeTrace)\n  }\n\n  if (data === Object(data)) {\n    return Object.fromEntries(\n      Object.entries(data)\n        .filter(([key]) => key !== \"trace\")\n        .map(([key, value]) => [key, removeTrace(value)]),\n    )\n  }\n\n  return data\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeRuntimeLibraries(data: any): any {\n  const cp = {\n    ...data,\n  }\n\n  if (data.runtime) {\n    cp.runtime = {\n      ...data.runtime,\n    }\n\n    delete cp.runtime.libraries\n  }\n\n  return cp\n}\n\nconst currentVersionRegex = /^(.+CurrentVersion[0-9A-F]{8})[0-9a-f]{32}$/\n\n/**\n * Match a resource created by `lambda.Function.currentVersion`, which\n * will include the asset hash as part of the reousrce name and return\n * a snapshot-friendly version of it if found.\n */\nfunction rewriteCurrentVersionIfFound(value: string): string | null {\n  const match = currentVersionRegex.exec(value)\n  return match ? `${match[1]}xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx` : null\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeAssetDetailsFromTemplate(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeAssetDetailsFromTemplate)\n  }\n\n  if (data === Object(data)) {\n    return Object.fromEntries(\n      Object.entries(data)\n        .map(([key, value]) => {\n          const newCurrentVersion = rewriteCurrentVersionIfFound(key)\n          if (newCurrentVersion) {\n            return [newCurrentVersion, removeAssetDetailsFromTemplate(value)]\n          } else if (key.includes(\"AssetParameter\")) {\n            return null\n          } else if (\n            key === \"Ref\" &&\n            typeof value === \"string\" &&\n            value.includes(\"AssetParameters\")\n          ) {\n            return [key, \"snapshot-value\"]\n          } else if (\n            key === \"aws:asset:path\" &&\n            typeof value === \"string\" &&\n            /asset\\.[0-9a-f]{64}/.test(value)\n          ) {\n            return [key, \"asset.snapshot-value\"]\n          } else {\n            return [key, removeAssetDetailsFromTemplate(value)]\n          }\n        })\n        .filter((it): it is [] => it != null),\n    )\n  }\n\n  if (typeof data === \"string\") {\n    const newCurrentVersion = rewriteCurrentVersionIfFound(data)\n    if (newCurrentVersion) {\n      return newCurrentVersion\n    }\n\n    // Handle typical content hashes.\n    return data.replace(/[0-9a-f]{64}/g, \"snapshot-value\")\n  }\n\n  return data\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeAssetDetailsFromManifest(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeAssetDetailsFromManifest)\n  }\n\n  if (data === Object(data)) {\n    // aws:cdk:asset in metadata\n    if (data[\"type\"] === \"aws:cdk:asset\" && \"data\" in data) {\n      return {\n        ...data,\n        data: \"snapshot-value\",\n      }\n    }\n\n    return Object.fromEntries(\n      Object.entries(data)\n        .map(([key, value]) => {\n          if (key.includes(\"AssetParameters\")) {\n            return null\n          } else {\n            return [key, removeAssetDetailsFromManifest(value)]\n          }\n        })\n        .filter((it): it is [] => it != null),\n    )\n  }\n\n  if (typeof data === \"string\") {\n    const newCurrentVersion = rewriteCurrentVersionIfFound(data)\n    if (newCurrentVersion) {\n      return newCurrentVersion\n    }\n\n    // Handle typical content hashes.\n    return data.replace(/[0-9a-f]{64}/g, \"snapshot-value\")\n  }\n\n  return data\n}\n/**\n * Remove the CDKMetadata resources that is part of the synthesized\n * template since CDK 1.63.0.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeCdkMetadataResourceFromTemplate(data: any): any {\n  const cp = {\n    ...data,\n    Resources: {\n      ...data.Resources,\n    },\n  }\n\n  delete cp.Resources.CDKMetadata\n  return cp\n}\n\nfunction prepareManifestForSnapshot(content: string): string {\n  const input = JSON.parse(content)\n  const output = [\n    removeVersion,\n    // Remove the runtime version information so it don't conflict with CI\n    // or other users generating the snapshots.\n    removeRuntimeLibraries,\n    // Remove the trace from manifest for now.\n    removeTrace,\n    // Avoid details (hashes) from assets.\n    removeAssetDetailsFromManifest,\n  ].reduce((acc, fn) => fn(acc), input)\n\n  return JSON.stringify(output, undefined, \"  \")\n}\n\n/**\n * Transform the Cloud Assembly manifest file so that it can be persisted\n * as a snapshot without causing invalidations for every synthesize.\n */\nasync function prepareManifestFileForSnapshot(file: string): Promise<void> {\n  const result = prepareManifestForSnapshot(\n    await fs.promises.readFile(file, \"utf8\"),\n  )\n\n  await fs.promises.writeFile(file, result)\n}\n\nfunction prepareTemplateForSnapshot(content: string): string {\n  const input = JSON.parse(content)\n  const output = [\n    removeCdkMetadataResourceFromTemplate,\n    // Avoid details (hashes) from assets.\n    removeAssetDetailsFromTemplate,\n  ].reduce((acc, fn) => fn(acc), input)\n\n  return JSON.stringify(output, undefined, \"  \")\n}\n\n/**\n * Transform a Cloud Assembly template file so that it can be persisted\n * as a snapshot without causing invalidations for minor changes.\n */\nasync function prepareTemplateFileForSnapshot(file: string): Promise<void> {\n  const result = prepareTemplateForSnapshot(\n    await fs.promises.readFile(file, \"utf8\"),\n  )\n\n  await fs.promises.writeFile(file, result)\n}\n\n/**\n * Convert a Cloud Assembly to a snapshot.\n */\nexport async function createCloudAssemblySnapshot(\n  src: string,\n  dst: string,\n): Promise<void> {\n  const base = path.join(process.cwd(), dst)\n\n  await cpy(\".\", base, {\n    parents: true,\n    cwd: path.join(process.cwd(), src),\n  })\n\n  // Don't keep track of manifest version.\n  await del(path.join(dst, \"**/cdk.out\"))\n\n  // The tree file doesn't give us much value as part of the snapshot.\n  await del(path.join(dst, \"tree.json\"))\n\n  // Remove asset contents for now.\n  await del(path.join(dst, \"**/asset.*\"))\n\n  // Remove asset configs so we don't have to update\n  // snapshots for asset changes.\n  await del(path.join(dst, \"**/*.assets.json\"))\n\n  // Transform the manifest to be more snapshot friendly.\n  for (const file of glob.sync(\"**/manifest.json\", { cwd: base })) {\n    await prepareManifestFileForSnapshot(path.join(base, file))\n  }\n\n  // Transform all templates.\n  for (const file of glob.sync(\"**/*.template.json\", { cwd: base })) {\n    await prepareTemplateFileForSnapshot(path.join(base, file))\n  }\n}\n"]}
212
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshots.js","sourceRoot":"","sources":["../src/snapshots.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-assignment */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-unsafe-return */\nimport * as cpy from \"cpy\"\nimport * as del from \"del\"\nimport * as fs from \"fs\"\nimport * as glob from \"glob\"\nimport * as path from \"path\"\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeVersion(data: any): any {\n  const cp = {\n    ...data,\n  }\n\n  delete cp.version\n  return cp\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeTrace(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeTrace)\n  }\n\n  if (data === Object(data)) {\n    return Object.fromEntries(\n      Object.entries(data as Record<string, unknown>)\n        .filter(([key]) => key !== \"trace\")\n        .map(([key, value]) => [key, removeTrace(value)]),\n    )\n  }\n\n  return data\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeRuntimeLibraries(data: any): any {\n  const cp = {\n    ...data,\n  }\n\n  if (data.runtime) {\n    cp.runtime = {\n      ...data.runtime,\n    }\n\n    delete cp.runtime.libraries\n  }\n\n  return cp\n}\n\nconst currentVersionRegex = /^(.+CurrentVersion[0-9A-F]{8})[0-9a-f]{32}$/\n\n/**\n * Match a resource created by `lambda.Function.currentVersion`, which\n * will include the asset hash as part of the reousrce name and return\n * a snapshot-friendly version of it if found.\n */\nfunction rewriteCurrentVersionIfFound(value: string): string | null {\n  const match = currentVersionRegex.exec(value)\n  return match ? `${match[1]}xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx` : null\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeAssetDetailsFromTemplate(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeAssetDetailsFromTemplate)\n  }\n\n  if (data === Object(data)) {\n    return Object.fromEntries(\n      Object.entries(data as Record<string, unknown>)\n        .map(([key, value]) => {\n          const newCurrentVersion = rewriteCurrentVersionIfFound(key)\n          if (newCurrentVersion) {\n            return [newCurrentVersion, removeAssetDetailsFromTemplate(value)]\n          } else if (key.includes(\"AssetParameter\")) {\n            return null\n          } else if (\n            key === \"Ref\" &&\n            typeof value === \"string\" &&\n            value.includes(\"AssetParameters\")\n          ) {\n            return [key, \"snapshot-value\"]\n          } else if (\n            key === \"aws:asset:path\" &&\n            typeof value === \"string\" &&\n            /asset\\.[0-9a-f]{64}/.test(value)\n          ) {\n            return [key, \"asset.snapshot-value\"]\n          } else {\n            return [key, removeAssetDetailsFromTemplate(value)]\n          }\n        })\n        .filter((it): it is [] => it != null),\n    )\n  }\n\n  if (typeof data === \"string\") {\n    const newCurrentVersion = rewriteCurrentVersionIfFound(data)\n    if (newCurrentVersion) {\n      return newCurrentVersion\n    }\n\n    // Handle typical content hashes.\n    return data.replace(/[0-9a-f]{64}/g, \"snapshot-value\")\n  }\n\n  return data\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeAssetDetailsFromManifest(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeAssetDetailsFromManifest)\n  }\n\n  if (data === Object(data)) {\n    // aws:cdk:asset in metadata\n    if (data[\"type\"] === \"aws:cdk:asset\" && \"data\" in data) {\n      return {\n        ...data,\n        data: \"snapshot-value\",\n      }\n    }\n\n    return Object.fromEntries(\n      Object.entries(data as Record<string, unknown>)\n        .map(([key, value]) => {\n          if (key.includes(\"AssetParameters\")) {\n            return null\n          } else {\n            return [key, removeAssetDetailsFromManifest(value)]\n          }\n        })\n        .filter((it): it is [] => it != null),\n    )\n  }\n\n  if (typeof data === \"string\") {\n    const newCurrentVersion = rewriteCurrentVersionIfFound(data)\n    if (newCurrentVersion) {\n      return newCurrentVersion\n    }\n\n    // Handle typical content hashes.\n    return data.replace(/[0-9a-f]{64}/g, \"snapshot-value\")\n  }\n\n  return data\n}\n\n/**\n * Remove the CDKMetadata resources that is part of the synthesized\n * template since CDK 1.63.0.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeCdkMetadataResourceFromTemplate(data: any): any {\n  const cp = {\n    ...data,\n    Resources: {\n      ...data.Resources,\n    },\n  }\n\n  delete cp.Resources.CDKMetadata\n  return cp\n}\n\nfunction prepareManifestForSnapshot(content: string): string {\n  const input = JSON.parse(content)\n  const output = [\n    removeVersion,\n    // Remove the runtime version information so it don't conflict with CI\n    // or other users generating the snapshots.\n    removeRuntimeLibraries,\n    // Remove the trace from manifest for now.\n    removeTrace,\n    // Avoid details (hashes) from assets.\n    removeAssetDetailsFromManifest,\n  ].reduce((acc, fn) => fn(acc), input)\n\n  return JSON.stringify(output, undefined, \"  \")\n}\n\n/**\n * Transform the Cloud Assembly manifest file so that it can be persisted\n * as a snapshot without causing invalidations for every synthesize.\n */\nasync function prepareManifestFileForSnapshot(file: string): Promise<void> {\n  const result = prepareManifestForSnapshot(\n    await fs.promises.readFile(file, \"utf8\"),\n  )\n\n  await fs.promises.writeFile(file, result)\n}\n\nfunction prepareTemplateForSnapshot(content: string): string {\n  const input = JSON.parse(content)\n  const output = [\n    removeCdkMetadataResourceFromTemplate,\n    // Avoid details (hashes) from assets.\n    removeAssetDetailsFromTemplate,\n  ].reduce((acc, fn) => fn(acc), input)\n\n  return JSON.stringify(output, undefined, \"  \")\n}\n\n/**\n * Transform a Cloud Assembly template file so that it can be persisted\n * as a snapshot without causing invalidations for minor changes.\n */\nasync function prepareTemplateFileForSnapshot(file: string): Promise<void> {\n  const result = prepareTemplateForSnapshot(\n    await fs.promises.readFile(file, \"utf8\"),\n  )\n\n  await fs.promises.writeFile(file, result)\n}\n\n/**\n * Convert a Cloud Assembly to a snapshot.\n */\nexport async function createCloudAssemblySnapshot(\n  src: string,\n  dst: string,\n): Promise<void> {\n  const base = path.join(process.cwd(), dst)\n\n  await cpy(\".\", base, {\n    parents: true,\n    cwd: path.join(process.cwd(), src),\n  })\n\n  // Don't keep track of manifest version.\n  await del(path.join(dst, \"**/cdk.out\"))\n\n  // The tree file doesn't give us much value as part of the snapshot.\n  await del(path.join(dst, \"tree.json\"))\n\n  // Remove asset contents for now.\n  await del(path.join(dst, \"**/asset.*\"))\n\n  // Remove asset configs so we don't have to update\n  // snapshots for asset changes.\n  await del(path.join(dst, \"**/*.assets.json\"))\n\n  // Transform the manifest to be more snapshot friendly.\n  for (const file of glob.sync(\"**/manifest.json\", { cwd: base })) {\n    await prepareManifestFileForSnapshot(path.join(base, file))\n  }\n\n  // Transform all templates.\n  for (const file of glob.sync(\"**/*.template.json\", { cwd: base })) {\n    await prepareTemplateFileForSnapshot(path.join(base, file))\n  }\n}\n"]}
@@ -0,0 +1,40 @@
1
+ import * as cloudfront from "@aws-cdk/aws-cloudfront";
2
+ import * as cdk from "@aws-cdk/core";
3
+ export interface FrameOptionsHeader {
4
+ value?: "DENY" | "SAMEORIGIN";
5
+ }
6
+ export interface ReferrerPolicyHeader {
7
+ value?: string;
8
+ }
9
+ export interface StrictTransportSecurityHeader {
10
+ maxAge?: number;
11
+ includeSubDomains?: boolean;
12
+ preload?: boolean;
13
+ }
14
+ export interface ContentSecurityPolicyHeader {
15
+ reportOnly?: boolean;
16
+ baseUri?: string;
17
+ childSrc?: string;
18
+ defaultSrc?: string;
19
+ fontSrc?: string;
20
+ frameSrc?: string;
21
+ formAction?: string;
22
+ frameAncestors?: string;
23
+ imgSrc?: string;
24
+ manifestSrc?: string;
25
+ mediaSrc?: string;
26
+ objectSrc?: string;
27
+ scriptSrc?: string;
28
+ styleSrc?: string;
29
+ connectSrc?: string;
30
+ }
31
+ export interface SecurityHeaders {
32
+ contentSecurityPolicy?: ContentSecurityPolicyHeader;
33
+ strictTransportSecurity?: StrictTransportSecurityHeader;
34
+ referrerPolicy?: ReferrerPolicyHeader;
35
+ frameOptions?: FrameOptionsHeader;
36
+ }
37
+ export declare class WebappSecurityHeaders extends cdk.Construct {
38
+ readonly securityHeadersFunction: cloudfront.Function;
39
+ constructor(scope: cdk.Construct, id: string, props: SecurityHeaders);
40
+ }
@@ -0,0 +1,126 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.WebappSecurityHeaders = void 0;
4
+ const cloudfront = require("@aws-cdk/aws-cloudfront");
5
+ const cdk = require("@aws-cdk/core");
6
+ function validateCspParam(param) {
7
+ if (param.indexOf('"') !== -1) {
8
+ throw Error('CSP override contains invalid character "');
9
+ }
10
+ if (param.indexOf(";") !== -1) {
11
+ throw Error("CSP override contains invalid character ;");
12
+ }
13
+ if (param.indexOf("\\") !== -1) {
14
+ throw Error("CSP override contains invalid character \\");
15
+ }
16
+ return param;
17
+ }
18
+ /* Replace all whitespace in a string with a single space */
19
+ function trim(value) {
20
+ return value.replace(/\s+/g, " ").trim();
21
+ }
22
+ function generateContentSecurityPolicyHeader(headerOptions) {
23
+ const defaultValues = {
24
+ baseUri: "'self'",
25
+ childSrc: "'none'",
26
+ connectSrc: "'self'",
27
+ defaultSrc: "'self'",
28
+ fontSrc: "'self'",
29
+ formAction: "'self'",
30
+ frameAncestors: "'none'",
31
+ frameSrc: "'self'",
32
+ imgSrc: "'self'",
33
+ manifestSrc: "'self'",
34
+ mediaSrc: "'self'",
35
+ objectSrc: "'none'",
36
+ scriptSrc: "'self'",
37
+ styleSrc: "'self'",
38
+ };
39
+ const options = {
40
+ ...defaultValues,
41
+ ...headerOptions,
42
+ };
43
+ Object.values(options).forEach((v) => typeof v === "string" && validateCspParam(v));
44
+ let headerValue = "";
45
+ headerValue += `base-uri ${trim(options.baseUri)};`;
46
+ headerValue += `child-src ${trim(options.childSrc)};`;
47
+ headerValue += `connect-src ${trim(options.connectSrc)};`;
48
+ headerValue += `default-src ${trim(options.defaultSrc)};`;
49
+ headerValue += `font-src ${trim(options.fontSrc)};`;
50
+ headerValue += `frame-src ${trim(options.frameSrc)};`;
51
+ headerValue += `img-src ${trim(options.imgSrc)};`;
52
+ headerValue += `manifest-src ${trim(options.manifestSrc)};`;
53
+ headerValue += `media-src ${trim(options.mediaSrc)};`;
54
+ headerValue += `object-src ${trim(options.objectSrc)};`;
55
+ headerValue += `script-src ${trim(options.scriptSrc)};`;
56
+ headerValue += `style-src ${trim(options.styleSrc)};`;
57
+ return trim(headerValue);
58
+ }
59
+ function generateStrictTransportSecurityHeader(headerOptions) {
60
+ const defaultValues = {
61
+ maxAge: 63072000,
62
+ includeSubDomains: false,
63
+ preload: false,
64
+ };
65
+ const options = {
66
+ ...defaultValues,
67
+ ...headerOptions,
68
+ };
69
+ let headerValue = "";
70
+ headerValue += `max-age=${options.maxAge};`;
71
+ headerValue += options.preload ? "preload;" : "";
72
+ headerValue += options.includeSubDomains ? "includeSubDomains;" : "";
73
+ return trim(headerValue);
74
+ }
75
+ function generateReferrerPolicyHeader(headerOptions) {
76
+ const defaultValues = {
77
+ value: "strict-origin-when-cross-origin",
78
+ };
79
+ const options = {
80
+ ...defaultValues,
81
+ ...headerOptions,
82
+ };
83
+ return options.value;
84
+ }
85
+ function generateFrameOptionsHeader(headerOptions) {
86
+ const defaultValues = {
87
+ value: "DENY",
88
+ };
89
+ const options = {
90
+ ...defaultValues,
91
+ ...headerOptions,
92
+ };
93
+ return trim(options.value);
94
+ }
95
+ class WebappSecurityHeaders extends cdk.Construct {
96
+ constructor(scope, id, props) {
97
+ var _a;
98
+ super(scope, id);
99
+ const cspHeaderName = ((_a = props.contentSecurityPolicy) === null || _a === void 0 ? void 0 : _a.reportOnly)
100
+ ? "content-security-policy-report-only"
101
+ : "content-security-policy";
102
+ const contentSecurityPolicy = generateContentSecurityPolicyHeader(props.contentSecurityPolicy);
103
+ const strictTransportSecurity = generateStrictTransportSecurityHeader(props.strictTransportSecurity);
104
+ const referrerPolicy = generateReferrerPolicyHeader(props.referrerPolicy);
105
+ const frameOptions = generateFrameOptionsHeader(props.frameOptions);
106
+ const lambdaCode = `function handler(event) {
107
+ var response = event.response;
108
+ var headers = response.headers;
109
+ headers['referrer-policy'] = {value: '${referrerPolicy}'};
110
+ headers['strict-transport-security'] = {value: '${strictTransportSecurity}'};
111
+ headers['x-content-type-options'] = {value: 'nosniff'};
112
+ headers['x-frame-options'] = {value: '${frameOptions}'};
113
+ headers['x-xss-protection'] = {value: '1; mode=block'};
114
+ headers['${cspHeaderName}'] = {value: "${contentSecurityPolicy}"};
115
+ return response;
116
+ }`;
117
+ // Hardcoded logical ID due to bug: https://github.com/aws/aws-cdk/issues/15523
118
+ const functionId = `Function${this.node.addr}`;
119
+ this.securityHeadersFunction = new cloudfront.Function(this, functionId, {
120
+ functionName: functionId,
121
+ code: cloudfront.FunctionCode.fromInline(lambdaCode),
122
+ });
123
+ }
124
+ }
125
+ exports.WebappSecurityHeaders = WebappSecurityHeaders;
126
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"security-headers.js","sourceRoot":"","sources":["../../src/webapp/security-headers.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cloudfront from \"@aws-cdk/aws-cloudfront\"\nimport * as cdk from \"@aws-cdk/core\"\n\nexport interface FrameOptionsHeader {\n  value?: \"DENY\" | \"SAMEORIGIN\"\n}\n\nexport interface ReferrerPolicyHeader {\n  value?: string\n}\n\nexport interface StrictTransportSecurityHeader {\n  maxAge?: number\n  includeSubDomains?: boolean\n  preload?: boolean\n}\n\nexport interface ContentSecurityPolicyHeader {\n  reportOnly?: boolean\n  baseUri?: string\n  childSrc?: string\n  defaultSrc?: string\n  fontSrc?: string\n  frameSrc?: string\n  formAction?: string\n  frameAncestors?: string\n  imgSrc?: string\n  manifestSrc?: string\n  mediaSrc?: string\n  objectSrc?: string\n  scriptSrc?: string\n  styleSrc?: string\n  connectSrc?: string\n}\n\nexport interface SecurityHeaders {\n  contentSecurityPolicy?: ContentSecurityPolicyHeader\n  strictTransportSecurity?: StrictTransportSecurityHeader\n  referrerPolicy?: ReferrerPolicyHeader\n  frameOptions?: FrameOptionsHeader\n}\n\nfunction validateCspParam(param: string): string {\n  if (param.indexOf('\"') !== -1) {\n    throw Error('CSP override contains invalid character \"')\n  }\n\n  if (param.indexOf(\";\") !== -1) {\n    throw Error(\"CSP override contains invalid character ;\")\n  }\n\n  if (param.indexOf(\"\\\\\") !== -1) {\n    throw Error(\"CSP override contains invalid character \\\\\")\n  }\n\n  return param\n}\n\n/* Replace all whitespace in a string with a single space */\nfunction trim(value: string): string {\n  return value.replace(/\\s+/g, \" \").trim()\n}\n\nfunction generateContentSecurityPolicyHeader(\n  headerOptions?: ContentSecurityPolicyHeader,\n) {\n  const defaultValues = {\n    baseUri: \"'self'\",\n    childSrc: \"'none'\",\n    connectSrc: \"'self'\",\n    defaultSrc: \"'self'\",\n    fontSrc: \"'self'\",\n    formAction: \"'self'\",\n    frameAncestors: \"'none'\",\n    frameSrc: \"'self'\",\n    imgSrc: \"'self'\",\n    manifestSrc: \"'self'\",\n    mediaSrc: \"'self'\",\n    objectSrc: \"'none'\",\n    scriptSrc: \"'self'\",\n    styleSrc: \"'self'\",\n  }\n\n  const options = {\n    ...defaultValues,\n    ...headerOptions,\n  }\n\n  Object.values(options).forEach(\n    (v) => typeof v === \"string\" && validateCspParam(v),\n  )\n\n  let headerValue = \"\"\n  headerValue += `base-uri ${trim(options.baseUri)};`\n  headerValue += `child-src ${trim(options.childSrc)};`\n  headerValue += `connect-src ${trim(options.connectSrc)};`\n  headerValue += `default-src ${trim(options.defaultSrc)};`\n  headerValue += `font-src ${trim(options.fontSrc)};`\n  headerValue += `frame-src ${trim(options.frameSrc)};`\n  headerValue += `img-src ${trim(options.imgSrc)};`\n  headerValue += `manifest-src ${trim(options.manifestSrc)};`\n  headerValue += `media-src ${trim(options.mediaSrc)};`\n  headerValue += `object-src ${trim(options.objectSrc)};`\n  headerValue += `script-src ${trim(options.scriptSrc)};`\n  headerValue += `style-src ${trim(options.styleSrc)};`\n\n  return trim(headerValue)\n}\n\nfunction generateStrictTransportSecurityHeader(\n  headerOptions?: StrictTransportSecurityHeader,\n) {\n  const defaultValues = {\n    maxAge: 63072000,\n    includeSubDomains: false,\n    preload: false,\n  }\n  const options = {\n    ...defaultValues,\n    ...headerOptions,\n  }\n  let headerValue = \"\"\n  headerValue += `max-age=${options.maxAge};`\n  headerValue += options.preload ? \"preload;\" : \"\"\n  headerValue += options.includeSubDomains ? \"includeSubDomains;\" : \"\"\n  return trim(headerValue)\n}\n\nfunction generateReferrerPolicyHeader(headerOptions?: ReferrerPolicyHeader) {\n  const defaultValues = {\n    value: \"strict-origin-when-cross-origin\",\n  }\n  const options = {\n    ...defaultValues,\n    ...headerOptions,\n  }\n  return options.value\n}\n\nfunction generateFrameOptionsHeader(headerOptions?: FrameOptionsHeader) {\n  const defaultValues = {\n    value: \"DENY\",\n  }\n  const options = {\n    ...defaultValues,\n    ...headerOptions,\n  }\n  return trim(options.value)\n}\n\nexport class WebappSecurityHeaders extends cdk.Construct {\n  public readonly securityHeadersFunction: cloudfront.Function\n\n  constructor(scope: cdk.Construct, id: string, props: SecurityHeaders) {\n    super(scope, id)\n\n    const cspHeaderName = props.contentSecurityPolicy?.reportOnly\n      ? \"content-security-policy-report-only\"\n      : \"content-security-policy\"\n\n    const contentSecurityPolicy = generateContentSecurityPolicyHeader(\n      props.contentSecurityPolicy,\n    )\n    const strictTransportSecurity = generateStrictTransportSecurityHeader(\n      props.strictTransportSecurity,\n    )\n    const referrerPolicy = generateReferrerPolicyHeader(props.referrerPolicy)\n    const frameOptions = generateFrameOptionsHeader(props.frameOptions)\n\n    const lambdaCode = `function handler(event) {\n      var response = event.response;\n      var headers = response.headers;\n      headers['referrer-policy'] = {value: '${referrerPolicy}'};\n      headers['strict-transport-security'] = {value: '${strictTransportSecurity}'};\n      headers['x-content-type-options'] = {value: 'nosniff'};\n      headers['x-frame-options'] = {value: '${frameOptions}'};\n      headers['x-xss-protection'] = {value: '1; mode=block'};\n      headers['${cspHeaderName}'] = {value: \"${contentSecurityPolicy}\"};\n      return response;\n    }`\n\n    // Hardcoded logical ID due to bug: https://github.com/aws/aws-cdk/issues/15523\n    const functionId = `Function${this.node.addr}`\n\n    this.securityHeadersFunction = new cloudfront.Function(this, functionId, {\n      functionName: functionId,\n      code: cloudfront.FunctionCode.fromInline(lambdaCode),\n    })\n  }\n}\n"]}
@@ -5,6 +5,7 @@ import * as r53 from "@aws-cdk/aws-route53";
5
5
  import * as s3 from "@aws-cdk/aws-s3";
6
6
  import * as cdk from "@aws-cdk/core";
7
7
  import * as webappDeploy from "@capraconsulting/webapp-deploy-lambda";
8
+ import { SecurityHeaders } from "./security-headers";
8
9
  export interface WebappProps {
9
10
  /**
10
11
  * ACM certificate that covers the specifeid domain names.
@@ -40,6 +41,28 @@ export interface WebappProps {
40
41
  * @default - No custom page for 403 errors.
41
42
  */
42
43
  webAclErrorPagePath?: string;
44
+ /**
45
+ * Enable adding common security headers to CloudFront responses using a CloudFront Function.
46
+ *
47
+ * If enabled, the default behavior is to add the following headers with fairly strict defaults. Most of the headers can be customized:
48
+ * - Content-Security-Policy
49
+ * - Referrer-Policy
50
+ * - Strict-Transport-Security
51
+ * - X-Content-Type-Options
52
+ * - X-Frame-Options
53
+ * - X-XSS-Protection
54
+ *
55
+ * @default - No security headers will be added to responses
56
+ */
57
+ enableSecurityHeaders?: boolean;
58
+ /**
59
+ * Security headers overrides.
60
+ *
61
+ * Used to override certain security header values if the webapp requires more lax settings compared to the defaults.
62
+ *
63
+ * @default - A set of strict security header values will be used
64
+ */
65
+ securityHeadersOverrides?: SecurityHeaders;
43
66
  }
44
67
  /**
45
68
  * CloudFront for a Single-Page-Application.