UNPKG

open-next-cdk

Version:

Deploy a NextJS app using OpenNext packaging to serverless AWS using CDK

123 lines 17.4 kB
"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"]}