@liflig/cdk 2.9.2 → 2.10.1

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.
@@ -1 +1,2 @@
1
1
  export { Webapp, WebappProps } from "./webapp";
2
+ export { generateContentSecurityPolicyHeader } from "./security-headers";
@@ -1,6 +1,8 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.Webapp = void 0;
3
+ exports.generateContentSecurityPolicyHeader = exports.Webapp = void 0;
4
4
  var webapp_1 = require("./webapp");
5
5
  Object.defineProperty(exports, "Webapp", { enumerable: true, get: function () { return webapp_1.Webapp; } });
6
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvd2ViYXBwL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLG1DQUE4QztBQUFyQyxnR0FBQSxNQUFNLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBXZWJhcHAsIFdlYmFwcFByb3BzIH0gZnJvbSBcIi4vd2ViYXBwXCJcbiJdfQ==
6
+ var security_headers_1 = require("./security-headers");
7
+ Object.defineProperty(exports, "generateContentSecurityPolicyHeader", { enumerable: true, get: function () { return security_headers_1.generateContentSecurityPolicyHeader; } });
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvd2ViYXBwL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLG1DQUE4QztBQUFyQyxnR0FBQSxNQUFNLE9BQUE7QUFDZix1REFBd0U7QUFBL0QsdUlBQUEsbUNBQW1DLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBXZWJhcHAsIFdlYmFwcFByb3BzIH0gZnJvbSBcIi4vd2ViYXBwXCJcbmV4cG9ydCB7IGdlbmVyYXRlQ29udGVudFNlY3VyaXR5UG9saWN5SGVhZGVyIH0gZnJvbSBcIi4vc2VjdXJpdHktaGVhZGVyc1wiXG4iXX0=
@@ -1,18 +1,15 @@
1
1
  import * as constructs from "constructs";
2
2
  import * as cloudfront from "aws-cdk-lib/aws-cloudfront";
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
- }
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
+ }>;
14
12
  export interface ContentSecurityPolicyHeader {
15
- reportOnly?: boolean;
16
13
  baseUri?: string;
17
14
  childSrc?: string;
18
15
  defaultSrc?: string;
@@ -28,13 +25,14 @@ export interface ContentSecurityPolicyHeader {
28
25
  styleSrc?: string;
29
26
  connectSrc?: string;
30
27
  }
31
- export interface SecurityHeaders {
32
- contentSecurityPolicy?: ContentSecurityPolicyHeader;
33
- strictTransportSecurity?: StrictTransportSecurityHeader;
34
- referrerPolicy?: ReferrerPolicyHeader;
35
- frameOptions?: FrameOptionsHeader;
36
- }
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;
37
35
  export declare class WebappSecurityHeaders extends constructs.Construct {
38
- readonly securityHeadersFunction: cloudfront.Function;
39
- constructor(scope: constructs.Construct, id: string, props: SecurityHeaders);
36
+ readonly responseHeadersPolicy: cloudfront.ResponseHeadersPolicy;
37
+ constructor(scope: constructs.Construct, id: string, props: WebappSecurityHeadersProps);
40
38
  }
@@ -1,6 +1,7 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.WebappSecurityHeaders = void 0;
3
+ exports.WebappSecurityHeaders = exports.generateContentSecurityPolicyHeader = void 0;
4
+ const cdk = require("aws-cdk-lib");
4
5
  const constructs = require("constructs");
5
6
  const cloudfront = require("aws-cdk-lib/aws-cloudfront");
6
7
  function validateCspParam(param) {
@@ -19,17 +20,23 @@ function validateCspParam(param) {
19
20
  function trim(value) {
20
21
  return value.replace(/\s+/g, " ").trim();
21
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
+ */
22
29
  function generateContentSecurityPolicyHeader(headerOptions) {
23
30
  const defaultValues = {
24
31
  baseUri: "'self'",
25
- childSrc: "'none'",
26
- connectSrc: "'self'",
27
- defaultSrc: "'self'",
32
+ childSrc: "'self'",
33
+ connectSrc: "'self' https:",
34
+ defaultSrc: "'none'",
28
35
  fontSrc: "'self'",
29
- formAction: "'self'",
36
+ formAction: "'none'",
30
37
  frameAncestors: "'none'",
31
38
  frameSrc: "'self'",
32
- imgSrc: "'self'",
39
+ imgSrc: "'self' data:",
33
40
  manifestSrc: "'self'",
34
41
  mediaSrc: "'self'",
35
42
  objectSrc: "'none'",
@@ -48,79 +55,75 @@ function generateContentSecurityPolicyHeader(headerOptions) {
48
55
  headerValue += `default-src ${trim(options.defaultSrc)};`;
49
56
  headerValue += `font-src ${trim(options.fontSrc)};`;
50
57
  headerValue += `frame-src ${trim(options.frameSrc)};`;
58
+ headerValue += `form-action ${trim(options.formAction)};`;
59
+ headerValue += `frame-ancestors ${trim(options.frameAncestors)};`;
51
60
  headerValue += `img-src ${trim(options.imgSrc)};`;
52
61
  headerValue += `manifest-src ${trim(options.manifestSrc)};`;
53
62
  headerValue += `media-src ${trim(options.mediaSrc)};`;
54
63
  headerValue += `object-src ${trim(options.objectSrc)};`;
55
64
  headerValue += `script-src ${trim(options.scriptSrc)};`;
56
- headerValue += `style-src ${trim(options.styleSrc)};`;
65
+ headerValue += `style-src ${trim(options.styleSrc)}`;
57
66
  return trim(headerValue);
58
67
  }
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
- }
68
+ exports.generateContentSecurityPolicyHeader = generateContentSecurityPolicyHeader;
95
69
  class WebappSecurityHeaders extends constructs.Construct {
96
70
  constructor(scope, id, props) {
97
- var _a;
98
71
  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),
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
+ }),
122
125
  });
123
126
  }
124
127
  }
