@downmoney/nextjs-lambda
v4.0.0
Published
Plug-and-play lambda for replacing default NextJS image optimization handler.
Downloads
2
Readme
NextJS Lambda Utils
This is a project allowing to deploy Next applications (standalone options turned on) to AWS Lambda without hassle.
This is an alternative to existing Lambda@Edge implementation (see) as it has too many limitations (primarily inability to use env vars) and deployments take too long.
This library uses Cloudfront, S3, ApiGateway and Lambdas to deploy easily in seconds (hotswap supported)
TL;DR
- In your NextJS project, set output to standalone.
- Run
npm install @sladg/nextjs-lambda
(will install dependency). - Run
next build
(will generate standalone next folder). - Run
npm run-script env -- next-utils pack
(will create ZIPs). - Run
npm run-script env -- next-utils deploy
(will deploy to AWS). - Profit 🎉
Usage
We need to create 2 lambdas in order to use NextJS. First one is handling pages/api rendering, second is solving image optimization.
This division makes it easier to control resources and specify sizes and timeouts as those operations are completely different.
Loading of assets and static content is handled via Cloudfront and S3 origin, so there is no need for specifying this behaviour in Lambda or handling it anyhow.
next.config.js
The only requirement is to change your Next12 config to produce standalone output via output: 'standalone'
.
This should work fine for single-repositories with yarn/npm/pnpm.
In case you are using monorepo/workspaces, be aware! Producing standalone build is tricky due to dependencies being spread out and not contained within single node_modules
folder, making it complicated for SWC
to properly produce required dependencies. This would most likely result in deployment failing with HTTP 500, internal error, as some required dependency is not in place.
See:
- https://github.com/vercel/next.js/issues/36386
- https://github.com/vercel/next.js/discussions/32223
Server handler
This is a Lambda entrypoint to handle non-asset requests. We need a way to start Next in lambda-friendly way and translate ApiGateway event into typical HTTP event. This is handled by server-handler, which sits alongside of next's server.js
in standalone output.
Image handler
Lambda consumes Api Gateway requests, so we need to create ApiGw proxy (v2) that will trigger Lambda.
Lambda is designed to serve _next/image*
route in NextJS stack and replaces the default handler so we can optimize caching and memory limits for page renders and image optimization.
Via CDK
See NextStandaloneStack
construct in cdk/example.ts
.
Or just use next-utils deploy
command so you don't have to manage CDK yourself.
Benchmark
Creation of stack: 385sec (6min 25sec)
Run #2 436sec (7min 16sec)
Run #3 383sec (6min 23sec)
Deletion of stack: 262sec (4min 22sec)
Run #2 319sec (5m 19sec)
Update of stack: 92sec (1min 32sec)
Run #2 5sec (no changes)
Run #3 3sec (no changes)
Run #4 164sec (2min 44sec)
Run #5 62sec (1min 2sec) (no changes, used
next-utils deploy
)
Sharp layer
Besides handler (wrapper) itself, underlying NextJS also requires sharp binaries.
To build those, we use npm install
with some extra parametes. Then we zip all sharp dependencies and compress it to easily importable zip file.
import { sharpLayerZipPath } from '@sladg/nextjs-lambda'
const sharpLayer = new LayerVersion(this, 'SharpDependenciesLayer', {
code: Code.fromAsset(sharpLayerZipPath)
})
Next layer
To provide both image and server handlers with all depdencies (next is using require.resolve
inside, so it cannot be bundled standalone for now).
We pre-package this layer so it can be included in Lambda without hassle.
import { nextLayerZipPath } from '@sladg/nextjs-lambda'
const nextLayer = new LayerVersion(this, 'NextDependenciesLayer', {
code: Code.fromAsset(nextLayerZipPath)
})
Packaging
In order to succefully deploy, you firstly need to include target: 'standalone'
in your next.config.js
setup.
Make sure to use NextJS in version 12 or above so this is properly supported.
Once target is set, you can go on and use your next build
command as you normally would.
To package everything, make sure to be in your project root folder and next folder .next
and public
exist. Packaging is done via NPM CLI command of @slack/nextjs-lambda package
.
It will create next.out/
folder with 3 zip packages. One zip Lambda's code, one is dependencies layer and one is assets layer.
- code zip: include all files generated by next that are required to run on Lambda behind ApiGateway. Original handler as well as new server handler are included. Use
server-handler.handler
for custom one orserver.handler
for original one. - dependencies layer: all transpilied
node_modules
. Next includes only used files, dramatically reducing overall size. - assets layer: your public folder together with generated assets. Keep in mind that to public refer file, you need to include it in
public/assets/
folder, not just in public. This limitation dramatically simplifies whole setup. This zip file is uploaded to S3, it's not included in Lambda code.
Server handler
Custom wrapper around NextServer to allow for passing ApiGateway events into Next server.
Cloudfront paths used:
default
_next/data/*
Static assets
Next uses multiple directories to determine which file should be served. By default next provides us with list of routes for API/images/assets/pages. To simplify the process as much as possible, we are tapping into resulting paths.
We are packaging those assets to simulate output structure and we are using S3 behind CloudFront to serve those files. Also, Image Handler is tapping into S3 to provide images, so correct folder structure is crucial.
Cloudfront paths used:
_next/*
assets/*
Keep in minda, Cloudfront does not allow for multiple regex patterns in single origin, so using extensions to distinguish image/server handlers is not doable.
Versioning
This package exposes two CLI functions intended to deal with versioning your application and releasing.
Motivation behind is to get rid of huge dependencies and over-kill implementations such as @auto-it, release-it or semver. Those are bulky and unncessarily complex.
Guess
Simple CLI command that takes commit message and current version and outputs (stdout) next version based on keywords inside commit message.
Shipit
Similar to guess command, however, it automatically tags a commit on current branch and creates release branch for you so hooking up pipelines is as simple as it can be. Version is automatically bumped in common NPM and PHP files (package.json, package-lock.json and composer.json).
Simply call @sladg/next-lambda shipit
on any branch and be done.
TODO
- Move Next into peer dependency and build layer manually via npx. Parse next version from parent (allow parameter) to ensure compatibility with our implementation.
- Move CDK into peer dependency and allow for CDK dependencies to not be used / to have version defined in upstream.
Disclaimer
At this point, advanced features were not tested with this setup. This includes:
GetServerSideProps
,- middleware,
- ISR and fallbacks,
- streaming,
- custom babel configuration.
I am looking for advanced projects implementing those features, so we can test them out! Reach out to me!