@liflig/cdk 1.48.0 → 1.50.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -54,7 +54,6 @@ const startDeployHandler = async (event) => {
54
54
  "status",
55
55
  "taskDefinitionArn",
56
56
  ];
57
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
58
57
  const updatedSpec = {
59
58
  ...Object.fromEntries(Object.entries(prevTaskDefinition).filter(([key]) => !exclude.includes(key))),
60
59
  containerDefinitions: [
@@ -102,4 +101,4 @@ const startDeployHandler = async (event) => {
102
101
  return {};
103
102
  };
104
103
  exports.startDeployHandler = startDeployHandler;
105
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"start-deploy-handler.js","sourceRoot":"","sources":["../../src/ecs-update-image/start-deploy-handler.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-assignment */\n/* eslint-disable @typescript-eslint/no-unsafe-call */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-unsafe-return */\n/* eslint-disable @typescript-eslint/no-var-requires */\nimport type { Handler } from \"aws-lambda\"\nimport type * as _AWS from \"aws-sdk\"\n\ninterface ExpectedInput {\n  tag: string\n}\n\n// This function is inline-compiled for the lambda.\n// It must be self-contained.\nexport const startDeployHandler: Handler<Partial<ExpectedInput>> = async (\n  event,\n) => {\n  const AWS = require(\"aws-sdk\")\n  const ecs = new AWS.ECS() as _AWS.ECS\n  const sm = new AWS.SecretsManager() as _AWS.SecretsManager\n\n  function requireEnv(name: string): string {\n    const value = process.env[name]\n    if (value === undefined) {\n      throw new Error(`Missing ${name}`)\n    }\n    return value\n  }\n\n  async function getService(\n    clusterName: string,\n    serviceName: string,\n  ): Promise<AWS.ECS.Service> {\n    const services = await ecs\n      .describeServices({\n        cluster: clusterName,\n        services: [serviceName],\n      })\n      .promise()\n\n    if (services.services?.length !== 1) {\n      throw new Error(`Service not found: ${clusterName}/${serviceName}`)\n    }\n\n    return services.services[0]\n  }\n\n  async function getTaskDefinition(\n    taskDefinition: string,\n  ): Promise<AWS.ECS.TaskDefinition> {\n    return (\n      await ecs\n        .describeTaskDefinition({\n          taskDefinition: taskDefinition,\n        })\n        .promise()\n    ).taskDefinition!\n  }\n\n  async function updateServiceToImage(\n    clusterName: string,\n    serviceName: string,\n    image: string,\n  ) {\n    console.log(`Cluster name: ${clusterName}`)\n    console.log(`Service name: ${serviceName}`)\n\n    const service = await getService(clusterName, serviceName)\n    const prevTaskDefinition = await getTaskDefinition(service.taskDefinition!)\n\n    // Don't bother updating the service if the image is already the latest.\n    const prevImage = prevTaskDefinition.containerDefinitions![0].image!\n    if (prevImage === image) {\n      return\n    }\n\n    console.log(\n      `Updating image for service '${serviceName}' from '${prevImage}' to '${image}'`,\n    )\n\n    const exclude = [\n      \"registeredAt\",\n      \"registeredBy\",\n      \"compatibilities\",\n      \"requiresAttributes\",\n      \"revision\",\n      \"status\",\n      \"taskDefinitionArn\",\n    ]\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const updatedSpec: any = {\n      ...Object.fromEntries(\n        Object.entries(prevTaskDefinition).filter(\n          ([key]) => !exclude.includes(key),\n        ),\n      ),\n      containerDefinitions: [\n        {\n          ...prevTaskDefinition.containerDefinitions![0],\n          image,\n        },\n      ],\n    }\n\n    const updatedTaskDefinition = (\n      await ecs.registerTaskDefinition(updatedSpec).promise()\n    ).taskDefinition!\n\n    await ecs\n      .updateService({\n        cluster: clusterName,\n        service: serviceName,\n        taskDefinition: updatedTaskDefinition.taskDefinitionArn,\n      })\n      .promise()\n\n    console.log(\"Service is updated\")\n  }\n\n  const clusterName = requireEnv(\"CLUSTER_NAME\")\n  const serviceName = requireEnv(\"SERVICE_NAME\")\n  const repositoryUrl = requireEnv(\"REPOSITORY_URL\")\n  const ecrTagSecretArn = requireEnv(\"ECR_TAG_SECRET_ARN\")\n\n  // Validate the input.\n  if (typeof event.tag !== \"string\") {\n    throw new Error(\"Input invalid: \" + JSON.stringify(event, undefined, \"  \"))\n  }\n\n  // Register tag as current target.\n  // This is needed so that CloudFormation deployments, e.g.\n  // updates to the Task Definition, will use the same image.\n  await sm\n    .updateSecret({\n      SecretId: ecrTagSecretArn,\n      SecretString: JSON.stringify({\n        tag: event.tag,\n      }),\n    })\n    .promise()\n\n  // Update the service if we know the service name. This is unknown\n  // during initial deployment of the stack.\n  if (serviceName !== \"\") {\n    const image = `${repositoryUrl}:${event.tag}`\n    await updateServiceToImage(clusterName, serviceName, image)\n  }\n\n  return {}\n}\n"]}
104
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"start-deploy-handler.js","sourceRoot":"","sources":["../../src/ecs-update-image/start-deploy-handler.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-assignment */\n/* eslint-disable @typescript-eslint/no-unsafe-call */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-unsafe-return */\n/* eslint-disable @typescript-eslint/no-var-requires */\nimport type { Handler } from \"aws-lambda\"\nimport type * as _AWS from \"aws-sdk\"\nimport { RegisterTaskDefinitionRequest } from \"aws-sdk/clients/ecs\"\n\ninterface ExpectedInput {\n  tag: string\n}\n\n// This function is inline-compiled for the lambda.\n// It must be self-contained.\nexport const startDeployHandler: Handler<Partial<ExpectedInput>> = async (\n  event,\n) => {\n  const AWS = require(\"aws-sdk\")\n  const ecs = new AWS.ECS() as _AWS.ECS\n  const sm = new AWS.SecretsManager() as _AWS.SecretsManager\n\n  function requireEnv(name: string): string {\n    const value = process.env[name]\n    if (value === undefined) {\n      throw new Error(`Missing ${name}`)\n    }\n    return value\n  }\n\n  async function getService(\n    clusterName: string,\n    serviceName: string,\n  ): Promise<AWS.ECS.Service> {\n    const services = await ecs\n      .describeServices({\n        cluster: clusterName,\n        services: [serviceName],\n      })\n      .promise()\n\n    if (services.services?.length !== 1) {\n      throw new Error(`Service not found: ${clusterName}/${serviceName}`)\n    }\n\n    return services.services[0]\n  }\n\n  async function getTaskDefinition(\n    taskDefinition: string,\n  ): Promise<AWS.ECS.TaskDefinition> {\n    return (\n      await ecs\n        .describeTaskDefinition({\n          taskDefinition: taskDefinition,\n        })\n        .promise()\n    ).taskDefinition!\n  }\n\n  async function updateServiceToImage(\n    clusterName: string,\n    serviceName: string,\n    image: string,\n  ) {\n    console.log(`Cluster name: ${clusterName}`)\n    console.log(`Service name: ${serviceName}`)\n\n    const service = await getService(clusterName, serviceName)\n    const prevTaskDefinition = await getTaskDefinition(service.taskDefinition!)\n\n    // Don't bother updating the service if the image is already the latest.\n    const prevImage = prevTaskDefinition.containerDefinitions![0].image!\n    if (prevImage === image) {\n      return\n    }\n\n    console.log(\n      `Updating image for service '${serviceName}' from '${prevImage}' to '${image}'`,\n    )\n\n    const exclude = [\n      \"registeredAt\",\n      \"registeredBy\",\n      \"compatibilities\",\n      \"requiresAttributes\",\n      \"revision\",\n      \"status\",\n      \"taskDefinitionArn\",\n    ]\n\n    const updatedSpec: RegisterTaskDefinitionRequest = {\n      ...Object.fromEntries(\n        Object.entries(prevTaskDefinition).filter(\n          ([key]) => !exclude.includes(key),\n        ),\n      ),\n      containerDefinitions: [\n        {\n          ...prevTaskDefinition.containerDefinitions![0],\n          image,\n        },\n      ],\n    } as RegisterTaskDefinitionRequest\n\n    const updatedTaskDefinition = (\n      await ecs.registerTaskDefinition(updatedSpec).promise()\n    ).taskDefinition!\n\n    await ecs\n      .updateService({\n        cluster: clusterName,\n        service: serviceName,\n        taskDefinition: updatedTaskDefinition.taskDefinitionArn,\n      })\n      .promise()\n\n    console.log(\"Service is updated\")\n  }\n\n  const clusterName = requireEnv(\"CLUSTER_NAME\")\n  const serviceName = requireEnv(\"SERVICE_NAME\")\n  const repositoryUrl = requireEnv(\"REPOSITORY_URL\")\n  const ecrTagSecretArn = requireEnv(\"ECR_TAG_SECRET_ARN\")\n\n  // Validate the input.\n  if (typeof event.tag !== \"string\") {\n    throw new Error(\"Input invalid: \" + JSON.stringify(event, undefined, \"  \"))\n  }\n\n  // Register tag as current target.\n  // This is needed so that CloudFormation deployments, e.g.\n  // updates to the Task Definition, will use the same image.\n  await sm\n    .updateSecret({\n      SecretId: ecrTagSecretArn,\n      SecretString: JSON.stringify({\n        tag: event.tag,\n      }),\n    })\n    .promise()\n\n  // Update the service if we know the service name. This is unknown\n  // during initial deployment of the stack.\n  if (serviceName !== \"\") {\n    const image = `${repositoryUrl}:${event.tag}`\n    await updateServiceToImage(clusterName, serviceName, image)\n  }\n\n  return {}\n}\n"]}
@@ -0,0 +1 @@
1
+ export { KinesisToDatadogStream, KinesisToDatadogStreamProps, } from "./kinesis-to-datadog-stream";
@@ -0,0 +1,6 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.KinesisToDatadogStream = void 0;
4
+ var kinesis_to_datadog_stream_1 = require("./kinesis-to-datadog-stream");
5
+ Object.defineProperty(exports, "KinesisToDatadogStream", { enumerable: true, get: function () { return kinesis_to_datadog_stream_1.KinesisToDatadogStream; } });
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMva2luZXNpcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5RUFHb0M7QUFGbEMsbUlBQUEsc0JBQXNCLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQge1xuICBLaW5lc2lzVG9EYXRhZG9nU3RyZWFtLFxuICBLaW5lc2lzVG9EYXRhZG9nU3RyZWFtUHJvcHMsXG59IGZyb20gXCIuL2tpbmVzaXMtdG8tZGF0YWRvZy1zdHJlYW1cIlxuIl19
@@ -0,0 +1,27 @@
1
+ import * as logs from "@aws-cdk/aws-logs";
2
+ import * as cdk from "@aws-cdk/core";
3
+ export interface KinesisToDatadogStreamProps {
4
+ /**
5
+ *
6
+ * The name of the SecretsManager secret where your Datadog API key is saved.
7
+ *
8
+ * The secret must be a JSON object on the format { "value": "SECRET" }
9
+ *
10
+ */
11
+ datadogApiKeySecretName: string;
12
+ /**
13
+ *
14
+ * The CloudWatch log groups from you are streaming to Datadog
15
+ *
16
+ */
17
+ logGroups: logs.LogGroup[];
18
+ }
19
+ /**
20
+ *
21
+ * Forwards logs from log-groups in CloudWatch to a Datadog account.
22
+ * The logs are delivered through a Firehose delivery stream, which is being subscribed to the log-groups in CloudWatch.
23
+ *
24
+ */
25
+ export declare class KinesisToDatadogStream extends cdk.Construct {
26
+ constructor(scope: cdk.Construct, id: string, props: KinesisToDatadogStreamProps);
27
+ }
@@ -0,0 +1,124 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.KinesisToDatadogStream = void 0;
4
+ const iam = require("@aws-cdk/aws-iam");
5
+ const firehose = require("@aws-cdk/aws-kinesisfirehose");
6
+ const logs = require("@aws-cdk/aws-logs");
7
+ const s3 = require("@aws-cdk/aws-s3");
8
+ const aws_s3_1 = require("@aws-cdk/aws-s3");
9
+ const secretsmanager = require("@aws-cdk/aws-secretsmanager");
10
+ const cdk = require("@aws-cdk/core");
11
+ /**
12
+ *
13
+ * Forwards logs from log-groups in CloudWatch to a Datadog account.
14
+ * The logs are delivered through a Firehose delivery stream, which is being subscribed to the log-groups in CloudWatch.
15
+ *
16
+ */
17
+ class KinesisToDatadogStream extends cdk.Construct {
18
+ constructor(scope, id, props) {
19
+ super(scope, id);
20
+ const deliveryStreamLogGroup = new logs.LogGroup(this, "DeliveryStreamLogGroup");
21
+ const deliveryStreamLogStream = new logs.LogStream(this, "DeliveryStreamLogStream", {
22
+ logGroup: deliveryStreamLogGroup,
23
+ });
24
+ const failedDataBucket = new s3.Bucket(this, "FailedDataBucket", {
25
+ blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL,
26
+ });
27
+ const cloudWatchLogsRole = new iam.Role(this, "CloudWatchLogsRole", {
28
+ assumedBy: new iam.ServicePrincipal(`logs.${cdk.Stack.of(this).region}.amazonaws.com`),
29
+ });
30
+ const firehoseLogsRole = new iam.Role(this, "FirehoseLogsRole", {
31
+ assumedBy: new iam.ServicePrincipal("firehose.amazonaws.com"),
32
+ });
33
+ const datadogDeliveryStream = new firehose.CfnDeliveryStream(this, "DeliveryStream", {
34
+ deliveryStreamType: "DirectPut",
35
+ httpEndpointDestinationConfiguration: {
36
+ roleArn: firehoseLogsRole.roleArn,
37
+ endpointConfiguration: {
38
+ url: "https://aws-kinesis-http-intake.logs.datadoghq.eu/v1/input",
39
+ accessKey: secretsmanager.Secret.fromSecretNameV2(scope, "DatadogApiKey", props.datadogApiKeySecretName)
40
+ .secretValueFromJson("value")
41
+ .toString(),
42
+ name: "datadog-logs-endpoint",
43
+ },
44
+ requestConfiguration: {
45
+ contentEncoding: "GZIP",
46
+ },
47
+ cloudWatchLoggingOptions: {
48
+ enabled: true,
49
+ logGroupName: deliveryStreamLogGroup.logGroupName,
50
+ logStreamName: deliveryStreamLogStream.logStreamName,
51
+ },
52
+ bufferingHints: {
53
+ intervalInSeconds: 60,
54
+ sizeInMBs: 4,
55
+ },
56
+ retryOptions: {
57
+ durationInSeconds: 60,
58
+ },
59
+ s3BackupMode: "FailedDataOnly",
60
+ s3Configuration: {
61
+ bucketArn: failedDataBucket.bucketArn,
62
+ compressionFormat: "UNCOMPRESSED",
63
+ roleArn: firehoseLogsRole.roleArn,
64
+ },
65
+ },
66
+ });
67
+ new iam.Policy(this, "CloudWatchLogsPolicy", {
68
+ document: new iam.PolicyDocument({
69
+ statements: [
70
+ new iam.PolicyStatement({
71
+ actions: ["firehose:PutRecord", "firehose:PutRecordBatch"],
72
+ resources: [datadogDeliveryStream.attrArn],
73
+ }),
74
+ ],
75
+ }),
76
+ roles: [cloudWatchLogsRole],
77
+ });
78
+ new iam.Policy(this, "FirehoseLogsPolicy", {
79
+ document: new iam.PolicyDocument({
80
+ statements: [
81
+ new iam.PolicyStatement({
82
+ actions: [
83
+ "s3:AbortMultipartUpload",
84
+ "s3:GetBucketLocation",
85
+ "s3:GetObject",
86
+ "s3:ListBucket",
87
+ "s3:ListBucketMultipartUploads",
88
+ "s3:PutObject",
89
+ ],
90
+ resources: [
91
+ failedDataBucket.bucketArn,
92
+ `${failedDataBucket.bucketArn}/*`,
93
+ ],
94
+ }),
95
+ new iam.PolicyStatement({
96
+ actions: ["logs:PutLogEvents"],
97
+ resources: [
98
+ `arn:aws:logs:${cdk.Stack.of(this).region}:${cdk.Stack.of(this).account}:log-group:${deliveryStreamLogGroup.logGroupName}:log-stream:${deliveryStreamLogStream.logStreamName}`,
99
+ ],
100
+ }),
101
+ new iam.PolicyStatement({
102
+ actions: [
103
+ "kinesis:DescribeStream",
104
+ "kinesis:GetShardIterator",
105
+ "kinesis:GetRecords",
106
+ ],
107
+ resources: [datadogDeliveryStream.attrArn],
108
+ }),
109
+ ],
110
+ }),
111
+ roles: [firehoseLogsRole],
112
+ });
113
+ props.logGroups.forEach((logGroup, index) => {
114
+ new logs.CfnSubscriptionFilter(this, `SubscriptionFilter${index}`, {
115
+ logGroupName: logGroup.logGroupName,
116
+ destinationArn: datadogDeliveryStream.attrArn,
117
+ filterPattern: logs.FilterPattern.allEvents().logPatternString,
118
+ roleArn: cloudWatchLogsRole.roleArn,
119
+ });
120
+ });
121
+ }
122
+ }
123
+ exports.KinesisToDatadogStream = KinesisToDatadogStream;
124
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kinesis-to-datadog-stream.js","sourceRoot":"","sources":["../../src/kinesis/kinesis-to-datadog-stream.ts"],"names":[],"mappings":"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","sourcesContent":["import * as iam from \"@aws-cdk/aws-iam\"\nimport * as firehose from \"@aws-cdk/aws-kinesisfirehose\"\nimport * as logs from \"@aws-cdk/aws-logs\"\nimport * as s3 from \"@aws-cdk/aws-s3\"\nimport { BlockPublicAccess } from \"@aws-cdk/aws-s3\"\nimport * as secretsmanager from \"@aws-cdk/aws-secretsmanager\"\nimport * as cdk from \"@aws-cdk/core\"\n\nexport interface KinesisToDatadogStreamProps {\n  /**\n   *\n   * The name of the SecretsManager secret where your Datadog API key is saved.\n   *\n   * The secret must be a JSON object on the format { \"value\": \"SECRET\" }\n   *\n   */\n  datadogApiKeySecretName: string\n  /**\n   *\n   * The CloudWatch log groups from you are streaming to Datadog\n   *\n   */\n  logGroups: logs.LogGroup[]\n}\n\n/**\n *\n * Forwards logs from log-groups in CloudWatch to a Datadog account.\n * The logs are delivered through a Firehose delivery stream, which is being subscribed to the log-groups in CloudWatch.\n *\n */\nexport class KinesisToDatadogStream extends cdk.Construct {\n  constructor(\n    scope: cdk.Construct,\n    id: string,\n    props: KinesisToDatadogStreamProps,\n  ) {\n    super(scope, id)\n\n    const deliveryStreamLogGroup = new logs.LogGroup(\n      this,\n      \"DeliveryStreamLogGroup\",\n    )\n\n    const deliveryStreamLogStream = new logs.LogStream(\n      this,\n      \"DeliveryStreamLogStream\",\n      {\n        logGroup: deliveryStreamLogGroup,\n      },\n    )\n\n    const failedDataBucket = new s3.Bucket(this, \"FailedDataBucket\", {\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n    })\n\n    const cloudWatchLogsRole = new iam.Role(this, \"CloudWatchLogsRole\", {\n      assumedBy: new iam.ServicePrincipal(\n        `logs.${cdk.Stack.of(this).region}.amazonaws.com`,\n      ),\n    })\n\n    const firehoseLogsRole = new iam.Role(this, \"FirehoseLogsRole\", {\n      assumedBy: new iam.ServicePrincipal(\"firehose.amazonaws.com\"),\n    })\n\n    const datadogDeliveryStream = new firehose.CfnDeliveryStream(\n      this,\n      \"DeliveryStream\",\n      {\n        deliveryStreamType: \"DirectPut\",\n        httpEndpointDestinationConfiguration: {\n          roleArn: firehoseLogsRole.roleArn,\n          endpointConfiguration: {\n            url: \"https://aws-kinesis-http-intake.logs.datadoghq.eu/v1/input\",\n            accessKey: secretsmanager.Secret.fromSecretNameV2(\n              scope,\n              \"DatadogApiKey\",\n              props.datadogApiKeySecretName,\n            )\n              .secretValueFromJson(\"value\")\n              .toString(),\n            name: \"datadog-logs-endpoint\",\n          },\n          requestConfiguration: {\n            contentEncoding: \"GZIP\",\n          },\n          cloudWatchLoggingOptions: {\n            enabled: true,\n            logGroupName: deliveryStreamLogGroup.logGroupName,\n            logStreamName: deliveryStreamLogStream.logStreamName,\n          },\n          bufferingHints: {\n            intervalInSeconds: 60,\n            sizeInMBs: 4,\n          },\n          retryOptions: {\n            durationInSeconds: 60,\n          },\n          s3BackupMode: \"FailedDataOnly\",\n          s3Configuration: {\n            bucketArn: failedDataBucket.bucketArn,\n            compressionFormat: \"UNCOMPRESSED\",\n            roleArn: firehoseLogsRole.roleArn,\n          },\n        },\n      },\n    )\n\n    new iam.Policy(this, \"CloudWatchLogsPolicy\", {\n      document: new iam.PolicyDocument({\n        statements: [\n          new iam.PolicyStatement({\n            actions: [\"firehose:PutRecord\", \"firehose:PutRecordBatch\"],\n            resources: [datadogDeliveryStream.attrArn],\n          }),\n        ],\n      }),\n      roles: [cloudWatchLogsRole],\n    })\n\n    new iam.Policy(this, \"FirehoseLogsPolicy\", {\n      document: new iam.PolicyDocument({\n        statements: [\n          new iam.PolicyStatement({\n            actions: [\n              \"s3:AbortMultipartUpload\",\n              \"s3:GetBucketLocation\",\n              \"s3:GetObject\",\n              \"s3:ListBucket\",\n              \"s3:ListBucketMultipartUploads\",\n              \"s3:PutObject\",\n            ],\n            resources: [\n              failedDataBucket.bucketArn,\n              `${failedDataBucket.bucketArn}/*`,\n            ],\n          }),\n          new iam.PolicyStatement({\n            actions: [\"logs:PutLogEvents\"],\n            resources: [\n              `arn:aws:logs:${cdk.Stack.of(this).region}:${\n                cdk.Stack.of(this).account\n              }:log-group:${deliveryStreamLogGroup.logGroupName}:log-stream:${\n                deliveryStreamLogStream.logStreamName\n              }`,\n            ],\n          }),\n          new iam.PolicyStatement({\n            actions: [\n              \"kinesis:DescribeStream\",\n              \"kinesis:GetShardIterator\",\n              \"kinesis:GetRecords\",\n            ],\n            resources: [datadogDeliveryStream.attrArn],\n          }),\n        ],\n      }),\n      roles: [firehoseLogsRole],\n    })\n\n    props.logGroups.forEach((logGroup, index) => {\n      new logs.CfnSubscriptionFilter(this, `SubscriptionFilter${index}`, {\n        logGroupName: logGroup.logGroupName,\n        destinationArn: datadogDeliveryStream.attrArn,\n        filterPattern: logs.FilterPattern.allEvents().logPatternString,\n        roleArn: cloudWatchLogsRole.roleArn,\n      })\n    })\n  }\n}\n"]}
package/lib/snapshots.js CHANGED
@@ -209,4 +209,4 @@ async function createCloudAssemblySnapshot(src, dst) {
209
209
  }
