Deploying Next.JS on AWS with Lambda, S3 & CloudFront
What is Next.js?
Next.js is a React framework for building web applications. It renders server-side by default but can also use static site generation (SSG) and API routes
How can NextJS run inside a lambda?
The "native" way to run Next.js is to deploy it on the vercel hosting provider but by using the AWS Lambda web adapter it can also be deployed in an AWS Lambda Function.
What infrastructure is required?
Lambda
This will act as the server
S3
Static files will be stored as objects in S3 so that they are not served directly by the server
Cloudfront
So that S3 objects are stored in regional edge caches all requests for static files will be made via the CloudFront CDN
Deploying to AWS
The following steps will walk you through how to deploy a Next.js application to AWS using Lambda, S3 and CloudFront
Create the Lambda Function
- Create a lambda function named aws-nextjs-lambda, ensure it has a function url
- Set the handler to run.sh
- Add the following environment variables:
AWS_LAMBDA_EXEC_WRAPPER: /opt/bootstrap
PORT: 8000
RUST_LOG: info
- Add the AWS Lambda Web Adapter layer arn:aws:lambda:eu-west-2 :753240598075:layer: LambdaAdapterLayerX86:24
Create the S3 bucket
-
Create an S3 bucket, I have named it named aws-nextjs-s3 but as bucket names must be globally unique you will have to choose your own name
-
Update the bucket policy, remember that the value CLOUDFRONT_DISTRIBUTION_ARN will have to be updated once the CloudFront distribution has been created
{
"Version": "2012-10-17",
"Statement": {
"Sid": "AllowCloudFrontServicePrincipalReadOnly",
"Effect": "Allow",
"Principal": {
"Service": "cloudfront.amazonaws.com"
},
"Action": "s3:GetObject",
"Resource": "arn:aws:s3:::aws-nextjs-s3/*",
"Condition": {
"StringEquals": {
"AWS:SourceArn": "CLOUDFRONT_DISTRIBUTION_ARN"
}
}
}
}
Create the CloudFront distribution
CloudFront is a CDN which we will use to serve all static content.
- Create a distribution with the following settings:
- Origin domain: aws-nextjs-s3 (the bucket we created earlier)
- Origin access control enabled
- Disable WAF
-
At this point the AWS console will prompt you to update the S3 bucket policy so that CloudFront can read from the bucket
-
Add an additional origin using the Lambda function url as the Origin Domain
-
Edit the default CloudFront Distribution behaviour to point to the Lambda origin that was created in the previous step
-
Add a CloudFront Distribution behaviour to route all requests for static files to the S3 bucket:
- Path pattern: _next/static/**
- Origin: aws-nextjs-s3.s3.eu-west-2.amazonaws.com
Create the Next.JS application
To create the Next.JS application run npx create-next-app@latest
Then we need to change the default settings in next.config.ts
const nextConfig = {
output: "standalone",
assetPrefix: "DISTRIBUTION_DOMAIN_NAME",
images: {
unoptimized: true,
},
};
output: "standalone" generates an output directory that can be deployed to AWS and is explained here, it excludes the public and .next/static directories as these will be deployed to the CloudFront distribution specified in the assetPrefix
value.
images -> unoptimized -> true is also set as this is a Vercel feature.
Deploy scripts
Next we need to create some scripts to bundle, deploy and run the application
In the root directory create run.sh which is the script that the Lambda will use to start the webserver.
#!/bin/bash
[ ! -d '/tmp/cache' ] && mkdir -p /tmp/cache
HOSTNAME=0.0.0.0 exec node server.js
Also in the root directory create prepare-build.sh
cp -r public/. .next/standalone/public
cp -r .next/static/. .next/standalone/.next/static
cp run.sh .next/standalone/run.sh```
Add the following scripts to package.json
"build": "next build && sh ./prepare-build.sh",
"zip": "cd .next/standalone/ && zip -r app.zip . ",
"cp": "aws s3 cp ./.next/static s3://aws-nextjs-s3/_next/static --recursive"
Build and deploy the Lambda
Run the following commands
- npm run build
- npm run zip
- Upload the file .next/standalone/app.zip to the Lambda
- npm run cp
Viewing the application
Now the Next.JS application can be accessed via the CloudFront Distribution domain name; all assets in the directory _next/static/** will be cached in cloudfront and all server requests will be handled by the Lambda function.