@scloud/cdk-patterns
Version:
Serverless CDK patterns for common infrastructure needs
106 lines • 19.4 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.WebApp = void 0;
const aws_cdk_lib_1 = require("aws-cdk-lib");
const aws_certificatemanager_1 = require("aws-cdk-lib/aws-certificatemanager");
const aws_route53_targets_1 = require("aws-cdk-lib/aws-route53-targets");
const aws_cloudfront_origins_1 = require("aws-cdk-lib/aws-cloudfront-origins");
const aws_cloudfront_1 = require("aws-cdk-lib/aws-cloudfront");
const aws_apigateway_1 = require("aws-cdk-lib/aws-apigateway");
const constructs_1 = require("constructs");
const aws_route53_1 = require("aws-cdk-lib/aws-route53");
const RedirectWww_1 = require("./RedirectWww");
const GithubActions_1 = require("./GithubActions");
const PrivateBucket_1 = require("./PrivateBucket");
const ZipFunction_1 = require("./ZipFunction");
/**
* Builds a dynamic web application, backed by a single Lambda function, also knowm as a "Lambda-lith" (https://github.com/cdk-patterns/serverless/blob/main/the-lambda-trilogy/README.md)
*
* This construct sends requests that don't have a file extension to the Lambda. Static content is handled by routing requests that match *.* (eg *.js. *.css) to an S3 bucket.
*/
class WebApp extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, `${id}WebApp`);
const domainName = props.domainName || `${props.zone.zoneName}`;
// Static content
this.bucket = PrivateBucket_1.PrivateBucket.expendable(scope, `${id}Static`);
(0, GithubActions_1.githubActions)(scope).addGhaBucket(id, this.bucket);
// Web app handler - default values can be overridden using lambdaProps
this.lambda = props.lambda;
this.api = new aws_apigateway_1.LambdaRestApi(scope, `${id}ApiGateway`, {
handler: this.lambda,
proxy: true,
description: `${aws_cdk_lib_1.Stack.of(scope).stackName} ${id}`,
binaryMediaTypes: ['multipart/form-data'],
});
this.certificate = new aws_certificatemanager_1.DnsValidatedCertificate(scope, `${id}Certificate`, {
domainName,
hostedZone: props.zone,
region: 'us-east-1',
subjectAlternativeNames: props.redirectWww !== false ? [`www.${domainName}`] : undefined,
});
// This enables us to separate out the defaultBehavior props (if any) from the distributionProps (if provided)
// See https://stackoverflow.com/a/34710102/723506 for an explanation of this destructuring
const { defaultBehavior, additionalBehaviors, ...distributionProps } = props.distributionProps || {};
this.distribution = new aws_cloudfront_1.Distribution(scope, `${id}Distribution`, {
domainNames: [domainName],
comment: domainName,
defaultRootObject: props.defaultIndex ? 'index.html' : undefined,
defaultBehavior: {
origin: new aws_cloudfront_origins_1.RestApiOrigin(this.api),
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_ALL,
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
cachePolicy: aws_cloudfront_1.CachePolicy.CACHING_DISABLED, // Assume dynamic content
originRequestPolicy: aws_cloudfront_1.OriginRequestPolicy.ALL_VIEWER_EXCEPT_HOST_HEADER,
...defaultBehavior,
},
// All requests for something with a file extension go to s3 (actually, any path that contains a period).
// The aim is to route *.css, *.js, *.jpeg, etc)
additionalBehaviors: {
'*.*': {
origin: aws_cloudfront_origins_1.S3BucketOrigin.withOriginAccessControl(this.bucket),
allowedMethods: aws_cloudfront_1.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
compress: true,
},
...additionalBehaviors,
},
certificate: this.certificate,
errorResponses: props.errorResponses,
...distributionProps,
});
(0, GithubActions_1.githubActions)(scope).addGhaDistribution(id, this.distribution);
// DNS record for the Cloudfront distribution
new aws_route53_1.ARecord(scope, `${id}ARecord`, {
recordName: domainName,
target: aws_route53_1.RecordTarget.fromAlias(new aws_route53_targets_1.CloudFrontTarget(this.distribution)),
zone: props.zone,
});
if (props.redirectWww !== false)
new RedirectWww_1.RedirectWww(scope, id, { zone: props.zone, certificate: this.certificate, domainName });
}
/**
* Creates a WebApp backed by a Node.js Lambda function.
*
* Memory defaults to 3008 MB because this has the effest of assigning more compute resource and therefore reduces latency.
*/
static node(scope, id, zone, domainName, defaultIndex, redirectWww, functionProps) {
const lambda = ZipFunction_1.ZipFunction.node(scope, id, { memorySize: 3008, ...functionProps });
return new WebApp(scope, id, {
lambda, zone, domainName, defaultIndex, redirectWww,
});
}
/**
* Creates a WebApp backed by a Python Lambda function.
*
* Memory defaults to 3008 MB because this has the effest of assigning more compute resource and therefore reduces latency.
*/
static python(scope, id, zone, domainName, defaultIndex, redirectWww, functionProps) {
const lambda = ZipFunction_1.ZipFunction.python(scope, id, { memorySize: 3008, ...functionProps });
return new WebApp(scope, id, {
lambda, zone, domainName, defaultIndex, redirectWww,
});
}
}
exports.WebApp = WebApp;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"WebApp.js","sourceRoot":"","sources":["../src/WebApp.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Stack,\n} from 'aws-cdk-lib';\nimport { DnsValidatedCertificate } from 'aws-cdk-lib/aws-certificatemanager';\nimport { CloudFrontTarget } from 'aws-cdk-lib/aws-route53-targets';\nimport { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { RestApiOrigin, S3BucketOrigin } from 'aws-cdk-lib/aws-cloudfront-origins';\nimport {\n  AllowedMethods, CachePolicy, Distribution,\n  DistributionProps,\n  ErrorResponse,\n  OriginRequestPolicy,\n  ViewerProtocolPolicy,\n} from 'aws-cdk-lib/aws-cloudfront';\nimport {\n  LambdaRestApi,\n} from 'aws-cdk-lib/aws-apigateway';\nimport { Function } from 'aws-cdk-lib/aws-lambda';\nimport { Construct } from 'constructs';\nimport { ARecord, IHostedZone, RecordTarget } from 'aws-cdk-lib/aws-route53';\nimport { RedirectWww } from './RedirectWww';\nimport { githubActions } from './GithubActions';\nimport { PrivateBucket } from './PrivateBucket';\nimport { ZipFunction, ZipFunctionProps } from './ZipFunction';\n/**\n * @param lambda The function which will respond to incoming request events.\n * @param zone The DNS zone for this web app.\n * @param domainName Optional: by default the zone name will be used (e.g. 'example.com') a different domain here (e.g. 'subdomain.example.com').\n * @param defaultIndex Default: false. If true, maps a viewer request for '/' to an s3 request for /index.html.\n * @param redirectWww Default: true. Redirects www requests to the bare domain name, e.g. www.example.com->example.com, www.sub.example.com->sub.example.com.\n * @param distributionProps Optional: If you want to add additional properties to the Cloudfront distribution, you can pass them here.\n * @param errorResponses Optional: If you want to add custom error responses to the Cloudfront distribution, you can pass them here.\n */\nexport interface WebAppProps {\n  lambda: Function,\n  zone: IHostedZone,\n  domainName?: string,\n  defaultIndex?: boolean,\n  redirectWww?: boolean,\n  distributionProps?: Partial<DistributionProps>,\n  errorResponses?: ErrorResponse[],\n}\n\n/**\n * Builds a dynamic web application, backed by a single Lambda function, also knowm as a \"Lambda-lith\" (https://github.com/cdk-patterns/serverless/blob/main/the-lambda-trilogy/README.md)\n *\n * This construct sends requests that don't have a file extension to the Lambda. Static content is handled by routing requests that match *.* (eg *.js. *.css) to an S3 bucket.\n */\nexport class WebApp extends Construct {\n  lambda: Function;\n\n  bucket: Bucket;\n\n  distribution: Distribution;\n\n  api: LambdaRestApi;\n\n  certificate: DnsValidatedCertificate;\n\n  constructor(\n    scope: Construct,\n    id: string,\n    props: WebAppProps,\n  ) {\n    super(scope, `${id}WebApp`);\n\n    const domainName = props.domainName || `${props.zone.zoneName}`;\n\n    // Static content\n    this.bucket = PrivateBucket.expendable(scope, `${id}Static`);\n    githubActions(scope).addGhaBucket(id, this.bucket);\n\n    // Web app handler - default values can be overridden using lambdaProps\n    this.lambda = props.lambda;\n\n    this.api = new LambdaRestApi(scope, `${id}ApiGateway`, {\n      handler: this.lambda,\n      proxy: true,\n      description: `${Stack.of(scope).stackName} ${id}`,\n      binaryMediaTypes: ['multipart/form-data'],\n    });\n\n    this.certificate = new DnsValidatedCertificate(scope, `${id}Certificate`, {\n      domainName,\n      hostedZone: props.zone,\n      region: 'us-east-1',\n      subjectAlternativeNames: props.redirectWww !== false ? [`www.${domainName}`] : undefined,\n    });\n\n    // This enables us to separate out the defaultBehavior props (if any) from the distributionProps (if provided)\n    // See https://stackoverflow.com/a/34710102/723506 for an explanation of this destructuring\n    const { defaultBehavior, additionalBehaviors, ...distributionProps } = props.distributionProps || ({} as Partial<DistributionProps>);\n    this.distribution = new Distribution(scope, `${id}Distribution`, {\n      domainNames: [domainName],\n      comment: domainName,\n      defaultRootObject: props.defaultIndex ? 'index.html' : undefined,\n      defaultBehavior: {\n        origin: new RestApiOrigin(this.api),\n        allowedMethods: AllowedMethods.ALLOW_ALL,\n        viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        cachePolicy: CachePolicy.CACHING_DISABLED, // Assume dynamic content\n        originRequestPolicy: OriginRequestPolicy.ALL_VIEWER_EXCEPT_HOST_HEADER,\n        ...defaultBehavior,\n      },\n      // All requests for something with a file extension go to s3 (actually, any path that contains a period).\n      // The aim is to route *.css, *.js, *.jpeg, etc)\n      additionalBehaviors: {\n        '*.*': {\n          origin: S3BucketOrigin.withOriginAccessControl(this.bucket),\n          allowedMethods: AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n          viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n          compress: true,\n        },\n        ...additionalBehaviors,\n      },\n      certificate: this.certificate,\n      errorResponses: props.errorResponses,\n      ...distributionProps,\n    });\n    githubActions(scope).addGhaDistribution(id, this.distribution);\n\n    // DNS record for the Cloudfront distribution\n    new ARecord(scope, `${id}ARecord`, {\n      recordName: domainName,\n      target: RecordTarget.fromAlias(new CloudFrontTarget(this.distribution)),\n      zone: props.zone,\n    });\n\n    if (props.redirectWww !== false) new RedirectWww(scope, id, { zone: props.zone, certificate: this.certificate, domainName });\n  }\n\n  /**\n   * Creates a WebApp backed by a Node.js Lambda function.\n   *\n   * Memory defaults to 3008 MB because this has the effest of assigning more compute resource and therefore reduces latency.\n   */\n  static node(\n    scope: Construct,\n    id: string,\n    zone: IHostedZone,\n    domainName?: string,\n    defaultIndex?: boolean,\n    redirectWww?: boolean,\n    functionProps?: ZipFunctionProps,\n  ): WebApp {\n    const lambda = ZipFunction.node(scope, id, { memorySize: 3008, ...functionProps });\n    return new WebApp(scope, id, {\n      lambda, zone, domainName, defaultIndex, redirectWww,\n    });\n  }\n\n  /**\n   * Creates a WebApp backed by a Python Lambda function.\n   *\n   * Memory defaults to 3008 MB because this has the effest of assigning more compute resource and therefore reduces latency.\n   */\n  static python(\n    scope: Construct,\n    id: string,\n    zone: IHostedZone,\n    domainName?: string,\n    defaultIndex?: boolean,\n    redirectWww?: boolean,\n    functionProps?: ZipFunctionProps,\n  ): WebApp {\n    const lambda = ZipFunction.python(scope, id, { memorySize: 3008, ...functionProps });\n    return new WebApp(scope, id, {\n      lambda, zone, domainName, defaultIndex, redirectWww,\n    });\n  }\n}\n"]}