210
210
  }
211
211
  exports.createCloudAssemblySnapshot = createCloudAssemblySnapshot;
212
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshots.js","sourceRoot":"","sources":["../src/snapshots.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-assignment */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-unsafe-return */\nimport * as cpy from \"cpy\"\nimport * as del from \"del\"\nimport * as fs from \"fs\"\nimport * as glob from \"glob\"\nimport * as path from \"path\"\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeVersion(data: any): any {\n  const cp = {\n    ...data,\n  }\n\n  delete cp.version\n  return cp\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeTrace(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeTrace)\n  }\n\n  if (data === Object(data)) {\n    return Object.fromEntries(\n      Object.entries(data)\n        .filter(([key]) => key !== \"trace\")\n        .map(([key, value]) => [key, removeTrace(value)]),\n    )\n  }\n\n  return data\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeRuntimeLibraries(data: any): any {\n  const cp = {\n    ...data,\n  }\n\n  if (data.runtime) {\n    cp.runtime = {\n      ...data.runtime,\n    }\n\n    delete cp.runtime.libraries\n  }\n\n  return cp\n}\n\nconst currentVersionRegex = /^(.+CurrentVersion[0-9A-F]{8})[0-9a-f]{32}$/\n\n/**\n * Match a resource created by `lambda.Function.currentVersion`, which\n * will include the asset hash as part of the reousrce name and return\n * a snapshot-friendly version of it if found.\n */\nfunction rewriteCurrentVersionIfFound(value: string): string | null {\n  const match = currentVersionRegex.exec(value)\n  return match ? `${match[1]}xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx` : null\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeAssetDetailsFromTemplate(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeAssetDetailsFromTemplate)\n  }\n\n  if (data === Object(data)) {\n    return Object.fromEntries(\n      Object.entries(data)\n        .map(([key, value]) => {\n          const newCurrentVersion = rewriteCurrentVersionIfFound(key)\n          if (newCurrentVersion) {\n            return [newCurrentVersion, removeAssetDetailsFromTemplate(value)]\n          } else if (key.includes(\"AssetParameter\")) {\n            return null\n          } else if (\n            key === \"Ref\" &&\n            typeof value === \"string\" &&\n            value.includes(\"AssetParameters\")\n          ) {\n            return [key, \"snapshot-value\"]\n          } else if (\n            key === \"aws:asset:path\" &&\n            typeof value === \"string\" &&\n            /asset\\.[0-9a-f]{64}/.test(value)\n          ) {\n            return [key, \"asset.snapshot-value\"]\n          } else {\n            return [key, removeAssetDetailsFromTemplate(value)]\n          }\n        })\n        .filter((it): it is [] => it != null),\n    )\n  }\n\n  if (typeof data === \"string\") {\n    const newCurrentVersion = rewriteCurrentVersionIfFound(data)\n    if (newCurrentVersion) {\n      return newCurrentVersion\n    }\n\n    // Handle typical content hashes.\n    return data.replace(/[0-9a-f]{64}/g, \"snapshot-value\")\n  }\n\n  return data\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeAssetDetailsFromManifest(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeAssetDetailsFromManifest)\n  }\n\n  if (data === Object(data)) {\n    // aws:cdk:asset in metadata\n    if (data[\"type\"] === \"aws:cdk:asset\" && \"data\" in data) {\n      return {\n        ...data,\n        data: \"snapshot-value\",\n      }\n    }\n\n    return Object.fromEntries(\n      Object.entries(data)\n        .map(([key, value]) => {\n          if (key.includes(\"AssetParameters\")) {\n            return null\n          } else {\n            return [key, removeAssetDetailsFromManifest(value)]\n          }\n        })\n        .filter((it): it is [] => it != null),\n    )\n  }\n\n  if (typeof data === \"string\") {\n    const newCurrentVersion = rewriteCurrentVersionIfFound(data)\n    if (newCurrentVersion) {\n      return newCurrentVersion\n    }\n\n    // Handle typical content hashes.\n    return data.replace(/[0-9a-f]{64}/g, \"snapshot-value\")\n  }\n\n  return data\n}\n/**\n * Remove the CDKMetadata resources that is part of the synthesized\n * template since CDK 1.63.0.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeCdkMetadataResourceFromTemplate(data: any): any {\n  const cp = {\n    ...data,\n    Resources: {\n      ...data.Resources,\n    },\n  }\n\n  delete cp.Resources.CDKMetadata\n  return cp\n}\n\nfunction prepareManifestForSnapshot(content: string): string {\n  const input = JSON.parse(content)\n  const output = [\n    removeVersion,\n    // Remove the runtime version information so it don't conflict with CI\n    // or other users generating the snapshots.\n    removeRuntimeLibraries,\n    // Remove the trace from manifest for now.\n    removeTrace,\n    // Avoid details (hashes) from assets.\n    removeAssetDetailsFromManifest,\n  ].reduce((acc, fn) => fn(acc), input)\n\n  return JSON.stringify(output, undefined, \"  \")\n}\n\n/**\n * Transform the Cloud Assembly manifest file so that it can be persisted\n * as a snapshot without causing invalidations for every synthesize.\n */\nasync function prepareManifestFileForSnapshot(file: string): Promise<void> {\n  const result = prepareManifestForSnapshot(\n    await fs.promises.readFile(file, \"utf8\"),\n  )\n\n  await fs.promises.writeFile(file, result)\n}\n\nfunction prepareTemplateForSnapshot(content: string): string {\n  const input = JSON.parse(content)\n  const output = [\n    removeCdkMetadataResourceFromTemplate,\n    // Avoid details (hashes) from assets.\n    removeAssetDetailsFromTemplate,\n  ].reduce((acc, fn) => fn(acc), input)\n\n  return JSON.stringify(output, undefined, \"  \")\n}\n\n/**\n * Transform a Cloud Assembly template file so that it can be persisted\n * as a snapshot without causing invalidations for minor changes.\n */\nasync function prepareTemplateFileForSnapshot(file: string): Promise<void> {\n  const result = prepareTemplateForSnapshot(\n    await fs.promises.readFile(file, \"utf8\"),\n  )\n\n  await fs.promises.writeFile(file, result)\n}\n\n/**\n * Convert a Cloud Assembly to a snapshot.\n */\nexport async function createCloudAssemblySnapshot(\n  src: string,\n  dst: string,\n): Promise<void> {\n  const base = path.join(process.cwd(), dst)\n\n  await cpy(\".\", base, {\n    parents: true,\n    cwd: path.join(process.cwd(), src),\n  })\n\n  // Don't keep track of manifest version.\n  await del(path.join(dst, \"**/cdk.out\"))\n\n  // The tree file doesn't give us much value as part of the snapshot.\n  await del(path.join(dst, \"tree.json\"))\n\n  // Remove asset contents for now.\n  await del(path.join(dst, \"**/asset.*\"))\n\n  // Remove asset configs so we don't have to update\n  // snapshots for asset changes.\n  await del(path.join(dst, \"**/*.assets.json\"))\n\n  // Transform the manifest to be more snapshot friendly.\n  for (const file of glob.sync(\"**/manifest.json\", { cwd: base })) {\n    await prepareManifestFileForSnapshot(path.join(base, file))\n  }\n\n  // Transform all templates.\n  for (const file of glob.sync(\"**/*.template.json\", { cwd: base })) {\n    await prepareTemplateFileForSnapshot(path.join(base, file))\n  }\n}\n"]}
212
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshots.js","sourceRoot":"","sources":["../src/snapshots.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-unsafe-assignment */\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-unsafe-return */\nimport * as cpy from \"cpy\"\nimport * as del from \"del\"\nimport * as fs from \"fs\"\nimport * as glob from \"glob\"\nimport * as path from \"path\"\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeVersion(data: any): any {\n  const cp = {\n    ...data,\n  }\n\n  delete cp.version\n  return cp\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeTrace(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeTrace)\n  }\n\n  if (data === Object(data)) {\n    return Object.fromEntries(\n      Object.entries(data as Record<string, unknown>)\n        .filter(([key]) => key !== \"trace\")\n        .map(([key, value]) => [key, removeTrace(value)]),\n    )\n  }\n\n  return data\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeRuntimeLibraries(data: any): any {\n  const cp = {\n    ...data,\n  }\n\n  if (data.runtime) {\n    cp.runtime = {\n      ...data.runtime,\n    }\n\n    delete cp.runtime.libraries\n  }\n\n  return cp\n}\n\nconst currentVersionRegex = /^(.+CurrentVersion[0-9A-F]{8})[0-9a-f]{32}$/\n\n/**\n * Match a resource created by `lambda.Function.currentVersion`, which\n * will include the asset hash as part of the reousrce name and return\n * a snapshot-friendly version of it if found.\n */\nfunction rewriteCurrentVersionIfFound(value: string): string | null {\n  const match = currentVersionRegex.exec(value)\n  return match ? `${match[1]}xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx` : null\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeAssetDetailsFromTemplate(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeAssetDetailsFromTemplate)\n  }\n\n  if (data === Object(data)) {\n    return Object.fromEntries(\n      Object.entries(data as Record<string, unknown>)\n        .map(([key, value]) => {\n          const newCurrentVersion = rewriteCurrentVersionIfFound(key)\n          if (newCurrentVersion) {\n            return [newCurrentVersion, removeAssetDetailsFromTemplate(value)]\n          } else if (key.includes(\"AssetParameter\")) {\n            return null\n          } else if (\n            key === \"Ref\" &&\n            typeof value === \"string\" &&\n            value.includes(\"AssetParameters\")\n          ) {\n            return [key, \"snapshot-value\"]\n          } else if (\n            key === \"aws:asset:path\" &&\n            typeof value === \"string\" &&\n            /asset\\.[0-9a-f]{64}/.test(value)\n          ) {\n            return [key, \"asset.snapshot-value\"]\n          } else {\n            return [key, removeAssetDetailsFromTemplate(value)]\n          }\n        })\n        .filter((it): it is [] => it != null),\n    )\n  }\n\n  if (typeof data === \"string\") {\n    const newCurrentVersion = rewriteCurrentVersionIfFound(data)\n    if (newCurrentVersion) {\n      return newCurrentVersion\n    }\n\n    // Handle typical content hashes.\n    return data.replace(/[0-9a-f]{64}/g, \"snapshot-value\")\n  }\n\n  return data\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeAssetDetailsFromManifest(data: any): any {\n  if (data instanceof Array) {\n    return data.map(removeAssetDetailsFromManifest)\n  }\n\n  if (data === Object(data)) {\n    // aws:cdk:asset in metadata\n    if (data[\"type\"] === \"aws:cdk:asset\" && \"data\" in data) {\n      return {\n        ...data,\n        data: \"snapshot-value\",\n      }\n    }\n\n    return Object.fromEntries(\n      Object.entries(data as Record<string, unknown>)\n        .map(([key, value]) => {\n          if (key.includes(\"AssetParameters\")) {\n            return null\n          } else {\n            return [key, removeAssetDetailsFromManifest(value)]\n          }\n        })\n        .filter((it): it is [] => it != null),\n    )\n  }\n\n  if (typeof data === \"string\") {\n    const newCurrentVersion = rewriteCurrentVersionIfFound(data)\n    if (newCurrentVersion) {\n      return newCurrentVersion\n    }\n\n    // Handle typical content hashes.\n    return data.replace(/[0-9a-f]{64}/g, \"snapshot-value\")\n  }\n\n  return data\n}\n\n/**\n * Remove the CDKMetadata resources that is part of the synthesized\n * template since CDK 1.63.0.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction removeCdkMetadataResourceFromTemplate(data: any): any {\n  const cp = {\n    ...data,\n    Resources: {\n      ...data.Resources,\n    },\n  }\n\n  delete cp.Resources.CDKMetadata\n  return cp\n}\n\nfunction prepareManifestForSnapshot(content: string): string {\n  const input = JSON.parse(content)\n  const output = [\n    removeVersion,\n    // Remove the runtime version information so it don't conflict with CI\n    // or other users generating the snapshots.\n    removeRuntimeLibraries,\n    // Remove the trace from manifest for now.\n    removeTrace,\n    // Avoid details (hashes) from assets.\n    removeAssetDetailsFromManifest,\n  ].reduce((acc, fn) => fn(acc), input)\n\n  return JSON.stringify(output, undefined, \"  \")\n}\n\n/**\n * Transform the Cloud Assembly manifest file so that it can be persisted\n * as a snapshot without causing invalidations for every synthesize.\n */\nasync function prepareManifestFileForSnapshot(file: string): Promise<void> {\n  const result = prepareManifestForSnapshot(\n    await fs.promises.readFile(file, \"utf8\"),\n  )\n\n  await fs.promises.writeFile(file, result)\n}\n\nfunction prepareTemplateForSnapshot(content: string): string {\n  const input = JSON.parse(content)\n  const output = [\n    removeCdkMetadataResourceFromTemplate,\n    // Avoid details (hashes) from assets.\n    removeAssetDetailsFromTemplate,\n  ].reduce((acc, fn) => fn(acc), input)\n\n  return JSON.stringify(output, undefined, \"  \")\n}\n\n/**\n * Transform a Cloud Assembly template file so that it can be persisted\n * as a snapshot without causing invalidations for minor changes.\n */\nasync function prepareTemplateFileForSnapshot(file: string): Promise<void> {\n  const result = prepareTemplateForSnapshot(\n    await fs.promises.readFile(file, \"utf8\"),\n  )\n\n  await fs.promises.writeFile(file, result)\n}\n\n/**\n * Convert a Cloud Assembly to a snapshot.\n */\nexport async function createCloudAssemblySnapshot(\n  src: string,\n  dst: string,\n): Promise<void> {\n  const base = path.join(process.cwd(), dst)\n\n  await cpy(\".\", base, {\n    parents: true,\n    cwd: path.join(process.cwd(), src),\n  })\n\n  // Don't keep track of manifest version.\n  await del(path.join(dst, \"**/cdk.out\"))\n\n  // The tree file doesn't give us much value as part of the snapshot.\n  await del(path.join(dst, \"tree.json\"))\n\n  // Remove asset contents for now.\n  await del(path.join(dst, \"**/asset.*\"))\n\n  // Remove asset configs so we don't have to update\n  // snapshots for asset changes.\n  await del(path.join(dst, \"**/*.assets.json\"))\n\n  // Transform the manifest to be more snapshot friendly.\n  for (const file of glob.sync(\"**/manifest.json\", { cwd: base })) {\n    await prepareManifestFileForSnapshot(path.join(base, file))\n  }\n\n  // Transform all templates.\n  for (const file of glob.sync(\"**/*.template.json\", { cwd: base })) {\n    await prepareTemplateFileForSnapshot(path.join(base, file))\n  }\n}\n"]}
@@ -0,0 +1,40 @@
1
+ import * as cloudfront from "@aws-cdk/aws-cloudfront";
2
+ import * as cdk from "@aws-cdk/core";
3
+ export interface FrameOptionsHeader {
4
+ value?: "DENY" | "SAMEORIGIN";
5
+ }
6
+ export interface ReferrerPolicyHeader {
7
+ value?: string;
8
+ }
9
+ export interface StrictTransportSecurityHeader {
10
+ maxAge?: number;
11
+ includeSubDomains?: boolean;
12
+ preload?: boolean;
13
+ }
14
+ export interface ContentSecurityPolicyHeader {
15
+ reportOnly?: boolean;
16
+ baseUri?: string;
17
+ childSrc?: string;
18
+ defaultSrc?: string;
19
+ fontSrc?: string;
20
+ frameSrc?: string;
21
+ formAction?: string;
22
+ frameAncestors?: string;
23
+ imgSrc?: string;
24
+ manifestSrc?: string;
25
+ mediaSrc?: string;
26
+ objectSrc?: string;
27
+ scriptSrc?: string;
28
+ styleSrc?: string;
29
+ connectSrc?: string;
30
+ }
31
+ export interface SecurityHeaders {
32
+ contentSecurityPolicy?: ContentSecurityPolicyHeader;
33
+ strictTransportSecurity?: StrictTransportSecurityHeader;
34
+ referrerPolicy?: ReferrerPolicyHeader;
35
+ frameOptions?: FrameOptionsHeader;
36
+ }
37
+ export declare class WebappSecurityHeaders extends cdk.Construct {
38
+ readonly securityHeadersFunction: cloudfront.Function;
39
+ constructor(scope: cdk.Construct, id: string, props: SecurityHeaders);
40
+ }
@@ -0,0 +1,126 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.WebappSecurityHeaders = void 0;
4
+ const cloudfront = require("@aws-cdk/aws-cloudfront");
5
+ const cdk = require("@aws-cdk/core");
6
+ function validateCspParam(param) {
7
+ if (param.indexOf('"') !== -1) {
8
+ throw Error('CSP override contains invalid character "');
9
+ }
10
+ if (param.indexOf(";") !== -1) {
11
+ throw Error("CSP override contains invalid character ;");
12
+ }
13
+ if (param.indexOf("\\") !== -1) {
14
+ throw Error("CSP override contains invalid character \\");
15
+ }
16
+ return param;
17
+ }
18
+ /* Replace all whitespace in a string with a single space */
19
+ function trim(value) {
20
+ return value.replace(/\s+/g, " ").trim();
21
+ }
22
+ function generateContentSecurityPolicyHeader(headerOptions) {
23
+ const defaultValues = {
24
+ baseUri: "'self'",
25
+ childSrc: "'none'",
26
+ connectSrc: "'self'",
27
+ defaultSrc: "'self'",
28
+ fontSrc: "'self'",
29
+ formAction: "'self'",
30
+ frameAncestors: "'none'",
31
+ frameSrc: "'self'",
32
+ imgSrc: "'self'",
33
+ manifestSrc: "'self'",
34
+ mediaSrc: "'self'",
35
+ objectSrc: "'none'",
36
+ scriptSrc: "'self'",
37
+ styleSrc: "'self'",
38
+ };
39
+ const options = {
40
+ ...defaultValues,
41
+ ...headerOptions,
42
+ };
43
+ Object.values(options).forEach((v) => typeof v === "string" && validateCspParam(v));
44
+ let headerValue = "";
45
+ headerValue += `base-uri ${trim(options.baseUri)};`;
46
+ headerValue += `child-src ${trim(options.childSrc)};`;
47
+ headerValue += `connect-src ${trim(options.connectSrc)};`;
48
+ headerValue += `default-src ${trim(options.defaultSrc)};`;
49
+ headerValue += `font-src ${trim(options.fontSrc)};`;
50
+ headerValue += `frame-src ${trim(options.frameSrc)};`;
51
+ headerValue += `img-src ${trim(options.imgSrc)};`;
52
+ headerValue += `manifest-src ${trim(options.manifestSrc)};`;
53
+ headerValue += `media-src ${trim(options.mediaSrc)};`;
54
+ headerValue += `object-src ${trim(options.objectSrc)};`;
55
+ headerValue += `script-src ${trim(options.scriptSrc)};`;
56
+ headerValue += `style-src ${trim(options.styleSrc)};`;
57
+ return trim(headerValue);
58
+ }
59
+ function generateStrictTransportSecurityHeader(headerOptions) {
60
+ const defaultValues = {
61
+ maxAge: 63072000,
62
+ includeSubDomains: false,
63
+ preload: false,
64
+ };
65
+ const options = {
66
+ ...defaultValues,
67
+ ...headerOptions,
68
+ };
69
+ let headerValue = "";
70
+ headerValue += `max-age=${options.maxAge};`;
71
+ headerValue += options.preload ? "preload;" : "";
72
+ headerValue += options.includeSubDomains ? "includeSubDomains;" : "";
73
+ return trim(headerValue);
74
+ }
75
+ function generateReferrerPolicyHeader(headerOptions) {
76
+ const defaultValues = {
77
+ value: "strict-origin-when-cross-origin",
78
+ };
79
+ const options = {
80
+ ...defaultValues,
81
+ ...headerOptions,
82
+ };
83
+ return options.value;
84
+ }
85
+ function generateFrameOptionsHeader(headerOptions) {
86
+ const defaultValues = {
87
+ value: "DENY",
88
+ };
89
+ const options = {
90
+ ...defaultValues,
91
+ ...headerOptions,
92
+ };
93
+ return trim(options.value);
94
+ }
95
+ class WebappSecurityHeaders extends cdk.Construct {
96
+ constructor(scope, id, props) {
97
+ var _a;
98
+ super(scope, id);
99
+ const cspHeaderName = ((_a = props.contentSecurityPolicy) === null || _a === void 0 ? void 0 : _a.reportOnly)
100
+ ? "content-security-policy-report-only"
101
+ : "content-security-policy";
102
+ const contentSecurityPolicy = generateContentSecurityPolicyHeader(props.contentSecurityPolicy);
103
+ const strictTransportSecurity = generateStrictTransportSecurityHeader(props.strictTransportSecurity);
104
+ const referrerPolicy = generateReferrerPolicyHeader(props.referrerPolicy);
105
+ const frameOptions = generateFrameOptionsHeader(props.frameOptions);
106
+ const lambdaCode = `function handler(event) {
107
+ var response = event.response;
108
+ var headers = response.headers;
109
+ headers['referrer-policy'] = {value: '${referrerPolicy}'};
110
+ headers['strict-transport-security'] = {value: '${strictTransportSecurity}'};
111
+ headers['x-content-type-options'] = {value: 'nosniff'};
112
+ headers['x-frame-options'] = {value: '${frameOptions}'};
113
+ headers['x-xss-protection'] = {value: '1; mode=block'};
114
+ headers['${cspHeaderName}'] = {value: "${contentSecurityPolicy}"};
115
+ return response;
116
+ }`;
117
+ // Hardcoded logical ID due to bug: https://github.com/aws/aws-cdk/issues/15523
118
+ const functionId = `Function${this.node.addr}`;
119
+ this.securityHeadersFunction = new cloudfront.Function(this, functionId, {
120
+ functionName: functionId,
121
+ code: cloudfront.FunctionCode.fromInline(lambdaCode),
122
+ });
123
+ }
124
+ }
125
+ exports.WebappSecurityHeaders = WebappSecurityHeaders;
126
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"security-headers.js","sourceRoot":"","sources":["../../src/webapp/security-headers.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cloudfront from \"@aws-cdk/aws-cloudfront\"\nimport * as cdk from \"@aws-cdk/core\"\n\nexport interface FrameOptionsHeader {\n  value?: \"DENY\" | \"SAMEORIGIN\"\n}\n\nexport interface ReferrerPolicyHeader {\n  value?: string\n}\n\nexport interface StrictTransportSecurityHeader {\n  maxAge?: number\n  includeSubDomains?: boolean\n  preload?: boolean\n}\n\nexport interface ContentSecurityPolicyHeader {\n  reportOnly?: boolean\n  baseUri?: string\n  childSrc?: string\n  defaultSrc?: string\n  fontSrc?: string\n  frameSrc?: string\n  formAction?: string\n  frameAncestors?: string\n  imgSrc?: string\n  manifestSrc?: string\n  mediaSrc?: string\n  objectSrc?: string\n  scriptSrc?: string\n  styleSrc?: string\n  connectSrc?: string\n}\n\nexport interface SecurityHeaders {\n  contentSecurityPolicy?: ContentSecurityPolicyHeader\n  strictTransportSecurity?: StrictTransportSecurityHeader\n  referrerPolicy?: ReferrerPolicyHeader\n  frameOptions?: FrameOptionsHeader\n}\n\nfunction validateCspParam(param: string): string {\n  if (param.indexOf('\"') !== -1) {\n    throw Error('CSP override contains invalid character \"')\n  }\n\n  if (param.indexOf(\";\") !== -1) {\n    throw Error(\"CSP override contains invalid character ;\")\n  }\n\n  if (param.indexOf(\"\\\\\") !== -1) {\n    throw Error(\"CSP override contains invalid character \\\\\")\n  }\n\n  return param\n}\n\n/* Replace all whitespace in a string with a single space */\nfunction trim(value: string): string {\n  return value.replace(/\\s+/g, \" \").trim()\n}\n\nfunction generateContentSecurityPolicyHeader(\n  headerOptions?: ContentSecurityPolicyHeader,\n) {\n  const defaultValues = {\n    baseUri: \"'self'\",\n    childSrc: \"'none'\",\n    connectSrc: \"'self'\",\n    defaultSrc: \"'self'\",\n    fontSrc: \"'self'\",\n    formAction: \"'self'\",\n    frameAncestors: \"'none'\",\n    frameSrc: \"'self'\",\n    imgSrc: \"'self'\",\n    manifestSrc: \"'self'\",\n    mediaSrc: \"'self'\",\n    objectSrc: \"'none'\",\n    scriptSrc: \"'self'\",\n    styleSrc: \"'self'\",\n  }\n\n  const options = {\n    ...defaultValues,\n    ...headerOptions,\n  }\n\n  Object.values(options).forEach(\n    (v) => typeof v === \"string\" && validateCspParam(v),\n  )\n\n  let headerValue = \"\"\n  headerValue += `base-uri ${trim(options.baseUri)};`\n  headerValue += `child-src ${trim(options.childSrc)};`\n  headerValue += `connect-src ${trim(options.connectSrc)};`\n  headerValue += `default-src ${trim(options.defaultSrc)};`\n  headerValue += `font-src ${trim(options.fontSrc)};`\n  headerValue += `frame-src ${trim(options.frameSrc)};`\n  headerValue += `img-src ${trim(options.imgSrc)};`\n  headerValue += `manifest-src ${trim(options.manifestSrc)};`\n  headerValue += `media-src ${trim(options.mediaSrc)};`\n  headerValue += `object-src ${trim(options.objectSrc)};`\n  headerValue += `script-src ${trim(options.scriptSrc)};`\n  headerValue += `style-src ${trim(options.styleSrc)};`\n\n  return trim(headerValue)\n}\n\nfunction generateStrictTransportSecurityHeader(\n  headerOptions?: StrictTransportSecurityHeader,\n) {\n  const defaultValues = {\n    maxAge: 63072000,\n    includeSubDomains: false,\n    preload: false,\n  }\n  const options = {\n    ...defaultValues,\n    ...headerOptions,\n  }\n  let headerValue = \"\"\n  headerValue += `max-age=${options.maxAge};`\n  headerValue += options.preload ? \"preload;\" : \"\"\n  headerValue += options.includeSubDomains ? \"includeSubDomains;\" : \"\"\n  return trim(headerValue)\n}\n\nfunction generateReferrerPolicyHeader(headerOptions?: ReferrerPolicyHeader) {\n  const defaultValues = {\n    value: \"strict-origin-when-cross-origin\",\n  }\n  const options = {\n    ...defaultValues,\n    ...headerOptions,\n  }\n  return options.value\n}\n\nfunction generateFrameOptionsHeader(headerOptions?: FrameOptionsHeader) {\n  const defaultValues = {\n    value: \"DENY\",\n  }\n  const options = {\n    ...defaultValues,\n    ...headerOptions,\n  }\n  return trim(options.value)\n}\n\nexport class WebappSecurityHeaders extends cdk.Construct {\n  public readonly securityHeadersFunction: cloudfront.Function\n\n  constructor(scope: cdk.Construct, id: string, props: SecurityHeaders) {\n    super(scope, id)\n\n    const cspHeaderName = props.contentSecurityPolicy?.reportOnly\n      ? \"content-security-policy-report-only\"\n      : \"content-security-policy\"\n\n    const contentSecurityPolicy = generateContentSecurityPolicyHeader(\n      props.contentSecurityPolicy,\n    )\n    const strictTransportSecurity = generateStrictTransportSecurityHeader(\n      props.strictTransportSecurity,\n    )\n    const referrerPolicy = generateReferrerPolicyHeader(props.referrerPolicy)\n    const frameOptions = generateFrameOptionsHeader(props.frameOptions)\n\n    const lambdaCode = `function handler(event) {\n      var response = event.response;\n      var headers = response.headers;\n      headers['referrer-policy'] = {value: '${referrerPolicy}'};\n      headers['strict-transport-security'] = {value: '${strictTransportSecurity}'};\n      headers['x-content-type-options'] = {value: 'nosniff'};\n      headers['x-frame-options'] = {value: '${frameOptions}'};\n      headers['x-xss-protection'] = {value: '1; mode=block'};\n      headers['${cspHeaderName}'] = {value: \"${contentSecurityPolicy}\"};\n      return response;\n    }`\n\n    // Hardcoded logical ID due to bug: https://github.com/aws/aws-cdk/issues/15523\n    const functionId = `Function${this.node.addr}`\n\n    this.securityHeadersFunction = new cloudfront.Function(this, functionId, {\n      functionName: functionId,\n      code: cloudfront.FunctionCode.fromInline(lambdaCode),\n    })\n  }\n}\n"]}
@@ -5,6 +5,7 @@ import * as r53 from "@aws-cdk/aws-route53";
5
5
  import * as s3 from "@aws-cdk/aws-s3";