125
128
  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 constructs from \"constructs\"\nimport * as cloudfront from \"aws-cdk-lib/aws-cloudfront\"\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 constructs.Construct {\n  public readonly securityHeadersFunction: cloudfront.Function\n\n  constructor(scope: constructs.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"]}
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"]}
@@ -5,7 +5,7 @@ import * as origins from "aws-cdk-lib/aws-cloudfront-origins";
5
5
  import * as r53 from "aws-cdk-lib/aws-route53";
6
6
  import * as s3 from "aws-cdk-lib/aws-s3";
7
7
  import * as webappDeploy from "@capraconsulting/webapp-deploy-lambda";
8
- import { SecurityHeaders } from "./security-headers";
8
+ import { WebappSecurityHeadersProps } from "./security-headers";
9
9
  export interface WebappProps {
10
10
  /**
11
11
  * ACM certificate that covers the specifeid domain names.
@@ -42,27 +42,44 @@ export interface WebappProps {
42
42
  */
43
43
  webAclErrorPagePath?: string;
44
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
45
+ * Enable, disable or configure security headers for the web application
46
+ * @default - a set of strict security headers are configured by default
56
47
  */
57
- enableSecurityHeaders?: boolean;
48
+ securityHeaders?: {
49
+ /**
50
+ * Enable adding common security headers to CloudFront responses
51
+ *
52
+ * If enabled, the default behavior is to add the following headers with fairly strict defaults. Most of the headers can be customized:
53
+ * - Content-Security-Policy
54
+ * - Referrer-Policy
55
+ * - Strict-Transport-Security
56
+ * - X-Content-Type-Options
57
+ * - X-Frame-Options
58
+ * - X-XSS-Protection
59
+ *
60
+ *
61
+ * @default true
62
+ */
63
+ enabled?: boolean;
64
+ /**
65
+ * Security headers overrides.
66
+ *
67
+ * Used to override certain default security header values if the webapp requires different settings than the defaults.
68
+ *
69
+ * NOTE: If you need to disable certain headers, you must explicitly set them to undefined
70
+ *
71
+ * @default - A set of strict security header values will be used
72
+ */
73
+ behaviorOverrides?: WebappSecurityHeadersProps;
74
+ };
58
75
  /**
59
- * Security headers overrides.
76
+ * Cloudfront behavior overrides.
60
77
  *
61
- * Used to override certain security header values if the webapp requires more lax settings compared to the defaults.
78
+ * Used to override cloudfront behavior
62
79
  *
63
- * @default - A set of strict security header values will be used
80
+ * NOTE: ResponseHeadersPolicy defined here will overwrite BOTH the default security headers policy and
81
+ * any values specified in securityHeaders.behaviorOverrides.
64
82
  */
65
- securityHeadersOverrides?: SecurityHeaders;
66
83
  overrideCloudFrontBehaviourOptions?: Partial<cloudfront.BehaviorOptions>;
67
84
  }
