@pinegraph/react-website-cdk-constructs
Advanced tools
Comparing version 0.0.4 to 0.0.5
@@ -35,1 +35,8 @@ import { RemovalPolicy } from "aws-cdk-lib"; | ||
} | ||
/** | ||
* @Deprecated | ||
* | ||
* This function is not meant to be used by others. For Pinegraph, reorganizing | ||
* resources into constructs causes the logical ids to change. See https://github.com/aws/aws-cdk-rfcs/issues/162 | ||
*/ | ||
export declare function constructConstructsFromProps(self: Construct, props: ReactWebsiteProps): void; |
251
lib/index.js
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.ReactWebsiteConstruct = void 0; | ||
exports.constructConstructsFromProps = exports.ReactWebsiteConstruct = void 0; | ||
const aws_cdk_lib_1 = require("aws-cdk-lib"); | ||
@@ -20,128 +20,139 @@ const aws_certificatemanager_1 = require("aws-cdk-lib/aws-certificatemanager"); | ||
super(scope, id); | ||
const domain = props.domainName; | ||
const allDomainNames = [domain]; | ||
const zones = {}; | ||
for (const d of allDomainNames) { | ||
const resourceId = getResourceId(d, "Zone"); | ||
zones[d] = aws_route53_1.HostedZone.fromLookup(this, resourceId, { | ||
domainName: domain, | ||
}); | ||
} | ||
const cloudfrontOAI = new aws_cloudfront_1.OriginAccessIdentity(this, "cloudfront-OAI", { | ||
comment: `OAI for ${domain}`, | ||
}); | ||
new aws_cdk_lib_1.CfnOutput(this, "Site", { value: "https://" + domain }); | ||
// Content bucket | ||
const siteBucket = new aws_s3_1.Bucket(this, "SiteBucket", { | ||
publicReadAccess: false, | ||
blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL, | ||
removalPolicy: props.removalPolicy, | ||
autoDeleteObjects: props.removalPolicy === aws_cdk_lib_1.RemovalPolicy.DESTROY, | ||
}); | ||
// Grant access to cloudfront | ||
siteBucket.addToResourcePolicy(new aws_iam_1.PolicyStatement({ | ||
actions: ["s3:GetObject"], | ||
resources: [siteBucket.arnForObjects("*")], | ||
principals: [ | ||
new aws_iam_1.CanonicalUserPrincipal(cloudfrontOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId), | ||
], | ||
})); | ||
new aws_cdk_lib_1.CfnOutput(this, "Bucket", { value: siteBucket.bucketName }); | ||
// TLS certificate | ||
const certificate = new aws_certificatemanager_1.DnsValidatedCertificate(this, "SiteCertificate", { | ||
constructConstructsFromProps(this, props); | ||
} | ||
} | ||
exports.ReactWebsiteConstruct = ReactWebsiteConstruct; | ||
/** | ||
* @Deprecated | ||
* | ||
* This function is not meant to be used by others. For Pinegraph, reorganizing | ||
* resources into constructs causes the logical ids to change. See https://github.com/aws/aws-cdk-rfcs/issues/162 | ||
*/ | ||
function constructConstructsFromProps(self, props) { | ||
const domain = props.domainName; | ||
const allDomainNames = [domain]; | ||
const zones = {}; | ||
for (const d of allDomainNames) { | ||
const resourceId = getResourceId(d, "Zone"); | ||
zones[d] = aws_route53_1.HostedZone.fromLookup(self, resourceId, { | ||
domainName: domain, | ||
subjectAlternativeNames: allDomainNames, | ||
hostedZone: zones[domain], | ||
region: "us-east-1", // Cloudfront only checks this region for certificates. | ||
}); | ||
new aws_cdk_lib_1.CfnOutput(this, "Certificate", { value: certificate.certificateArn }); | ||
const responseHeaderPolicy = props.responseHeaderPolicy || | ||
new aws_cloudfront_1.ResponseHeadersPolicy(this, "SecurityHeadersResponsePolicy", { | ||
comment: "Security headers response policy", | ||
securityHeadersBehavior: { | ||
contentSecurityPolicy: { | ||
override: true, | ||
contentSecurityPolicy: `default-src 'self';`, | ||
}, | ||
strictTransportSecurity: { | ||
override: true, | ||
accessControlMaxAge: aws_cdk_lib_1.Duration.days(365), | ||
includeSubdomains: true, | ||
preload: true, | ||
}, | ||
contentTypeOptions: { | ||
override: true, | ||
}, | ||
referrerPolicy: { | ||
override: true, | ||
referrerPolicy: aws_cloudfront_1.HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN, | ||
}, | ||
xssProtection: { | ||
override: true, | ||
protection: true, | ||
modeBlock: true, | ||
}, | ||
frameOptions: { | ||
override: true, | ||
frameOption: aws_cloudfront_1.HeadersFrameOption.DENY, | ||
}, | ||
} | ||
const cloudfrontOAI = new aws_cloudfront_1.OriginAccessIdentity(self, "cloudfront-OAI", { | ||
comment: `OAI for ${domain}`, | ||
}); | ||
new aws_cdk_lib_1.CfnOutput(self, "Site", { value: "https://" + domain }); | ||
// Content bucket | ||
const siteBucket = new aws_s3_1.Bucket(self, "SiteBucket", { | ||
bucketName: domain, | ||
publicReadAccess: false, | ||
blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL, | ||
removalPolicy: props.removalPolicy, | ||
autoDeleteObjects: props.removalPolicy === aws_cdk_lib_1.RemovalPolicy.DESTROY, | ||
}); | ||
// Grant access to cloudfront | ||
siteBucket.addToResourcePolicy(new aws_iam_1.PolicyStatement({ | ||
actions: ["s3:GetObject"], | ||
resources: [siteBucket.arnForObjects("*")], | ||
principals: [ | ||
new aws_iam_1.CanonicalUserPrincipal(cloudfrontOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId), | ||
], | ||
})); | ||
new aws_cdk_lib_1.CfnOutput(self, "Bucket", { value: siteBucket.bucketName }); | ||
// TLS certificate | ||
const certificate = new aws_certificatemanager_1.DnsValidatedCertificate(self, "SiteCertificate", { | ||
domainName: domain, | ||
subjectAlternativeNames: allDomainNames, | ||
hostedZone: zones[domain], | ||
region: "us-east-1", // Cloudfront only checks this region for certificates. | ||
}); | ||
new aws_cdk_lib_1.CfnOutput(self, "Certificate", { value: certificate.certificateArn }); | ||
const responseHeaderPolicy = props.responseHeaderPolicy || | ||
new aws_cloudfront_1.ResponseHeadersPolicy(self, "SecurityHeadersResponsePolicy", { | ||
comment: "Security headers response policy", | ||
securityHeadersBehavior: { | ||
contentSecurityPolicy: { | ||
override: true, | ||
contentSecurityPolicy: `default-src 'self';`, | ||
}, | ||
}); | ||
// CloudFront distribution | ||
const distribution = new aws_cloudfront_1.Distribution(this, "SiteDistribution", { | ||
certificate: certificate, | ||
domainNames: allDomainNames, | ||
comment: props.domainName, | ||
minimumProtocolVersion: aws_cloudfront_1.SecurityPolicyProtocol.TLS_V1_2016, | ||
defaultBehavior: { | ||
origin: new aws_cloudfront_origins_1.S3Origin(siteBucket, { | ||
originAccessIdentity: cloudfrontOAI, | ||
}), | ||
edgeLambdas: props.edgeOriginRequestFunction | ||
? [ | ||
{ | ||
functionVersion: props.edgeOriginRequestFunction.currentVersion, | ||
eventType: aws_cloudfront_1.LambdaEdgeEventType.ORIGIN_REQUEST, | ||
}, | ||
] | ||
: [], | ||
compress: true, | ||
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_GET_HEAD_OPTIONS, | ||
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, | ||
responseHeadersPolicy: responseHeaderPolicy, | ||
strictTransportSecurity: { | ||
override: true, | ||
accessControlMaxAge: aws_cdk_lib_1.Duration.days(365), | ||
includeSubdomains: true, | ||
preload: true, | ||
}, | ||
contentTypeOptions: { | ||
override: true, | ||
}, | ||
referrerPolicy: { | ||
override: true, | ||
referrerPolicy: aws_cloudfront_1.HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN, | ||
}, | ||
xssProtection: { | ||
override: true, | ||
protection: true, | ||
modeBlock: true, | ||
}, | ||
frameOptions: { | ||
override: true, | ||
frameOption: aws_cloudfront_1.HeadersFrameOption.DENY, | ||
}, | ||
}, | ||
/** | ||
* This ensures that all routes end up using index.html for client side rendering. | ||
*/ | ||
errorResponses: [ | ||
{ | ||
httpStatus: 403, | ||
responseHttpStatus: 200, | ||
responsePagePath: "/index.html", | ||
}, | ||
], | ||
defaultRootObject: "index.html", | ||
}); | ||
new aws_cdk_lib_1.CfnOutput(this, "DistributionId", { | ||
value: distribution.distributionId, | ||
// CloudFront distribution | ||
const distribution = new aws_cloudfront_1.Distribution(self, "SiteDistribution", { | ||
certificate: certificate, | ||
domainNames: allDomainNames, | ||
comment: props.domainName, | ||
minimumProtocolVersion: aws_cloudfront_1.SecurityPolicyProtocol.TLS_V1_2016, | ||
defaultBehavior: { | ||
origin: new aws_cloudfront_origins_1.S3Origin(siteBucket, { | ||
originAccessIdentity: cloudfrontOAI, | ||
}), | ||
edgeLambdas: props.edgeOriginRequestFunction | ||
? [ | ||
{ | ||
functionVersion: props.edgeOriginRequestFunction.currentVersion, | ||
eventType: aws_cloudfront_1.LambdaEdgeEventType.ORIGIN_REQUEST, | ||
}, | ||
] | ||
: [], | ||
compress: true, | ||
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_GET_HEAD_OPTIONS, | ||
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS, | ||
responseHeadersPolicy: responseHeaderPolicy, | ||
}, | ||
/** | ||
* This ensures that all routes end up using index.html for client side rendering. | ||
*/ | ||
errorResponses: [ | ||
{ | ||
httpStatus: 403, | ||
responseHttpStatus: 200, | ||
responsePagePath: "/index.html", | ||
}, | ||
], | ||
defaultRootObject: "index.html", | ||
}); | ||
new aws_cdk_lib_1.CfnOutput(self, "DistributionId", { | ||
value: distribution.distributionId, | ||
}); | ||
// Route53 alias record for the CloudFront distribution] | ||
for (const d of allDomainNames) { | ||
const resourceId = getResourceId(d, "SiteAliasRecord"); | ||
new aws_route53_1.ARecord(self, resourceId, { | ||
recordName: d, | ||
target: aws_route53_1.RecordTarget.fromAlias(new aws_route53_targets_1.CloudFrontTarget(distribution)), | ||
zone: zones[d], | ||
}); | ||
// Route53 alias record for the CloudFront distribution] | ||
for (const d of allDomainNames) { | ||
const resourceId = getResourceId(d, "SiteAliasRecord"); | ||
new aws_route53_1.ARecord(this, resourceId, { | ||
recordName: d, | ||
target: aws_route53_1.RecordTarget.fromAlias(new aws_route53_targets_1.CloudFrontTarget(distribution)), | ||
zone: zones[d], | ||
}); | ||
} | ||
// Deploy site contents to S3 bucket | ||
new aws_s3_deployment_1.BucketDeployment(this, "DeployWithInvalidation", { | ||
sources: [props.sourceAsset], | ||
destinationBucket: siteBucket, | ||
distribution, | ||
distributionPaths: ["/*"], | ||
}); | ||
} | ||
// Deploy site contents to S3 bucket | ||
new aws_s3_deployment_1.BucketDeployment(self, "DeployWithInvalidation", { | ||
sources: [props.sourceAsset], | ||
destinationBucket: siteBucket, | ||
distribution, | ||
distributionPaths: ["/*"], | ||
}); | ||
} | ||
exports.ReactWebsiteConstruct = ReactWebsiteConstruct; | ||
exports.constructConstructsFromProps = constructConstructsFromProps; | ||
function capitalizeFirstLetter(str) { | ||
@@ -159,2 +170,2 @@ return str.charAt(0).toUpperCase() + str.slice(1); | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { CfnOutput, Duration, RemovalPolicy } from \"aws-cdk-lib\";\nimport { DnsValidatedCertificate } from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n  AllowedMethods,\n  Distribution,\n  experimental,\n  HeadersFrameOption,\n  HeadersReferrerPolicy,\n  LambdaEdgeEventType,\n  OriginAccessIdentity,\n  ResponseHeadersPolicy,\n  SecurityPolicyProtocol,\n  ViewerProtocolPolicy,\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport { S3Origin } from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport { CanonicalUserPrincipal, PolicyStatement } from \"aws-cdk-lib/aws-iam\";\nimport {\n  ARecord,\n  HostedZone,\n  IHostedZone,\n  RecordTarget,\n} from \"aws-cdk-lib/aws-route53\";\nimport { CloudFrontTarget } from \"aws-cdk-lib/aws-route53-targets\";\nimport { BlockPublicAccess, Bucket } from \"aws-cdk-lib/aws-s3\";\nimport { BucketDeployment, ISource } from \"aws-cdk-lib/aws-s3-deployment\";\nimport { Construct } from \"constructs\";\nexport interface ReactWebsiteProps {\n  /**\n   * The domain name of the website.\n   */\n  readonly domainName: string;\n\n  /**\n   * The built react source code to deploy.\n   */\n  readonly sourceAsset: ISource;\n\n  /**\n   * If not specified, it will default to a strict and secure CSP that will score\n   * well on https://observatory.mozilla.org/.\n   */\n  readonly responseHeaderPolicy?: ResponseHeadersPolicy;\n\n  /**\n   * An optional edge function to transform S3 responses. This is useful\n   * for doing things like injecting metatags for search optimizations.\n   */\n  readonly edgeOriginRequestFunction?: experimental.EdgeFunction;\n\n  /**\n   * Policy for deleting resources. Defaults to destroy.\n   */\n  readonly removalPolicy?: RemovalPolicy;\n}\n\n/**\n * A construct to create a secure single page application react website.\n */\nexport class ReactWebsiteConstruct extends Construct {\n  constructor(scope: Construct, id: string, props: ReactWebsiteProps) {\n    super(scope, id);\n    const domain = props.domainName;\n    const allDomainNames = [domain];\n\n    const zones: { [name: string]: IHostedZone } = {};\n    for (const d of allDomainNames) {\n      const resourceId = getResourceId(d, \"Zone\");\n      zones[d] = HostedZone.fromLookup(this, resourceId, {\n        domainName: domain,\n      });\n    }\n    const cloudfrontOAI = new OriginAccessIdentity(this, \"cloudfront-OAI\", {\n      comment: `OAI for ${domain}`,\n    });\n\n    new CfnOutput(this, \"Site\", { value: \"https://\" + domain });\n\n    // Content bucket\n    const siteBucket = new Bucket(this, \"SiteBucket\", {\n      publicReadAccess: false,\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n      removalPolicy: props.removalPolicy,\n      autoDeleteObjects: props.removalPolicy === RemovalPolicy.DESTROY,\n    });\n    // Grant access to cloudfront\n    siteBucket.addToResourcePolicy(\n      new PolicyStatement({\n        actions: [\"s3:GetObject\"],\n        resources: [siteBucket.arnForObjects(\"*\")],\n        principals: [\n          new CanonicalUserPrincipal(\n            cloudfrontOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId\n          ),\n        ],\n      })\n    );\n    new CfnOutput(this, \"Bucket\", { value: siteBucket.bucketName });\n\n    // TLS certificate\n    const certificate = new DnsValidatedCertificate(this, \"SiteCertificate\", {\n      domainName: domain,\n      subjectAlternativeNames: allDomainNames,\n      hostedZone: zones[domain],\n      region: \"us-east-1\", // Cloudfront only checks this region for certificates.\n    });\n    new CfnOutput(this, \"Certificate\", { value: certificate.certificateArn });\n    const responseHeaderPolicy =\n      props.responseHeaderPolicy ||\n      new ResponseHeadersPolicy(this, \"SecurityHeadersResponsePolicy\", {\n        comment: \"Security headers response policy\",\n        securityHeadersBehavior: {\n          contentSecurityPolicy: {\n            override: true,\n            contentSecurityPolicy: `default-src 'self';`,\n          },\n          strictTransportSecurity: {\n            override: true,\n            accessControlMaxAge: Duration.days(365),\n            includeSubdomains: true,\n            preload: true,\n          },\n          contentTypeOptions: {\n            override: true,\n          },\n          referrerPolicy: {\n            override: true,\n            referrerPolicy:\n              HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN,\n          },\n          xssProtection: {\n            override: true,\n            protection: true,\n            modeBlock: true,\n          },\n          frameOptions: {\n            override: true,\n            frameOption: HeadersFrameOption.DENY,\n          },\n        },\n      });\n\n    // CloudFront distribution\n    const distribution = new Distribution(this, \"SiteDistribution\", {\n      certificate: certificate,\n      domainNames: allDomainNames,\n      comment: props.domainName,\n      minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2016,\n      defaultBehavior: {\n        origin: new S3Origin(siteBucket, {\n          originAccessIdentity: cloudfrontOAI,\n        }),\n        edgeLambdas: props.edgeOriginRequestFunction\n          ? [\n              {\n                functionVersion: props.edgeOriginRequestFunction.currentVersion,\n                eventType: LambdaEdgeEventType.ORIGIN_REQUEST,\n              },\n            ]\n          : [],\n        compress: true,\n        allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n        viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        responseHeadersPolicy: responseHeaderPolicy,\n      },\n\n      /**\n       * This ensures that all routes end up using index.html for client side rendering.\n       */\n      errorResponses: [\n        {\n          httpStatus: 403,\n          responseHttpStatus: 200,\n          responsePagePath: \"/index.html\",\n        },\n      ],\n      defaultRootObject: \"index.html\",\n    });\n\n    new CfnOutput(this, \"DistributionId\", {\n      value: distribution.distributionId,\n    });\n\n    // Route53 alias record for the CloudFront distribution]\n    for (const d of allDomainNames) {\n      const resourceId = getResourceId(d, \"SiteAliasRecord\");\n      new ARecord(this, resourceId, {\n        recordName: d,\n        target: RecordTarget.fromAlias(new CloudFrontTarget(distribution)),\n        zone: zones[d],\n      });\n    }\n\n    // Deploy site contents to S3 bucket\n    new BucketDeployment(this, \"DeployWithInvalidation\", {\n      sources: [props.sourceAsset],\n      destinationBucket: siteBucket,\n      distribution,\n      distributionPaths: [\"/*\"],\n    });\n  }\n}\n\nfunction capitalizeFirstLetter(str: string) {\n  return str.charAt(0).toUpperCase() + str.slice(1);\n}\n\n/**\n * A simple function to make the resource id human readable.\n */\nfunction getResourceId(domain: string, resourceType: string) {\n  if (!domain.includes(\".\")) {\n    return resourceType;\n  }\n  return capitalizeFirstLetter(`${domain.split(\".\")[0]}${resourceType}`);\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { CfnOutput, Duration, RemovalPolicy } from \"aws-cdk-lib\";\nimport { DnsValidatedCertificate } from \"aws-cdk-lib/aws-certificatemanager\";\nimport {\n  AllowedMethods,\n  Distribution,\n  experimental,\n  HeadersFrameOption,\n  HeadersReferrerPolicy,\n  LambdaEdgeEventType,\n  OriginAccessIdentity,\n  ResponseHeadersPolicy,\n  SecurityPolicyProtocol,\n  ViewerProtocolPolicy,\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport { S3Origin } from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport { CanonicalUserPrincipal, PolicyStatement } from \"aws-cdk-lib/aws-iam\";\nimport {\n  ARecord,\n  HostedZone,\n  IHostedZone,\n  RecordTarget,\n} from \"aws-cdk-lib/aws-route53\";\nimport { CloudFrontTarget } from \"aws-cdk-lib/aws-route53-targets\";\nimport { BlockPublicAccess, Bucket } from \"aws-cdk-lib/aws-s3\";\nimport { BucketDeployment, ISource } from \"aws-cdk-lib/aws-s3-deployment\";\nimport { Construct } from \"constructs\";\nexport interface ReactWebsiteProps {\n  /**\n   * The domain name of the website.\n   */\n  readonly domainName: string;\n\n  /**\n   * The built react source code to deploy.\n   */\n  readonly sourceAsset: ISource;\n\n  /**\n   * If not specified, it will default to a strict and secure CSP that will score\n   * well on https://observatory.mozilla.org/.\n   */\n  readonly responseHeaderPolicy?: ResponseHeadersPolicy;\n\n  /**\n   * An optional edge function to transform S3 responses. This is useful\n   * for doing things like injecting metatags for search optimizations.\n   */\n  readonly edgeOriginRequestFunction?: experimental.EdgeFunction;\n\n  /**\n   * Policy for deleting resources. Defaults to destroy.\n   */\n  readonly removalPolicy?: RemovalPolicy;\n}\n\n/**\n * A construct to create a secure single page application react website.\n */\nexport class ReactWebsiteConstruct extends Construct {\n  constructor(scope: Construct, id: string, props: ReactWebsiteProps) {\n    super(scope, id);\n    constructConstructsFromProps(this, props);\n  }\n}\n\n/**\n * @Deprecated\n *\n * This function is not meant to be used by others. For Pinegraph, reorganizing\n * resources into constructs causes the logical ids to change. See https://github.com/aws/aws-cdk-rfcs/issues/162\n */\nexport function constructConstructsFromProps(\n  self: Construct,\n  props: ReactWebsiteProps\n) {\n  const domain = props.domainName;\n  const allDomainNames = [domain];\n\n  const zones: { [name: string]: IHostedZone } = {};\n  for (const d of allDomainNames) {\n    const resourceId = getResourceId(d, \"Zone\");\n    zones[d] = HostedZone.fromLookup(self, resourceId, {\n      domainName: domain,\n    });\n  }\n  const cloudfrontOAI = new OriginAccessIdentity(self, \"cloudfront-OAI\", {\n    comment: `OAI for ${domain}`,\n  });\n\n  new CfnOutput(self, \"Site\", { value: \"https://\" + domain });\n\n  // Content bucket\n  const siteBucket = new Bucket(self, \"SiteBucket\", {\n    bucketName: domain,\n    publicReadAccess: false,\n    blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n    removalPolicy: props.removalPolicy,\n    autoDeleteObjects: props.removalPolicy === RemovalPolicy.DESTROY,\n  });\n  // Grant access to cloudfront\n  siteBucket.addToResourcePolicy(\n    new PolicyStatement({\n      actions: [\"s3:GetObject\"],\n      resources: [siteBucket.arnForObjects(\"*\")],\n      principals: [\n        new CanonicalUserPrincipal(\n          cloudfrontOAI.cloudFrontOriginAccessIdentityS3CanonicalUserId\n        ),\n      ],\n    })\n  );\n  new CfnOutput(self, \"Bucket\", { value: siteBucket.bucketName });\n\n  // TLS certificate\n  const certificate = new DnsValidatedCertificate(self, \"SiteCertificate\", {\n    domainName: domain,\n    subjectAlternativeNames: allDomainNames,\n    hostedZone: zones[domain],\n    region: \"us-east-1\", // Cloudfront only checks this region for certificates.\n  });\n  new CfnOutput(self, \"Certificate\", { value: certificate.certificateArn });\n  const responseHeaderPolicy =\n    props.responseHeaderPolicy ||\n    new ResponseHeadersPolicy(self, \"SecurityHeadersResponsePolicy\", {\n      comment: \"Security headers response policy\",\n      securityHeadersBehavior: {\n        contentSecurityPolicy: {\n          override: true,\n          contentSecurityPolicy: `default-src 'self';`,\n        },\n        strictTransportSecurity: {\n          override: true,\n          accessControlMaxAge: Duration.days(365),\n          includeSubdomains: true,\n          preload: true,\n        },\n        contentTypeOptions: {\n          override: true,\n        },\n        referrerPolicy: {\n          override: true,\n          referrerPolicy: HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN,\n        },\n        xssProtection: {\n          override: true,\n          protection: true,\n          modeBlock: true,\n        },\n        frameOptions: {\n          override: true,\n          frameOption: HeadersFrameOption.DENY,\n        },\n      },\n    });\n\n  // CloudFront distribution\n  const distribution = new Distribution(self, \"SiteDistribution\", {\n    certificate: certificate,\n    domainNames: allDomainNames,\n    comment: props.domainName,\n    minimumProtocolVersion: SecurityPolicyProtocol.TLS_V1_2016,\n    defaultBehavior: {\n      origin: new S3Origin(siteBucket, {\n        originAccessIdentity: cloudfrontOAI,\n      }),\n      edgeLambdas: props.edgeOriginRequestFunction\n        ? [\n            {\n              functionVersion: props.edgeOriginRequestFunction.currentVersion,\n              eventType: LambdaEdgeEventType.ORIGIN_REQUEST,\n            },\n          ]\n        : [],\n      compress: true,\n      allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n      viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      responseHeadersPolicy: responseHeaderPolicy,\n    },\n\n    /**\n     * This ensures that all routes end up using index.html for client side rendering.\n     */\n    errorResponses: [\n      {\n        httpStatus: 403,\n        responseHttpStatus: 200,\n        responsePagePath: \"/index.html\",\n      },\n    ],\n    defaultRootObject: \"index.html\",\n  });\n\n  new CfnOutput(self, \"DistributionId\", {\n    value: distribution.distributionId,\n  });\n\n  // Route53 alias record for the CloudFront distribution]\n  for (const d of allDomainNames) {\n    const resourceId = getResourceId(d, \"SiteAliasRecord\");\n    new ARecord(self, resourceId, {\n      recordName: d,\n      target: RecordTarget.fromAlias(new CloudFrontTarget(distribution)),\n      zone: zones[d],\n    });\n  }\n\n  // Deploy site contents to S3 bucket\n  new BucketDeployment(self, \"DeployWithInvalidation\", {\n    sources: [props.sourceAsset],\n    destinationBucket: siteBucket,\n    distribution,\n    distributionPaths: [\"/*\"],\n  });\n}\n\nfunction capitalizeFirstLetter(str: string) {\n  return str.charAt(0).toUpperCase() + str.slice(1);\n}\n\n/**\n * A simple function to make the resource id human readable.\n */\nfunction getResourceId(domain: string, resourceType: string) {\n  if (!domain.includes(\".\")) {\n    return resourceType;\n  }\n  return capitalizeFirstLetter(`${domain.split(\".\")[0]}${resourceType}`);\n}\n"]} |
@@ -21,3 +21,3 @@ { | ||
"license": "Apache-2.0", | ||
"version": "0.0.4", | ||
"version": "0.0.5", | ||
"jest": { | ||
@@ -24,0 +24,0 @@ "testMatch": [ |
40659
209