6
6
  import * as cdk from "@aws-cdk/core";
7
7
  import * as webappDeploy from "@capraconsulting/webapp-deploy-lambda";
8
+ import { SecurityHeaders } from "./security-headers";
8
9
  export interface WebappProps {
9
10
  /**
10
11
  * ACM certificate that covers the specifeid domain names.
@@ -40,6 +41,28 @@ export interface WebappProps {
40
41
  * @default - No custom page for 403 errors.
41
42
  */
42
43
  webAclErrorPagePath?: string;
44
+ /**
45
+ * Enable adding common security headers to CloudFront responses using a CloudFront Function.
46
+ *
47
+ * If enabled, the default behavior is to add the following headers with fairly strict defaults. Most of the headers can be customized:
48
+ * - Content-Security-Policy
49
+ * - Referrer-Policy
50
+ * - Strict-Transport-Security
51
+ * - X-Content-Type-Options
52
+ * - X-Frame-Options
53
+ * - X-XSS-Protection
54
+ *
55
+ * @default - No security headers will be added to responses
56
+ */
57
+ enableSecurityHeaders?: boolean;
58
+ /**
59
+ * Security headers overrides.
60
+ *
61
+ * Used to override certain security header values if the webapp requires more lax settings compared to the defaults.
62
+ *
63
+ * @default - A set of strict security header values will be used
64
+ */
65
+ securityHeadersOverrides?: SecurityHeaders;
43
66
  }
