Skip to content

UrosNikolic/serverless-with-next

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

27 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Serverless with Next

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.


Getting started

  • 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 run sls 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 node 6.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)

Why?

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.

Features

  • 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)

Routing workflow

Scenario 1 - GET /

Here is how a standard GET request will flow, assuming we call / (https://swn.dev.vadorequest.fr/ in our example):

  1. Hit server function route because of path: / (serverless.yml)
  2. Hit server.js:handler proxy (/src/functions/server/server.js)
  3. Hit the rule app.get('*') which proxies the Next app and run nextProxy(req, res);, which is then treated by the Next app
  4. Next app will resolve the / path by resolving it to /pages/index.js

Scenario 2 - GET /status

Here is how a standard GET request will flow, assuming we call /status (https://swn.dev.vadorequest.fr/status in our example):

  1. Hit status function route because of path: status (serverless.yml)
  2. Hit status.js:handler proxy (/src/functions/status/status.js)

Scenario 3 - GET /whatever/nested

Here is how a standard GET request will flow, assuming we call /whatever/nested (https://swn.dev.vadorequest.fr/whatever/nested in our example):

  1. Hit server function route because of path: /{any+} (serverless.yml)
  2. Hit server.js:handler proxy (/src/functions/server/server.js)
  3. Hit the rule app.get('/:level1/:level2') which will return a JSON response {"level1":"whatever","level2":"nested"}

Scenario 4 - GET /whatever/nested/deep

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):

  1. Hit server function route because of path: /{any+} (serverless.yml)
  2. Hit server.js:handler proxy (/src/functions/server/server.js)
  3. Hit the rule app.get('*') which proxies the Next app and run nextProxy(req, res);, which is then treated by the Next app
  4. Next app will fail to resolve the /whatever/nested/deep path and display a 404 because no page match for this URL

Routing summary

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.

Development vs production (local vs AWS) stages differences

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.

AWS

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

Local

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.

Requirements

This project assume:

Known issues

  1. 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 the staging part of the url entirely, which fixes the issue):

    See issue

  2. Useless files are packaged and uploaded to AWS:

    The .next and static 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 use serverless-webpack which isn't compatible), I don't know how to ignore those folders for certain functions.

    See

  3. 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 got TypeError: 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 doing app.use('/_next/', proxy('http://localhost:3001/_next/')); but then I get 404 for all js scripts like http://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 complains Could 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.

    See issue


In-depth diving of the configuration

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.

Serverless plugins

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 advanced

  1. 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.

Babel config (.babelrc)

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.

Next

next.config.js

Due to a webpack's bug/unwanted behavior, we get warnings/errors due to missing fs module. See evanw/node-source-map-support#155

Static

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.

Pages

Next "Pages" are in the /pages folder and can't be moved in another folder.

Version

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.

Serverless

TODO

Functions

TODO How to catch all routes (main handler)

Utils

Logging

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.


Acknowledgements

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.


DEPRECATED TUTORIAL, WON'T MAINTAIN - Steps (tutorial, from scratch)

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.

  1. Run sls create --template hello-world --path serverless-with-next (optionally ignore .idea folder)

  2. Test using sls deploy should print something like this:

  3. Let's add ES6 using webpack and serverless-webpack

    1. Run npm init -y

    2. Ignore .webpack folder

    3. Update serverless.yml

      plugins:
        - serverless-webpack
      

      We use the serverless-webpack plugin to build our serverless app. The build is then uploaded to aws

    4. Add .babelrc config

      {
        "plugins": ["source-map-support", "transform-runtime"],
        "presets": ["env", "stage-3"]
      }
      
    5. 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

    6. Test if it works correctly!

      1. 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": ""
        }
    7. Test source maps too

      1. 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,
        }))
        
      2. Run sls invoke local -f helloWorld

      3. 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!)

  4. Add serverless-offline support for ease of development (see serverless-offline)

    1. Run npm install serverless-offline --save-dev

    2. Update serverless.yml

      plugins:
        - serverless-webpack
        - serverless-offline
      
    3. Run sls offline, should print:

    4. Go to http://localhost:3000/, it should print (on the browser)

    5. Go to http://localhost:3000/hello-world, it should print (on the server) (The web page should be blank)

    6. 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.

  5. Redirecting all requests to our handler entrypoint

    1. 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
    2. Now, go to:
    3. You'll notice all of them return the same thing (on the server)
  6. Make Next work with Serverless and display "Hello world!"

    1. Move server.js to lambdas/server.js and rename the hello function to handler

    2. Create pages/index.js with the following content:

      import React from 'react'
      
      export default () => {
        return (
          <div>Hello world!</div>
        );
      };
    3. Run npm i -D concurrently jest cross-env serverless-jest-plugin

      serverless-jest-plugin is a nice helper to generate tests

    4. Run npm i -S aws-serverless-express next react react-dom

    5. 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 use sls 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 run npm 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 respectively
      • npm run emulate: To emulate the production environment in local
      • npm run deploy: To deploy the application on the cloud provider (AWS, through serverless)
      • npm run test:create: Run npm run test:create -- --function server, where server 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...)
    6. Update .babelrc and add the preset "next/babel"

    7. 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 and module, see webpack-contrib/css-loader#447

    8. 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.

About

Serverless.js with Next.js (using AWS provider)

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • JavaScript 100.0%