68
85
  /**
@@ -19,6 +19,7 @@ const security_headers_1 = require("./security-headers");
19
19
  */
20
20
  class Webapp extends constructs.Construct {
21
21
  constructor(scope, id, props) {
22
+ var _a, _b, _c;
22
23
  super(scope, id);
23
24
  if (props.webAclErrorPagePath != null && props.webAclId == null) {
24
25
  throw new Error("webAclErrorPagePath set but webAclId is missing");
@@ -62,23 +63,18 @@ class Webapp extends constructs.Construct {
62
63
  responsePagePath: props.webAclErrorPagePath,
63
64
  });
64
65
  }
65
- let functionAssociations;
66
- if (props.enableSecurityHeaders) {
66
+ let responseHeadersPolicy;
67
+ if ((_b = (_a = props.securityHeaders) === null || _a === void 0 ? void 0 : _a.enabled) !== null && _b !== void 0 ? _b : true) {
67
68
  const securityHeaders = new security_headers_1.WebappSecurityHeaders(this, "SecurityHeaders", {
68
- ...props.securityHeadersOverrides,
69
+ ...(_c = props.securityHeaders) === null || _c === void 0 ? void 0 : _c.behaviorOverrides,
69
70
  });
70
- functionAssociations = [
71
- {
72
- function: securityHeaders.securityHeadersFunction,
73
- eventType: cloudfront.FunctionEventType.VIEWER_RESPONSE,
74
- },
75
- ];
71
+ responseHeadersPolicy = securityHeaders.responseHeadersPolicy;
76
72
  }
77
73
  this.distribution = new cloudfront.Distribution(this, "Distribution", {
78
74
  defaultBehavior: {
79
75
  origin: this.webappOrigin,
80
76
  viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
81
- functionAssociations: functionAssociations,
77
+ responseHeadersPolicy: responseHeadersPolicy,
82
78
  ...props.overrideCloudFrontBehaviourOptions,
83
79
  },
84
80
  defaultRootObject: "index.html",
@@ -118,4 +114,4 @@ class Webapp extends constructs.Construct {
118
114
  }
119
115
  }
120
116
  exports.Webapp = Webapp;
121
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webapp.js","sourceRoot":"","sources":["../../src/webapp/webapp.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as certificatemanager from \"aws-cdk-lib/aws-certificatemanager\"\nimport * as cloudfront from \"aws-cdk-lib/aws-cloudfront\"\nimport * as origins from \"aws-cdk-lib/aws-cloudfront-origins\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as r53 from \"aws-cdk-lib/aws-route53\"\nimport * as r53t from \"aws-cdk-lib/aws-route53-targets\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as webappDeploy from \"@capraconsulting/webapp-deploy-lambda\"\nimport { WebappSecurityHeaders, SecurityHeaders } from \"./security-headers\"\n\nexport interface WebappProps {\n  /**\n   * ACM certificate that covers the specifeid domain names.\n   *\n   * This certificate must be created in the region us-east-1.\n   *\n   * @default - The CloudFront wildcard certificate (*.cloudfront.net) will be used.\n   */\n  cloudfrontCertificate?: certificatemanager.ICertificate\n  /**\n   * List of domain names the CloudFront distribution should use.\n   *\n   * @default - Generated name (e.g., d111111abcdef8.cloudfront.net)\n   */\n  domainNames?: string[]\n  /**\n   * AWS WAF web ACL to associate with the CloudFront distribution.\n   *\n   * To specify a web ACL created using the latest version of AWS WAF, use the ACL ARN, for example\n   * `arn:aws:wafv2:us-east-1:123456789012:global/webacl/ExampleWebACL/473e64fd-f30b-4765-81a0-62ad96dd167a`.\n   * To specify a web ACL created using AWS WAF Classic, use the ACL ID, for example `473e64fd-f30b-4765-81a0-62ad96dd167a`.\n   *\n   * @default - No AWS Web Application Firewall web access control list (web ACL).\n   */\n  webAclId?: string\n  /**\n   * The path to the page that will be served for users not allowed to access\n   * the site when using WAF. E.g. \"/4xx-errors/403-forbidden.html\".\n   *\n   * Note that this wil catch any 403 errors from the origin(s), that might\n   * cover any other behaviors is added.\n   *\n   * @default - No custom page for 403 errors.\n   */\n  webAclErrorPagePath?: string\n  /**\n   * Enable adding common security headers to CloudFront responses using a CloudFront Function.\n   *\n   * If enabled, the default behavior is to add the following headers with fairly strict defaults. Most of the headers can be customized:\n   * - Content-Security-Policy\n   * - Referrer-Policy\n   * - Strict-Transport-Security\n   * - X-Content-Type-Options\n   * - X-Frame-Options\n   * - X-XSS-Protection\n   *\n   * @default - No security headers will be added to responses\n   */\n  enableSecurityHeaders?: boolean\n  /**\n   * Security headers overrides.\n   *\n   * Used to override certain security header values if the webapp requires more lax settings compared to the defaults.\n   *\n   * @default - A set of strict security header values will be used\n   */\n  securityHeadersOverrides?: SecurityHeaders\n  overrideCloudFrontBehaviourOptions?: Partial<cloudfront.BehaviorOptions>\n}\n\n/**\n * CloudFront for a Single-Page-Application.\n *\n * A bucket will be created and its prefix \"web\" is used to\n * serve files. Use the addDeployment method to automatically\n * deploy files as part the the CDK deployment.\n */\nexport class Webapp extends constructs.Construct {\n  public readonly distribution: cloudfront.Distribution\n  public readonly webappBucket: s3.Bucket\n  public readonly webappOrigin: origins.S3Origin\n\n  constructor(scope: constructs.Construct, id: string, props: WebappProps) {\n    super(scope, id)\n\n    if (props.webAclErrorPagePath != null && props.webAclId == null) {\n      throw new Error(\"webAclErrorPagePath set but webAclId is missing\")\n    }\n\n    this.webappBucket = new s3.Bucket(this, \"Bucket\", {\n      encryption: s3.BucketEncryption.S3_MANAGED,\n    })\n\n    const originAccessIdentity = new cloudfront.OriginAccessIdentity(\n      this,\n      \"OriginAccessIdentity\",\n    )\n\n    this.webappBucket.addToResourcePolicy(\n      new iam.PolicyStatement({\n        resources: [this.webappBucket.arnForObjects(\"*\")],\n        actions: [\"s3:GetObject\"],\n        principals: [originAccessIdentity.grantPrincipal],\n      }),\n    )\n\n    this.webappBucket.addToResourcePolicy(\n      new iam.PolicyStatement({\n        resources: [this.webappBucket.bucketArn],\n        // Grant s3:ListBucket so that CloudFront receives 404 from\n        // the origin rather than 403 when accessing files that\n        // does not exist. We cannot fallback to index.html for 403\n        // errors since it would also be served if using a WAF.\n        // See https://aws.amazon.com/premiumsupport/knowledge-center/s3-website-cloudfront-error-403/#The_requested_objects_must_exist_in_the_bucket\n        actions: [\"s3:ListBucket\"],\n        principals: [originAccessIdentity.grantPrincipal],\n      }),\n    )\n\n    this.webappOrigin = new origins.S3Origin(this.webappBucket, {\n      // webapp-deploy-lambda will upload files to this folder\n      // since it keeps some other administrative files outside.\n      originPath: \"/web\",\n      originAccessIdentity,\n    })\n\n    const errorResponses: cloudfront.ErrorResponse[] = [\n      {\n        httpStatus: 404,\n        responseHttpStatus: 200,\n        responsePagePath: \"/index.html\",\n      },\n    ]\n\n    if (props.webAclErrorPagePath != null) {\n      errorResponses.push({\n        httpStatus: 403,\n        responseHttpStatus: 403,\n        responsePagePath: props.webAclErrorPagePath,\n      })\n    }\n\n    let functionAssociations: cloudfront.FunctionAssociation[] | undefined\n    if (props.enableSecurityHeaders) {\n      const securityHeaders = new WebappSecurityHeaders(\n        this,\n        \"SecurityHeaders\",\n        {\n          ...props.securityHeadersOverrides,\n        },\n      )\n      functionAssociations = [\n        {\n          function: securityHeaders.securityHeadersFunction,\n          eventType: cloudfront.FunctionEventType.VIEWER_RESPONSE,\n        },\n      ]\n    }\n\n    this.distribution = new cloudfront.Distribution(this, \"Distribution\", {\n      defaultBehavior: {\n        origin: this.webappOrigin,\n        viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        functionAssociations: functionAssociations,\n        ...props.overrideCloudFrontBehaviourOptions,\n      },\n      defaultRootObject: \"index.html\",\n      priceClass: cloudfront.PriceClass.PRICE_CLASS_100,\n      certificate: props.cloudfrontCertificate,\n      domainNames: props.domainNames,\n      errorResponses,\n      webAclId: props.webAclId,\n    })\n  }\n\n  addDnsRecord(hostedZone: r53.IHostedZone, domainName: string): void {\n    new r53.ARecord(this, `DnsRecord${domainName}`, {\n      zone: hostedZone,\n      recordName: `${domainName}.`,\n      target: r53.RecordTarget.fromAlias(\n        new r53t.CloudFrontTarget(this.distribution),\n      ),\n    })\n  }\n\n  /**\n   * Add a deployment using webapp-deploy-lambda.\n   *\n   * See https://github.com/capraconsulting/webapp-deploy-lambda\n   * for details about how this works.\n   */\n  addDeployment(\n    /**\n     * The deployment source.\n     */\n    source: webappDeploy.ISource,\n    props?: {\n      /**\n       * Include source maps in the deployment.\n       *\n       * @default false\n       */\n      deploySourceMaps?: boolean\n    },\n  ): void {\n    const deploySourceMaps = props?.deploySourceMaps ?? false\n\n    new webappDeploy.WebappDeploy(this, \"Deploy\", {\n      source: source,\n      webBucket: this.webappBucket,\n      distribution: this.distribution,\n      excludePattern: deploySourceMaps ? undefined : \"\\\\.map$\",\n    })\n  }\n}\n"]}
117
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webapp.js","sourceRoot":"","sources":["../../src/webapp/webapp.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as certificatemanager from \"aws-cdk-lib/aws-certificatemanager\"\nimport * as cloudfront from \"aws-cdk-lib/aws-cloudfront\"\nimport * as origins from \"aws-cdk-lib/aws-cloudfront-origins\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as r53 from \"aws-cdk-lib/aws-route53\"\nimport * as r53t from \"aws-cdk-lib/aws-route53-targets\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as webappDeploy from \"@capraconsulting/webapp-deploy-lambda\"\nimport {\n  WebappSecurityHeaders,\n  WebappSecurityHeadersProps,\n} from \"./security-headers\"\n\nexport interface WebappProps {\n  /**\n   * ACM certificate that covers the specifeid domain names.\n   *\n   * This certificate must be created in the region us-east-1.\n   *\n   * @default - The CloudFront wildcard certificate (*.cloudfront.net) will be used.\n   */\n  cloudfrontCertificate?: certificatemanager.ICertificate\n  /**\n   * List of domain names the CloudFront distribution should use.\n   *\n   * @default - Generated name (e.g., d111111abcdef8.cloudfront.net)\n   */\n  domainNames?: string[]\n  /**\n   * AWS WAF web ACL to associate with the CloudFront distribution.\n   *\n   * To specify a web ACL created using the latest version of AWS WAF, use the ACL ARN, for example\n   * `arn:aws:wafv2:us-east-1:123456789012:global/webacl/ExampleWebACL/473e64fd-f30b-4765-81a0-62ad96dd167a`.\n   * To specify a web ACL created using AWS WAF Classic, use the ACL ID, for example `473e64fd-f30b-4765-81a0-62ad96dd167a`.\n   *\n   * @default - No AWS Web Application Firewall web access control list (web ACL).\n   */\n  webAclId?: string\n  /**\n   * The path to the page that will be served for users not allowed to access\n   * the site when using WAF. E.g. \"/4xx-errors/403-forbidden.html\".\n   *\n   * Note that this wil catch any 403 errors from the origin(s), that might\n   * cover any other behaviors is added.\n   *\n   * @default - No custom page for 403 errors.\n   */\n  webAclErrorPagePath?: string\n  /**\n   *  Enable, disable or configure security headers for the web application\n   * @default - a set of strict security headers are configured by default\n   */\n  securityHeaders?: {\n    /**\n     * Enable adding common security headers to CloudFront responses\n     *\n     * If enabled, the default behavior is to add the following headers with fairly strict defaults. Most of the headers can be customized:\n     * - Content-Security-Policy\n     * - Referrer-Policy\n     * - Strict-Transport-Security\n     * - X-Content-Type-Options\n     * - X-Frame-Options\n     * - X-XSS-Protection\n     *\n     *\n     * @default true\n     */\n    enabled?: boolean\n    /**\n     * Security headers overrides.\n     *\n     * Used to override certain default security header values if the webapp requires different settings than the defaults.\n     *\n     * NOTE: If you need to disable certain headers, you must explicitly set them to undefined\n     *\n     * @default - A set of strict security header values will be used\n     */\n    behaviorOverrides?: WebappSecurityHeadersProps\n  }\n  /**\n   * Cloudfront behavior overrides.\n   *\n   * Used to override cloudfront behavior\n   *\n   * NOTE: ResponseHeadersPolicy defined here will overwrite BOTH the default security headers policy and\n   * any values specified in securityHeaders.behaviorOverrides.\n   */\n  overrideCloudFrontBehaviourOptions?: Partial<cloudfront.BehaviorOptions>\n}\n\n/**\n * CloudFront for a Single-Page-Application.\n *\n * A bucket will be created and its prefix \"web\" is used to\n * serve files. Use the addDeployment method to automatically\n * deploy files as part the the CDK deployment.\n */\nexport class Webapp extends constructs.Construct {\n  public readonly distribution: cloudfront.Distribution\n  public readonly webappBucket: s3.Bucket\n  public readonly webappOrigin: origins.S3Origin\n\n  constructor(scope: constructs.Construct, id: string, props: WebappProps) {\n    super(scope, id)\n\n    if (props.webAclErrorPagePath != null && props.webAclId == null) {\n      throw new Error(\"webAclErrorPagePath set but webAclId is missing\")\n    }\n\n    this.webappBucket = new s3.Bucket(this, \"Bucket\", {\n      encryption: s3.BucketEncryption.S3_MANAGED,\n    })\n\n    const originAccessIdentity = new cloudfront.OriginAccessIdentity(\n      this,\n      \"OriginAccessIdentity\",\n    )\n\n    this.webappBucket.addToResourcePolicy(\n      new iam.PolicyStatement({\n        resources: [this.webappBucket.arnForObjects(\"*\")],\n        actions: [\"s3:GetObject\"],\n        principals: [originAccessIdentity.grantPrincipal],\n      }),\n    )\n\n    this.webappBucket.addToResourcePolicy(\n      new iam.PolicyStatement({\n        resources: [this.webappBucket.bucketArn],\n        // Grant s3:ListBucket so that CloudFront receives 404 from\n        // the origin rather than 403 when accessing files that\n        // does not exist. We cannot fallback to index.html for 403\n        // errors since it would also be served if using a WAF.\n        // See https://aws.amazon.com/premiumsupport/knowledge-center/s3-website-cloudfront-error-403/#The_requested_objects_must_exist_in_the_bucket\n        actions: [\"s3:ListBucket\"],\n        principals: [originAccessIdentity.grantPrincipal],\n      }),\n    )\n\n    this.webappOrigin = new origins.S3Origin(this.webappBucket, {\n      // webapp-deploy-lambda will upload files to this folder\n      // since it keeps some other administrative files outside.\n      originPath: \"/web\",\n      originAccessIdentity,\n    })\n\n    const errorResponses: cloudfront.ErrorResponse[] = [\n      {\n        httpStatus: 404,\n        responseHttpStatus: 200,\n        responsePagePath: \"/index.html\",\n      },\n    ]\n\n    if (props.webAclErrorPagePath != null) {\n      errorResponses.push({\n        httpStatus: 403,\n        responseHttpStatus: 403,\n        responsePagePath: props.webAclErrorPagePath,\n      })\n    }\n\n    let responseHeadersPolicy: cloudfront.IResponseHeadersPolicy | undefined\n\n    if (props.securityHeaders?.enabled ?? true) {\n      const securityHeaders = new WebappSecurityHeaders(\n        this,\n        \"SecurityHeaders\",\n        {\n          ...props.securityHeaders?.behaviorOverrides,\n        },\n      )\n      responseHeadersPolicy = securityHeaders.responseHeadersPolicy\n    }\n\n    this.distribution = new cloudfront.Distribution(this, \"Distribution\", {\n      defaultBehavior: {\n        origin: this.webappOrigin,\n        viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        responseHeadersPolicy: responseHeadersPolicy,\n        ...props.overrideCloudFrontBehaviourOptions,\n      },\n      defaultRootObject: \"index.html\",\n      priceClass: cloudfront.PriceClass.PRICE_CLASS_100,\n      certificate: props.cloudfrontCertificate,\n      domainNames: props.domainNames,\n      errorResponses,\n      webAclId: props.webAclId,\n    })\n  }\n\n  addDnsRecord(hostedZone: r53.IHostedZone, domainName: string): void {\n    new r53.ARecord(this, `DnsRecord${domainName}`, {\n      zone: hostedZone,\n      recordName: `${domainName}.`,\n      target: r53.RecordTarget.fromAlias(\n        new r53t.CloudFrontTarget(this.distribution),\n      ),\n    })\n  }\n\n  /**\n   * Add a deployment using webapp-deploy-lambda.\n   *\n   * See https://github.com/capraconsulting/webapp-deploy-lambda\n   * for details about how this works.\n   */\n  addDeployment(\n    /**\n     * The deployment source.\n     */\n    source: webappDeploy.ISource,\n    props?: {\n      /**\n       * Include source maps in the deployment.\n       *\n       * @default false\n       */\n      deploySourceMaps?: boolean\n    },\n  ): void {\n    const deploySourceMaps = props?.deploySourceMaps ?? false\n\n    new webappDeploy.WebappDeploy(this, \"Deploy\", {\n      source: source,\n      webBucket: this.webappBucket,\n      distribution: this.distribution,\n      excludePattern: deploySourceMaps ? undefined : \"\\\\.map$\",\n    })\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@liflig/cdk",
3
- "version": "2.9.2",
3
+ "version": "2.10.1",
4
4
  "description": "CDK library for Liflig",
5
5
  "repository": {
6
6
  "type": "git",
@@ -36,15 +36,15 @@
36
36
  "devDependencies": {
37
37
  "@commitlint/cli": "17.4.2",
38
38
  "@commitlint/config-conventional": "17.4.2",
39
- "@aws-cdk/assert": "2.61.1",
40
- "@types/aws-lambda": "8.10.109",
39
+ "@aws-cdk/assert": "2.62.0",
40
+ "@types/aws-lambda": "8.10.110",
41
41
  "@types/jest": "27.5.2",
42
42
  "@types/node": "18.11.18",
43
43
  "@typescript-eslint/eslint-plugin": "5.49.0",
44
44
  "@typescript-eslint/parser": "5.49.0",
45
- "aws-cdk": "2.61.1",
46
- "aws-cdk-lib": "2.61.1",
47
- "constructs": "10.1.229",
45
+ "aws-cdk": "2.62.0",
46
+ "aws-cdk-lib": "2.62.0",
47
+ "constructs": "10.1.232",
48
48
  "eslint": "8.32.0",
49
49
  "eslint-config-prettier": "8.6.0",
50
50
  "eslint-plugin-prettier": "4.2.1",
@@ -52,7 +52,7 @@
52
52
  "jest": "27.5.1",
53
53
  "jest-cdk-snapshot": "2.0.1",
54
54
  "prettier": "2.8.3",
55
- "semantic-release": "20.0.4",
55
+ "semantic-release": "20.1.0",
56
56
  "ts-jest": "27.1.5",
57
57
  "ts-node": "10.9.1",
58
58
  "typescript": "4.9.4"