@liflig/cdk 3.7.1 → 3.9.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.
@@ -26,11 +26,11 @@ export interface FargateServiceProps {
26
26
  */
27
27
  logsRetention?: logs.RetentionDays;
28
28
  /**
29
- * @default 15 seconds
29
+ * @default AwsLogDriverMode.BLOCKING
30
30
  */
31
31
  logDriverMode?: ecs.AwsLogDriverMode;
32
32
  /**
33
- * @default AwsLogDriverMode.BLOCKING
33
+ * @default 15 seconds
34
34
  */
35
35
  deregistrationDelay?: cdk.Duration;
36
36
  /**
@@ -113,4 +113,4 @@ export class FargateService extends constructs.Construct {
113
113
  }
114
114
  }
115
115
  }
116
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate-service.js","sourceRoot":"","sources":["../../src/ecs/fargate-service.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport { Duration } from \"aws-cdk-lib\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport * as ecs from \"aws-cdk-lib/aws-ecs\"\nimport { CfnService } from \"aws-cdk-lib/aws-ecs\"\nimport * as elb from \"aws-cdk-lib/aws-elasticloadbalancingv2\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as constructs from \"constructs\"\nimport { ConfigureParameters } from \"../configure-parameters\"\nimport { Parameter } from \"../configure-parameters/configure-parameters\"\n\nexport interface FargateServiceProps {\n  serviceName: string\n  vpc: ec2.IVpc\n  cluster: ecs.ICluster\n  desiredCount: number\n  ecsImage: ecs.ContainerImage\n  portMappings?: ecs.PortMapping[]\n  containerHealthCheck?: ecs.HealthCheck\n  /**\n   * @default 256\n   */\n  cpu?: number\n  /**\n   * @default 512\n   */\n  memoryLimitMiB?: number\n  /**\n   * @default 2 weeks\n   */\n  logsRetention?: logs.RetentionDays\n  /**\n   * @default 15 seconds\n   */\n  logDriverMode?: ecs.AwsLogDriverMode\n  /**\n   * @default AwsLogDriverMode.BLOCKING\n   */\n  deregistrationDelay?: cdk.Duration\n  /**\n   * @default 8080\n   */\n  containerPort?: number\n  /**\n   * @default 60 seconds\n   */\n  healthCheckGracePeriod?: cdk.Duration\n  /**\n   * Use this as workaround when adding the service to a load balancer after\n   * it has been created. For avoiding 'Health check grace period is only valid for services configured to use load balancers'\n   * Link to GitHub issue: https://github.com/aws/aws-cdk/issues/19842\n   */\n  skipHealthCheckGracePeriod?: boolean\n  parameters?: Parameter[]\n  overrideFargateServiceProps?: Partial<ecs.FargateServiceProps>\n  overrideHealthCheck?: Partial<elb.HealthCheck>\n  overrideTargetGroupProps?: Partial<elb.ApplicationTargetGroupProps>\n  overrideContainerProps?: Partial<ecs.ContainerDefinitionOptions>\n  secrets?: Record<string, ecs.Secret>\n  environment?: Record<string, string>\n  /**\n   * @default false\n   */\n  skipTargetGroup?: boolean\n  /**\n   * @default false\n   */\n  enableCircuitBreaker?: boolean\n}\n\nexport class FargateService extends constructs.Construct {\n  public readonly fargateService: ecs.FargateService\n  public readonly securityGroup: ec2.SecurityGroup\n  public readonly taskDefinition: ecs.TaskDefinition\n  public readonly targetGroup: elb.ApplicationTargetGroup | undefined\n  public readonly logGroup: logs.LogGroup\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: FargateServiceProps,\n  ) {\n    super(scope, id)\n\n    /**\n     * Set this flag to disable this stack creating a completely new service and attempting replace when enabling circuit breakers\n     * Mitigating the deployment error: 'a service with the name <...> already exists'\n     * See: github.com/aws/aws-cdk/pull/22467\n     */\n    this.node.setContext(\n      \"@aws-cdk/aws-ecs:disableExplicitDeploymentControllerForCircuitBreaker\",\n      true,\n    )\n\n    const parameters = new ConfigureParameters(this, {\n      ssmPrefix: `/liflig-cdk/${cdk.Stack.of(this).stackName}/${\n        this.node.addr\n      }/parameters`,\n      parameters: props.parameters ?? [],\n    })\n\n    this.logGroup = new logs.LogGroup(this, \"LogGroup\", {\n      retention: props.logsRetention ?? logs.RetentionDays.TWO_WEEKS,\n    })\n\n    this.securityGroup = new ec2.SecurityGroup(this, \"SecurityGroup\", {\n      vpc: props.vpc,\n    })\n\n    this.taskDefinition = new ecs.FargateTaskDefinition(\n      this,\n      \"TaskDefinition\",\n      {\n        cpu: props.cpu ?? 256,\n        memoryLimitMiB: props.memoryLimitMiB ?? 512,\n      },\n    )\n\n    parameters.grantRead(this.taskDefinition.taskRole)\n\n    const port = props.containerPort ?? 8080\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    const container = this.taskDefinition.addContainer(\"Container\", {\n      logging: ecs.LogDriver.awsLogs({\n        logGroup: this.logGroup,\n        streamPrefix: \"ecs\",\n        datetimeFormat: \"%Y-%m-%dT%H:%M:%S\",\n        mode: props.logDriverMode ?? ecs.AwsLogDriverMode.BLOCKING,\n      }),\n      image: props.ecsImage,\n      secrets: props.secrets,\n      healthCheck: props.containerHealthCheck,\n      portMappings: props.portMappings ?? [\n        {\n          containerPort: port,\n          hostPort: port,\n        },\n      ],\n      environment: {\n        SSM_PREFIX: parameters.ssmPrefix,\n        // Not read by the application, only used to help with redeployments.\n        PARAMS_HASH: parameters.hashValue,\n        ...(props.environment ?? {}),\n      },\n      linuxParameters: new ecs.LinuxParameters(this, \"LinuxParameters\", {\n        initProcessEnabled: true,\n      }),\n      ...props.overrideContainerProps,\n    })\n\n    const enableCircuitBreaker = props.enableCircuitBreaker ?? false\n\n    this.fargateService = new ecs.FargateService(this, \"Service\", {\n      serviceName: props.serviceName,\n      vpcSubnets: {\n        subnetType: ec2.SubnetType.PUBLIC,\n      },\n      taskDefinition: this.taskDefinition,\n      cluster: props.cluster,\n      minHealthyPercent: 100,\n      healthCheckGracePeriod: props.healthCheckGracePeriod,\n      desiredCount: props.desiredCount,\n      assignPublicIp: true,\n      securityGroups: [this.securityGroup],\n      platformVersion: ecs.FargatePlatformVersion.VERSION1_4,\n      enableExecuteCommand: true,\n      circuitBreaker: enableCircuitBreaker\n        ? {\n            enable: true,\n            rollback: true,\n          }\n        : undefined,\n      ...props.overrideFargateServiceProps,\n    })\n\n    if (props.skipHealthCheckGracePeriod) {\n      ;(\n        this.fargateService.node.defaultChild as CfnService\n      ).healthCheckGracePeriodSeconds = undefined\n    }\n\n    for (const param of parameters.parameters) {\n      this.fargateService.node.addDependency(param)\n    }\n\n    if (!props.skipTargetGroup) {\n      this.targetGroup = new elb.ApplicationTargetGroup(this, \"TargetGroup\", {\n        protocol: elb.ApplicationProtocol.HTTP,\n        port: port,\n        vpc: props.vpc,\n        targetType: elb.TargetType.IP,\n        targets: [this.fargateService],\n        deregistrationDelay:\n          props.deregistrationDelay ?? cdk.Duration.seconds(15),\n        ...props.overrideTargetGroupProps,\n      })\n\n      this.targetGroup.configureHealthCheck({\n        interval: Duration.seconds(10),\n        path: \"/health\",\n        healthyThresholdCount: 2,\n        ...props.overrideHealthCheck,\n      })\n    }\n  }\n}\n"]}
116
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate-service.js","sourceRoot":"","sources":["../../src/ecs/fargate-service.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\"\nimport { Duration } from \"aws-cdk-lib\"\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\"\nimport * as ecs from \"aws-cdk-lib/aws-ecs\"\nimport { CfnService } from \"aws-cdk-lib/aws-ecs\"\nimport * as elb from \"aws-cdk-lib/aws-elasticloadbalancingv2\"\nimport * as logs from \"aws-cdk-lib/aws-logs\"\nimport * as constructs from \"constructs\"\nimport { ConfigureParameters } from \"../configure-parameters\"\nimport { Parameter } from \"../configure-parameters/configure-parameters\"\n\nexport interface FargateServiceProps {\n  serviceName: string\n  vpc: ec2.IVpc\n  cluster: ecs.ICluster\n  desiredCount: number\n  ecsImage: ecs.ContainerImage\n  portMappings?: ecs.PortMapping[]\n  containerHealthCheck?: ecs.HealthCheck\n  /**\n   * @default 256\n   */\n  cpu?: number\n  /**\n   * @default 512\n   */\n  memoryLimitMiB?: number\n  /**\n   * @default 2 weeks\n   */\n  logsRetention?: logs.RetentionDays\n  /**\n   * @default AwsLogDriverMode.BLOCKING\n   */\n  logDriverMode?: ecs.AwsLogDriverMode\n  /**\n   * @default 15 seconds\n   */\n  deregistrationDelay?: cdk.Duration\n  /**\n   * @default 8080\n   */\n  containerPort?: number\n  /**\n   * @default 60 seconds\n   */\n  healthCheckGracePeriod?: cdk.Duration\n  /**\n   * Use this as workaround when adding the service to a load balancer after\n   * it has been created. For avoiding 'Health check grace period is only valid for services configured to use load balancers'\n   * Link to GitHub issue: https://github.com/aws/aws-cdk/issues/19842\n   */\n  skipHealthCheckGracePeriod?: boolean\n  parameters?: Parameter[]\n  overrideFargateServiceProps?: Partial<ecs.FargateServiceProps>\n  overrideHealthCheck?: Partial<elb.HealthCheck>\n  overrideTargetGroupProps?: Partial<elb.ApplicationTargetGroupProps>\n  overrideContainerProps?: Partial<ecs.ContainerDefinitionOptions>\n  secrets?: Record<string, ecs.Secret>\n  environment?: Record<string, string>\n  /**\n   * @default false\n   */\n  skipTargetGroup?: boolean\n  /**\n   * @default false\n   */\n  enableCircuitBreaker?: boolean\n}\n\nexport class FargateService extends constructs.Construct {\n  public readonly fargateService: ecs.FargateService\n  public readonly securityGroup: ec2.SecurityGroup\n  public readonly taskDefinition: ecs.TaskDefinition\n  public readonly targetGroup: elb.ApplicationTargetGroup | undefined\n  public readonly logGroup: logs.LogGroup\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: FargateServiceProps,\n  ) {\n    super(scope, id)\n\n    /**\n     * Set this flag to disable this stack creating a completely new service and attempting replace when enabling circuit breakers\n     * Mitigating the deployment error: 'a service with the name <...> already exists'\n     * See: github.com/aws/aws-cdk/pull/22467\n     */\n    this.node.setContext(\n      \"@aws-cdk/aws-ecs:disableExplicitDeploymentControllerForCircuitBreaker\",\n      true,\n    )\n\n    const parameters = new ConfigureParameters(this, {\n      ssmPrefix: `/liflig-cdk/${cdk.Stack.of(this).stackName}/${\n        this.node.addr\n      }/parameters`,\n      parameters: props.parameters ?? [],\n    })\n\n    this.logGroup = new logs.LogGroup(this, \"LogGroup\", {\n      retention: props.logsRetention ?? logs.RetentionDays.TWO_WEEKS,\n    })\n\n    this.securityGroup = new ec2.SecurityGroup(this, \"SecurityGroup\", {\n      vpc: props.vpc,\n    })\n\n    this.taskDefinition = new ecs.FargateTaskDefinition(\n      this,\n      \"TaskDefinition\",\n      {\n        cpu: props.cpu ?? 256,\n        memoryLimitMiB: props.memoryLimitMiB ?? 512,\n      },\n    )\n\n    parameters.grantRead(this.taskDefinition.taskRole)\n\n    const port = props.containerPort ?? 8080\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    const container = this.taskDefinition.addContainer(\"Container\", {\n      logging: ecs.LogDriver.awsLogs({\n        logGroup: this.logGroup,\n        streamPrefix: \"ecs\",\n        datetimeFormat: \"%Y-%m-%dT%H:%M:%S\",\n        mode: props.logDriverMode ?? ecs.AwsLogDriverMode.BLOCKING,\n      }),\n      image: props.ecsImage,\n      secrets: props.secrets,\n      healthCheck: props.containerHealthCheck,\n      portMappings: props.portMappings ?? [\n        {\n          containerPort: port,\n          hostPort: port,\n        },\n      ],\n      environment: {\n        SSM_PREFIX: parameters.ssmPrefix,\n        // Not read by the application, only used to help with redeployments.\n        PARAMS_HASH: parameters.hashValue,\n        ...(props.environment ?? {}),\n      },\n      linuxParameters: new ecs.LinuxParameters(this, \"LinuxParameters\", {\n        initProcessEnabled: true,\n      }),\n      ...props.overrideContainerProps,\n    })\n\n    const enableCircuitBreaker = props.enableCircuitBreaker ?? false\n\n    this.fargateService = new ecs.FargateService(this, \"Service\", {\n      serviceName: props.serviceName,\n      vpcSubnets: {\n        subnetType: ec2.SubnetType.PUBLIC,\n      },\n      taskDefinition: this.taskDefinition,\n      cluster: props.cluster,\n      minHealthyPercent: 100,\n      healthCheckGracePeriod: props.healthCheckGracePeriod,\n      desiredCount: props.desiredCount,\n      assignPublicIp: true,\n      securityGroups: [this.securityGroup],\n      platformVersion: ecs.FargatePlatformVersion.VERSION1_4,\n      enableExecuteCommand: true,\n      circuitBreaker: enableCircuitBreaker\n        ? {\n            enable: true,\n            rollback: true,\n          }\n        : undefined,\n      ...props.overrideFargateServiceProps,\n    })\n\n    if (props.skipHealthCheckGracePeriod) {\n      ;(\n        this.fargateService.node.defaultChild as CfnService\n      ).healthCheckGracePeriodSeconds = undefined\n    }\n\n    for (const param of parameters.parameters) {\n      this.fargateService.node.addDependency(param)\n    }\n\n    if (!props.skipTargetGroup) {\n      this.targetGroup = new elb.ApplicationTargetGroup(this, \"TargetGroup\", {\n        protocol: elb.ApplicationProtocol.HTTP,\n        port: port,\n        vpc: props.vpc,\n        targetType: elb.TargetType.IP,\n        targets: [this.fargateService],\n        deregistrationDelay:\n          props.deregistrationDelay ?? cdk.Duration.seconds(15),\n        ...props.overrideTargetGroupProps,\n      })\n\n      this.targetGroup.configureHealthCheck({\n        interval: Duration.seconds(10),\n        path: \"/health\",\n        healthyThresholdCount: 2,\n        ...props.overrideHealthCheck,\n      })\n    }\n  }\n}\n"]}
@@ -7,6 +7,7 @@ export interface LoadBalancerProps {
7
7
  certificates: certificatemanager.ICertificate[];
8
8
  vpc: ec2.IVpc;
9
9
  overrideLoadBalancerProps?: Partial<elb.ApplicationLoadBalancerProps>;
10
+ overrideHttpsListenerProps?: Partial<elb.BaseApplicationListenerProps>;
10
11
  }
