cdk-nextjs-standalone
Version:
Deploy a NextJS app to AWS using CDK and OpenNext.
79 lines • 14.1 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.NextjsBucketDeployment = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const path = require("node:path");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
const constructs_1 = require("constructs");
const common_lambda_props_1 = require("./utils/common-lambda-props");
/**
* Similar to CDK's `BucketDeployment` construct, but with a focus on replacing
* template placeholders (i.e. environment variables) and configuring PUT
* options like cache control.
*/
class NextjsBucketDeployment extends constructs_1.Construct {
/**
* Formats a string as a template value so custom resource knows to replace.
*/
static getSubstitutionValue(v) {
return `{{ ${v} }}`;
}
/**
* Creates `substitutionConfig` an object by extracting unresolved tokens.
*/
static getSubstitutionConfig(env) {
const substitutionConfig = {};
for (const [k, v] of Object.entries(env)) {
if (aws_cdk_lib_1.Token.isUnresolved(v)) {
substitutionConfig[NextjsBucketDeployment.getSubstitutionValue(k)] = v;
}
}
return substitutionConfig;
}
constructor(scope, id, props) {
super(scope, id);
this.props = props;
this.function = this.createFunction();
this.createCustomResource(this.function.functionArn);
}
createFunction() {
const fn = new aws_lambda_1.Function(this, 'Fn', {
...(0, common_lambda_props_1.getCommonFunctionProps)(this),
code: aws_lambda_1.Code.fromAsset(path.resolve(__dirname, '..', 'assets', 'lambdas', 'nextjs-bucket-deployment')),
handler: 'index.handler',
timeout: aws_cdk_lib_1.Duration.minutes(5),
...this.props.overrides?.functionProps,
});
if (this.props.debug) {
fn.addEnvironment('DEBUG', '1');
}
this.props.asset.grantRead(fn);
this.props.destinationBucket.grantReadWrite(fn);
return fn;
}
createCustomResource(serviceToken) {
const properties = {
sourceBucketName: this.props.asset.s3BucketName,
sourceKeyPrefix: this.props.asset.s3ObjectKey,
destinationBucketName: this.props.destinationBucket.bucketName,
destinationKeyPrefix: this.props.destinationKeyPrefix,
putConfig: this.props.putConfig,
prune: this.props.prune ?? false,
substitutionConfig: this.props.substitutionConfig,
zip: this.props.zip,
queueSize: this.props.queueSize,
};
return new aws_cdk_lib_1.CustomResource(this, 'CustomResource', {
properties,
resourceType: 'Custom::NextjsBucketDeployment',
serviceToken,
...this.props.overrides?.customResourceProps,
});
}
}
exports.NextjsBucketDeployment = NextjsBucketDeployment;
_a = JSII_RTTI_SYMBOL_1;
NextjsBucketDeployment[_a] = { fqn: "cdk-nextjs-standalone.NextjsBucketDeployment", version: "4.3.0" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NextjsBucketDeployment.js","sourceRoot":"","sources":["../src/NextjsBucketDeployment.ts"],"names":[],"mappings":";;;;;AAAA,kCAAkC;AAClC,6CAA8D;AAC9D,uDAAwD;AAGxD,2CAAuC;AAEvC,qEAAqE;AAsFrE;;;;GAIG;AACH,MAAa,sBAAuB,SAAQ,sBAAS;IACnD;;OAEG;IACH,MAAM,CAAC,oBAAoB,CAAC,CAAS;QACnC,OAAO,MAAM,CAAC,KAAK,CAAC;IACtB,CAAC;IACD;;OAEG;IACH,MAAM,CAAC,qBAAqB,CAAC,GAA2B;QACtD,MAAM,kBAAkB,GAA2B,EAAE,CAAC;QACtD,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;YACzC,IAAI,mBAAK,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,CAAC;gBAC1B,kBAAkB,CAAC,sBAAsB,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;YACzE,CAAC;QACH,CAAC;QACD,OAAO,kBAAkB,CAAC;IAC5B,CAAC;IAOD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAkC;QAC1E,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACjB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;QACtC,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;IACvD,CAAC;IAEO,cAAc;QACpB,MAAM,EAAE,GAAG,IAAI,qBAAQ,CAAC,IAAI,EAAE,IAAI,EAAE;YAClC,GAAG,IAAA,4CAAsB,EAAC,IAAI,CAAC;YAC/B,IAAI,EAAE,iBAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,IAAI,EAAE,QAAQ,EAAE,SAAS,EAAE,0BAA0B,CAAC,CAAC;YACpG,OAAO,EAAE,eAAe;YACxB,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAC5B,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,aAAa;SACvC,CAAC,CAAC;QACH,IAAI,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC;YACrB,EAAE,CAAC,cAAc,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;QAClC,CAAC;QACD,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;QAC/B,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,cAAc,CAAC,EAAE,CAAC,CAAC;QAChD,OAAO,EAAE,CAAC;IACZ,CAAC;IAEO,oBAAoB,CAAC,YAAoB;QAC/C,MAAM,UAAU,GAA6B;YAC3C,gBAAgB,EAAE,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,YAAY;YAC/C,eAAe,EAAE,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,WAAW;YAC7C,qBAAqB,EAAE,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,UAAU;YAC9D,oBAAoB,EAAE,IAAI,CAAC,KAAK,CAAC,oBAAoB;YACrD,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,SAAS;YAC/B,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,KAAK,IAAI,KAAK;YAChC,kBAAkB,EAAE,IAAI,CAAC,KAAK,CAAC,kBAAkB;YACjD,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,GAAG;YACnB,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,SAAS;SAChC,CAAC;QACF,OAAO,IAAI,4BAAc,CAAC,IAAI,EAAE,gBAAgB,EAAE;YAChD,UAAU;YACV,YAAY,EAAE,gCAAgC;YAC9C,YAAY;YACZ,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,mBAAmB;SAC7C,CAAC,CAAC;IACL,CAAC;;AAlEH,wDAmEC","sourcesContent":["import * as path from 'node:path';\nimport { CustomResource, Duration, Token } from 'aws-cdk-lib';\nimport { Code, Function } from 'aws-cdk-lib/aws-lambda';\nimport { IBucket } from 'aws-cdk-lib/aws-s3';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { Construct } from 'constructs';\nimport { OptionalCustomResourceProps, OptionalFunctionProps } from './generated-structs';\nimport { getCommonFunctionProps } from './utils/common-lambda-props';\n\nexport interface NextjsBucketDeploymentOverrides {\n  readonly functionProps?: OptionalFunctionProps;\n  readonly customResourceProps?: OptionalCustomResourceProps;\n}\n\nexport interface NextjsBucketDeploymentProps {\n  /**\n   * Source `Asset`\n   */\n  readonly asset: Asset;\n  /**\n   * Enable verbose output of Custom Resource Lambda\n   * @default false\n   */\n  readonly debug?: boolean | undefined;\n  /**\n   * If `true`, then delete old objects in `destinationBucket`/`destinationKeyPrefix`\n   * **after** uploading new objects. Only applies if `zip` is `false`.\n   *\n   * Old objects are determined by listing objects\n   * in bucket before creating new objects and finding the objects that aren't in\n   * the new objects.\n   *\n   * Note, if this is set to true then clients who have old HTML files (browser tabs opened before deployment)\n   * will reference JS, CSS files that do not exist in S3 reslting in 404s.\n   * @default false\n   */\n  readonly prune?: boolean | undefined;\n  /**\n   * Mapping of files to PUT options for `PutObjectCommand`. Keys of\n   * record must be a glob pattern (uses micromatch). Values of record are options\n   * for PUT command for AWS SDK JS V3. See [here](https://docs.aws.amazon.com/AWSJavaScriptSDK/v3/latest/Package/-aws-sdk-client-s3/Interface/PutObjectRequest/)\n   * for options. If a file matches multiple globs, configuration will be\n   * merged. Later entries override earlier entries.\n   *\n   * `Bucket`, `Key`, and `Body` PUT options cannot be set.\n   */\n  readonly putConfig?: Record<string, Record<string, string>>;\n  /**\n   * Destination S3 Bucket\n   */\n  readonly destinationBucket: IBucket;\n  /**\n   * Destination S3 Bucket Key Prefix\n   */\n  readonly destinationKeyPrefix?: string | undefined;\n  /**\n   * Override props for every construct.\n   */\n  readonly overrides?: NextjsBucketDeploymentOverrides;\n  /**\n   * Replace placeholders in all files in `asset`. Placeholder targets are\n   * defined by keys of record. Values to replace placeholders with are defined\n   * by values of record.\n   */\n  readonly substitutionConfig?: Record<string, string>;\n  /**\n   * If `true` then files will be zipped before writing to destination bucket.\n   *\n   * Useful for Lambda functions.\n   * @default false\n   */\n  readonly zip?: boolean | undefined;\n  /**\n   * The number of files to upload in parallel.\n   */\n  readonly queueSize?: number | undefined;\n}\n\n/**\n * @internal\n */\nexport interface CustomResourceProperties {\n  destinationBucketName: string;\n  destinationKeyPrefix?: string;\n  prune?: boolean | undefined;\n  putConfig?: NextjsBucketDeploymentProps['putConfig'];\n  queueSize?: number | undefined;\n  substitutionConfig?: NextjsBucketDeploymentProps['substitutionConfig'];\n  sourceBucketName: string;\n  sourceKeyPrefix?: string | undefined;\n  zip?: boolean | undefined;\n}\n\n/**\n * Similar to CDK's `BucketDeployment` construct, but with a focus on replacing\n * template placeholders (i.e. environment variables) and configuring PUT\n * options like cache control.\n */\nexport class NextjsBucketDeployment extends Construct {\n  /**\n   * Formats a string as a template value so custom resource knows to replace.\n   */\n  static getSubstitutionValue(v: string): string {\n    return `{{ ${v} }}`;\n  }\n  /**\n   * Creates `substitutionConfig` an object by extracting unresolved tokens.\n   */\n  static getSubstitutionConfig(env: Record<string, string>): Record<string, string> {\n    const substitutionConfig: Record<string, string> = {};\n    for (const [k, v] of Object.entries(env)) {\n      if (Token.isUnresolved(v)) {\n        substitutionConfig[NextjsBucketDeployment.getSubstitutionValue(k)] = v;\n      }\n    }\n    return substitutionConfig;\n  }\n  /**\n   * Lambda Function Provider for Custom Resource\n   */\n  function: Function;\n  private props: NextjsBucketDeploymentProps;\n\n  constructor(scope: Construct, id: string, props: NextjsBucketDeploymentProps) {\n    super(scope, id);\n    this.props = props;\n    this.function = this.createFunction();\n    this.createCustomResource(this.function.functionArn);\n  }\n\n  private createFunction() {\n    const fn = new Function(this, 'Fn', {\n      ...getCommonFunctionProps(this),\n      code: Code.fromAsset(path.resolve(__dirname, '..', 'assets', 'lambdas', 'nextjs-bucket-deployment')),\n      handler: 'index.handler',\n      timeout: Duration.minutes(5),\n      ...this.props.overrides?.functionProps,\n    });\n    if (this.props.debug) {\n      fn.addEnvironment('DEBUG', '1');\n    }\n    this.props.asset.grantRead(fn);\n    this.props.destinationBucket.grantReadWrite(fn);\n    return fn;\n  }\n\n  private createCustomResource(serviceToken: string) {\n    const properties: CustomResourceProperties = {\n      sourceBucketName: this.props.asset.s3BucketName,\n      sourceKeyPrefix: this.props.asset.s3ObjectKey,\n      destinationBucketName: this.props.destinationBucket.bucketName,\n      destinationKeyPrefix: this.props.destinationKeyPrefix,\n      putConfig: this.props.putConfig,\n      prune: this.props.prune ?? false,\n      substitutionConfig: this.props.substitutionConfig,\n      zip: this.props.zip,\n      queueSize: this.props.queueSize,\n    };\n    return new CustomResource(this, 'CustomResource', {\n      properties,\n      resourceType: 'Custom::NextjsBucketDeployment',\n      serviceToken,\n      ...this.props.overrides?.customResourceProps,\n    });\n  }\n}\n"]}