@liflig/cdk 2.18.5 → 2.18.6
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/assets/cloudtrail-slack-integration-lambda/main.py +267 -0
- package/assets/pipeline-slack-notification-lambda/index.py +300 -0
- package/assets/prepare-cdk-source-lambda/index.py +159 -0
- package/assets/slack-alarm-lambda/index.py +103 -0
- package/lib/alarms/database-alarms.d.ts +125 -0
- package/lib/alarms/database-alarms.js +171 -0
- package/lib/alarms/index.d.ts +3 -0
- package/lib/alarms/index.js +10 -0
- package/lib/alarms/service-alarms.d.ts +145 -0
- package/lib/alarms/service-alarms.js +148 -0
- package/lib/alarms/ses-alarms.d.ts +67 -0
- package/lib/alarms/ses-alarms.js +49 -0
- package/lib/alarms/slack-alarm.d.ts +25 -0
- package/lib/alarms/slack-alarm.js +47 -0
- package/lib/bastion-host.d.ts +41 -0
- package/lib/bastion-host.js +86 -0
- package/lib/bin/cdk-create-snapshots.d.ts +2 -0
- package/lib/bin/fetch-pipeline-variables.d.ts +2 -0
- package/lib/build-artifacts/index.d.ts +68 -0
- package/lib/build-artifacts/index.js +118 -0
- package/lib/cdk-deploy/cdk-deploy.d.ts +63 -0
- package/lib/cdk-deploy/cdk-deploy.js +175 -0
- package/lib/cdk-deploy/index.d.ts +1 -0
- package/lib/cdk-deploy/index.js +6 -0
- package/lib/cdk-deploy/start-deploy-handler.d.ts +8 -0
- package/lib/cdk-deploy/start-deploy-handler.js +72 -0
- package/lib/cdk-deploy/status-handler.d.ts +6 -0
- package/lib/cdk-deploy/status-handler.js +83 -0
- package/lib/cdk-pipelines/cloud-assembly-lookup-handler.d.ts +6 -0
- package/lib/cdk-pipelines/cloud-assembly-lookup-handler.js +63 -0
- package/lib/cdk-pipelines/index.d.ts +3 -0
- package/lib/cdk-pipelines/index.js +10 -0
- package/lib/cdk-pipelines/liflig-cdk-pipeline.d.ts +110 -0
- package/lib/cdk-pipelines/liflig-cdk-pipeline.js +232 -0
- package/lib/cdk-pipelines/slack-notification.d.ts +51 -0
- package/lib/cdk-pipelines/slack-notification.js +54 -0
- package/lib/cdk-pipelines/variables.d.ts +15 -0
- package/lib/cdk-pipelines/variables.js +80 -0
- package/lib/cloudtrail-slack-integration/cloudtrail-slack-integration.d.ts +47 -0
- package/lib/cloudtrail-slack-integration/cloudtrail-slack-integration.js +211 -0
- package/lib/cloudtrail-slack-integration/index.d.ts +1 -0
- package/lib/cloudtrail-slack-integration/index.js +6 -0
- package/lib/configure-parameters/configure-parameters.d.ts +61 -0
- package/lib/configure-parameters/configure-parameters.js +94 -0
- package/lib/configure-parameters/index.d.ts +1 -0
- package/lib/configure-parameters/index.js +6 -0
- package/lib/cross-region-ssm-parameter.d.ts +13 -0
- package/lib/cross-region-ssm-parameter.js +46 -0
- package/lib/ecs/cluster.d.ts +25 -0
- package/lib/ecs/cluster.js +70 -0
- package/lib/ecs/fargate-service.d.ts +63 -0
- package/lib/ecs/fargate-service.js +98 -0
- package/lib/ecs/index.d.ts +3 -0
- package/lib/ecs/index.js +10 -0
- package/lib/ecs/listener-rule.d.ts +25 -0
- package/lib/ecs/listener-rule.js +27 -0
- package/lib/ecs-update-image/artifact-status.d.ts +39 -0
- package/lib/ecs-update-image/artifact-status.js +41 -0
- package/lib/ecs-update-image/ecs-update-image.d.ts +41 -0
- package/lib/ecs-update-image/ecs-update-image.js +98 -0
- package/lib/ecs-update-image/index.d.ts +3 -0
- package/lib/ecs-update-image/index.js +10 -0
- package/lib/ecs-update-image/start-deploy-handler.d.ts +6 -0
- package/lib/ecs-update-image/start-deploy-handler.js +104 -0
- package/lib/ecs-update-image/status-handler.d.ts +11 -0
- package/lib/ecs-update-image/status-handler.js +74 -0
- package/lib/ecs-update-image/tag.d.ts +47 -0
- package/lib/ecs-update-image/tag.js +67 -0
- package/lib/feature-flags.d.ts +18 -0
- package/lib/feature-flags.js +48 -0
- package/lib/griid/artefact-bucket.d.ts +7 -0
- package/lib/griid/artefact-bucket.js +30 -0
- package/lib/griid/index.d.ts +4 -0
- package/lib/griid/index.js +18 -0
- package/lib/hosted-zone-with-param.d.ts +29 -0
- package/lib/hosted-zone-with-param.js +65 -0
- package/lib/index.d.ts +32 -0
- package/lib/kinesis/index.d.ts +1 -0
- package/lib/kinesis/index.js +6 -0
- package/lib/kinesis/kinesis-to-datadog-stream.d.ts +28 -0
- package/lib/kinesis/kinesis-to-datadog-stream.js +126 -0
- package/lib/load-balancer/index.d.ts +1 -0
- package/lib/load-balancer/index.js +6 -0
- package/lib/load-balancer/load-balancer.d.ts +16 -0
- package/lib/load-balancer/load-balancer.js +60 -0
- package/lib/pipelines/conventions.d.ts +14 -0
- package/lib/pipelines/conventions.js +24 -0
- package/lib/pipelines/deploy-env.d.ts +18 -0
- package/lib/pipelines/deploy-env.js +96 -0
- package/lib/pipelines/index.d.ts +2 -0
- package/lib/pipelines/index.js +8 -0
- package/lib/pipelines/liflig-cdk-deployer-deps.d.ts +13 -0
- package/lib/pipelines/liflig-cdk-deployer-deps.js +35 -0
- package/lib/pipelines/pipeline.d.ts +78 -0
- package/lib/pipelines/pipeline.js +224 -0
- package/lib/platform/index.d.ts +1 -0
- package/lib/platform/index.js +7 -0
- package/lib/platform/platform.d.ts +37 -0
- package/lib/platform/platform.js +57 -0
- package/lib/rds/database.d.ts +49 -0
- package/lib/rds/database.js +60 -0
- package/lib/rds/index.d.ts +1 -0
- package/lib/rds/index.js +6 -0
- package/lib/ses/configurationsetdeliveryoptions/index.d.ts +26 -0
- package/lib/ses/configurationsetdeliveryoptions/index.js +48 -0
- package/lib/ses/configurationsetsnsdestination/handler.d.ts +17 -0
- package/lib/ses/configurationsetsnsdestination/handler.js +75 -0
- package/lib/ses/configurationsetsnsdestination/index.d.ts +29 -0
- package/lib/ses/configurationsetsnsdestination/index.js +75 -0
- package/lib/ses/index.d.ts +4 -0
- package/lib/ses/index.js +12 -0
- package/lib/ses/sesdomain/handler.d.ts +10 -0
- package/lib/ses/sesdomain/handler.js +82 -0
- package/lib/ses/sesdomain/index.d.ts +57 -0
- package/lib/ses/sesdomain/index.js +94 -0
- package/lib/ses/sesverifyemail/handler.d.ts +9 -0
- package/lib/ses/sesverifyemail/handler.js +25 -0
- package/lib/ses/sesverifyemail/index.d.ts +13 -0
- package/lib/ses/sesverifyemail/index.js +51 -0
- package/lib/snapshots.d.ts +4 -0
- package/lib/snapshots.js +214 -0
- package/lib/ssm-parameter-backed-resource.d.ts +45 -0
- package/lib/ssm-parameter-backed-resource.js +67 -0
- package/lib/ssm-parameter-reader.d.ts +21 -0
- package/lib/ssm-parameter-reader.js +48 -0
- package/lib/tags.d.ts +8 -0
- package/lib/tags.js +36 -0
- package/lib/utils.d.ts +2 -0
- package/lib/utils.js +17 -0
- package/lib/webapp/index.d.ts +3 -0
- package/lib/webapp/index.js +10 -0
- package/lib/webapp/monitor.d.ts +187 -0
- package/lib/webapp/monitor.js +156 -0
- package/lib/webapp/security-headers.d.ts +38 -0
- package/lib/webapp/security-headers.js +129 -0
- package/lib/webapp/webapp.d.ts +116 -0
- package/lib/webapp/webapp.js +118 -0
- package/lib/webapp-deploy-via-role.d.ts +25 -0
- package/lib/webapp-deploy-via-role.js +32 -0
- package/package.json +3 -2
|
@@ -0,0 +1,187 @@
|
|
|
1
|
+
import * as cdk from "aws-cdk-lib";
|
|
2
|
+
import * as cloudwatch from "aws-cdk-lib/aws-cloudwatch";
|
|
3
|
+
import * as cognito from "aws-cdk-lib/aws-cognito";
|
|
4
|
+
import * as iam from "aws-cdk-lib/aws-iam";
|
|
5
|
+
import * as rum from "aws-cdk-lib/aws-rum";
|
|
6
|
+
import * as constructs from "constructs";
|
|
7
|
+
type CwRumTelemetries = ("http" | "performance" | "errors")[];
|
|
8
|
+
type EventsAlarm = {
|
|
9
|
+
/**
|
|
10
|
+
* An action to use for CloudWatch alarm state changes instead of the default action
|
|
11
|
+
*/
|
|
12
|
+
action?: cloudwatch.IAlarmAction;
|
|
13
|
+
/**
|
|
14
|
+
* The threshold defined as the number of CloudWatch RUM events that determines if the alarm should trigger or not.
|
|
15
|
+
* @default 1000
|
|
16
|
+
*/
|
|
17
|
+
threshold?: number;
|
|
18
|
+
/**
|
|
19
|
+
* @default 1
|
|
20
|
+
*/
|
|
21
|
+
evaluationPeriods?: number;
|
|
22
|
+
/**
|
|
23
|
+
* @default cdk.Duration.minutes(1)
|
|
24
|
+
*/
|
|
25
|
+
period?: cdk.Duration;
|
|
26
|
+
/**
|
|
27
|
+
* @default false
|
|
28
|
+
*/
|
|
29
|
+
enableOkAlarm?: boolean;
|
|
30
|
+
};
|
|
31
|
+
type ErrorsAlarm = {
|
|
32
|
+
/**
|
|
33
|
+
* An action to use for CloudWatch alarm state changes instead of the default action
|
|
34
|
+
*/
|
|
35
|
+
action?: cloudwatch.IAlarmAction;
|
|
36
|
+
/**
|
|
37
|
+
* The threshold defined as the number of JavaScript errors recorded by CloudWatch RUM that determines if the alarm should trigger or not.
|
|
38
|
+
* @default 0
|
|
39
|
+
*/
|
|
40
|
+
threshold?: number;
|
|
41
|
+
/**
|
|
42
|
+
* @default 1
|
|
43
|
+
*/
|
|
44
|
+
evaluationPeriods?: number;
|
|
45
|
+
/**
|
|
46
|
+
* @default cdk.Duration.minutes(1)
|
|
47
|
+
*/
|
|
48
|
+
period?: cdk.Duration;
|
|
49
|
+
/**
|
|
50
|
+
* @default false
|
|
51
|
+
*/
|
|
52
|
+
enableOkAlarms: boolean;
|
|
53
|
+
};
|
|
54
|
+
interface WebappMonitorProps {
|
|
55
|
+
/**
|
|
56
|
+
* The name of the CloudWatch RUM App Monitor.
|
|
57
|
+
*
|
|
58
|
+
* NOTE: This name needs to be unique within a given AWS region.
|
|
59
|
+
*/
|
|
60
|
+
appMonitorName: string;
|
|
61
|
+
/**
|
|
62
|
+
* The domain the web application to monitor is served on.
|
|
63
|
+
*
|
|
64
|
+
* Only web applications served on this domain can publish data to CloudWatch RUM.
|
|
65
|
+
*/
|
|
66
|
+
webappDomainName: string;
|
|
67
|
+
/**
|
|
68
|
+
* Which telemetries to allow CloudWatch RUM to collect.
|
|
69
|
+
*
|
|
70
|
+
* Currently these telemetries are supported:
|
|
71
|
+
* - "http" report the http fetch calles made by browser to CW RUM
|
|
72
|
+
* - "errors" report the js errors which occur, passed via rumConfig.recordError
|
|
73
|
+
* - "performance" report the app performance
|
|
74
|
+
*
|
|
75
|
+
* @default: ["errors", "http", "performance"]
|
|
76
|
+
*/
|
|
77
|
+
telemetries?: CwRumTelemetries;
|
|
78
|
+
/**
|
|
79
|
+
* The default CloudWatch alarm action to use.
|
|
80
|
+
*
|
|
81
|
+
* By default, an alarm is created for a high number of events being sent to CloudWatch RUM. This alarm action is used for the default alarm,
|
|
82
|
+
* and for other alarms if no other action is explicitly set when configuring or enabling alarms through this construct.
|
|
83
|
+
*
|
|
84
|
+
*/
|
|
85
|
+
defaultAlarmAction: cloudwatch.IAlarmAction;
|
|
86
|
+
/**
|
|
87
|
+
* Allow CloudWatch RUM to record custom events.
|
|
88
|
+
*
|
|
89
|
+
* These events need to be defined in and sent from your web application.
|
|
90
|
+
*
|
|
91
|
+
* @default false
|
|
92
|
+
*/
|
|
93
|
+
enableXRay?: boolean;
|
|
94
|
+
/**
|
|
95
|
+
* Allow CloudWatch RUM to use cookies.
|
|
96
|
+
*
|
|
97
|
+
* CloudWatch RUM does not require cookies in order to record errors, but some features require cookies to be enabled.
|
|
98
|
+
* @default false
|
|
99
|
+
*/
|
|
100
|
+
enableCustomEvents?: boolean;
|
|
101
|
+
/**
|
|
102
|
+
* Allow CloudWatch RUM to use cookies.
|
|
103
|
+
*
|
|
104
|
+
* CloudWatch RUM does not require cookies in order to record errors, but some features require cookies to be enabled.
|
|
105
|
+
*
|
|
106
|
+
* See: https://aws.amazon.com/blogs/mt/how-and-when-to-enable-session-cookies-with-amazon-cloudwatch-rum/
|
|
107
|
+
*
|
|
108
|
+
* NOTE: You'll likely need to introduce a cookie consent popup in your web application
|
|
109
|
+
* before enabling cookies in your client-side configuration.
|
|
110
|
+
*
|
|
111
|
+
* @default true
|
|
112
|
+
*/
|
|
113
|
+
allowCookies?: boolean;
|
|
114
|
+
/**
|
|
115
|
+
* The rate of client-side CloudWatch RUM sessions to sample.
|
|
116
|
+
*
|
|
117
|
+
* Values are between 0 and 1, where 1 is 100% of sessions.
|
|
118
|
+
*
|
|
119
|
+
* NOTE: You'll likely want to set this to a number below 1 in production environments with
|
|
120
|
+
* a lot of traffic.
|
|
121
|
+
*
|
|
122
|
+
* @default 1
|
|
123
|
+
*/
|
|
124
|
+
sessionSampleRate?: number;
|
|
125
|
+
/**
|
|
126
|
+
*
|
|
127
|
+
* Whether to store a copy of CloudWatch RUM events in CloudWatch Logs.
|
|
128
|
+
*
|
|
129
|
+
* Enabling this can improve the ergonomics of finding relevant CloudWatch RUM events compared to
|
|
130
|
+
* using the CloudWatch RUM console as it makes it possible to search using CloudWatch Logs Insights etc.
|
|
131
|
+
*
|
|
132
|
+
* @default false
|
|
133
|
+
*/
|
|
134
|
+
exportLogs?: boolean;
|
|
135
|
+
/**
|
|
136
|
+
* Override top-level properties for the rum.CfnAppMonitor construct
|
|
137
|
+
*
|
|
138
|
+
* @default - no overrides are used.
|
|
139
|
+
*/
|
|
140
|
+
overrideAppMonitorProps?: Partial<rum.CfnAppMonitorProps>;
|
|
141
|
+
/**
|
|
142
|
+
* Override App Monitor configuration properties for the rum.CfnAppMonitor construct
|
|
143
|
+
*
|
|
144
|
+
* @default - no overrides are used.
|
|
145
|
+
*/
|
|
146
|
+
overrideAppMonitorConfiguration?: Partial<rum.CfnAppMonitor.AppMonitorConfigurationProperty>;
|
|
147
|
+
}
|
|
148
|
+
/**
|
|
149
|
+
* A construct for creating a CloudWatch Real User Monitoring (RUM) App Monitor.
|
|
150
|
+
*
|
|
151
|
+
* An app monitor allows you to centrally collect and view client-side data (JavaScript and HTTP errors, performance metrics, etc.)
|
|
152
|
+
* recorded from a web application.
|
|
153
|
+
*
|
|
154
|
+
* NOTE: You'll need to add a JavaScript snippet to your web application and configure CloudWatch RUM using
|
|
155
|
+
* AWS's client-side SDK after deploying the construct to enable your web application to start collecting data.
|
|
156
|
+
* The App Monitor controls which CloudWatch RUM functionality the web application can use, and in order to
|
|
157
|
+
* leverage given functionality you'll need to configure it both in the App Monitor as well as in the
|
|
158
|
+
* configuration of the client-side SDK used by your web application.
|
|
159
|
+
*
|
|
160
|
+
*/
|
|
161
|
+
export declare class WebappMonitor extends constructs.Construct {
|
|
162
|
+
readonly identityPool: cognito.CfnIdentityPool;
|
|
163
|
+
readonly guestRole: iam.IRole;
|
|
164
|
+
readonly appMonitor: rum.CfnAppMonitor;
|
|
165
|
+
private readonly _webappDomainName;
|
|
166
|
+
private readonly _defaultAlarmAction;
|
|
167
|
+
private _eventsAlarm;
|
|
168
|
+
private _eventsAlarmId;
|
|
169
|
+
constructor(scope: constructs.Construct, id: string, props: WebappMonitorProps);
|
|
170
|
+
/**
|
|
171
|
+
*
|
|
172
|
+
* Configure an alarm used to notify on an unexpectedly high number of CloudWatch RUM events being recorded.
|
|
173
|
+
*
|
|
174
|
+
* This alarm is required when using this construct.
|
|
175
|
+
*
|
|
176
|
+
* @default - an alarm is created with sane defaults.
|
|
177
|
+
*
|
|
178
|
+
*/
|
|
179
|
+
configureEventsAlarm(props?: EventsAlarm): void;
|
|
180
|
+
private _createEventsAlarm;
|
|
181
|
+
/**
|
|
182
|
+
* Add an alarm that alerts on JavaScript errors
|
|
183
|
+
* sent to CloudWatch RUM.
|
|
184
|
+
*/
|
|
185
|
+
addErrorsAlarm(props?: ErrorsAlarm): void;
|
|
186
|
+
}
|
|
187
|
+
export {};
|
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.WebappMonitor = void 0;
|
|
4
|
+
const cdk = require("aws-cdk-lib");
|
|
5
|
+
const cloudwatch = require("aws-cdk-lib/aws-cloudwatch");
|
|
6
|
+
const cognito = require("aws-cdk-lib/aws-cognito");
|
|
7
|
+
const iam = require("aws-cdk-lib/aws-iam");
|
|
8
|
+
const rum = require("aws-cdk-lib/aws-rum");
|
|
9
|
+
const constructs = require("constructs");
|
|
10
|
+
/**
|
|
11
|
+
* A construct for creating a CloudWatch Real User Monitoring (RUM) App Monitor.
|
|
12
|
+
*
|
|
13
|
+
* An app monitor allows you to centrally collect and view client-side data (JavaScript and HTTP errors, performance metrics, etc.)
|
|
14
|
+
* recorded from a web application.
|
|
15
|
+
*
|
|
16
|
+
* NOTE: You'll need to add a JavaScript snippet to your web application and configure CloudWatch RUM using
|
|
17
|
+
* AWS's client-side SDK after deploying the construct to enable your web application to start collecting data.
|
|
18
|
+
* The App Monitor controls which CloudWatch RUM functionality the web application can use, and in order to
|
|
19
|
+
* leverage given functionality you'll need to configure it both in the App Monitor as well as in the
|
|
20
|
+
* configuration of the client-side SDK used by your web application.
|
|
21
|
+
*
|
|
22
|
+
*/
|
|
23
|
+
class WebappMonitor extends constructs.Construct {
|
|
24
|
+
constructor(scope, id, props) {
|
|
25
|
+
var _a, _b, _c, _d, _e;
|
|
26
|
+
super(scope, id);
|
|
27
|
+
this._eventsAlarmId = "EventsAlarm";
|
|
28
|
+
if (props.sessionSampleRate &&
|
|
29
|
+
(props.sessionSampleRate < 0 || props.sessionSampleRate > 1)) {
|
|
30
|
+
throw new Error("sessionSampleRate needs to be a number between 0 and 1");
|
|
31
|
+
}
|
|
32
|
+
this._webappDomainName = props.webappDomainName;
|
|
33
|
+
const appMonitorName = props.appMonitorName;
|
|
34
|
+
this.identityPool = new cognito.CfnIdentityPool(this, "RumIdentityPool", {
|
|
35
|
+
allowUnauthenticatedIdentities: true,
|
|
36
|
+
});
|
|
37
|
+
this.guestRole = new iam.Role(this, "RumRole", {
|
|
38
|
+
assumedBy: new iam.WebIdentityPrincipal("cognito-identity.amazonaws.com", {
|
|
39
|
+
StringEquals: {
|
|
40
|
+
"cognito-identity.amazonaws.com:aud": this.identityPool.ref,
|
|
41
|
+
},
|
|
42
|
+
"ForAnyValue:StringEquals": {
|
|
43
|
+
"cognito-identity.amazonaws.com:amr": "unauthenticated",
|
|
44
|
+
},
|
|
45
|
+
}),
|
|
46
|
+
inlinePolicies: {
|
|
47
|
+
"allow-rum": new iam.PolicyDocument({
|
|
48
|
+
statements: [
|
|
49
|
+
new iam.PolicyStatement({
|
|
50
|
+
actions: ["rum:PutRumEvents"],
|
|
51
|
+
resources: [
|
|
52
|
+
cdk.Stack.of(this).formatArn({
|
|
53
|
+
service: "rum",
|
|
54
|
+
resource: "appmonitor",
|
|
55
|
+
resourceName: appMonitorName,
|
|
56
|
+
}),
|
|
57
|
+
],
|
|
58
|
+
}),
|
|
59
|
+
],
|
|
60
|
+
}),
|
|
61
|
+
},
|
|
62
|
+
});
|
|
63
|
+
// attach role to identity pool
|
|
64
|
+
new cognito.CfnIdentityPoolRoleAttachment(this, "RoleAttachment", {
|
|
65
|
+
identityPoolId: this.identityPool.ref,
|
|
66
|
+
roles: {
|
|
67
|
+
unauthenticated: this.guestRole.roleArn,
|
|
68
|
+
},
|
|
69
|
+
});
|
|
70
|
+
this.appMonitor = new rum.CfnAppMonitor(this, "AppMonitor", {
|
|
71
|
+
name: appMonitorName,
|
|
72
|
+
domain: props.webappDomainName,
|
|
73
|
+
cwLogEnabled: (_a = props.exportLogs) !== null && _a !== void 0 ? _a : false,
|
|
74
|
+
appMonitorConfiguration: {
|
|
75
|
+
enableXRay: (_b = props.enableXRay) !== null && _b !== void 0 ? _b : false,
|
|
76
|
+
allowCookies: (_c = props.allowCookies) !== null && _c !== void 0 ? _c : true,
|
|
77
|
+
telemetries: (_d = props.telemetries) !== null && _d !== void 0 ? _d : ["errors", "http", "performance"],
|
|
78
|
+
sessionSampleRate: (_e = props.sessionSampleRate) !== null && _e !== void 0 ? _e : 1,
|
|
79
|
+
identityPoolId: this.identityPool.ref,
|
|
80
|
+
guestRoleArn: this.guestRole.roleArn,
|
|
81
|
+
...props.overrideAppMonitorConfiguration,
|
|
82
|
+
},
|
|
83
|
+
customEvents: {
|
|
84
|
+
status: props.enableCustomEvents ? "ENABLED" : "DISABLED",
|
|
85
|
+
},
|
|
86
|
+
...props.overrideAppMonitorProps,
|
|
87
|
+
});
|
|
88
|
+
this._defaultAlarmAction = props.defaultAlarmAction;
|
|
89
|
+
this._eventsAlarm = this._createEventsAlarm();
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
*
|
|
93
|
+
* Configure an alarm used to notify on an unexpectedly high number of CloudWatch RUM events being recorded.
|
|
94
|
+
*
|
|
95
|
+
* This alarm is required when using this construct.
|
|
96
|
+
*
|
|
97
|
+
* @default - an alarm is created with sane defaults.
|
|
98
|
+
*
|
|
99
|
+
*/
|
|
100
|
+
configureEventsAlarm(props) {
|
|
101
|
+
this.node.tryRemoveChild(this._eventsAlarmId);
|
|
102
|
+
this._eventsAlarm = this._createEventsAlarm(props);
|
|
103
|
+
}
|
|
104
|
+
_createEventsAlarm(props) {
|
|
105
|
+
var _a, _b, _c, _d, _e, _f;
|
|
106
|
+
const eventsAlarm = new cloudwatch.Metric({
|
|
107
|
+
metricName: "RumEventPayloadSize",
|
|
108
|
+
namespace: "AWS/RUM",
|
|
109
|
+
statistic: cloudwatch.Stats.SAMPLE_COUNT,
|
|
110
|
+
period: (_a = props === null || props === void 0 ? void 0 : props.period) !== null && _a !== void 0 ? _a : cdk.Duration.seconds(60),
|
|
111
|
+
dimensionsMap: {
|
|
112
|
+
application_name: this.appMonitor.name,
|
|
113
|
+
},
|
|
114
|
+
}).createAlarm(this, this._eventsAlarmId, {
|
|
115
|
+
alarmDescription: `A high number of events are being sent to CloudWatch RUM for app monitor '${this.appMonitor.name}'. This may lead to high costs, so you'll likely want to investigate this.`,
|
|
116
|
+
comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
117
|
+
threshold: (_b = props === null || props === void 0 ? void 0 : props.threshold) !== null && _b !== void 0 ? _b : 1000,
|
|
118
|
+
evaluationPeriods: (_c = props === null || props === void 0 ? void 0 : props.evaluationPeriods) !== null && _c !== void 0 ? _c : 1,
|
|
119
|
+
treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
|
|
120
|
+
});
|
|
121
|
+
// These alarms are mandatory as costs should be mandator to monitor
|
|
122
|
+
eventsAlarm.addAlarmAction((_d = props === null || props === void 0 ? void 0 : props.action) !== null && _d !== void 0 ? _d : this._defaultAlarmAction);
|
|
123
|
+
if ((_e = props === null || props === void 0 ? void 0 : props.enableOkAlarm) !== null && _e !== void 0 ? _e : false) {
|
|
124
|
+
eventsAlarm.addOkAction((_f = props === null || props === void 0 ? void 0 : props.action) !== null && _f !== void 0 ? _f : this._defaultAlarmAction);
|
|
125
|
+
}
|
|
126
|
+
return eventsAlarm;
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* Add an alarm that alerts on JavaScript errors
|
|
130
|
+
* sent to CloudWatch RUM.
|
|
131
|
+
*/
|
|
132
|
+
addErrorsAlarm(props) {
|
|
133
|
+
var _a, _b, _c, _d, _e, _f;
|
|
134
|
+
const errorsAlarm = new cloudwatch.Metric({
|
|
135
|
+
metricName: "JsErrorCount",
|
|
136
|
+
namespace: "AWS/RUM",
|
|
137
|
+
statistic: cloudwatch.Stats.SAMPLE_COUNT,
|
|
138
|
+
period: (_a = props === null || props === void 0 ? void 0 : props.period) !== null && _a !== void 0 ? _a : cdk.Duration.seconds(60),
|
|
139
|
+
dimensionsMap: {
|
|
140
|
+
application_name: this.appMonitor.name,
|
|
141
|
+
},
|
|
142
|
+
}).createAlarm(this, "RumJsErrorsAlarm", {
|
|
143
|
+
alarmDescription: `CloudWatch RUM has recorded JavaScript errors for app monitor '${this.appMonitor.name}' on domain '${this._webappDomainName}'`,
|
|
144
|
+
comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
145
|
+
threshold: (_b = props === null || props === void 0 ? void 0 : props.threshold) !== null && _b !== void 0 ? _b : 0,
|
|
146
|
+
evaluationPeriods: (_c = props === null || props === void 0 ? void 0 : props.evaluationPeriods) !== null && _c !== void 0 ? _c : 1,
|
|
147
|
+
treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
|
|
148
|
+
});
|
|
149
|
+
errorsAlarm.addAlarmAction((_d = props === null || props === void 0 ? void 0 : props.action) !== null && _d !== void 0 ? _d : this._defaultAlarmAction);
|
|
150
|
+
if ((_e = props === null || props === void 0 ? void 0 : props.enableOkAlarms) !== null && _e !== void 0 ? _e : false) {
|
|
151
|
+
errorsAlarm.addOkAction((_f = props === null || props === void 0 ? void 0 : props.action) !== null && _f !== void 0 ? _f : this._defaultAlarmAction);
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
exports.WebappMonitor = WebappMonitor;
|
|
156
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"monitor.js","sourceRoot":"","sources":["../../src/webapp/monitor.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport * as cloudwatch from \"aws-cdk-lib/aws-cloudwatch\"\nimport * as cognito from \"aws-cdk-lib/aws-cognito\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as rum from \"aws-cdk-lib/aws-rum\"\nimport * as constructs from \"constructs\"\n\ntype CwRumTelemetries = (\"http\" | \"performance\" | \"errors\")[]\n\ntype EventsAlarm = {\n  /**\n   * An action to use for CloudWatch alarm state changes instead of the default action\n   */\n  action?: cloudwatch.IAlarmAction\n  /**\n   * The threshold defined as the number of CloudWatch RUM events that determines if the alarm should trigger or not.\n   * @default 1000\n   */\n  threshold?: number\n  /**\n   * @default 1\n   */\n  evaluationPeriods?: number\n  /**\n   * @default cdk.Duration.minutes(1)\n   */\n  period?: cdk.Duration\n  /**\n   * @default false\n   */\n  enableOkAlarm?: boolean\n}\n\ntype ErrorsAlarm = {\n  /**\n   * An action to use for CloudWatch alarm state changes instead of the default action\n   */\n  action?: cloudwatch.IAlarmAction\n  /**\n   * The threshold defined as the number of JavaScript errors recorded by CloudWatch RUM that determines if the alarm should trigger or not.\n   * @default 0\n   */\n  threshold?: number\n  /**\n   * @default 1\n   */\n  evaluationPeriods?: number\n  /**\n   * @default cdk.Duration.minutes(1)\n   */\n  period?: cdk.Duration\n  /**\n   * @default false\n   */\n  enableOkAlarms: boolean\n}\n\ninterface WebappMonitorProps {\n  /**\n   * The name of the CloudWatch RUM App Monitor.\n   *\n   * NOTE: This name needs to be unique within a given AWS region.\n   */\n  appMonitorName: string\n  /**\n   * The domain the web application to monitor is served on.\n   *\n   * Only web applications served on this domain can publish data to CloudWatch RUM.\n   */\n  webappDomainName: string\n  /**\n   * Which telemetries to allow CloudWatch RUM to collect.\n   *\n   * Currently these telemetries are supported:\n   * - \"http\" report the http fetch calles made by browser to CW RUM\n   * - \"errors\" report the js errors which occur, passed via rumConfig.recordError\n   * - \"performance\" report the app performance\n   *\n   * @default: [\"errors\", \"http\", \"performance\"]\n   */\n  telemetries?: CwRumTelemetries\n  /**\n   * The default CloudWatch alarm action to use.\n   *\n   * By default, an alarm is created for a high number of events being sent to CloudWatch RUM. This alarm action is used for the default alarm,\n   * and for other alarms if no other action is explicitly set when configuring or enabling alarms through this construct.\n   *\n   */\n  defaultAlarmAction: cloudwatch.IAlarmAction\n  /**\n   * Allow CloudWatch RUM to record custom events.\n   *\n   * These events need to be defined in and sent from your web application.\n   *\n   * @default false\n   */\n  enableXRay?: boolean\n  /**\n   * Allow CloudWatch RUM to use cookies.\n   *\n   * CloudWatch RUM does not require cookies in order to record errors, but some features require cookies to be enabled.\n   * @default false\n   */\n  enableCustomEvents?: boolean\n  /**\n   * Allow CloudWatch RUM to use cookies.\n   *\n   * CloudWatch RUM does not require cookies in order to record errors, but some features require cookies to be enabled.\n   *\n   * See: https://aws.amazon.com/blogs/mt/how-and-when-to-enable-session-cookies-with-amazon-cloudwatch-rum/\n   *\n   * NOTE: You'll likely need to introduce a cookie consent popup in your web application\n   * before enabling cookies in your client-side configuration.\n   *\n   * @default true\n   */\n  allowCookies?: boolean\n  /**\n   * The rate of client-side CloudWatch RUM sessions to sample.\n   *\n   * Values are between 0 and 1, where 1 is 100% of sessions.\n   *\n   * NOTE: You'll likely want to set this to a number below 1 in production environments with\n   * a lot of traffic.\n   *\n   * @default 1\n   */\n  sessionSampleRate?: number\n  /**\n   *\n   * Whether to store a copy of CloudWatch RUM events in CloudWatch Logs.\n   *\n   * Enabling this can improve the ergonomics of finding relevant CloudWatch RUM events compared to\n   * using the CloudWatch RUM console as it makes it possible to search using CloudWatch Logs Insights etc.\n   *\n   * @default false\n   */\n  exportLogs?: boolean\n  /**\n   * Override top-level properties for the rum.CfnAppMonitor construct\n   *\n   * @default - no overrides are used.\n   */\n  overrideAppMonitorProps?: Partial<rum.CfnAppMonitorProps>\n  /**\n   * Override App Monitor configuration properties for the rum.CfnAppMonitor construct\n   *\n   * @default - no overrides are used.\n   */\n  overrideAppMonitorConfiguration?: Partial<rum.CfnAppMonitor.AppMonitorConfigurationProperty>\n}\n\n/**\n *  A construct for creating a CloudWatch Real User Monitoring (RUM) App Monitor.\n *\n * An app monitor allows you to centrally collect and view client-side data (JavaScript and HTTP errors, performance metrics, etc.)\n * recorded from a web application.\n *\n * NOTE: You'll need to add a JavaScript snippet to your web application and configure CloudWatch RUM using\n * AWS's client-side SDK after deploying the construct to enable your web application to start collecting data.\n * The App Monitor controls which CloudWatch RUM functionality the web application can use, and in order to\n * leverage given functionality you'll need to configure it both in the App Monitor as well as in the\n * configuration of the client-side SDK used by your web application.\n *\n */\nexport class WebappMonitor extends constructs.Construct {\n  readonly identityPool: cognito.CfnIdentityPool\n  readonly guestRole: iam.IRole\n  readonly appMonitor: rum.CfnAppMonitor\n  private readonly _webappDomainName: string\n\n  private readonly _defaultAlarmAction: cloudwatch.IAlarmAction\n\n  private _eventsAlarm: cloudwatch.IAlarm\n  private _eventsAlarmId = \"EventsAlarm\"\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: WebappMonitorProps,\n  ) {\n    super(scope, id)\n\n    if (\n      props.sessionSampleRate &&\n      (props.sessionSampleRate < 0 || props.sessionSampleRate > 1)\n    ) {\n      throw new Error(\"sessionSampleRate needs to be a number between 0 and 1\")\n    }\n\n    this._webappDomainName = props.webappDomainName\n\n    const appMonitorName = props.appMonitorName\n\n    this.identityPool = new cognito.CfnIdentityPool(this, \"RumIdentityPool\", {\n      allowUnauthenticatedIdentities: true,\n    })\n\n    this.guestRole = new iam.Role(this, \"RumRole\", {\n      assumedBy: new iam.WebIdentityPrincipal(\n        \"cognito-identity.amazonaws.com\",\n        {\n          StringEquals: {\n            \"cognito-identity.amazonaws.com:aud\": this.identityPool.ref,\n          },\n          \"ForAnyValue:StringEquals\": {\n            \"cognito-identity.amazonaws.com:amr\": \"unauthenticated\",\n          },\n        },\n      ),\n      inlinePolicies: {\n        \"allow-rum\": new iam.PolicyDocument({\n          statements: [\n            new iam.PolicyStatement({\n              actions: [\"rum:PutRumEvents\"],\n              resources: [\n                cdk.Stack.of(this).formatArn({\n                  service: \"rum\",\n                  resource: \"appmonitor\",\n                  resourceName: appMonitorName,\n                }),\n              ],\n            }),\n          ],\n        }),\n      },\n    })\n\n    // attach role to identity pool\n    new cognito.CfnIdentityPoolRoleAttachment(this, \"RoleAttachment\", {\n      identityPoolId: this.identityPool.ref,\n      roles: {\n        unauthenticated: this.guestRole.roleArn,\n      },\n    })\n\n    this.appMonitor = new rum.CfnAppMonitor(this, \"AppMonitor\", {\n      name: appMonitorName,\n      domain: props.webappDomainName,\n      cwLogEnabled: props.exportLogs ?? false,\n      appMonitorConfiguration: {\n        enableXRay: props.enableXRay ?? false,\n        allowCookies: props.allowCookies ?? true,\n        telemetries: props.telemetries ?? [\"errors\", \"http\", \"performance\"],\n        sessionSampleRate: props.sessionSampleRate ?? 1,\n        identityPoolId: this.identityPool.ref,\n        guestRoleArn: this.guestRole.roleArn,\n        ...props.overrideAppMonitorConfiguration,\n      },\n      customEvents: {\n        status: props.enableCustomEvents ? \"ENABLED\" : \"DISABLED\",\n      },\n      ...props.overrideAppMonitorProps,\n    })\n\n    this._defaultAlarmAction = props.defaultAlarmAction\n    this._eventsAlarm = this._createEventsAlarm()\n  }\n\n  /**\n   *\n   * Configure an alarm used to notify on an unexpectedly high number of CloudWatch RUM events being recorded.\n   *\n   * This alarm is required when using this construct.\n   *\n   * @default - an alarm is created with sane defaults.\n   *\n   */\n  configureEventsAlarm(props?: EventsAlarm) {\n    this.node.tryRemoveChild(this._eventsAlarmId)\n    this._eventsAlarm = this._createEventsAlarm(props)\n  }\n\n  private _createEventsAlarm(props?: EventsAlarm): cloudwatch.IAlarm {\n    const eventsAlarm = new cloudwatch.Metric({\n      metricName: \"RumEventPayloadSize\",\n      namespace: \"AWS/RUM\",\n      statistic: cloudwatch.Stats.SAMPLE_COUNT,\n      period: props?.period ?? cdk.Duration.seconds(60),\n      dimensionsMap: {\n        application_name: this.appMonitor.name,\n      },\n    }).createAlarm(this, this._eventsAlarmId, {\n      alarmDescription: `A high number of events are being sent to CloudWatch RUM for app monitor '${this.appMonitor.name}'. This may lead to high costs, so you'll likely want to investigate this.`,\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: props?.threshold ?? 1000,\n      evaluationPeriods: props?.evaluationPeriods ?? 1,\n      treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n    })\n\n    // These alarms are mandatory as costs should be mandator to monitor\n    eventsAlarm.addAlarmAction(props?.action ?? this._defaultAlarmAction)\n    if (props?.enableOkAlarm ?? false) {\n      eventsAlarm.addOkAction(props?.action ?? this._defaultAlarmAction)\n    }\n\n    return eventsAlarm\n  }\n\n  /**\n   * Add an alarm that alerts on JavaScript errors\n   * sent to CloudWatch RUM.\n   */\n  addErrorsAlarm(props?: ErrorsAlarm) {\n    const errorsAlarm = new cloudwatch.Metric({\n      metricName: \"JsErrorCount\",\n      namespace: \"AWS/RUM\",\n      statistic: cloudwatch.Stats.SAMPLE_COUNT,\n      period: props?.period ?? cdk.Duration.seconds(60),\n      dimensionsMap: {\n        application_name: this.appMonitor.name,\n      },\n    }).createAlarm(this, \"RumJsErrorsAlarm\", {\n      alarmDescription: `CloudWatch RUM has recorded JavaScript errors for app monitor '${this.appMonitor.name}' on domain '${this._webappDomainName}'`,\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: props?.threshold ?? 0,\n      evaluationPeriods: props?.evaluationPeriods ?? 1,\n      treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n    })\n\n    errorsAlarm.addAlarmAction(props?.action ?? this._defaultAlarmAction)\n\n    if (props?.enableOkAlarms ?? false) {\n      errorsAlarm.addOkAction(props?.action ?? this._defaultAlarmAction)\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as cloudfront from "aws-cdk-lib/aws-cloudfront";
|
|
3
|
+
export type WebappSecurityHeadersProps = Partial<cloudfront.ResponseSecurityHeadersBehavior & {
|
|
4
|
+
contentSecurityPolicy?: cloudfront.ResponseSecurityHeadersBehavior["contentSecurityPolicy"] & {
|
|
5
|
+
/**
|
|
6
|
+
* Whether to only monitor the effects of the content security policy without actually blocking anything.
|
|
7
|
+
* @default false
|
|
8
|
+
*/
|
|
9
|
+
reportOnly?: boolean;
|
|
10
|
+
};
|
|
11
|
+
}>;
|
|
12
|
+
export interface ContentSecurityPolicyHeader {
|
|
13
|
+
baseUri?: string;
|
|
14
|
+
childSrc?: string;
|
|
15
|
+
defaultSrc?: string;
|
|
16
|
+
fontSrc?: string;
|
|
17
|
+
frameSrc?: string;
|
|
18
|
+
formAction?: string;
|
|
19
|
+
frameAncestors?: string;
|
|
20
|
+
imgSrc?: string;
|
|
21
|
+
manifestSrc?: string;
|
|
22
|
+
mediaSrc?: string;
|
|
23
|
+
objectSrc?: string;
|
|
24
|
+
scriptSrc?: string;
|
|
25
|
+
styleSrc?: string;
|
|
26
|
+
connectSrc?: string;
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Helper function that generates a string containing a Content Security Policy that can be
|
|
30
|
+
* used in a security header.
|
|
31
|
+
*
|
|
32
|
+
* NOTE: The string can be further extended using string concatenation for directives that aren't currently supported by the function.
|
|
33
|
+
*/
|
|
34
|
+
export declare function generateContentSecurityPolicyHeader(headerOptions?: ContentSecurityPolicyHeader): string;
|
|
35
|
+
export declare class WebappSecurityHeaders extends constructs.Construct {
|
|
36
|
+
readonly responseHeadersPolicy: cloudfront.ResponseHeadersPolicy;
|
|
37
|
+
constructor(scope: constructs.Construct, id: string, props: WebappSecurityHeadersProps);
|
|
38
|
+
}
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.WebappSecurityHeaders = exports.generateContentSecurityPolicyHeader = void 0;
|
|
4
|
+
const cdk = require("aws-cdk-lib");
|
|
5
|
+
const constructs = require("constructs");
|
|
6
|
+
const cloudfront = require("aws-cdk-lib/aws-cloudfront");
|
|
7
|
+
function validateCspParam(param) {
|
|
8
|
+
if (param.indexOf('"') !== -1) {
|
|
9
|
+
throw Error('CSP override contains invalid character "');
|
|
10
|
+
}
|
|
11
|
+
if (param.indexOf(";") !== -1) {
|
|
12
|
+
throw Error("CSP override contains invalid character ;");
|
|
13
|
+
}
|
|
14
|
+
if (param.indexOf("\\") !== -1) {
|
|
15
|
+
throw Error("CSP override contains invalid character \\");
|
|
16
|
+
}
|
|
17
|
+
return param;
|
|
18
|
+
}
|
|
19
|
+
/* Replace all whitespace in a string with a single space */
|
|
20
|
+
function trim(value) {
|
|
21
|
+
return value.replace(/\s+/g, " ").trim();
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Helper function that generates a string containing a Content Security Policy that can be
|
|
25
|
+
* used in a security header.
|
|
26
|
+
*
|
|
27
|
+
* NOTE: The string can be further extended using string concatenation for directives that aren't currently supported by the function.
|
|
28
|
+
*/
|
|
29
|
+
function generateContentSecurityPolicyHeader(headerOptions) {
|
|
30
|
+
const defaultValues = {
|
|
31
|
+
baseUri: "'self'",
|
|
32
|
+
childSrc: "'self'",
|
|
33
|
+
connectSrc: "'self' https:",
|
|
34
|
+
defaultSrc: "'none'",
|
|
35
|
+
fontSrc: "'self'",
|
|
36
|
+
formAction: "'none'",
|
|
37
|
+
frameAncestors: "'none'",
|
|
38
|
+
frameSrc: "'self'",
|
|
39
|
+
imgSrc: "'self' data:",
|
|
40
|
+
manifestSrc: "'self'",
|
|
41
|
+
mediaSrc: "'self'",
|
|
42
|
+
objectSrc: "'none'",
|
|
43
|
+
scriptSrc: "'self'",
|
|
44
|
+
styleSrc: "'self'",
|
|
45
|
+
};
|
|
46
|
+
const options = {
|
|
47
|
+
...defaultValues,
|
|
48
|
+
...headerOptions,
|
|
49
|
+
};
|
|
50
|
+
Object.values(options).forEach((v) => typeof v === "string" && validateCspParam(v));
|
|
51
|
+
let headerValue = "";
|
|
52
|
+
headerValue += `base-uri ${trim(options.baseUri)};`;
|
|
53
|
+
headerValue += `child-src ${trim(options.childSrc)};`;
|
|
54
|
+
headerValue += `connect-src ${trim(options.connectSrc)};`;
|
|
55
|
+
headerValue += `default-src ${trim(options.defaultSrc)};`;
|
|
56
|
+
headerValue += `font-src ${trim(options.fontSrc)};`;
|
|
57
|
+
headerValue += `frame-src ${trim(options.frameSrc)};`;
|
|
58
|
+
headerValue += `form-action ${trim(options.formAction)};`;
|
|
59
|
+
headerValue += `frame-ancestors ${trim(options.frameAncestors)};`;
|
|
60
|
+
headerValue += `img-src ${trim(options.imgSrc)};`;
|
|
61
|
+
headerValue += `manifest-src ${trim(options.manifestSrc)};`;
|
|
62
|
+
headerValue += `media-src ${trim(options.mediaSrc)};`;
|
|
63
|
+
headerValue += `object-src ${trim(options.objectSrc)};`;
|
|
64
|
+
headerValue += `script-src ${trim(options.scriptSrc)};`;
|
|
65
|
+
headerValue += `style-src ${trim(options.styleSrc)}`;
|
|
66
|
+
return trim(headerValue);
|
|
67
|
+
}
|
|
68
|
+
exports.generateContentSecurityPolicyHeader = generateContentSecurityPolicyHeader;
|
|
69
|
+
class WebappSecurityHeaders extends constructs.Construct {
|
|
70
|
+
constructor(scope, id, props) {
|
|
71
|
+
super(scope, id);
|
|
72
|
+
const { contentSecurityPolicy: contentSecurityPolicyOverride, ...overrides } = props;
|
|
73
|
+
const contentSecurityPolicyCustomHeader = contentSecurityPolicyOverride || {
|
|
74
|
+
reportOnly: false,
|
|
75
|
+
contentSecurityPolicy: generateContentSecurityPolicyHeader(),
|
|
76
|
+
override: true,
|
|
77
|
+
};
|
|
78
|
+
const defaultValues = {
|
|
79
|
+
contentTypeOptions: {
|
|
80
|
+
override: true,
|
|
81
|
+
},
|
|
82
|
+
referrerPolicy: {
|
|
83
|
+
override: true,
|
|
84
|
+
referrerPolicy: cloudfront.HeadersReferrerPolicy.SAME_ORIGIN,
|
|
85
|
+
},
|
|
86
|
+
frameOptions: {
|
|
87
|
+
frameOption: cloudfront.HeadersFrameOption.DENY,
|
|
88
|
+
override: true,
|
|
89
|
+
},
|
|
90
|
+
strictTransportSecurity: {
|
|
91
|
+
override: true,
|
|
92
|
+
accessControlMaxAge: cdk.Duration.days(182.5),
|
|
93
|
+
includeSubdomains: false,
|
|
94
|
+
preload: false,
|
|
95
|
+
},
|
|
96
|
+
xssProtection: {
|
|
97
|
+
override: true,
|
|
98
|
+
protection: true,
|
|
99
|
+
modeBlock: true,
|
|
100
|
+
},
|
|
101
|
+
};
|
|
102
|
+
this.responseHeadersPolicy = new cloudfront.ResponseHeadersPolicy(this, "ResponseHeadersPolicy", {
|
|
103
|
+
securityHeadersBehavior: {
|
|
104
|
+
...defaultValues,
|
|
105
|
+
...overrides,
|
|
106
|
+
...(!contentSecurityPolicyCustomHeader.reportOnly && {
|
|
107
|
+
contentSecurityPolicy: {
|
|
108
|
+
contentSecurityPolicy: contentSecurityPolicyCustomHeader.contentSecurityPolicy,
|
|
109
|
+
override: contentSecurityPolicyCustomHeader.override,
|
|
110
|
+
},
|
|
111
|
+
}),
|
|
112
|
+
},
|
|
113
|
+
...(contentSecurityPolicyCustomHeader.reportOnly && {
|
|
114
|
+
customHeadersBehavior: {
|
|
115
|
+
// Report only is not supported by securityHeadersBehavior in AWS and must be defined as custom header
|
|
116
|
+
customHeaders: [
|
|
117
|
+
{
|
|
118
|
+
header: "Content-Security-Policy-Report-Only",
|
|
119
|
+
value: contentSecurityPolicyCustomHeader.contentSecurityPolicy,
|
|
120
|
+
override: contentSecurityPolicyCustomHeader.override,
|
|
121
|
+
},
|
|
122
|
+
],
|
|
123
|
+
},
|
|
124
|
+
}),
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
exports.WebappSecurityHeaders = WebappSecurityHeaders;
|
|
129
|
+
//# 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 cdk from \"aws-cdk-lib\"\nimport * as constructs from \"constructs\"\nimport * as cloudfront from \"aws-cdk-lib/aws-cloudfront\"\n\nexport type WebappSecurityHeadersProps = Partial<\n  cloudfront.ResponseSecurityHeadersBehavior & {\n    contentSecurityPolicy?: cloudfront.ResponseSecurityHeadersBehavior[\"contentSecurityPolicy\"] & {\n      /**\n       * Whether to only monitor the effects of the content security policy without actually blocking anything.\n       * @default false\n       */\n      reportOnly?: boolean\n    }\n  }\n>\n\nexport interface ContentSecurityPolicyHeader {\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\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\n/**\n * Helper function that generates a string containing a Content Security Policy that can be\n * used in a security header.\n *\n * NOTE: The string can be further extended using string concatenation for directives that aren't currently supported by the function.\n */\nexport function generateContentSecurityPolicyHeader(\n  headerOptions?: ContentSecurityPolicyHeader,\n) {\n  const defaultValues = {\n    baseUri: \"'self'\",\n    childSrc: \"'self'\",\n    connectSrc: \"'self' https:\",\n    defaultSrc: \"'none'\",\n    fontSrc: \"'self'\",\n    formAction: \"'none'\",\n    frameAncestors: \"'none'\",\n    frameSrc: \"'self'\",\n    imgSrc: \"'self' data:\",\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 += `form-action ${trim(options.formAction)};`\n  headerValue += `frame-ancestors ${trim(options.frameAncestors)};`\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}\nexport class WebappSecurityHeaders extends constructs.Construct {\n  public readonly responseHeadersPolicy: cloudfront.ResponseHeadersPolicy\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: WebappSecurityHeadersProps,\n  ) {\n    super(scope, id)\n\n    const {\n      contentSecurityPolicy: contentSecurityPolicyOverride,\n      ...overrides\n    } = props\n\n    const contentSecurityPolicyCustomHeader = contentSecurityPolicyOverride || {\n      reportOnly: false,\n      contentSecurityPolicy: generateContentSecurityPolicyHeader(),\n      override: true,\n    }\n\n    const defaultValues: Omit<\n      cloudfront.ResponseSecurityHeadersBehavior,\n      \"contentSecurityPolicy\"\n    > = {\n      contentTypeOptions: {\n        override: true,\n      },\n      referrerPolicy: {\n        override: true,\n        referrerPolicy: cloudfront.HeadersReferrerPolicy.SAME_ORIGIN,\n      },\n      frameOptions: {\n        frameOption: cloudfront.HeadersFrameOption.DENY,\n        override: true,\n      },\n      strictTransportSecurity: {\n        override: true,\n        accessControlMaxAge: cdk.Duration.days(182.5),\n        includeSubdomains: false,\n        preload: false,\n      },\n      xssProtection: {\n        override: true,\n        protection: true,\n        modeBlock: true,\n      },\n    }\n\n    this.responseHeadersPolicy = new cloudfront.ResponseHeadersPolicy(\n      this,\n      \"ResponseHeadersPolicy\",\n      {\n        securityHeadersBehavior: {\n          ...defaultValues,\n          ...overrides,\n          ...(!contentSecurityPolicyCustomHeader.reportOnly && {\n            contentSecurityPolicy: {\n              contentSecurityPolicy:\n                contentSecurityPolicyCustomHeader.contentSecurityPolicy,\n              override: contentSecurityPolicyCustomHeader.override,\n            },\n          }),\n        },\n        ...(contentSecurityPolicyCustomHeader.reportOnly && {\n          customHeadersBehavior: {\n            // Report only is not supported by securityHeadersBehavior in AWS and must be defined as custom header\n            customHeaders: [\n              {\n                header: \"Content-Security-Policy-Report-Only\",\n                value: contentSecurityPolicyCustomHeader.contentSecurityPolicy,\n                override: contentSecurityPolicyCustomHeader.override,\n              },\n            ],\n          },\n        }),\n      },\n    )\n  }\n}\n"]}
|