open-next-cdk
Version:
Deploy a NextJS app using OpenNext packaging to serverless AWS using CDK
129 lines • 22.1 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.listDirectory = exports.NextJsAssetsDeployment = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const os = require("os");
const path = require("path");
const aws_s3_deployment_1 = require("aws-cdk-lib/aws-s3-deployment");
const constructs_1 = require("constructs");
const fs = require("fs-extra");
const micromatch = require("micromatch");
const constants_1 = require("./constants");
const NextjsBuild_1 = require("./NextjsBuild");
const NextjsS3EnvRewriter_1 = require("./NextjsS3EnvRewriter");
/**
* Uploads NextJS-built static and public files to S3.
*
* Will rewrite CloudFormation references with their resolved values after uploading.
*/
class NextJsAssetsDeployment extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
this.props = props;
this.bucket = props.bucket;
this.staticTempDir = this.prepareArchiveDirectory();
this.deployments = this.uploadS3Assets(this.staticTempDir);
// do rewrites of unresolved CDK tokens in static files
if (this.props.environment && !this.props.isPlaceholder) {
this.rewriter = new NextjsS3EnvRewriter_1.NextjsS3EnvRewriter(this, 'NextjsS3EnvRewriter', {
...props,
s3Bucket: this.bucket,
s3keys: this._getStaticFilesForRewrite(),
replacementConfig: {
env: NextjsS3EnvRewriter_1.getS3ReplaceValues(this.props.environment, true),
},
debug: false,
cloudfrontDistributionId: this.props.distribution?.distributionId,
});
// wait for s3 assets to be uploaded first before running
this.rewriter.node.addDependency(...this.deployments);
}
}
// arrange directory structure for S3 asset deployments
// should contain _next/static and ./ for public files
prepareArchiveDirectory() {
const archiveDir = this.props.tempBuildDir
? path.resolve(path.join(this.props.tempBuildDir, 'static'))
: fs.mkdtempSync(path.join(os.tmpdir(), 'static-'));
fs.mkdirpSync(archiveDir);
// theoretically we could move the files instead of copy for speed...
// path to public folder; root static assets
const staticDir = this.props.nextBuild.nextStaticDir;
if (!this.props.isPlaceholder && fs.existsSync(staticDir)) {
// copy public+static files to root
fs.copySync(this.props.nextBuild.nextStaticDir, archiveDir, {
recursive: true,
dereference: true,
preserveTimestamps: true,
});
}
return archiveDir;
}
uploadS3Assets(archiveDir) {
// zip up bucket contents and upload to bucket
const archiveZipFilePath = NextjsBuild_1.createArchive({
directory: archiveDir,
zipFileName: 'assets.zip',
zipOutDir: path.join(this.staticTempDir, 'assets'),
compressionLevel: this.props.compressionLevel,
quiet: this.props.quiet,
});
if (!archiveZipFilePath)
return [];
const maxAge = this.props.cachePolicies?.staticMaxAgeDefault?.toSeconds() ?? constants_1.DEFAULT_STATIC_MAX_AGE;
const staleWhileRevalidate = this.props.cachePolicies?.staticStaleWhileRevalidateDefault?.toSeconds() ?? constants_1.DEFAULT_STATIC_STALE_WHILE_REVALIDATE;
const cacheControl = aws_s3_deployment_1.CacheControl.fromString(`public,max-age=${maxAge},stale-while-revalidate=${staleWhileRevalidate},immutable`);
const deployment = new aws_s3_deployment_1.BucketDeployment(this, 'NextStaticAssetsS3Deployment', {
destinationBucket: this.bucket,
cacheControl: [cacheControl],
sources: [aws_s3_deployment_1.Source.asset(archiveZipFilePath)],
distribution: this.props.distribution,
prune: this.props.prune,
useEfs: this.props.useEfs,
vpc: this.props.vpc,
memoryLimit: this.props.memoryLimit,
ephemeralStorageSize: this.props.ephemeralStorageSize,
});
return [deployment];
}
_getStaticFilesForRewrite() {
const staticDir = this.staticTempDir;
const s3keys = [];
if (!fs.existsSync(staticDir)) {
return [];
}
listDirectory(staticDir).forEach((file) => {
const relativePath = path.relative(staticDir, file);
// skip bogus system files
if (relativePath.endsWith('.DS_Store'))
return;
// is this file a glob match?
if (!micromatch.isMatch(relativePath, NextjsS3EnvRewriter_1.replaceTokenGlobs, { dot: true })) {
return;
}
s3keys.push(relativePath);
});
return s3keys;
}
}
exports.NextJsAssetsDeployment = NextJsAssetsDeployment;
_a = JSII_RTTI_SYMBOL_1;
NextJsAssetsDeployment[_a] = { fqn: "open-next-cdk.NextJsAssetsDeployment", version: "0.0.10" };
function listDirectory(dir) {
const fileList = [];
const publicFiles = fs.readdirSync(dir);
for (const filename of publicFiles) {
const filepath = path.join(dir, filename);
const stat = fs.statSync(filepath);
if (stat.isDirectory()) {
fileList.push(...listDirectory(filepath));
}
else {
fileList.push(filepath);
}
}
return fileList;
}
exports.listDirectory = listDirectory;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NextjsAssetsDeployment.js","sourceRoot":"","sources":["../src/NextjsAssetsDeployment.ts"],"names":[],"mappings":"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","sourcesContent":["import * as os from 'os';\nimport * as path from 'path';\nimport { Duration, Size } from 'aws-cdk-lib';\nimport * as cloudfront from 'aws-cdk-lib/aws-cloudfront';\nimport { IVpc } from 'aws-cdk-lib/aws-ec2';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { BucketDeployment, CacheControl, Source } from 'aws-cdk-lib/aws-s3-deployment';\nimport { Construct } from 'constructs';\nimport * as fs from 'fs-extra';\nimport * as micromatch from 'micromatch';\nimport { DEFAULT_STATIC_MAX_AGE, DEFAULT_STATIC_STALE_WHILE_REVALIDATE } from './constants';\nimport { NextjsBaseProps } from './NextjsBase';\nimport { createArchive, NextjsBuild } from './NextjsBuild';\nimport { getS3ReplaceValues, NextjsS3EnvRewriter, replaceTokenGlobs } from './NextjsS3EnvRewriter';\n\nexport interface NextjsAssetsCachePolicyProps {\n  /**\n   * Cache-control max-age default for S3 static assets.\n   * Default: 30 days.\n   */\n  readonly staticMaxAgeDefault?: Duration;\n  /**\n   * Cache-control stale-while-revalidate default for S3 static assets.\n   * Default: 1 day.\n   */\n  readonly staticStaleWhileRevalidateDefault?: Duration;\n}\n\nexport interface NextjsAssetsDeploymentProps extends NextjsBaseProps {\n  /**\n   * The `NextjsBuild` instance representing the built Nextjs application.\n   */\n  readonly nextBuild: NextjsBuild;\n\n  /**\n   * Properties for the S3 bucket containing the NextJS assets.\n   */\n  readonly bucket: s3.IBucket;\n\n  /**\n   * Distribution to invalidate when assets change.\n   */\n  readonly distribution?: cloudfront.IDistribution;\n\n  /**\n   * Override the default S3 cache policies created internally.\n   */\n  readonly cachePolicies?: NextjsAssetsCachePolicyProps;\n\n  /**\n   * Set to true to delete old assets (defaults to false).\n   * Recommended to only set to true if you don't need the ability to roll back deployments.\n   */\n  readonly prune?: boolean;\n\n  /**\n   * In case of useEfs, vpc is required\n   */\n  readonly vpc?: IVpc;\n\n  /**\n   * In case of useEfs, vpc is required\n   */\n  readonly useEfs?: boolean;\n\n  /**\n   * memoryLimit for lambda function which been run by BucketDeployment\n   */\n  readonly memoryLimit?: number;\n\n  /**\n   * ephemeralStorageSize for lambda function which been run by BucketDeployment\n   */\n  readonly ephemeralStorageSize?: Size;\n}\n\n/**\n * Effectively a Partial<NextjsAssetsCachePolicyProps> to satisfy JSII\n */\nexport interface NextjsAssetsDeploymentPropsDefaults extends NextjsBaseProps {\n  /**\n   * The `NextjsBuild` instance representing the built Nextjs application.\n   */\n  readonly nextBuild?: NextjsBuild;\n\n  /**\n   * Properties for the S3 bucket containing the NextJS assets.\n   */\n  readonly bucket?: s3.IBucket;\n\n  /**\n   * Distribution to invalidate when assets change.\n   */\n  readonly distribution?: cloudfront.IDistribution;\n\n  /**\n   * Override the default S3 cache policies created internally.\n   */\n  readonly cachePolicies?: NextjsAssetsCachePolicyProps;\n\n  /**\n   * Set to true to delete old assets (defaults to false).\n   * Recommended to only set to true if you don't need the ability to roll back deployments.\n   */\n  readonly prune?: boolean;\n\n  /**\n   * In case of useEfs, vpc is required\n   */\n  readonly vpc?: IVpc;\n\n  /**\n   * In case of useEfs, vpc is required\n   */\n  readonly useEfs?: boolean;\n\n  /**\n   * memoryLimit for lambda function which been run by BucketDeployment\n   */\n  readonly memoryLimit?: number;\n\n  /**\n   * ephemeralStorageSize for lambda function which been run by BucketDeployment\n   */\n  readonly ephemeralStorageSize?: Size;\n}\n\n/**\n * Uploads NextJS-built static and public files to S3.\n *\n * Will rewrite CloudFormation references with their resolved values after uploading.\n */\nexport class NextJsAssetsDeployment extends Construct {\n  /**\n   * Bucket containing assets.\n   */\n  bucket: s3.IBucket;\n\n  /**\n   * Asset deployments to S3.\n   */\n  public deployments: BucketDeployment[];\n  public rewriter?: NextjsS3EnvRewriter;\n\n  public staticTempDir: string;\n\n  protected props: NextjsAssetsDeploymentProps;\n\n  constructor(scope: Construct, id: string, props: NextjsAssetsDeploymentProps) {\n    super(scope, id);\n\n    this.props = props;\n\n    this.bucket = props.bucket;\n    this.staticTempDir = this.prepareArchiveDirectory();\n    this.deployments = this.uploadS3Assets(this.staticTempDir);\n\n    // do rewrites of unresolved CDK tokens in static files\n    if (this.props.environment && !this.props.isPlaceholder) {\n      this.rewriter = new NextjsS3EnvRewriter(this, 'NextjsS3EnvRewriter', {\n        ...props,\n        s3Bucket: this.bucket,\n        s3keys: this._getStaticFilesForRewrite(),\n        replacementConfig: {\n          env: getS3ReplaceValues(this.props.environment, true),\n        },\n        debug: false,\n        cloudfrontDistributionId: this.props.distribution?.distributionId,\n      });\n      // wait for s3 assets to be uploaded first before running\n      this.rewriter.node.addDependency(...this.deployments);\n    }\n  }\n\n  // arrange directory structure for S3 asset deployments\n  // should contain _next/static and ./ for public files\n  protected prepareArchiveDirectory(): string {\n    const archiveDir = this.props.tempBuildDir\n      ? path.resolve(path.join(this.props.tempBuildDir, 'static'))\n      : fs.mkdtempSync(path.join(os.tmpdir(), 'static-'));\n    fs.mkdirpSync(archiveDir);\n\n    // theoretically we could move the files instead of copy for speed...\n\n    // path to public folder; root static assets\n    const staticDir = this.props.nextBuild.nextStaticDir;\n\n    if (!this.props.isPlaceholder && fs.existsSync(staticDir)) {\n      // copy public+static files to root\n      fs.copySync(this.props.nextBuild.nextStaticDir, archiveDir, {\n        recursive: true,\n        dereference: true,\n        preserveTimestamps: true,\n      });\n    }\n\n    return archiveDir;\n  }\n\n  private uploadS3Assets(archiveDir: string) {\n    // zip up bucket contents and upload to bucket\n    const archiveZipFilePath = createArchive({\n      directory: archiveDir,\n      zipFileName: 'assets.zip',\n      zipOutDir: path.join(this.staticTempDir, 'assets'),\n      compressionLevel: this.props.compressionLevel,\n      quiet: this.props.quiet,\n    });\n    if (!archiveZipFilePath) return [];\n\n    const maxAge = this.props.cachePolicies?.staticMaxAgeDefault?.toSeconds() ?? DEFAULT_STATIC_MAX_AGE;\n    const staleWhileRevalidate =\n      this.props.cachePolicies?.staticStaleWhileRevalidateDefault?.toSeconds() ?? DEFAULT_STATIC_STALE_WHILE_REVALIDATE;\n    const cacheControl = CacheControl.fromString(\n      `public,max-age=${maxAge},stale-while-revalidate=${staleWhileRevalidate},immutable`\n    );\n    const deployment = new BucketDeployment(this, 'NextStaticAssetsS3Deployment', {\n      destinationBucket: this.bucket,\n      cacheControl: [cacheControl],\n      sources: [Source.asset(archiveZipFilePath)],\n      distribution: this.props.distribution,\n      prune: this.props.prune,\n      useEfs: this.props.useEfs,\n      vpc: this.props.vpc,\n      memoryLimit: this.props.memoryLimit,\n      ephemeralStorageSize: this.props.ephemeralStorageSize,\n    });\n\n    return [deployment];\n  }\n\n  private _getStaticFilesForRewrite() {\n    const staticDir = this.staticTempDir;\n    const s3keys: string[] = [];\n    if (!fs.existsSync(staticDir)) {\n      return [];\n    }\n    listDirectory(staticDir).forEach((file) => {\n      const relativePath = path.relative(staticDir, file);\n\n      // skip bogus system files\n      if (relativePath.endsWith('.DS_Store')) return;\n\n      // is this file a glob match?\n      if (!micromatch.isMatch(relativePath, replaceTokenGlobs, { dot: true })) {\n        return;\n      }\n      s3keys.push(relativePath);\n    });\n    return s3keys;\n  }\n}\n\nexport function listDirectory(dir: string) {\n  const fileList: string[] = [];\n  const publicFiles = fs.readdirSync(dir);\n  for (const filename of publicFiles) {\n    const filepath = path.join(dir, filename);\n    const stat = fs.statSync(filepath);\n    if (stat.isDirectory()) {\n      fileList.push(...listDirectory(filepath));\n    } else {\n      fileList.push(filepath);\n    }\n  }\n\n  return fileList;\n}\n"]}