open-next-cdk
Version:
Deploy a NextJS app using OpenNext packaging to serverless AWS using CDK
123 lines • 17.4 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.getS3ReplaceValues = exports.NextjsS3EnvRewriter = exports.replaceTokenGlobs = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const fs = require("fs");
const os = require("os");
const path = require("path");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const iam = require("aws-cdk-lib/aws-iam");
const lambda = require("aws-cdk-lib/aws-lambda");
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
const cr = require("aws-cdk-lib/custom-resources");
const constructs_1 = require("constructs");
const BundleFunction_1 = require("./BundleFunction");
const NextjsBuild_1 = require("./NextjsBuild");
// files to rewrite CloudFormation tokens in environment variables
exports.replaceTokenGlobs = ['**/*.html', '**/*.js', '**/*.cjs', '**/*.mjs', '**/*.json'];
/**
* Rewrites variables in S3 objects after a deployment happens to
* replace CloudFormation tokens with their values.
* These values are not resolved at build time because they are
* only known at deploy time.
*/
class NextjsS3EnvRewriter extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
const { s3Bucket, s3keys, replacementConfig, debug, cloudfrontDistributionId } = props;
if (s3keys.length === 0)
return;
const app = aws_cdk_lib_1.App.of(this);
const tmpDir = props.tempBuildDir
? path.resolve(path.join(props.tempBuildDir, 'static'))
: fs.mkdtempSync(path.join(os.tmpdir(), 'static-'));
// create a custom resource to find and replace tokenized strings in static files
// must happen after deployment when tokens can be resolved
// compile function
const inputPath = path.resolve(__dirname, '../assets/lambda/S3EnvRewriter.ts');
const outputPath = path.join(tmpDir, 'deployment-scripts', 'S3EnvRewriter.cjs');
const handlerDir = BundleFunction_1.bundleFunction({
inputPath,
outputPath,
bundleOptions: {
bundle: true,
sourcemap: true,
external: ['aws-sdk'],
target: 'node16',
platform: 'node',
format: 'cjs',
},
});
// rewriter lambda function
const rewriteFn = new lambda.Function(this, 'RewriteOnEventHandler', {
runtime: aws_lambda_1.Runtime.NODEJS_16_X,
memorySize: 1024,
timeout: aws_cdk_lib_1.Duration.minutes(5),
handler: 'S3EnvRewriter.handler',
code: lambda.Code.fromAsset(handlerDir),
initialPolicy: [
new iam.PolicyStatement({
actions: ['s3:GetObject', 's3:PutObject'],
resources: [s3Bucket.arnForObjects('*')],
}),
...(cloudfrontDistributionId
? [
new iam.PolicyStatement({
actions: ['cloudfront:CreateInvalidation'],
resources: [`arn:aws:cloudfront::${app.account}:distribution/${cloudfrontDistributionId}`],
}),
]
: []),
],
});
// grant permission to read env var config if provided
if (replacementConfig.jsonS3Bucket && replacementConfig.jsonS3Key) {
const bucket = typeof replacementConfig.jsonS3Bucket === 'string'
? aws_s3_1.Bucket.fromBucketName(this, 'EnvConfigBucket', replacementConfig.jsonS3Bucket)
: replacementConfig.jsonS3Bucket;
rewriteFn.addToRolePolicy(new iam.PolicyStatement({
actions: ['s3:GetObject'],
resources: [bucket.arnForObjects(replacementConfig.jsonS3Key)],
}));
}
// custom resource to run the rewriter after files are copied and we can resolve token values
const provider = new cr.Provider(this, 'RewriteStaticProvider', {
onEventHandler: rewriteFn,
});
// params for the rewriter function
const properties = {
bucket: s3Bucket.bucketName,
s3keys,
replacementConfig: {
...replacementConfig,
jsonS3Bucket: replacementConfig.jsonS3Bucket?.bucketName,
},
debug,
cloudfrontDistributionId,
};
this.rewriteNode = new aws_cdk_lib_1.CustomResource(this, 'RewriteStatic', {
serviceToken: provider.serviceToken,
properties,
});
}
}
exports.NextjsS3EnvRewriter = NextjsS3EnvRewriter;
_a = JSII_RTTI_SYMBOL_1;
NextjsS3EnvRewriter[_a] = { fqn: "open-next-cdk.NextjsS3EnvRewriter", version: "0.0.10" };
// inline env vars for client and server code
// these are values to replace in built code after it's deployed to S3/lambda
function getS3ReplaceValues(environment, publicOnly) {
const replacements = {};
Object.entries(environment || {})
.filter(([, value]) => aws_cdk_lib_1.Token.isUnresolved(value))
.filter(([key]) => !publicOnly || key.startsWith('NEXT_PUBLIC_')) // don't replace server-only env vars
.forEach(([key, value]) => {
const token = NextjsBuild_1.makeTokenPlaceholder(key);
replacements[token] = value.toString();
});
return replacements;
}
exports.getS3ReplaceValues = getS3ReplaceValues;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NextjsS3EnvRewriter.js","sourceRoot":"","sources":["../src/NextjsS3EnvRewriter.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { App, CustomResource, Duration, Token } from 'aws-cdk-lib';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport { Runtime } from 'aws-cdk-lib/aws-lambda';\nimport { Bucket, IBucket } from 'aws-cdk-lib/aws-s3';\nimport * as cr from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\nimport { bundleFunction } from './BundleFunction';\nimport { NextjsBaseProps } from './NextjsBase';\nimport { makeTokenPlaceholder } from './NextjsBuild';\n\n// files to rewrite CloudFormation tokens in environment variables\nexport const replaceTokenGlobs = ['**/*.html', '**/*.js', '**/*.cjs', '**/*.mjs', '**/*.json'];\n\nexport interface RewriteReplacementsConfig {\n  readonly env?: Record<string, string>; // replace keys with values in files\n  readonly jsonS3Bucket?: IBucket;\n  readonly jsonS3Key?: string;\n}\nexport interface RewriterParams {\n  readonly s3Bucket: IBucket;\n  readonly s3keys: string[]; // files to rewrite\n  readonly replacementConfig: RewriteReplacementsConfig;\n  readonly debug?: boolean;\n  readonly cloudfrontDistributionId?: string;\n}\n\nexport interface NextjsS3EnvRewriterProps extends NextjsBaseProps, RewriterParams {}\n\n/**\n * Rewrites variables in S3 objects after a deployment happens to\n * replace CloudFormation tokens with their values.\n * These values are not resolved at build time because they are\n * only known at deploy time.\n */\nexport class NextjsS3EnvRewriter extends Construct {\n  public rewriteNode?: Construct;\n\n  constructor(scope: Construct, id: string, props: NextjsS3EnvRewriterProps) {\n    super(scope, id);\n\n    const { s3Bucket, s3keys, replacementConfig, debug, cloudfrontDistributionId } = props;\n\n    if (s3keys.length === 0) return;\n\n    const app = App.of(this) as App;\n\n    const tmpDir = props.tempBuildDir\n      ? path.resolve(path.join(props.tempBuildDir, 'static'))\n      : fs.mkdtempSync(path.join(os.tmpdir(), 'static-'));\n\n    // create a custom resource to find and replace tokenized strings in static files\n    // must happen after deployment when tokens can be resolved\n    // compile function\n    const inputPath = path.resolve(__dirname, '../assets/lambda/S3EnvRewriter.ts');\n    const outputPath = path.join(tmpDir, 'deployment-scripts', 'S3EnvRewriter.cjs');\n    const handlerDir = bundleFunction({\n      inputPath,\n      outputPath,\n      bundleOptions: {\n        bundle: true,\n        sourcemap: true,\n        external: ['aws-sdk'],\n        target: 'node16',\n        platform: 'node',\n        format: 'cjs',\n      },\n    });\n\n    // rewriter lambda function\n    const rewriteFn = new lambda.Function(this, 'RewriteOnEventHandler', {\n      runtime: Runtime.NODEJS_16_X,\n      memorySize: 1024,\n      timeout: Duration.minutes(5),\n      handler: 'S3EnvRewriter.handler',\n      code: lambda.Code.fromAsset(handlerDir),\n      initialPolicy: [\n        new iam.PolicyStatement({\n          actions: ['s3:GetObject', 's3:PutObject'],\n          resources: [s3Bucket.arnForObjects('*')],\n        }),\n        ...(cloudfrontDistributionId\n          ? [\n              new iam.PolicyStatement({\n                actions: ['cloudfront:CreateInvalidation'],\n                resources: [`arn:aws:cloudfront::${app.account}:distribution/${cloudfrontDistributionId}`],\n              }),\n            ]\n          : []),\n      ],\n    });\n    // grant permission to read env var config if provided\n    if (replacementConfig.jsonS3Bucket && replacementConfig.jsonS3Key) {\n      const bucket: IBucket =\n        typeof replacementConfig.jsonS3Bucket === 'string'\n          ? Bucket.fromBucketName(this, 'EnvConfigBucket', replacementConfig.jsonS3Bucket)\n          : replacementConfig.jsonS3Bucket;\n      rewriteFn.addToRolePolicy(\n        new iam.PolicyStatement({\n          actions: ['s3:GetObject'],\n          resources: [bucket.arnForObjects(replacementConfig.jsonS3Key)],\n        })\n      );\n    }\n\n    // custom resource to run the rewriter after files are copied and we can resolve token values\n    const provider = new cr.Provider(this, 'RewriteStaticProvider', {\n      onEventHandler: rewriteFn,\n    });\n    // params for the rewriter function\n    const properties = {\n      bucket: s3Bucket.bucketName,\n      s3keys,\n      replacementConfig: {\n        ...replacementConfig,\n        jsonS3Bucket: replacementConfig.jsonS3Bucket?.bucketName,\n      },\n      debug,\n      cloudfrontDistributionId,\n    };\n    this.rewriteNode = new CustomResource(this, 'RewriteStatic', {\n      serviceToken: provider.serviceToken,\n      properties,\n    });\n  }\n}\n\n// inline env vars for client and server code\n// these are values to replace in built code after it's deployed to S3/lambda\nexport function getS3ReplaceValues(environment: Record<string, string>, publicOnly: boolean): Record<string, string> {\n  const replacements: Record<string, string> = {};\n\n  Object.entries(environment || {})\n    .filter(([, value]) => Token.isUnresolved(value))\n    .filter(([key]) => !publicOnly || key.startsWith('NEXT_PUBLIC_')) // don't replace server-only env vars\n    .forEach(([key, value]) => {\n      const token = makeTokenPlaceholder(key);\n      replacements[token] = value.toString();\n    });\n\n  return replacements;\n}\n"]}