This is a showcase to make Serverless (https://serverless.com) work with Next.js (https://github.com/zeit/next.js/). It's also an in-depth explanation of what are the steps to put those two together. The goal being to make a Serverless template for ease of use.
Notice: This project has reached a maturity where it can be used for production application (March 10, 2018). I will personally use it as such, but it is still very young and issues will likely arise.
git clone git@github.com:Vadorequest/serverless-with-next.git
- Disable
serverless.yml:custom:customDomain
or configure your own custom domain on AWS and then runsls create_domain
(can take 20-40 minutes) [See "Known issues"] See SLS Tutorial - (optional)
nvm use
if using nvm, or make sure you are using node6.10
npm i
npm start
(starts development server, powered by serverless-offline) (Note: serverless-offline only support AWS at the moment)- Go to:
http://localhost:3000/ko
(json) [serverless-offline powering express server]http://localhost:3000/status
(json) [serverless-offline powering provider function "status"]http://localhost:3000/
(hello world) [next.js app]http://localhost:3000/page2
(hello world 2) [next.js app]http://localhost:3000/test
(404) [next.js app]http://localhost:3000/event
(example of AWS API Gateway event data) [serverless-offline powering express server]
- (optional) npm run deploy (should work on any provider, only tested against AWS though [need changes on
serverless.yml
]) - You can check that the AWS-hosted app behaves exactly the same as the local app at https://swn.dev.vadorequest.fr
You can check the SSR by looking at the browser console "Network" panel when going on http://localhost:3000/page2
from http://localhost:3000
through the link (client-side redirection, no SSR)
or directly by pasting/typing the url (SSR)
Because Next.js helps building SSR react applications and serverless helps to deploy them on any cloud provider. (AWS, Google Cloud, etc.)
In my case, I need to render my homepage based on settings I must fetch from a DB. Hence the fact I need a server-side application if I want to have a good SEO.
We could use create-react-app
and just deploy the bundled version, but SEO wouldn't be great.
- ES6 (with source map support)
- Development ease, identical behaviours between local and AWS environments (using serverless-offline)
- Stages (production, staging, development)
- Static assets (but I recommend against using heavy static assets, it increases the build size, and the upload time to AWS since they are deployed at every
npm run deploy
, better to use a separated S3 bucket) - Express server, powering Next application but also potentially whatever else you need
- HTTP/2, this is just standard AWS behaviour (nothing particular has been done to enable this)
Here is how a standard GET request will flow, assuming we call /
(https://swn.dev.vadorequest.fr/
in our example):
- Hit
server
function route because ofpath: /
(serverless.yml
) - Hit
server.js:handler
proxy (/src/functions/server/server.js
) - Hit the rule
app.get('*')
which proxies the Next app and runnextProxy(req, res);
, which is then treated by the Next app - Next app will resolve the
/
path by resolving it to/pages/index.js
Here is how a standard GET request will flow, assuming we call /status
(https://swn.dev.vadorequest.fr/status
in our example):
- Hit
status
function route because ofpath: status
(serverless.yml
) - Hit
status.js:handler
proxy (/src/functions/status/status.js
)
Here is how a standard GET request will flow, assuming we call /whatever/nested
(https://swn.dev.vadorequest.fr/whatever/nested
in our example):
- Hit
server
function route because ofpath: /{any+}
(serverless.yml
) - Hit
server.js:handler
proxy (/src/functions/server/server.js
) - Hit the rule
app.get('/:level1/:level2')
which will return a JSON response{"level1":"whatever","level2":"nested"}
Here is how a standard GET request will flow, assuming we call /whatever/nested/deep
(https://swn.dev.vadorequest.fr/whatever/nested/deep
in our example):
- Hit
server
function route because ofpath: /{any+}
(serverless.yml
) - Hit
server.js:handler
proxy (/src/functions/server/server.js
) - Hit the rule
app.get('*')
which proxies the Next app and runnextProxy(req, res);
, which is then treated by the Next app - Next app will fail to resolve the
/whatever/nested/deep
path and display a 404 because no page match for this URL
With the previous examples, we can see that our functions routes have the most important priority.
Then, when redirected to our main handler, it's the Express framework who deals with the routing.
And then, depending on our Express routing, the Next app will handle the request, or not.
I tried to limit as much as possible the behaviours differences between the local and AWS environments. (For obvious reasons) I did all my tests against AWS and I therefore use it as example, but it's also valid for other providers.
On AWS, we upload a package which contains:
- `/.next`: Next.js build folder
- `/src`: Our sources, basically our functions in subfolders
When we hit an endpoint on AWS, it goes straight to our functions defined in serverless.yml
. We only have 2 functions:
- `status`: Simple `/status` endpoint to display AWS status and data
- `server`: All other AWS paths are catched and redirected to our `server` function, which uses Express
On local environment, we get the same path structure, with our /.next
and /src
folders at the root.
We have serverless-offline
running on port 3000, which handles the function calls. It will also proxy everything to our server
function.
We also have our Next.js application running on port 3001.
This project assume:
- a basic knowledge of Serverless, with the
serverless
cli installed. (see https://serverless.com/learn/quick-start/) - a basic knowledge of Next.js. (see https://learnnextjs.com)
- an AWS account,
sls deploy
commands will deploy on AWS (another provider is possible, but theserverless.yml
will need to be modified) - node <
6.9.3
installed, I personally used8.9.4
, doesn't matter so much because we use webpack. (See supported-languages) - (optional) The use of a custom domain to fix a Known issue (see https://github.com/amplify-education/serverless-domain-manager), can simply be disabled to play around
-
On AWS, I can't get Next.js to work correctly because of the Serverless
staging
path rewrite:The main page (
https://11lwiykejg.execute-api.us-east-1.amazonaws.com/development/
) works fine, but:- when clicking on a "Page 2" link, it goes to the wrong URL:
https://11lwiykejg.execute-api.us-east-1.amazonaws.com/page2
, it's missing the/development
part and the browser will display{"message":"Forbidden"}
- Current workaround: I used a custom domain, it fixes the missing
development
part (by removing thestaging
part of the url entirely, which fixes the issue):
- when clicking on a "Page 2" link, it goes to the wrong URL:
-
Useless files are packaged and uploaded to AWS:
The
.next
andstatic
folders are packaged for all functions, which is useless because only the server handler will use them. Since I'm using Webpack to copy both those folders (and not SLS native packaging because we useserverless-webpack
which isn't compatible), I don't know how to ignore those folders for certain functions. -
HMR not working on http://localhost:3000 for Next.js:
Next.js comes with HMR, which is great. But it doesn't work on http://localhost:3000 yet. It works on http://localhost:3001 though
But it would be a better developer experience to have everything working seamlessly on http://localhost:3000
- I tried to simply use
nextProxy(req, res)
but gotTypeError: Cannot read property 'waitUntilReloaded' of undefined at HotReloader._callee7$ (/Users/vadorequest/dev/serverless-with-next/node_modules/next/dist/server/hot-reloader.js:658:44)
- Then, I decided to proxy requests that Express doesn't want to handle to 3001, so that Next.js app handles them. But the proxy messes up with HMR and I haven't been able to fix it:
- I tried to proxy all
/_next
by doingapp.use('/_next/', proxy('http://localhost:3001/_next/'));
but then I get 404 for all js scripts likehttp://localhost:3000/_next/-/main.js
- I tried to proxy them all one by one but then they return HTML content instead of JS (basically the index page), ex:
app.use('/_next/-/main.js', proxy('http://localhost:3001/_next/-/main.js'));
- If you manually browse to
http://localhost:3001/_next/-/main.js
it works okay and return the actual JS file - I tried to disable HMR by setting
dev: false
but then the Next.js app complainsCould not find a valid build in the '.next' directory!
- I tried to force contentType to
text/event-stream
when proxying/_next/webpack-hmr
and it seem to work okay as long as Express doesn't catch the request first (which is the case with GET/:level1/:level2
route), and it does display[HMR] connected
but nothing happens when a file is changed.
- I tried to proxy all
- I tried to simply use
This part aims at giving you explanations about why is the project configured this way, we'll go deep in the configuration in order to explain the choices and understand the reasons behind.
It's perfect if you want to understand how all the pieces are working together, just skip it if you're not interested.
Used to be able to use the latest JS version, in combination with Babel.
One downside of using this plugin is the fact we can't rely on the official SLS documentation about how to package anymore. Source
Since the packaging is done using serverless-webpack
, we can't follow https://serverless.com/framework/docs/providers/aws/guide/packaging/ doc to do the packaging.
On the other hand, we don't (usually) have to worry about what node module to include for each function, since the plugin does it for us using some kind of smart scan to detect what are the needed dependencies.
Nevertheless, in some case you may need to override the default behavior and forceInclude/forceExclude some packages.
In addition, we use the CopyWebpackPlugin
, to copy the .next
and static
folder during packaging.
1. serverless-offline - AWS provider only!
Must-needed for local development. Kind of simulate lambda functions with local endpoints for ease of development. Time saver.
Read its doc is a must-do.
Plugin for Serverless Framework which adds support for test-driven development using Jest
Note: Not really used but can be a nice addition, I'm thinking about removing it. Not important.
Serverless plugin for managing custom domains with API Gateways.
Custom domain is kind of a must-have in any production application.
Especially because when you delete your stack and recreate it, or change the region, it'll change the endpoint url. You need a fixed url that doesn't change for production usage. (I do)
webpack-node-externals
Read more at https://github.com/serverless-heaven/serverless-webpack#node-modules--externals
Basically, stuff like aws-sdk
are automatically removed and not bundled/uploaded to AWS.
We enabled next/babel
preset as explained in the official documentation at https://github.com/zeit/next.js#customizing-babel-config
Additionally, we force to transpile the code to node 6.10 version to avoid any issue in the AWS environment
We also enable source map support.
Important: babel-runtime
and source-map-support
must be in the package.json:dependencies
or your build will fail on AWS.
Both those modules are needed at runtime and you'll run into issues if you move them to devDependencies
.
On the other hand, moving a casual package from dependencies
to devDependencies
like moment
will have no side effect since serverless-webpack
should resolve it and bundle it anyway.
But for the sake of understanding, better split packages correctly between both.
Due to a webpack's bug/unwanted behavior, we get warnings/errors due to missing fs module. See evanw/node-source-map-support#155
Next looks for static files in the ./static
folder. We kept the folder in the root folder for the sake of simplicity.
You can customize it a bit following Next documentation
I highly recommend not to use static folder, and prefer using an external S3 bucket or CDN for that purpose.
The main reason is to speedup deployment, since Serverless/Webpack will bundle those static assets every time you use sls deploy
.
If you have too many static assets, it'll make it last longer, and if your internet connection is weak, upload can become quite long.
And I don't think Next serves files faster than S3 does.
Also, you pay for files you upload on Lambda, so...
But for playing around, it's perfectly fine.
Next "Pages" are in the /pages
folder and can't be moved in another folder.
We use the non-stable version 5.0.1-canary.9
because they fix a webpack bug in that particular version and we can't use an older one.
Feel free to update to a more recent version though. I'm waiting for "canary" version to be released.
TODO
TODO How to catch all routes (main handler)
I put together a not-so-great logging helper. It does resolve webpack source map on AWS and that's its most interesting feature.
I also used stacktrace-js
for better stacktrace, it looked interesting but I never used it before.
Anyway, if you don't like it, just throw it away. Suggestions/improvements are welcome.
I am just a beginner with Serverless and Next.js
https://github.com/geovanisouza92/serverless-next was my main source of inspiration to put this together, but it was overcomplicated to my taste for a "getting started" and I couldn't understand how to decompose it all into smaller pieces.
I started this repo with this tutorial, to write down the steps I went through, but I don't actually maintain it anymore, too much has happened and it doesn't really match between those examples and the current version. I'm keeping it in case somebody would want to do the same. Most of the knowledge I've acquired from it is now explained in the previous "Deep dive" part.
-
Run
sls create --template hello-world --path serverless-with-next
(optionally ignore.idea
folder) -
Test using
sls deploy
should print something like this: -
Let's add ES6 using webpack and serverless-webpack
-
Run
npm init -y
-
Ignore
.webpack
folder -
Update
serverless.yml
plugins: - serverless-webpack
We use the
serverless-webpack
plugin to build our serverless app. The build is then uploaded to aws -
Add
.babelrc
config{ "plugins": ["source-map-support", "transform-runtime"], "presets": ["env", "stage-3"] }
-
Add the following npm dependencies:
"devDependencies": { "babel-core": "6.26.0", "babel-loader": "7.1.2", "babel-plugin-source-map-support": "2.0.0", "babel-plugin-transform-runtime": "6.23.0", "babel-preset-env": "1.6.1", "babel-preset-stage-3": "6.24.1", "serverless-webpack": "4.3.0", "webpack": "3.11.0", "webpack-node-externals": "1.6.0" }, "dependencies": { "aws-sdk": "2.194.0", "babel-runtime": "6.26.0", "source-map-support": "0.5.3" }
aws-sdk
isn't needed for this tutorial, but will be for any real application -
Test if it works correctly!
- Run
sls invoke local -f helloWorld
, should print:Time: 685ms Asset Size Chunks Chunk Names handler.js 3.58 kB 0 [emitted] handler handler.js.map 3.82 kB 0 [emitted] handler [0] ./handler.js 796 bytes {0} [built] [1] external "babel-runtime/core-js/promise" 42 bytes {0} [not cacheable] [2] external "source-map-support/register" 42 bytes {0} [not cacheable] { "message": "Go Serverless Webpack (Ecma Script) v1.0! First module!", "event": "" }
- Run
-
Test source maps too
-
Change
./handler.js
and add a syntax error.then(() => callback(null, { throw 'bouh' // Here message: 'Go Serverless Webpack (Ecma Script) v1.0! First module!', event, }))
-
Run
sls invoke local -f helloWorld
-
It should print (on the server)
We can see
ERROR in ./handler.js
with the line number. The stacktrace doesn't show the right line though. (if you know how to fix that, let met know!)
-
-
-
Add
serverless-offline
support for ease of development (see serverless-offline)-
Run
npm install serverless-offline --save-dev
-
Update
serverless.yml
plugins: - serverless-webpack - serverless-offline
-
Go to http://localhost:3000/, it should print (on the browser)
-
Go to http://localhost:3000/hello-world, it should print (on the server) (The web page should be blank)
-
Serverless offline is a great tool to do the dev locally, by running a local node server to handle request and mock AWS lambda behavior for quick development. It isn't perfect (can't mock everything) but does help quite a lot.
-
-
Redirecting all requests to our handler entrypoint
- Update the
serverless.yml
:functions: helloWorld: handler: handler.helloWorld # The `events` block defines how to trigger the handler.helloWorld code events: - http: method: get path: /{proxy+} # This is what captures all get requests and redirect them to our handler.helloWorld function
- Now, go to:
- You'll notice all of them return the same thing (on the server)
- Update the
-
Make Next work with Serverless and display "Hello world!"
-
Move
server.js
tolambdas/server.js
and rename thehello
function tohandler
-
Create
pages/index.js
with the following content:import React from 'react' export default () => { return ( <div>Hello world!</div> ); };
-
Run
npm i -D concurrently jest cross-env serverless-jest-plugin
serverless-jest-plugin
is a nice helper to generate tests -
Run
npm i -S aws-serverless-express next react react-dom
-
Update the npm scripts as follow in package.json`:
"scripts": { "start": "concurrently -p '{name}' -n 'next,serverless' -c 'gray.bgWhite,yellow.bgBlue' \"next\" \"serverless offline --port 3000\"", "build": "cross-env-shell NODE_ENV=production \"next build && serverless package\"", "emulate": "cross-env-shell NODE_ENV=production \"next build && serverless offline\"", "deploy": "serverless deploy", "test:create": "sls create test --path {function}", "test": "jest" },
npm start
: is for development mode, it runs both next and serverless in concurrency, and will display both logs in different color to help debugging. You can still usesls offline
but it will be extremely slow (even though it works) and will do a big rebuild at every request. It is therefore STRONGLY advised to runnpm start
instead from now on.npm start
:npm run build
: To build the app for production environment (both Next and SLS) in.next
and.serverless
respectivelynpm run emulate
: To emulate the production environment in localnpm run deploy
: To deploy the application on the cloud provider (AWS, through serverless)npm run test:create
: Runnpm run test:create -- --function server
, whereserver
is your function file name, note that you need to run this script within the function directory (haven't found a workaround about that yet)npm run test
: Run the tests (TODO: Make it work...)
-
Update
.babelrc
and add thepreset
"next/babel"
-
Create
next.config.js
with the following:module.exports = { webpack: (config, { buildId, dev, isServer, defaultLoaders }) => { config.node = { fs: 'empty', module: "empty", }; return config; }, };
Fixes webpack compilation for
fs
andmodule
, see webpack-contrib/css-loader#447 -
Update
serverless.yml
with the following:# Welcome to serverless. Read the docs # https://serverless.com/framework/docs/ service: serverless-with-next plugins: - serverless-webpack - serverless-offline - serverless-jest-plugin # Enable auto-packing of external modules # See https://serverless-stack.com/chapters/add-support-for-es6-es7-javascript.html custom: webpackIncludeModules: true # The `provider` block defines where your service will be deployed provider: name: aws runtime: nodejs6.10 package: individually: true # The `functions` block defines what code to deploy functions: server: handler: lambdas/server.handler events: - http: method: get path: / - http: method: get path: /_next/{proxy+} package: include: - ../.next/**
We package each function individually (doesn't change anything now because we only have one) But we basically don't want to package the
.next
build with our other endpoints.
-