44
67
  /**
45
68
  * CloudFront for a Single-Page-Application.
@@ -9,6 +9,7 @@ const r53t = require("@aws-cdk/aws-route53-targets");
9
9
  const s3 = require("@aws-cdk/aws-s3");
10
10
  const cdk = require("@aws-cdk/core");
11
11
  const webappDeploy = require("@capraconsulting/webapp-deploy-lambda");
12
+ const security_headers_1 = require("./security-headers");
12
13
  /**
13
14
  * CloudFront for a Single-Page-Application.
14
15
  *
@@ -61,10 +62,23 @@ class Webapp extends cdk.Construct {
61
62
  responsePagePath: props.webAclErrorPagePath,
62
63
  });
63
64
  }
65
+ let functionAssociations;
66
+ if (props.enableSecurityHeaders) {
67
+ const securityHeaders = new security_headers_1.WebappSecurityHeaders(this, "SecurityHeaders", {
68
+ ...props.securityHeadersOverrides,
69
+ });
70
+ functionAssociations = [
71
+ {
72
+ function: securityHeaders.securityHeadersFunction,
73
+ eventType: cloudfront.FunctionEventType.VIEWER_RESPONSE,
74
+ },
75
+ ];
76
+ }
64
77
  this.distribution = new cloudfront.Distribution(this, "Distribution", {
65
78
  defaultBehavior: {
66
79
  origin: this.webappOrigin,
67
80
  viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
81
+ functionAssociations: functionAssociations,
68
82
  },
69
83
  defaultRootObject: "index.html",
70
84
  priceClass: cloudfront.PriceClass.PRICE_CLASS_100,
@@ -103,4 +117,4 @@ class Webapp extends cdk.Construct {
103
117
  }
104
118
  }
105
119
  exports.Webapp = Webapp;
106
- //# 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 certificatemanager from \"@aws-cdk/aws-certificatemanager\"\nimport * as cloudfront from \"@aws-cdk/aws-cloudfront\"\nimport * as origins from \"@aws-cdk/aws-cloudfront-origins\"\nimport * as iam from \"@aws-cdk/aws-iam\"\nimport * as r53 from \"@aws-cdk/aws-route53\"\nimport * as r53t from \"@aws-cdk/aws-route53-targets\"\nimport * as s3 from \"@aws-cdk/aws-s3\"\nimport * as cdk from \"@aws-cdk/core\"\nimport * as webappDeploy from \"@capraconsulting/webapp-deploy-lambda\"\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\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 cdk.Construct {\n  public readonly distribution: cloudfront.Distribution\n  public readonly webappBucket: s3.Bucket\n  public readonly webappOrigin: origins.S3Origin\n\n  constructor(scope: cdk.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    this.distribution = new cloudfront.Distribution(this, \"Distribution\", {\n      defaultBehavior: {\n        origin: this.webappOrigin,\n        viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\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"]}
120
+ //# 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 certificatemanager from \"@aws-cdk/aws-certificatemanager\"\nimport * as cloudfront from \"@aws-cdk/aws-cloudfront\"\nimport * as origins from \"@aws-cdk/aws-cloudfront-origins\"\nimport * as iam from \"@aws-cdk/aws-iam\"\nimport * as r53 from \"@aws-cdk/aws-route53\"\nimport * as r53t from \"@aws-cdk/aws-route53-targets\"\nimport * as s3 from \"@aws-cdk/aws-s3\"\nimport * as cdk from \"@aws-cdk/core\"\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}\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 cdk.Construct {\n  public readonly distribution: cloudfront.Distribution\n  public readonly webappBucket: s3.Bucket\n  public readonly webappOrigin: origins.S3Origin\n\n  constructor(scope: cdk.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      },\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": "1.48.0",
3
+ "version": "1.50.0",
4
4
  "description": "Experimental CDK library for Liflig",
5
5
  "repository": {
6
6
  "type": "git",
@@ -34,59 +34,58 @@
34
34
  "access": "public"
35
35
  },
36
36
  "devDependencies": {
37
- "@aws-cdk/assert": "1.122.0",
38
- "@aws-cdk/aws-certificatemanager": "1.122.0",
39
- "@aws-cdk/aws-cloudfront": "1.122.0",
40
- "@aws-cdk/aws-cloudfront-origins": "1.122.0",
41
- "@aws-cdk/aws-cloudwatch": "1.122.0",
42
- "@aws-cdk/aws-cloudwatch-actions": "1.122.0",
43
- "@aws-cdk/aws-codebuild": "1.122.0",
44
- "@aws-cdk/aws-codepipeline": "1.122.0",
45
- "@aws-cdk/aws-codepipeline-actions": "1.122.0",
46
- "@aws-cdk/aws-ecs": "1.122.0",
47
- "@aws-cdk/aws-events-targets": "1.122.0",
48
- "@aws-cdk/aws-iam": "1.122.0",
49
- "@aws-cdk/aws-lambda": "1.122.0",
50
- "@aws-cdk/aws-logs": "1.122.0",
51
- "@aws-cdk/aws-rds": "1.122.0",
52
- "@aws-cdk/aws-route53": "1.122.0",
53
- "@aws-cdk/aws-route53-targets": "1.122.0",
54
- "@aws-cdk/aws-s3": "1.122.0",
55
- "@aws-cdk/aws-ses": "1.122.0",
56
- "@aws-cdk/aws-sns": "1.122.0",
57
- "@aws-cdk/aws-stepfunctions": "1.122.0",
58
- "@aws-cdk/aws-stepfunctions-tasks": "1.122.0",
59
- "@aws-cdk/core": "1.122.0",
60
- "@aws-cdk/custom-resources": "1.122.0",
61
- "@aws-cdk/pipelines": "1.122.0",
62
- "@commitlint/cli": "12.1.4",
63
- "@commitlint/config-conventional": "12.1.4",
64
- "@types/aws-lambda": "8.10.83",
65
- "@types/jest": "27.0.2",
66
- "@types/node": "14.17.18",
67
- "@typescript-eslint/eslint-plugin": "4.31.2",
68
- "@typescript-eslint/parser": "4.31.2",
69
- "aws-cdk": "1.122.0",
70
- "eslint": "7.32.0",
37
+ "@aws-cdk/assert": "1.128.0",
38
+ "@aws-cdk/aws-certificatemanager": "1.128.0",
39
+ "@aws-cdk/aws-cloudfront": "1.128.0",
40
+ "@aws-cdk/aws-cloudfront-origins": "1.128.0",
41
+ "@aws-cdk/aws-cloudwatch": "1.128.0",
42
+ "@aws-cdk/aws-cloudwatch-actions": "1.128.0",
43
+ "@aws-cdk/aws-codebuild": "1.128.0",
44
+ "@aws-cdk/aws-codepipeline": "1.128.0",
45
+ "@aws-cdk/aws-codepipeline-actions": "1.128.0",
46
+ "@aws-cdk/aws-ecs": "1.128.0",
47
+ "@aws-cdk/aws-events-targets": "1.128.0",
48
+ "@aws-cdk/aws-iam": "1.128.0",
49
+ "@aws-cdk/aws-lambda": "1.128.0",
50
+ "@aws-cdk/aws-logs": "1.128.0",
51
+ "@aws-cdk/aws-rds": "1.128.0",
52
+ "@aws-cdk/aws-route53": "1.128.0",
53
+ "@aws-cdk/aws-route53-targets": "1.128.0",
54
+ "@aws-cdk/aws-s3": "1.128.0",
55
+ "@aws-cdk/aws-ses": "1.128.0",
56
+ "@aws-cdk/aws-sns": "1.128.0",
57
+ "@aws-cdk/aws-stepfunctions": "1.128.0",
58
+ "@aws-cdk/aws-stepfunctions-tasks": "1.128.0",
59
+ "@aws-cdk/core": "1.128.0",
60
+ "@aws-cdk/custom-resources": "1.128.0",
61
+ "@aws-cdk/pipelines": "1.128.0",
62
+ "@commitlint/cli": "13.2.1",
63
+ "@commitlint/config-conventional": "13.2.0",
64
+ "@types/aws-lambda": "8.10.85",
65
+ "@types/jest": "27.0.3",
66
+ "@types/node": "16.11.11",
67
+ "@typescript-eslint/eslint-plugin": "5.6.0",
68
+ "@typescript-eslint/parser": "5.6.0",
69
+ "aws-cdk": "1.128.0",
70
+ "eslint": "8.4.1",
71
71
  "eslint-config-prettier": "8.3.0",
72
- "eslint-plugin-deprecation": "1.2.1",
73
72
  "eslint-plugin-prettier": "4.0.0",
74
- "husky": "7.0.2",
75
- "jest": "27.2.1",
73
+ "husky": "7.0.4",
74
+ "jest": "27.3.1",
76
75
  "jest-cdk-snapshot": "1.4.2",
77
76
  "prettier": "2.4.1",
78
- "semantic-release": "18.0.0",
79
- "ts-jest": "27.0.5",
80
- "ts-node": "10.2.1",
81
- "typescript": "4.4.3"
77
+ "semantic-release": "18.0.1",
78
+ "ts-jest": "27.0.7",
79
+ "ts-node": "10.4.0",
80
+ "typescript": "4.5.2"
82
81
  },
83
82
  "dependencies": {
84
83
  "@capraconsulting/webapp-deploy-lambda": "^1.2.1",
85
- "aws-sdk": "^2.799.0",
86
- "cpy": "^8.1.0",
84
+ "aws-sdk": "^2.1012.0",
85
+ "cpy": "^8.1.2",
87
86
  "del": "^6.0.0",
88
87
  "execa": "^5.0.0",
89
- "glob": "7.1.7",
88
+ "glob": "7.2.0",
90
89
  "source-map-support": "^0.5.19"
91
90
  },
92
91
  "peerDependencies": {