UNPKG

cdk-nextjs-standalone

Version:

Deploy a NextJS app to AWS using CDK and OpenNext.

139 lines 22.5 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.NextjsRevalidation = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const fs = require("fs"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_dynamodb_1 = require("aws-cdk-lib/aws-dynamodb"); const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const aws_lambda_1 = require("aws-cdk-lib/aws-lambda"); const aws_lambda_event_sources_1 = require("aws-cdk-lib/aws-lambda-event-sources"); const aws_logs_1 = require("aws-cdk-lib/aws-logs"); const aws_sqs_1 = require("aws-cdk-lib/aws-sqs"); const custom_resources_1 = require("aws-cdk-lib/custom-resources"); const constructs_1 = require("constructs"); const common_lambda_props_1 = require("./utils/common-lambda-props"); /** * Builds the system for revalidating Next.js resources. This includes a Lambda function handler and queue system as well * as the DynamoDB table and provider function. * * @see {@link https://github.com/serverless-stack/open-next/blob/main/README.md?plain=1#L65} * */ class NextjsRevalidation extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); this.props = props; this.queue = this.createQueue(); this.queueFunction = this.createQueueFunction(); this.table = this.createRevalidationTable(); this.tableFunction = this.createRevalidationInsertFunction(this.table); this.props.serverFunction.lambdaFunction.addEnvironment('CACHE_DYNAMO_TABLE', this.table.tableName); if (this.props.serverFunction.lambdaFunction.role) { this.table.grantReadWriteData(this.props.serverFunction.lambdaFunction.role); } this.props.serverFunction.lambdaFunction // allow server fn to send messages to queue ?.addEnvironment('REVALIDATION_QUEUE_URL', this.queue.queueUrl); props.serverFunction.lambdaFunction?.addEnvironment('REVALIDATION_QUEUE_REGION', aws_cdk_lib_1.Stack.of(this).region); } createQueue() { const queue = new aws_sqs_1.Queue(this, 'Queue', { fifo: true, receiveMessageWaitTime: aws_cdk_lib_1.Duration.seconds(20), ...this.props.overrides?.queueProps, }); // https://docs.aws.amazon.com/AWSSimpleQueueService/latest/SQSDeveloperGuide/sqs-least-privilege-policy.html queue.addToResourcePolicy(new aws_iam_1.PolicyStatement({ sid: 'DenyUnsecureTransport', actions: ['sqs:*'], effect: aws_iam_1.Effect.DENY, principals: [new aws_iam_1.AnyPrincipal()], resources: [queue.queueArn], conditions: { Bool: { 'aws:SecureTransport': 'false' }, }, })); // Allow server to send messages to the queue queue.grantSendMessages(this.props.serverFunction.lambdaFunction); return queue; } createQueueFunction() { const commonFnProps = (0, common_lambda_props_1.getCommonFunctionProps)(this); const fn = new aws_lambda_1.Function(this, 'QueueFn', { ...commonFnProps, // open-next revalidation-function // see: https://github.com/serverless-stack/open-next/blob/274d446ed7e940cfbe7ce05a21108f4c854ee37a/README.md?plain=1#L65 code: aws_lambda_1.Code.fromAsset(this.props.nextBuild.nextRevalidateFnDir), handler: 'index.handler', description: 'Next.js Queue Revalidation Function', timeout: aws_cdk_lib_1.Duration.seconds(30), ...this.props.overrides?.queueFunctionProps, }); fn.addEventSource(new aws_lambda_event_sources_1.SqsEventSource(this.queue, { batchSize: 5 })); return fn; } createRevalidationTable() { return new aws_dynamodb_1.TableV2(this, 'Table', { partitionKey: { name: 'tag', type: aws_dynamodb_1.AttributeType.STRING }, sortKey: { name: 'path', type: aws_dynamodb_1.AttributeType.STRING }, billing: aws_dynamodb_1.Billing.onDemand(), globalSecondaryIndexes: [ { indexName: 'revalidate', partitionKey: { name: 'path', type: aws_dynamodb_1.AttributeType.STRING }, sortKey: { name: 'revalidatedAt', type: aws_dynamodb_1.AttributeType.NUMBER }, }, ], removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, ...this.props.overrides?.tableProps, }); } /** * This function will insert the initial batch of tag / path / revalidation data into the DynamoDB table during deployment. * @see: {@link https://open-next.js.org/inner_workings/isr#tags} * * @param revalidationTable table to grant function access to * @returns the revalidation insert provider function */ createRevalidationInsertFunction(revalidationTable) { const dynamodbProviderPath = this.props.nextBuild.nextRevalidateDynamoDBProviderFnDir; // note the function may not exist - it only exists if there are cache tags values defined in Next.js build meta files to be inserted // see: https://github.com/sst/open-next/blob/c2b05e3a5f82de40da1181e11c087265983c349d/packages/open-next/src/build.ts#L426-L458 if (fs.existsSync(dynamodbProviderPath)) { const commonFnProps = (0, common_lambda_props_1.getCommonFunctionProps)(this); const insertFn = new aws_lambda_1.Function(this, 'DynamoDBProviderFn', { ...commonFnProps, // open-next revalidation-function // see: https://github.com/serverless-stack/open-next/blob/274d446ed7e940cfbe7ce05a21108f4c854ee37a/README.md?plain=1#L65 code: aws_lambda_1.Code.fromAsset(this.props.nextBuild.nextRevalidateDynamoDBProviderFnDir), handler: 'index.handler', description: 'Next.js Revalidation DynamoDB Provider', timeout: aws_cdk_lib_1.Duration.minutes(1), environment: { CACHE_DYNAMO_TABLE: revalidationTable.tableName, }, ...this.props.overrides?.insertFunctionProps, }); revalidationTable.grantReadWriteData(insertFn); const provider = new custom_resources_1.Provider(this, 'DynamoDBProvider', { onEventHandler: insertFn, logRetention: aws_logs_1.RetentionDays.ONE_DAY, ...this.props.overrides?.insertProviderProps, }); new aws_cdk_lib_1.CustomResource(this, 'DynamoDBResource', { serviceToken: provider.serviceToken, properties: { version: Date.now().toString(), }, ...this.props.overrides?.insertCustomResourceProps, }); return insertFn; } return undefined; } } exports.NextjsRevalidation = NextjsRevalidation; _a = JSII_RTTI_SYMBOL_1; NextjsRevalidation[_a] = { fqn: "cdk-nextjs-standalone.NextjsRevalidation", version: "4.2.3" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NextjsRevalidation.js","sourceRoot":"","sources":["../src/NextjsRevalidation.ts"],"names":[],"mappings":";;;;;AAAA,yBAAyB;AACzB,6CAA6E;AAC7E,2DAAoF;AACpF,iDAA4E;AAC5E,uDAA2F;AAC3F,mFAAsE;AACtE,mDAAqD;AACrD,iDAAwD;AACxD,mEAAwD;AACxD,2CAAuC;AASvC,qEAAqE;AA8BrE;;;;;;GAMG;AACH,MAAa,kBAAmB,SAAQ,sBAAS;IAO/C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA8B;QACtE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACjB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QAEnB,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAChC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAEhD,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAC5C,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,gCAAgC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAEvE,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,cAAc,CAAC,cAAc,CAAC,oBAAoB,EAAE,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAEpG,IAAI,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC;YAClD,IAAI,CAAC,KAAK,CAAC,kBAAkB,CAAC,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;QAC/E,CAAC;QAED,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,cAAc,CAAC,4CAA4C;YACnF,EAAE,cAAc,CAAC,wBAAwB,EAAE,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;QAClE,KAAK,CAAC,cAAc,CAAC,cAAc,EAAE,cAAc,CAAC,2BAA2B,EAAE,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,CAAC;IAC1G,CAAC;IAEO,WAAW;QACjB,MAAM,KAAK,GAAG,IAAI,eAAK,CAAC,IAAI,EAAE,OAAO,EAAE;YACrC,IAAI,EAAE,IAAI;YACV,sBAAsB,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YAC5C,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,UAAU;SACpC,CAAC,CAAC;QACH,6GAA6G;QAC7G,KAAK,CAAC,mBAAmB,CACvB,IAAI,yBAAe,CAAC;YAClB,GAAG,EAAE,uBAAuB;YAC5B,OAAO,EAAE,CAAC,OAAO,CAAC;YAClB,MAAM,EAAE,gBAAM,CAAC,IAAI;YACnB,UAAU,EAAE,CAAC,IAAI,sBAAY,EAAE,CAAC;YAChC,SAAS,EAAE,CAAC,KAAK,CAAC,QAAQ,CAAC;YAC3B,UAAU,EAAE;gBACV,IAAI,EAAE,EAAE,qBAAqB,EAAE,OAAO,EAAE;aACzC;SACF,CAAC,CACH,CAAC;QACF,6CAA6C;QAC7C,KAAK,CAAC,iBAAiB,CAAC,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;QAClE,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,mBAAmB;QACzB,MAAM,aAAa,GAAG,IAAA,4CAAsB,EAAC,IAAI,CAAC,CAAC;QACnD,MAAM,EAAE,GAAG,IAAI,qBAAc,CAAC,IAAI,EAAE,SAAS,EAAE;YAC7C,GAAG,aAAa;YAChB,kCAAkC;YAClC,yHAAyH;YACzH,IAAI,EAAE,iBAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,mBAAmB,CAAC;YAC9D,OAAO,EAAE,eAAe;YACxB,WAAW,EAAE,qCAAqC;YAClD,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YAC7B,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,kBAAkB;SAC5C,CAAC,CAAC;QACH,EAAE,CAAC,cAAc,CAAC,IAAI,yCAAc,CAAC,IAAI,CAAC,KAAK,EAAE,EAAE,SAAS,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QACpE,OAAO,EAAE,CAAC;IACZ,CAAC;IAEO,uBAAuB;QAC7B,OAAO,IAAI,sBAAK,CAAC,IAAI,EAAE,OAAO,EAAE;YAC9B,YAAY,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,4BAAa,CAAC,MAAM,EAAE;YACzD,OAAO,EAAE,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,4BAAa,CAAC,MAAM,EAAE;YACrD,OAAO,EAAE,sBAAO,CAAC,QAAQ,EAAE;YAC3B,sBAAsB,EAAE;gBACtB;oBACE,SAAS,EAAE,YAAY;oBACvB,YAAY,EAAE,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,4BAAa,CAAC,MAAM,EAAE;oBAC1D,OAAO,EAAE,EAAE,IAAI,EAAE,eAAe,EAAE,IAAI,EAAE,4BAAa,CAAC,MAAM,EAAE;iBAC/D;aACF;YACD,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,UAAU;SACpC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;OAMG;IACK,gCAAgC,CAAC,iBAAwB;QAC/D,MAAM,oBAAoB,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,mCAAmC,CAAC;QAEtF,qIAAqI;QACrI,gIAAgI;QAChI,IAAI,EAAE,CAAC,UAAU,CAAC,oBAAoB,CAAC,EAAE,CAAC;YACxC,MAAM,aAAa,GAAG,IAAA,4CAAsB,EAAC,IAAI,CAAC,CAAC;YACnD,MAAM,QAAQ,GAAG,IAAI,qBAAc,CAAC,IAAI,EAAE,oBAAoB,EAAE;gBAC9D,GAAG,aAAa;gBAChB,kCAAkC;gBAClC,yHAAyH;gBACzH,IAAI,EAAE,iBAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,mCAAmC,CAAC;gBAC9E,OAAO,EAAE,eAAe;gBACxB,WAAW,EAAE,wCAAwC;gBACrD,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;gBAC5B,WAAW,EAAE;oBACX,kBAAkB,EAAE,iBAAiB,CAAC,SAAS;iBAChD;gBACD,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,mBAAmB;aAC7C,CAAC,CAAC;YAEH,iBAAiB,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;YAE/C,MAAM,QAAQ,GAAG,IAAI,2BAAQ,CAAC,IAAI,EAAE,kBAAkB,EAAE;gBACtD,cAAc,EAAE,QAAQ;gBACxB,YAAY,EAAE,wBAAa,CAAC,OAAO;gBACnC,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,mBAAmB;aAC7C,CAAC,CAAC;YAEH,IAAI,4BAAc,CAAC,IAAI,EAAE,kBAAkB,EAAE;gBAC3C,YAAY,EAAE,QAAQ,CAAC,YAAY;gBACnC,UAAU,EAAE;oBACV,OAAO,EAAE,IAAI,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE;iBAC/B;gBACD,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,yBAAyB;aACnD,CAAC,CAAC;YAEH,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,OAAO,SAAS,CAAC;IACnB,CAAC;;AArIH,gDAsIC","sourcesContent":["import * as fs from 'fs';\nimport { CustomResource, Duration, RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport { AttributeType, Billing, TableV2 as Table } from 'aws-cdk-lib/aws-dynamodb';\nimport { AnyPrincipal, Effect, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Code, Function as LambdaFunction, FunctionOptions } from 'aws-cdk-lib/aws-lambda';\nimport { SqsEventSource } from 'aws-cdk-lib/aws-lambda-event-sources';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { Queue, QueueProps } from 'aws-cdk-lib/aws-sqs';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\nimport {\n  OptionalCustomResourceProps,\n  OptionalFunctionProps,\n  OptionalProviderProps,\n  OptionalTablePropsV2,\n} from './generated-structs';\nimport { NextjsBuild } from './NextjsBuild';\nimport { NextjsServer } from './NextjsServer';\nimport { getCommonFunctionProps } from './utils/common-lambda-props';\n\nexport interface NextjsRevalidationOverrides {\n  readonly queueProps?: QueueProps;\n  readonly queueFunctionProps?: OptionalFunctionProps;\n  readonly tableProps?: OptionalTablePropsV2;\n  readonly insertFunctionProps?: OptionalFunctionProps;\n  readonly insertProviderProps?: OptionalProviderProps;\n  readonly insertCustomResourceProps?: OptionalCustomResourceProps;\n}\n\nexport interface NextjsRevalidationProps {\n  /**\n   * Override function properties.\n   */\n  readonly lambdaOptions?: FunctionOptions;\n  /**\n   * @see {@link NextjsBuild}\n   */\n  readonly nextBuild: NextjsBuild;\n  /**\n   * Override props for every construct.\n   */\n  readonly overrides?: NextjsRevalidationOverrides;\n  /**\n   * @see {@link NextjsServer}\n   */\n  readonly serverFunction: NextjsServer;\n}\n\n/**\n * Builds the system for revalidating Next.js resources. This includes a Lambda function handler and queue system as well\n * as the DynamoDB table and provider function.\n *\n * @see {@link https://github.com/serverless-stack/open-next/blob/main/README.md?plain=1#L65}\n *\n */\nexport class NextjsRevalidation extends Construct {\n  queue: Queue;\n  table: Table;\n  queueFunction: LambdaFunction;\n  tableFunction: LambdaFunction | undefined;\n  private props: NextjsRevalidationProps;\n\n  constructor(scope: Construct, id: string, props: NextjsRevalidationProps) {\n    super(scope, id);\n    this.props = props;\n\n    this.queue = this.createQueue();\n    this.queueFunction = this.createQueueFunction();\n\n    this.table = this.createRevalidationTable();\n    this.tableFunction = this.createRevalidationInsertFunction(this.table);\n\n    this.props.serverFunction.lambdaFunction.addEnvironment('CACHE_DYNAMO_TABLE', this.table.tableName);\n\n    if (this.props.serverFunction.lambdaFunction.role) {\n      this.table.grantReadWriteData(this.props.serverFunction.lambdaFunction.role);\n    }\n\n    this.props.serverFunction.lambdaFunction // allow server fn to send messages to queue\n      ?.addEnvironment('REVALIDATION_QUEUE_URL', this.queue.queueUrl);\n    props.serverFunction.lambdaFunction?.addEnvironment('REVALIDATION_QUEUE_REGION', Stack.of(this).region);\n  }\n\n  private createQueue(): Queue {\n    const queue = new Queue(this, 'Queue', {\n      fifo: true,\n      receiveMessageWaitTime: Duration.seconds(20),\n      ...this.props.overrides?.queueProps,\n    });\n    // https://docs.aws.amazon.com/AWSSimpleQueueService/latest/SQSDeveloperGuide/sqs-least-privilege-policy.html\n    queue.addToResourcePolicy(\n      new PolicyStatement({\n        sid: 'DenyUnsecureTransport',\n        actions: ['sqs:*'],\n        effect: Effect.DENY,\n        principals: [new AnyPrincipal()],\n        resources: [queue.queueArn],\n        conditions: {\n          Bool: { 'aws:SecureTransport': 'false' },\n        },\n      })\n    );\n    // Allow server to send messages to the queue\n    queue.grantSendMessages(this.props.serverFunction.lambdaFunction);\n    return queue;\n  }\n\n  private createQueueFunction(): LambdaFunction {\n    const commonFnProps = getCommonFunctionProps(this);\n    const fn = new LambdaFunction(this, 'QueueFn', {\n      ...commonFnProps,\n      // open-next revalidation-function\n      // see: https://github.com/serverless-stack/open-next/blob/274d446ed7e940cfbe7ce05a21108f4c854ee37a/README.md?plain=1#L65\n      code: Code.fromAsset(this.props.nextBuild.nextRevalidateFnDir),\n      handler: 'index.handler',\n      description: 'Next.js Queue Revalidation Function',\n      timeout: Duration.seconds(30),\n      ...this.props.overrides?.queueFunctionProps,\n    });\n    fn.addEventSource(new SqsEventSource(this.queue, { batchSize: 5 }));\n    return fn;\n  }\n\n  private createRevalidationTable() {\n    return new Table(this, 'Table', {\n      partitionKey: { name: 'tag', type: AttributeType.STRING },\n      sortKey: { name: 'path', type: AttributeType.STRING },\n      billing: Billing.onDemand(),\n      globalSecondaryIndexes: [\n        {\n          indexName: 'revalidate',\n          partitionKey: { name: 'path', type: AttributeType.STRING },\n          sortKey: { name: 'revalidatedAt', type: AttributeType.NUMBER },\n        },\n      ],\n      removalPolicy: RemovalPolicy.DESTROY,\n      ...this.props.overrides?.tableProps,\n    });\n  }\n\n  /**\n   * This function will insert the initial batch of tag / path / revalidation data into the DynamoDB table during deployment.\n   * @see: {@link https://open-next.js.org/inner_workings/isr#tags}\n   *\n   * @param revalidationTable table to grant function access to\n   * @returns the revalidation insert provider function\n   */\n  private createRevalidationInsertFunction(revalidationTable: Table) {\n    const dynamodbProviderPath = this.props.nextBuild.nextRevalidateDynamoDBProviderFnDir;\n\n    // note the function may not exist - it only exists if there are cache tags values defined in Next.js build meta files to be inserted\n    // see: https://github.com/sst/open-next/blob/c2b05e3a5f82de40da1181e11c087265983c349d/packages/open-next/src/build.ts#L426-L458\n    if (fs.existsSync(dynamodbProviderPath)) {\n      const commonFnProps = getCommonFunctionProps(this);\n      const insertFn = new LambdaFunction(this, 'DynamoDBProviderFn', {\n        ...commonFnProps,\n        // open-next revalidation-function\n        // see: https://github.com/serverless-stack/open-next/blob/274d446ed7e940cfbe7ce05a21108f4c854ee37a/README.md?plain=1#L65\n        code: Code.fromAsset(this.props.nextBuild.nextRevalidateDynamoDBProviderFnDir),\n        handler: 'index.handler',\n        description: 'Next.js Revalidation DynamoDB Provider',\n        timeout: Duration.minutes(1),\n        environment: {\n          CACHE_DYNAMO_TABLE: revalidationTable.tableName,\n        },\n        ...this.props.overrides?.insertFunctionProps,\n      });\n\n      revalidationTable.grantReadWriteData(insertFn);\n\n      const provider = new Provider(this, 'DynamoDBProvider', {\n        onEventHandler: insertFn,\n        logRetention: RetentionDays.ONE_DAY,\n        ...this.props.overrides?.insertProviderProps,\n      });\n\n      new CustomResource(this, 'DynamoDBResource', {\n        serviceToken: provider.serviceToken,\n        properties: {\n          version: Date.now().toString(),\n        },\n        ...this.props.overrides?.insertCustomResourceProps,\n      });\n\n      return insertFn;\n    }\n\n    return undefined;\n  }\n}\n"]}