11
12
  export declare class LoadBalancer extends constructs.Construct {
12
13
  readonly loadBalancer: elb.ApplicationLoadBalancer;
@@ -43,6 +43,7 @@ export class LoadBalancer extends constructs.Construct {
43
43
  port: 443,
44
44
  certificates: props.certificates,
45
45
  defaultTargetGroups: [defaultTargetGroup],
46
+ ...props.overrideHttpsListenerProps,
46
47
  });
47
48
  this.accessLogsBucket = new s3.Bucket(this, "AccessLogsBucket", {
48
49
  encryption: s3.BucketEncryption.S3_MANAGED,
@@ -56,4 +57,4 @@ export class LoadBalancer extends constructs.Construct {
56
57
  this.loadBalancer.logAccessLogs(this.accessLogsBucket);
57
58
  }
58
59
  }
59
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9hZC1iYWxhbmNlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9sb2FkLWJhbGFuY2VyL2xvYWQtYmFsYW5jZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLFVBQVUsTUFBTSxZQUFZLENBQUE7QUFFeEMsT0FBTyxLQUFLLEdBQUcsTUFBTSxxQkFBcUIsQ0FBQTtBQUMxQyxPQUFPLEtBQUssR0FBRyxNQUFNLHdDQUF3QyxDQUFBO0FBQzdELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSx3Q0FBd0MsQ0FBQTtBQUN2RSxPQUFPLEtBQUssRUFBRSxNQUFNLG9CQUFvQixDQUFBO0FBQ3hDLE9BQU8sS0FBSyxHQUFHLE1BQU0sYUFBYSxDQUFBO0FBUWxDLE1BQU0sT0FBTyxZQUFhLFNBQVEsVUFBVSxDQUFDLFNBQVM7SUFDcEMsWUFBWSxDQUE2QjtJQUN6QyxhQUFhLENBQXlCO0lBQ3RDLGdCQUFnQixDQUFXO0lBRTNDLFlBQ0UsS0FBMkIsRUFDM0IsRUFBVSxFQUNWLEtBQXdCO1FBRXhCLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsSUFBSSxDQUFDLFlBQVksR0FBRyxJQUFJLEdBQUcsQ0FBQyx1QkFBdUIsQ0FBQyxJQUFJLEVBQUUsY0FBYyxFQUFFO1lBQ3hFLEdBQUcsRUFBRSxLQUFLLENBQUMsR0FBRztZQUNkLGNBQWMsRUFBRSxJQUFJO1lBQ3BCLFVBQVUsRUFBRSxLQUFLLENBQUMsR0FBRyxDQUFDLGFBQWEsQ0FBQztnQkFDbEMsVUFBVSxFQUFFLEdBQUcsQ0FBQyxVQUFVLENBQUMsTUFBTTthQUNsQyxDQUFDO1lBQ0YsR0FBRyxLQUFLLENBQUMseUJBQXlCO1NBQ25DLENBQUMsQ0FBQTtRQUVGLElBQUksQ0FBQyxZQUFZO2FBQ2QsV0FBVyxDQUFDLGNBQWMsRUFBRTtZQUMzQixJQUFJLEVBQUUsRUFBRTtTQUNULENBQUM7YUFDRCxTQUFTLENBQUMsZUFBZSxFQUFFO1lBQzFCLE1BQU0sRUFBRSxjQUFjLENBQUMsUUFBUSxDQUFDO2dCQUM5QixJQUFJLEVBQUUsS0FBSztnQkFDWCxRQUFRLEVBQUUsT0FBTztnQkFDakIsU0FBUyxFQUFFLElBQUk7YUFDaEIsQ0FBQztTQUNILENBQUMsQ0FBQTtRQUVKLG9EQUFvRDtRQUNwRCw0REFBNEQ7UUFDNUQsTUFBTSxrQkFBa0IsR0FBRyxJQUFJLEdBQUcsQ0FBQyxzQkFBc0IsQ0FDdkQsSUFBSSxFQUNKLG9CQUFvQixFQUNwQjtZQUNFLFFBQVEsRUFBRSxHQUFHLENBQUMsbUJBQW1CLENBQUMsSUFBSTtZQUN0QyxJQUFJLEVBQUUsRUFBRTtZQUNSLEdBQUcsRUFBRSxLQUFLLENBQUMsR0FBRztZQUNkLFVBQVUsRUFBRSxHQUFHLENBQUMsVUFBVSxDQUFDLFFBQVE7U0FDcEMsQ0FDRixDQUFBO1FBRUQsSUFBSSxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFdBQVcsQ0FBQyxlQUFlLEVBQUU7WUFDbEUsU0FBUyxFQUFFLEdBQUcsQ0FBQyxTQUFTLENBQUMsS0FBSztZQUM5QixRQUFRLEVBQUUsR0FBRyxDQUFDLG1CQUFtQixDQUFDLEtBQUs7WUFDdkMsSUFBSSxFQUFFLEdBQUc7WUFDVCxZQUFZLEVBQUUsS0FBSyxDQUFDLFlBQVk7WUFDaEMsbUJBQW1CLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQztTQUMxQyxDQUFDLENBQUE7UUFFRixJQUFJLENBQUMsZ0JBQWdCLEdBQUcsSUFBSSxFQUFFLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxrQkFBa0IsRUFBRTtZQUM5RCxVQUFVLEVBQUUsRUFBRSxDQUFDLGdCQUFnQixDQUFDLFVBQVU7WUFDMUMsaUJBQWlCLEVBQUUsRUFBRSxDQUFDLGlCQUFpQixDQUFDLFNBQVM7WUFDakQsY0FBYyxFQUFFO2dCQUNkO29CQUNFLFVBQVUsRUFBRSxHQUFHLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUM7aUJBQ2xDO2FBQ0Y7U0FDRixDQUFDLENBQUE7UUFFRixJQUFJLENBQUMsWUFBWSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsQ0FBQTtJQUN4RCxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBjb25zdHJ1Y3RzIGZyb20gXCJjb25zdHJ1Y3RzXCJcbmltcG9ydCAqIGFzIGNlcnRpZmljYXRlbWFuYWdlciBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNlcnRpZmljYXRlbWFuYWdlclwiXG5pbXBvcnQgKiBhcyBlYzIgZnJvbSBcImF3cy1jZGstbGliL2F3cy1lYzJcIlxuaW1wb3J0ICogYXMgZWxiIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtZWxhc3RpY2xvYWRiYWxhbmNpbmd2MlwiXG5pbXBvcnQgeyBMaXN0ZW5lckFjdGlvbiB9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtZWxhc3RpY2xvYWRiYWxhbmNpbmd2MlwiXG5pbXBvcnQgKiBhcyBzMyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXMzXCJcbmltcG9ydCAqIGFzIGNkayBmcm9tIFwiYXdzLWNkay1saWJcIlxuXG5leHBvcnQgaW50ZXJmYWNlIExvYWRCYWxhbmNlclByb3BzIHtcbiAgY2VydGlmaWNhdGVzOiBjZXJ0aWZpY2F0ZW1hbmFnZXIuSUNlcnRpZmljYXRlW11cbiAgdnBjOiBlYzIuSVZwY1xuICBvdmVycmlkZUxvYWRCYWxhbmNlclByb3BzPzogUGFydGlhbDxlbGIuQXBwbGljYXRpb25Mb2FkQmFsYW5jZXJQcm9wcz5cbn1cblxuZXhwb3J0IGNsYXNzIExvYWRCYWxhbmNlciBleHRlbmRzIGNvbnN0cnVjdHMuQ29uc3RydWN0IHtcbiAgcHVibGljIHJlYWRvbmx5IGxvYWRCYWxhbmNlcjogZWxiLkFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyXG4gIHB1YmxpYyByZWFkb25seSBodHRwc0xpc3RlbmVyOiBlbGIuQXBwbGljYXRpb25MaXN0ZW5lclxuICBwdWJsaWMgcmVhZG9ubHkgYWNjZXNzTG9nc0J1Y2tldDogczMuQnVja2V0XG5cbiAgY29uc3RydWN0b3IoXG4gICAgc2NvcGU6IGNvbnN0cnVjdHMuQ29uc3RydWN0LFxuICAgIGlkOiBzdHJpbmcsXG4gICAgcHJvcHM6IExvYWRCYWxhbmNlclByb3BzLFxuICApIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICB0aGlzLmxvYWRCYWxhbmNlciA9IG5ldyBlbGIuQXBwbGljYXRpb25Mb2FkQmFsYW5jZXIodGhpcywgXCJMb2FkQmFsYW5jZXJcIiwge1xuICAgICAgdnBjOiBwcm9wcy52cGMsXG4gICAgICBpbnRlcm5ldEZhY2luZzogdHJ1ZSxcbiAgICAgIHZwY1N1Ym5ldHM6IHByb3BzLnZwYy5zZWxlY3RTdWJuZXRzKHtcbiAgICAgICAgc3VibmV0VHlwZTogZWMyLlN1Ym5ldFR5cGUuUFVCTElDLFxuICAgICAgfSksXG4gICAgICAuLi5wcm9wcy5vdmVycmlkZUxvYWRCYWxhbmNlclByb3BzLFxuICAgIH0pXG5cbiAgICB0aGlzLmxvYWRCYWxhbmNlclxuICAgICAgLmFkZExpc3RlbmVyKFwiSHR0cExpc3RlbmVyXCIsIHtcbiAgICAgICAgcG9ydDogODAsXG4gICAgICB9KVxuICAgICAgLmFkZEFjdGlvbihcIkh0dHBzUmVkaXJlY3RcIiwge1xuICAgICAgICBhY3Rpb246IExpc3RlbmVyQWN0aW9uLnJlZGlyZWN0KHtcbiAgICAgICAgICBwb3J0OiBcIjQ0M1wiLFxuICAgICAgICAgIHByb3RvY29sOiBcIkhUVFBTXCIsXG4gICAgICAgICAgcGVybWFuZW50OiB0cnVlLFxuICAgICAgICB9KSxcbiAgICAgIH0pXG5cbiAgICAvLyBUaGUgTG9hZCBCYWxhbmNlciByZXF1aXJlIGEgZGVmYXVsdCB0YXJnZXQgZ3JvdXAuXG4gICAgLy8gV2Ugd2lsbCBub3QgY29ubmVjdCBhbnl0aGluZyB0byB0aGUgZGVmYXVsdCB0YXJnZXQgZ3JvdXAuXG4gICAgY29uc3QgZGVmYXVsdFRhcmdldEdyb3VwID0gbmV3IGVsYi5BcHBsaWNhdGlvblRhcmdldEdyb3VwKFxuICAgICAgdGhpcyxcbiAgICAgIFwiRGVmYXVsdFRhcmdldEdyb3VwXCIsXG4gICAgICB7XG4gICAgICAgIHByb3RvY29sOiBlbGIuQXBwbGljYXRpb25Qcm90b2NvbC5IVFRQLFxuICAgICAgICBwb3J0OiA4MCxcbiAgICAgICAgdnBjOiBwcm9wcy52cGMsXG4gICAgICAgIHRhcmdldFR5cGU6IGVsYi5UYXJnZXRUeXBlLklOU1RBTkNFLFxuICAgICAgfSxcbiAgICApXG5cbiAgICB0aGlzLmh0dHBzTGlzdGVuZXIgPSB0aGlzLmxvYWRCYWxhbmNlci5hZGRMaXN0ZW5lcihcIkh0dHBzTGlzdGVuZXJcIiwge1xuICAgICAgc3NsUG9saWN5OiBlbGIuU3NsUG9saWN5LlRMUzEyLFxuICAgICAgcHJvdG9jb2w6IGVsYi5BcHBsaWNhdGlvblByb3RvY29sLkhUVFBTLFxuICAgICAgcG9ydDogNDQzLFxuICAgICAgY2VydGlmaWNhdGVzOiBwcm9wcy5jZXJ0aWZpY2F0ZXMsXG4gICAgICBkZWZhdWx0VGFyZ2V0R3JvdXBzOiBbZGVmYXVsdFRhcmdldEdyb3VwXSxcbiAgICB9KVxuXG4gICAgdGhpcy5hY2Nlc3NMb2dzQnVja2V0ID0gbmV3IHMzLkJ1Y2tldCh0aGlzLCBcIkFjY2Vzc0xvZ3NCdWNrZXRcIiwge1xuICAgICAgZW5jcnlwdGlvbjogczMuQnVja2V0RW5jcnlwdGlvbi5TM19NQU5BR0VELFxuICAgICAgYmxvY2tQdWJsaWNBY2Nlc3M6IHMzLkJsb2NrUHVibGljQWNjZXNzLkJMT0NLX0FMTCxcbiAgICAgIGxpZmVjeWNsZVJ1bGVzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICBleHBpcmF0aW9uOiBjZGsuRHVyYXRpb24uZGF5cygzMCksXG4gICAgICAgIH0sXG4gICAgICBdLFxuICAgIH0pXG5cbiAgICB0aGlzLmxvYWRCYWxhbmNlci5sb2dBY2Nlc3NMb2dzKHRoaXMuYWNjZXNzTG9nc0J1Y2tldClcbiAgfVxufVxuIl19
60
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9hZC1iYWxhbmNlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9sb2FkLWJhbGFuY2VyL2xvYWQtYmFsYW5jZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLFVBQVUsTUFBTSxZQUFZLENBQUE7QUFFeEMsT0FBTyxLQUFLLEdBQUcsTUFBTSxxQkFBcUIsQ0FBQTtBQUMxQyxPQUFPLEtBQUssR0FBRyxNQUFNLHdDQUF3QyxDQUFBO0FBQzdELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSx3Q0FBd0MsQ0FBQTtBQUN2RSxPQUFPLEtBQUssRUFBRSxNQUFNLG9CQUFvQixDQUFBO0FBQ3hDLE9BQU8sS0FBSyxHQUFHLE1BQU0sYUFBYSxDQUFBO0FBU2xDLE1BQU0sT0FBTyxZQUFhLFNBQVEsVUFBVSxDQUFDLFNBQVM7SUFDcEMsWUFBWSxDQUE2QjtJQUN6QyxhQUFhLENBQXlCO0lBQ3RDLGdCQUFnQixDQUFXO0lBRTNDLFlBQ0UsS0FBMkIsRUFDM0IsRUFBVSxFQUNWLEtBQXdCO1FBRXhCLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUE7UUFFaEIsSUFBSSxDQUFDLFlBQVksR0FBRyxJQUFJLEdBQUcsQ0FBQyx1QkFBdUIsQ0FBQyxJQUFJLEVBQUUsY0FBYyxFQUFFO1lBQ3hFLEdBQUcsRUFBRSxLQUFLLENBQUMsR0FBRztZQUNkLGNBQWMsRUFBRSxJQUFJO1lBQ3BCLFVBQVUsRUFBRSxLQUFLLENBQUMsR0FBRyxDQUFDLGFBQWEsQ0FBQztnQkFDbEMsVUFBVSxFQUFFLEdBQUcsQ0FBQyxVQUFVLENBQUMsTUFBTTthQUNsQyxDQUFDO1lBQ0YsR0FBRyxLQUFLLENBQUMseUJBQXlCO1NBQ25DLENBQUMsQ0FBQTtRQUVGLElBQUksQ0FBQyxZQUFZO2FBQ2QsV0FBVyxDQUFDLGNBQWMsRUFBRTtZQUMzQixJQUFJLEVBQUUsRUFBRTtTQUNULENBQUM7YUFDRCxTQUFTLENBQUMsZUFBZSxFQUFFO1lBQzFCLE1BQU0sRUFBRSxjQUFjLENBQUMsUUFBUSxDQUFDO2dCQUM5QixJQUFJLEVBQUUsS0FBSztnQkFDWCxRQUFRLEVBQUUsT0FBTztnQkFDakIsU0FBUyxFQUFFLElBQUk7YUFDaEIsQ0FBQztTQUNILENBQUMsQ0FBQTtRQUVKLG9EQUFvRDtRQUNwRCw0REFBNEQ7UUFDNUQsTUFBTSxrQkFBa0IsR0FBRyxJQUFJLEdBQUcsQ0FBQyxzQkFBc0IsQ0FDdkQsSUFBSSxFQUNKLG9CQUFvQixFQUNwQjtZQUNFLFFBQVEsRUFBRSxHQUFHLENBQUMsbUJBQW1CLENBQUMsSUFBSTtZQUN0QyxJQUFJLEVBQUUsRUFBRTtZQUNSLEdBQUcsRUFBRSxLQUFLLENBQUMsR0FBRztZQUNkLFVBQVUsRUFBRSxHQUFHLENBQUMsVUFBVSxDQUFDLFFBQVE7U0FDcEMsQ0FDRixDQUFBO1FBRUQsSUFBSSxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLFdBQVcsQ0FBQyxlQUFlLEVBQUU7WUFDbEUsU0FBUyxFQUFFLEdBQUcsQ0FBQyxTQUFTLENBQUMsS0FBSztZQUM5QixRQUFRLEVBQUUsR0FBRyxDQUFDLG1CQUFtQixDQUFDLEtBQUs7WUFDdkMsSUFBSSxFQUFFLEdBQUc7WUFDVCxZQUFZLEVBQUUsS0FBSyxDQUFDLFlBQVk7WUFDaEMsbUJBQW1CLEVBQUUsQ0FBQyxrQkFBa0IsQ0FBQztZQUN6QyxHQUFHLEtBQUssQ0FBQywwQkFBMEI7U0FDcEMsQ0FBQyxDQUFBO1FBRUYsSUFBSSxDQUFDLGdCQUFnQixHQUFHLElBQUksRUFBRSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsa0JBQWtCLEVBQUU7WUFDOUQsVUFBVSxFQUFFLEVBQUUsQ0FBQyxnQkFBZ0IsQ0FBQyxVQUFVO1lBQzFDLGlCQUFpQixFQUFFLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQyxTQUFTO1lBQ2pELGNBQWMsRUFBRTtnQkFDZDtvQkFDRSxVQUFVLEVBQUUsR0FBRyxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO2lCQUNsQzthQUNGO1NBQ0YsQ0FBQyxDQUFBO1FBRUYsSUFBSSxDQUFDLFlBQVksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLGdCQUFnQixDQUFDLENBQUE7SUFDeEQsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY29uc3RydWN0cyBmcm9tIFwiY29uc3RydWN0c1wiXG5pbXBvcnQgKiBhcyBjZXJ0aWZpY2F0ZW1hbmFnZXIgZnJvbSBcImF3cy1jZGstbGliL2F3cy1jZXJ0aWZpY2F0ZW1hbmFnZXJcIlxuaW1wb3J0ICogYXMgZWMyIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtZWMyXCJcbmltcG9ydCAqIGFzIGVsYiBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWVsYXN0aWNsb2FkYmFsYW5jaW5ndjJcIlxuaW1wb3J0IHsgTGlzdGVuZXJBY3Rpb24gfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWVsYXN0aWNsb2FkYmFsYW5jaW5ndjJcIlxuaW1wb3J0ICogYXMgczMgZnJvbSBcImF3cy1jZGstbGliL2F3cy1zM1wiXG5pbXBvcnQgKiBhcyBjZGsgZnJvbSBcImF3cy1jZGstbGliXCJcblxuZXhwb3J0IGludGVyZmFjZSBMb2FkQmFsYW5jZXJQcm9wcyB7XG4gIGNlcnRpZmljYXRlczogY2VydGlmaWNhdGVtYW5hZ2VyLklDZXJ0aWZpY2F0ZVtdXG4gIHZwYzogZWMyLklWcGNcbiAgb3ZlcnJpZGVMb2FkQmFsYW5jZXJQcm9wcz86IFBhcnRpYWw8ZWxiLkFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyUHJvcHM+XG4gIG92ZXJyaWRlSHR0cHNMaXN0ZW5lclByb3BzPzogUGFydGlhbDxlbGIuQmFzZUFwcGxpY2F0aW9uTGlzdGVuZXJQcm9wcz5cbn1cblxuZXhwb3J0IGNsYXNzIExvYWRCYWxhbmNlciBleHRlbmRzIGNvbnN0cnVjdHMuQ29uc3RydWN0IHtcbiAgcHVibGljIHJlYWRvbmx5IGxvYWRCYWxhbmNlcjogZWxiLkFwcGxpY2F0aW9uTG9hZEJhbGFuY2VyXG4gIHB1YmxpYyByZWFkb25seSBodHRwc0xpc3RlbmVyOiBlbGIuQXBwbGljYXRpb25MaXN0ZW5lclxuICBwdWJsaWMgcmVhZG9ubHkgYWNjZXNzTG9nc0J1Y2tldDogczMuQnVja2V0XG5cbiAgY29uc3RydWN0b3IoXG4gICAgc2NvcGU6IGNvbnN0cnVjdHMuQ29uc3RydWN0LFxuICAgIGlkOiBzdHJpbmcsXG4gICAgcHJvcHM6IExvYWRCYWxhbmNlclByb3BzLFxuICApIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICB0aGlzLmxvYWRCYWxhbmNlciA9IG5ldyBlbGIuQXBwbGljYXRpb25Mb2FkQmFsYW5jZXIodGhpcywgXCJMb2FkQmFsYW5jZXJcIiwge1xuICAgICAgdnBjOiBwcm9wcy52cGMsXG4gICAgICBpbnRlcm5ldEZhY2luZzogdHJ1ZSxcbiAgICAgIHZwY1N1Ym5ldHM6IHByb3BzLnZwYy5zZWxlY3RTdWJuZXRzKHtcbiAgICAgICAgc3VibmV0VHlwZTogZWMyLlN1Ym5ldFR5cGUuUFVCTElDLFxuICAgICAgfSksXG4gICAgICAuLi5wcm9wcy5vdmVycmlkZUxvYWRCYWxhbmNlclByb3BzLFxuICAgIH0pXG5cbiAgICB0aGlzLmxvYWRCYWxhbmNlclxuICAgICAgLmFkZExpc3RlbmVyKFwiSHR0cExpc3RlbmVyXCIsIHtcbiAgICAgICAgcG9ydDogODAsXG4gICAgICB9KVxuICAgICAgLmFkZEFjdGlvbihcIkh0dHBzUmVkaXJlY3RcIiwge1xuICAgICAgICBhY3Rpb246IExpc3RlbmVyQWN0aW9uLnJlZGlyZWN0KHtcbiAgICAgICAgICBwb3J0OiBcIjQ0M1wiLFxuICAgICAgICAgIHByb3RvY29sOiBcIkhUVFBTXCIsXG4gICAgICAgICAgcGVybWFuZW50OiB0cnVlLFxuICAgICAgICB9KSxcbiAgICAgIH0pXG5cbiAgICAvLyBUaGUgTG9hZCBCYWxhbmNlciByZXF1aXJlIGEgZGVmYXVsdCB0YXJnZXQgZ3JvdXAuXG4gICAgLy8gV2Ugd2lsbCBub3QgY29ubmVjdCBhbnl0aGluZyB0byB0aGUgZGVmYXVsdCB0YXJnZXQgZ3JvdXAuXG4gICAgY29uc3QgZGVmYXVsdFRhcmdldEdyb3VwID0gbmV3IGVsYi5BcHBsaWNhdGlvblRhcmdldEdyb3VwKFxuICAgICAgdGhpcyxcbiAgICAgIFwiRGVmYXVsdFRhcmdldEdyb3VwXCIsXG4gICAgICB7XG4gICAgICAgIHByb3RvY29sOiBlbGIuQXBwbGljYXRpb25Qcm90b2NvbC5IVFRQLFxuICAgICAgICBwb3J0OiA4MCxcbiAgICAgICAgdnBjOiBwcm9wcy52cGMsXG4gICAgICAgIHRhcmdldFR5cGU6IGVsYi5UYXJnZXRUeXBlLklOU1RBTkNFLFxuICAgICAgfSxcbiAgICApXG5cbiAgICB0aGlzLmh0dHBzTGlzdGVuZXIgPSB0aGlzLmxvYWRCYWxhbmNlci5hZGRMaXN0ZW5lcihcIkh0dHBzTGlzdGVuZXJcIiwge1xuICAgICAgc3NsUG9saWN5OiBlbGIuU3NsUG9saWN5LlRMUzEyLFxuICAgICAgcHJvdG9jb2w6IGVsYi5BcHBsaWNhdGlvblByb3RvY29sLkhUVFBTLFxuICAgICAgcG9ydDogNDQzLFxuICAgICAgY2VydGlmaWNhdGVzOiBwcm9wcy5jZXJ0aWZpY2F0ZXMsXG4gICAgICBkZWZhdWx0VGFyZ2V0R3JvdXBzOiBbZGVmYXVsdFRhcmdldEdyb3VwXSxcbiAgICAgIC4uLnByb3BzLm92ZXJyaWRlSHR0cHNMaXN0ZW5lclByb3BzLFxuICAgIH0pXG5cbiAgICB0aGlzLmFjY2Vzc0xvZ3NCdWNrZXQgPSBuZXcgczMuQnVja2V0KHRoaXMsIFwiQWNjZXNzTG9nc0J1Y2tldFwiLCB7XG4gICAgICBlbmNyeXB0aW9uOiBzMy5CdWNrZXRFbmNyeXB0aW9uLlMzX01BTkFHRUQsXG4gICAgICBibG9ja1B1YmxpY0FjY2VzczogczMuQmxvY2tQdWJsaWNBY2Nlc3MuQkxPQ0tfQUxMLFxuICAgICAgbGlmZWN5Y2xlUnVsZXM6IFtcbiAgICAgICAge1xuICAgICAgICAgIGV4cGlyYXRpb246IGNkay5EdXJhdGlvbi5kYXlzKDMwKSxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgfSlcblxuICAgIHRoaXMubG9hZEJhbGFuY2VyLmxvZ0FjY2Vzc0xvZ3ModGhpcy5hY2Nlc3NMb2dzQnVja2V0KVxuICB9XG59XG4iXX0=
@@ -81,6 +81,12 @@ export interface WebappProps {
81
81
  * any values specified in securityHeaders.behaviorOverrides.
82
82
  */
83
83
  overrideCloudFrontBehaviourOptions?: Partial<cloudfront.BehaviorOptions>;
84
+ /**
85
+ * Distribution props overrides.
86
+ *
87
+ * Used to override default (AWS or Liflig) configuration
88
+ */
89
+ overrideDistributionProps?: Partial<cloudfront.DistributionProps>;
84
90
  }
85
91
  /**
86
92
  * CloudFront for a Single-Page-Application.
@@ -84,6 +84,7 @@ export class Webapp extends constructs.Construct {
84
84
  domainNames: props.domainNames,
85
85
  errorResponses,
86
86
  webAclId: props.webAclId,
87
+ ...props.overrideDistributionProps,
87
88
  });
88
89
  }
89
90
  addDnsRecord(hostedZone, domainName) {
@@ -113,4 +114,4 @@ export class Webapp extends constructs.Construct {
113
114
  });
114
115
  }
115
116
  }
116
- //# 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 (\n      (props.securityHeaders?.enabled ?? true) &&\n      !props.overrideCloudFrontBehaviourOptions?.responseHeadersPolicy\n        ?.responseHeadersPolicyId\n    ) {\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"]}
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   * Distribution props overrides.\n   *\n   * Used to override default (AWS or Liflig) configuration\n   */\n  overrideDistributionProps?: Partial<cloudfront.DistributionProps>\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 (\n      (props.securityHeaders?.enabled ?? true) &&\n      !props.overrideCloudFrontBehaviourOptions?.responseHeadersPolicy\n        ?.responseHeadersPolicyId\n    ) {\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      ...props.overrideDistributionProps,\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": "3.7.1",
3
+ "version": "3.9.0",
4
4
  "description": "CDK library for Liflig",
5
5
  "type": "module",
6
6
  "repository": {
@@ -42,34 +42,34 @@
42
42
  },
43
43
  "devDependencies": {
44
44
  "@aws-cdk/assert": "2.68.0",
45
- "@aws-sdk/client-cloudwatch-logs": "3.830.0",
45
+ "@aws-sdk/client-cloudwatch-logs": "3.832.0",
46
46
  "@aws-sdk/client-codebuild": "3.830.0",
47
47
  "@aws-sdk/client-codepipeline": "3.830.0",
48
- "@aws-sdk/client-ecs": "3.830.0",
49
- "@aws-sdk/client-s3": "3.830.0",
48
+ "@aws-sdk/client-ecs": "3.834.0",
49
+ "@aws-sdk/client-s3": "3.832.0",
50
50
  "@aws-sdk/client-secrets-manager": "3.830.0",
51
51
  "@aws-sdk/client-ses": "3.830.0",
52
52
  "@aws-sdk/client-sesv2": "3.830.0",
53
53
  "@aws-sdk/client-sfn": "3.830.0",
54
- "@aws-sdk/lib-storage": "3.830.0",
54
+ "@aws-sdk/lib-storage": "3.832.0",
55
55
  "@commitlint/cli": "19.8.1",
56
56
  "@commitlint/config-conventional": "19.8.1",
57
57
  "@eslint/eslintrc": "3.3.1",
58
58
  "@eslint/js": "9.29.0",
59
59
  "@types/aws-lambda": "8.10.150",
60
- "@types/jest": "29.5.14",
60
+ "@types/jest": "30.0.0",
61
61
  "@types/node": "22.15.32",
62
62
  "@typescript-eslint/eslint-plugin": "8.34.1",
63
63
  "@typescript-eslint/parser": "8.34.1",
64
- "aws-cdk": "2.1018.1",
65
- "aws-cdk-lib": "2.196.1",
64
+ "aws-cdk": "2.1019.1",
65
+ "aws-cdk-lib": "2.202.0",
66
66
  "constructs": "10.4.2",
67
67
  "esbuild": "0.25.5",
68
68
  "eslint": "9.29.0",
69
69
  "eslint-config-prettier": "10.1.5",
70
- "eslint-plugin-prettier": "5.4.1",
70
+ "eslint-plugin-prettier": "5.5.0",
71
71
  "husky": "9.1.7",
72
- "jest": "29.7.0",
72
+ "jest": "30.0.2",
73
73
  "jest-cdk-snapshot": "2.3.6",
74
74
  "prettier": "3.5.3",
75
75
  "semantic-release": "24.2.5",