UNPKG

@aws/pdk

Version:

All documentation is located at: https://aws.github.io/aws-pdk

228 lines 38.4 kB
"use strict"; var _a, _b; Object.defineProperty(exports, "__esModule", { value: true }); exports.StaticWebsiteOrigin = exports.StaticWebsite = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); /*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved. SPDX-License-Identifier: Apache-2.0 */ const monorepo_1 = require("../monorepo"); const pdk_nag_1 = require("../pdk-nag"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_cloudfront_1 = require("aws-cdk-lib/aws-cloudfront"); const aws_cloudfront_origins_1 = require("aws-cdk-lib/aws-cloudfront-origins"); const aws_s3_1 = require("aws-cdk-lib/aws-s3"); const aws_s3_deployment_1 = require("aws-cdk-lib/aws-s3-deployment"); const cdk_nag_1 = require("cdk-nag"); const constructs_1 = require("constructs"); const cloudfront_web_acl_1 = require("./cloudfront-web-acl"); const lazy_token_renderer_1 = require("./lazy-token-renderer"); const DEFAULT_RUNTIME_CONFIG_FILENAME = "runtime-config.json"; /** * Deploys a Static Website using by default a private S3 bucket as an origin and Cloudfront as the entrypoint. * * This construct configures a webAcl containing rules that are generally applicable to web applications. This * provides protection against exploitation of a wide range of vulnerabilities, including some of the high risk * and commonly occurring vulnerabilities described in OWASP publications such as OWASP Top 10. * */ class StaticWebsite extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); this.validateProps = (props) => { this.validateEncryptionSettings(props); props.runtimeOptions && this.validateRuntimeConfig(props.runtimeOptions); props.websiteBucket && this.validateBucketConfig(props.websiteBucket); }; this.validateRuntimeConfig = (config) => { if (!config) { throw new Error("validateRuntimeConfig only accepts non-null RuntimeOptions."); } if (config.jsonFileName && !config.jsonFileName.endsWith(".json")) { throw new Error("RuntimeOptions.jsonFileName must be a json file."); } }; this.validateBucketConfig = (bucket) => { if (bucket.isWebsite) { throw new Error("Website buckets cannot be configured as websites as this will break Cloudfront hosting!"); } }; this.validateEncryptionSettings = ({ defaultWebsiteBucketEncryption, defaultWebsiteBucketEncryptionKey, }) => { if (defaultWebsiteBucketEncryptionKey && defaultWebsiteBucketEncryption !== aws_s3_1.BucketEncryption.KMS) { throw new Error("Bucket encryption should be set to KMS if providing a defaultWebsiteBucketEncryptionKey."); } if (defaultWebsiteBucketEncryption && defaultWebsiteBucketEncryption !== aws_s3_1.BucketEncryption.KMS && defaultWebsiteBucketEncryption !== aws_s3_1.BucketEncryption.S3_MANAGED) { throw new Error("Only KMS and S3_MANAGED encryption are supported on the default bucket."); } }; this.suppressCDKNagViolations = (props) => { const stack = aws_cdk_lib_1.Stack.of(this); !props.distributionProps?.certificate && [ "AwsSolutions-CFR4", "AwsPrototyping-CloudFrontDistributionHttpsViewerNoOutdatedSSL", ].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(this.cloudFrontDistribution, [ { id: RuleId, reason: "Certificate is not mandatory therefore the Cloudfront certificate will be used.", }, ]); }); ["AwsSolutions-L1", "AwsPrototyping-LambdaLatestVersion"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(this, [ { id: RuleId, reason: "Latest runtime cannot be configured. CDK will need to upgrade the BucketDeployment construct accordingly.", }, ], true); }); ["AwsSolutions-IAM5", "AwsPrototyping-IAMNoWildcardPermissions"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(this, [ { id: RuleId, reason: "All Policies have been scoped to a Bucket. Given Buckets can contain arbitrary content, wildcard resources with bucket scope are required.", appliesTo: [ { regex: "/^Action::s3:.*$/g", }, { regex: `/^Resource::.*$/g`, }, ], }, ], true); }); ["AwsSolutions-IAM4", "AwsPrototyping-IAMNoManagedPolicies"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(this, [ { id: RuleId, reason: "Buckets can contain arbitrary content, therefore wildcard resources under a bucket are required.", appliesTo: [ { regex: `/^Policy::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole$/g`, }, ], }, ], true); }); ["AwsSolutions-S1", "AwsPrototyping-S3BucketLoggingEnabled"].forEach((RuleId) => { cdk_nag_1.NagSuppressions.addResourceSuppressions(this, [ { id: RuleId, reason: "Access Log buckets should not have s3 bucket logging", }, ], true); }); }; (0, monorepo_1.addMetric)(scope, "static-website"); this.node.setContext("@aws-cdk/aws-s3:serverAccessLogsUseBucketPolicy", true); this.validateProps(props); const accessLogsBucket = new aws_s3_1.Bucket(this, "AccessLogsBucket", { versioned: false, enforceSSL: true, autoDeleteObjects: true, removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, encryption: aws_s3_1.BucketEncryption.S3_MANAGED, objectOwnership: aws_s3_1.ObjectOwnership.OBJECT_WRITER, publicReadAccess: false, blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL, }); // S3 Bucket to hold website files this.websiteBucket = props.websiteBucket ?? new aws_s3_1.Bucket(this, "WebsiteBucket", { versioned: true, enforceSSL: true, autoDeleteObjects: true, removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, encryption: props.defaultWebsiteBucketEncryption ?? aws_s3_1.BucketEncryption.S3_MANAGED, objectOwnership: aws_s3_1.ObjectOwnership.BUCKET_OWNER_ENFORCED, encryptionKey: props.defaultWebsiteBucketEncryptionKey, publicReadAccess: false, blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL, serverAccessLogsPrefix: "website-access-logs", serverAccessLogsBucket: accessLogsBucket, }); // Web ACL const { distributionProps } = props; const webAclArn = distributionProps?.webAclId ?? (props.webAclProps?.disable ? undefined : new cloudfront_web_acl_1.CloudfrontWebAcl(this, "WebsiteAcl", props.webAclProps) .webAclArn); // Cloudfront Distribution const logBucket = props.distributionProps?.logBucket || new aws_s3_1.Bucket(this, "DistributionLogBucket", { enforceSSL: true, autoDeleteObjects: true, removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, encryption: props.defaultWebsiteBucketEncryption ?? aws_s3_1.BucketEncryption.S3_MANAGED, objectOwnership: aws_s3_1.ObjectOwnership.BUCKET_OWNER_PREFERRED, encryptionKey: props.defaultWebsiteBucketEncryptionKey, publicReadAccess: false, blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL, serverAccessLogsPrefix: "distribution-access-logs", serverAccessLogsBucket: accessLogsBucket, }); const defaultRootObject = distributionProps?.defaultRootObject ?? "index.html"; this.cloudFrontDistribution = new aws_cloudfront_1.Distribution(this, "CloudfrontDistribution", { ...distributionProps, webAclId: webAclArn, enableLogging: true, logBucket: logBucket, defaultBehavior: { ...distributionProps?.defaultBehavior, origin: aws_cloudfront_origins_1.S3BucketOrigin.withOriginAccessControl(this.websiteBucket), viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, }, defaultRootObject, // We need to redirect "key not found errors" to index.html for single page apps errorResponses: distributionProps?.errorResponses ?? [403, 404].map((httpStatus) => ({ httpStatus, responseHttpStatus: 200, responsePagePath: `/${defaultRootObject}`, })), }); // Deploy Website this.bucketDeployment = new aws_s3_deployment_1.BucketDeployment(this, "WebsiteDeployment", { memoryLimit: 2048, ...props.bucketDeploymentProps, sources: [ aws_s3_deployment_1.Source.asset(props.websiteContentPath), ...(props.runtimeOptions ? [ aws_s3_deployment_1.Source.jsonData(props.runtimeOptions?.jsonFileName || DEFAULT_RUNTIME_CONFIG_FILENAME, (0, lazy_token_renderer_1.lazilyRender)(this, props.runtimeOptions.jsonPayload)), ] : []), ], destinationBucket: this.websiteBucket, // Files in the distribution's edge caches will be invalidated after files are uploaded to the destination bucket. distribution: this.cloudFrontDistribution, }); new aws_cdk_lib_1.CfnOutput(this, "DistributionDomainName", { value: this.cloudFrontDistribution.domainName, }); this.suppressCDKNagViolations(props); } } exports.StaticWebsite = StaticWebsite; _a = JSII_RTTI_SYMBOL_1; StaticWebsite[_a] = { fqn: "@aws/pdk.static_website.StaticWebsite", version: "0.26.14" }; /** * If passing in distributionProps, the default behaviour.origin is a required parameter. An instance of this class can be passed in * to make the compiler happy. */ class StaticWebsiteOrigin { bind(_scope, _options) { throw new Error("This should never be called"); } } exports.StaticWebsiteOrigin = StaticWebsiteOrigin; _b = JSII_RTTI_SYMBOL_1; StaticWebsiteOrigin[_b] = { fqn: "@aws/pdk.static_website.StaticWebsiteOrigin", version: "0.26.14" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"static-website.js","sourceRoot":"","sources":["static-website.ts"],"names":[],"mappings":"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","sourcesContent":["/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.\nSPDX-License-Identifier: Apache-2.0 */\nimport { addMetric } from \"@aws/monorepo\";\nimport { PDKNag } from \"@aws/pdk-nag\";\nimport { CfnOutput, RemovalPolicy, Stack } from \"aws-cdk-lib\";\nimport {\n  Distribution,\n  IOrigin,\n  OriginBindConfig,\n  OriginBindOptions,\n  ViewerProtocolPolicy,\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport { S3BucketOrigin } from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport { Key } from \"aws-cdk-lib/aws-kms\";\nimport {\n  BlockPublicAccess,\n  Bucket,\n  BucketEncryption,\n  IBucket,\n  ObjectOwnership,\n} from \"aws-cdk-lib/aws-s3\";\nimport { BucketDeployment, Source } from \"aws-cdk-lib/aws-s3-deployment\";\nimport { NagSuppressions } from \"cdk-nag\";\nimport { Construct } from \"constructs\";\nimport { BucketDeploymentProps } from \"./bucket-deployment-props\";\nimport { CloudfrontWebAcl, CloudFrontWebAclProps } from \"./cloudfront-web-acl\";\nimport { DistributionProps } from \"./distribution-props\";\nimport { lazilyRender } from \"./lazy-token-renderer\";\n\nconst DEFAULT_RUNTIME_CONFIG_FILENAME = \"runtime-config.json\";\n\n/**\n * Dynamic configuration which gets resolved only during deployment.\n *\n * @example\n *\n * // Will store a JSON file called runtime-config.json in the root of the StaticWebsite S3 bucket containing any\n * // and all resolved values.\n * const runtimeConfig = {jsonPayload: {bucketArn: s3Bucket.bucketArn}};\n * new StaticWebsite(scope, 'StaticWebsite', {websiteContentPath: 'path/to/website', runtimeConfig});\n */\nexport interface RuntimeOptions {\n  /**\n   * File name to store runtime configuration (jsonPayload).\n   *\n   * Must follow pattern: '*.json'\n   *\n   * @default \"runtime-config.json\"\n   */\n  readonly jsonFileName?: string;\n\n  /**\n   * Arbitrary JSON payload containing runtime values to deploy. Typically this contains resourceArns, etc which\n   * are only known at deploy time.\n   *\n   * @example { userPoolId: some.userPool.userPoolId, someResourceArn: some.resource.Arn }\n   */\n  readonly jsonPayload: any;\n}\n\n/**\n * Properties for configuring the StaticWebsite.\n */\nexport interface StaticWebsiteProps {\n  /**\n   * Path to the directory containing the static website files and assets. This directory must contain an index.html file.\n   */\n  readonly websiteContentPath: string;\n\n  /**\n   * Dynamic configuration which gets resolved only during deployment.\n   */\n  readonly runtimeOptions?: RuntimeOptions;\n\n  /**\n   * Bucket encryption to use for the default bucket.\n   *\n   * Supported options are KMS or S3MANAGED.\n   *\n   * Note: If planning to use KMS, ensure you associate a Lambda Edge function to sign requests to S3 as OAI does not currently support KMS encryption. Refer to {@link https://aws.amazon.com/blogs/networking-and-content-delivery/serving-sse-kms-encrypted-content-from-s3-using-cloudfront/}\n   *\n   * @default - \"S3MANAGED\"\n   */\n  readonly defaultWebsiteBucketEncryption?: BucketEncryption;\n\n  /**\n   * A predefined KMS customer encryption key to use for the default bucket that gets created.\n   *\n   * Note: This is only used if the websiteBucket is left undefined, otherwise all settings from the provided websiteBucket will be used.\n   */\n  readonly defaultWebsiteBucketEncryptionKey?: Key;\n\n  /**\n   * Predefined bucket to deploy the website into.\n   */\n  readonly websiteBucket?: IBucket;\n\n  /**\n   * Custom distribution properties.\n   *\n   * Note: defaultBehaviour.origin is a required parameter, however it will not be used as this construct will wire it on your behalf.\n   * You will need to pass in an instance of StaticWebsiteOrigin (NoOp) to keep the compiler happy.\n   */\n  readonly distributionProps?: DistributionProps;\n\n  /**\n   * Custom bucket deployment properties.\n   *\n   * ```\n   */\n  readonly bucketDeploymentProps?: BucketDeploymentProps;\n\n  /**\n   * Limited configuration settings for the generated webAcl. For more advanced settings, create your own ACL and pass in the webAclId as a param to distributionProps.\n   *\n   * Note: If pass in your own ACL, make sure the SCOPE is CLOUDFRONT and it is created in us-east-1.\n   */\n  readonly webAclProps?: CloudFrontWebAclProps;\n}\n\n/**\n * Deploys a Static Website using by default a private S3 bucket as an origin and Cloudfront as the entrypoint.\n *\n * This construct configures a webAcl containing rules that are generally applicable to web applications. This\n * provides protection against exploitation of a wide range of vulnerabilities, including some of the high risk\n * and commonly occurring vulnerabilities described in OWASP publications such as OWASP Top 10.\n *\n */\nexport class StaticWebsite extends Construct {\n  public readonly websiteBucket: IBucket;\n  public readonly cloudFrontDistribution: Distribution;\n  public readonly bucketDeployment: BucketDeployment;\n\n  constructor(scope: Construct, id: string, props: StaticWebsiteProps) {\n    super(scope, id);\n\n    addMetric(scope, \"static-website\");\n\n    this.node.setContext(\n      \"@aws-cdk/aws-s3:serverAccessLogsUseBucketPolicy\",\n      true\n    );\n\n    this.validateProps(props);\n\n    const accessLogsBucket = new Bucket(this, \"AccessLogsBucket\", {\n      versioned: false,\n      enforceSSL: true,\n      autoDeleteObjects: true,\n      removalPolicy: RemovalPolicy.DESTROY,\n      encryption: BucketEncryption.S3_MANAGED,\n      objectOwnership: ObjectOwnership.OBJECT_WRITER,\n      publicReadAccess: false,\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n    });\n\n    // S3 Bucket to hold website files\n    this.websiteBucket =\n      props.websiteBucket ??\n      new Bucket(this, \"WebsiteBucket\", {\n        versioned: true,\n        enforceSSL: true,\n        autoDeleteObjects: true,\n        removalPolicy: RemovalPolicy.DESTROY,\n        encryption:\n          props.defaultWebsiteBucketEncryption ?? BucketEncryption.S3_MANAGED,\n        objectOwnership: ObjectOwnership.BUCKET_OWNER_ENFORCED,\n        encryptionKey: props.defaultWebsiteBucketEncryptionKey,\n        publicReadAccess: false,\n        blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n        serverAccessLogsPrefix: \"website-access-logs\",\n        serverAccessLogsBucket: accessLogsBucket,\n      });\n\n    // Web ACL\n    const { distributionProps } = props;\n    const webAclArn =\n      distributionProps?.webAclId ??\n      (props.webAclProps?.disable\n        ? undefined\n        : new CloudfrontWebAcl(this, \"WebsiteAcl\", props.webAclProps)\n            .webAclArn);\n\n    // Cloudfront Distribution\n    const logBucket =\n      props.distributionProps?.logBucket ||\n      new Bucket(this, \"DistributionLogBucket\", {\n        enforceSSL: true,\n        autoDeleteObjects: true,\n        removalPolicy: RemovalPolicy.DESTROY,\n        encryption:\n          props.defaultWebsiteBucketEncryption ?? BucketEncryption.S3_MANAGED,\n        objectOwnership: ObjectOwnership.BUCKET_OWNER_PREFERRED,\n        encryptionKey: props.defaultWebsiteBucketEncryptionKey,\n        publicReadAccess: false,\n        blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n        serverAccessLogsPrefix: \"distribution-access-logs\",\n        serverAccessLogsBucket: accessLogsBucket,\n      });\n\n    const defaultRootObject =\n      distributionProps?.defaultRootObject ?? \"index.html\";\n    this.cloudFrontDistribution = new Distribution(\n      this,\n      \"CloudfrontDistribution\",\n      {\n        ...distributionProps,\n        webAclId: webAclArn,\n        enableLogging: true,\n        logBucket: logBucket,\n        defaultBehavior: {\n          ...distributionProps?.defaultBehavior,\n          origin: S3BucketOrigin.withOriginAccessControl(this.websiteBucket),\n          viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        },\n        defaultRootObject,\n        // We need to redirect \"key not found errors\" to index.html for single page apps\n        errorResponses:\n          distributionProps?.errorResponses ??\n          [403, 404].map((httpStatus) => ({\n            httpStatus,\n            responseHttpStatus: 200,\n            responsePagePath: `/${defaultRootObject}`,\n          })),\n      }\n    );\n\n    // Deploy Website\n    this.bucketDeployment = new BucketDeployment(this, \"WebsiteDeployment\", {\n      memoryLimit: 2048,\n      ...props.bucketDeploymentProps,\n      sources: [\n        Source.asset(props.websiteContentPath),\n        ...(props.runtimeOptions\n          ? [\n              Source.jsonData(\n                props.runtimeOptions?.jsonFileName ||\n                  DEFAULT_RUNTIME_CONFIG_FILENAME,\n                lazilyRender(this, props.runtimeOptions.jsonPayload)\n              ),\n            ]\n          : []),\n      ],\n      destinationBucket: this.websiteBucket,\n      // Files in the distribution's edge caches will be invalidated after files are uploaded to the destination bucket.\n      distribution: this.cloudFrontDistribution,\n    });\n\n    new CfnOutput(this, \"DistributionDomainName\", {\n      value: this.cloudFrontDistribution.domainName,\n    });\n\n    this.suppressCDKNagViolations(props);\n  }\n\n  private validateProps = (props: StaticWebsiteProps) => {\n    this.validateEncryptionSettings(props);\n    props.runtimeOptions && this.validateRuntimeConfig(props.runtimeOptions);\n    props.websiteBucket && this.validateBucketConfig(props.websiteBucket);\n  };\n\n  private validateRuntimeConfig = (config: RuntimeOptions) => {\n    if (!config) {\n      throw new Error(\n        \"validateRuntimeConfig only accepts non-null RuntimeOptions.\"\n      );\n    }\n\n    if (config.jsonFileName && !config.jsonFileName.endsWith(\".json\")) {\n      throw new Error(\"RuntimeOptions.jsonFileName must be a json file.\");\n    }\n  };\n\n  private validateBucketConfig = (bucket: IBucket) => {\n    if (bucket.isWebsite) {\n      throw new Error(\n        \"Website buckets cannot be configured as websites as this will break Cloudfront hosting!\"\n      );\n    }\n  };\n\n  private validateEncryptionSettings = ({\n    defaultWebsiteBucketEncryption,\n    defaultWebsiteBucketEncryptionKey,\n  }: StaticWebsiteProps) => {\n    if (\n      defaultWebsiteBucketEncryptionKey &&\n      defaultWebsiteBucketEncryption !== BucketEncryption.KMS\n    ) {\n      throw new Error(\n        \"Bucket encryption should be set to KMS if providing a defaultWebsiteBucketEncryptionKey.\"\n      );\n    }\n\n    if (\n      defaultWebsiteBucketEncryption &&\n      defaultWebsiteBucketEncryption !== BucketEncryption.KMS &&\n      defaultWebsiteBucketEncryption !== BucketEncryption.S3_MANAGED\n    ) {\n      throw new Error(\n        \"Only KMS and S3_MANAGED encryption are supported on the default bucket.\"\n      );\n    }\n  };\n\n  private suppressCDKNagViolations = (props: StaticWebsiteProps) => {\n    const stack = Stack.of(this);\n    !props.distributionProps?.certificate &&\n      [\n        \"AwsSolutions-CFR4\",\n        \"AwsPrototyping-CloudFrontDistributionHttpsViewerNoOutdatedSSL\",\n      ].forEach((RuleId) => {\n        NagSuppressions.addResourceSuppressions(this.cloudFrontDistribution, [\n          {\n            id: RuleId,\n            reason:\n              \"Certificate is not mandatory therefore the Cloudfront certificate will be used.\",\n          },\n        ]);\n      });\n\n    [\"AwsSolutions-L1\", \"AwsPrototyping-LambdaLatestVersion\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          this,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Latest runtime cannot be configured. CDK will need to upgrade the BucketDeployment construct accordingly.\",\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    [\"AwsSolutions-IAM5\", \"AwsPrototyping-IAMNoWildcardPermissions\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          this,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"All Policies have been scoped to a Bucket. Given Buckets can contain arbitrary content, wildcard resources with bucket scope are required.\",\n              appliesTo: [\n                {\n                  regex: \"/^Action::s3:.*$/g\",\n                },\n                {\n                  regex: `/^Resource::.*$/g`,\n                },\n              ],\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    [\"AwsSolutions-IAM4\", \"AwsPrototyping-IAMNoManagedPolicies\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          this,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Buckets can contain arbitrary content, therefore wildcard resources under a bucket are required.\",\n              appliesTo: [\n                {\n                  regex: `/^Policy::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole$/g`,\n                },\n              ],\n            },\n          ],\n          true\n        );\n      }\n    );\n\n    [\"AwsSolutions-S1\", \"AwsPrototyping-S3BucketLoggingEnabled\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          this,\n          [\n            {\n              id: RuleId,\n              reason: \"Access Log buckets should not have s3 bucket logging\",\n            },\n          ],\n          true\n        );\n      }\n    );\n  };\n}\n\n/**\n * If passing in distributionProps, the default behaviour.origin is a required parameter. An instance of this class can be passed in\n * to make the compiler happy.\n */\nexport class StaticWebsiteOrigin implements IOrigin {\n  bind(_scope: Construct, _options: OriginBindOptions): OriginBindConfig {\n    throw new Error(\"This should never be called\");\n  }\n}\n"]}