Lightweight boilerplate project to setup a React 16 web application on AWS Lambda using the Serverless Framework.
- Universal app; server-side rendering with dynamic configuration context passed from backend to browser.
- Self-contained; no additional setup steps necessary other than running
npx sls deploy
. - Lightweight; no mandatory
redux
,react-router
,sass
,less
or any other 3rd party dependency for full flexibility. - Full TypeScript support using Babel 7 and Webpack 4.
- Working Jest test environment.
Looking for the plain JavaScript version of this boilerplate?
The idea is that we use AWS Lambda to serve the dynamic part of our app, the server-side logic, and perform the server-side rendering. For all static data like images, stylesheets and even the app's index.tsx
that is loaded in the browser, we use an S3 bucket for public hosting.
This combination makes our app fast and incredibly scalable. AWS will spin up new Lambda instances once your number of users increases, handling even the largest spikes fully automatically, while incurring virtually no costs when your app isn't used. At the same time S3 provides a robust and fast platform for your static content so you don't have to waste your own computing resources.
All resources, including the S3 bucket for hosting static content, are created and configured automatically when your app is deployed the first time. You can make changes to the default setup by updating your serverless.yml
to your linking.
serverless-react-boilerplate/
│
├── public/ - Public assets such as images
│ ├── favicon.ico - Favicon
│ └── manifest.json - Web page manifest
│
├── src/
│ ├── browser/
│ │ └── ... - Client-side code running in the browser as well as during server-side rendering
│ ├── components/
│ │ └── ... - React components
│ └── server/
│ └── ... - Server-side code running on AWS Lambda
│
├── handler.js - AWS Lambda function handler
├── serverless.yaml - Project configuration
├── webpack.browser.config.js - Webpack configuration for client-side code
├── webpack.server.config.js - Webpack configuration for the Lambda backend
└── ...
The project is based on the Serverless Framework and makes use of a number of plugins:
- Webpack Plugin - We use Webpack for packaging our sources.
- Offline Plugin - The Serverless Offline Plugin allows you to run Serverless applications locally as if they would be deployed on AWS. This is especially helpful for testing web applications and APIs without having to deploy them anywhere.
- Scripts Plugin - Run shell scripts as part of your Serverless workflow
- S3 Deploy Plugin - Deploy files to S3 buckets. This is used for uploading static content like images and the generated
index.js
.
Though we use the same source code for both the server-side and browser rendering, the project will be packaged into two distinct bundles:
- Backend code running on AWS Lambda. The main entry point is
./src/server/render.tsx
. It contains the handler function that is invoked by AWS Lambda.The packaging is controlled bywebpack.server.config.js
and optimized for Node.js 12. - Frontend code hosted in an S3 bucket and loaded by the browser. Main entry point is
./src/browser/index.tsx
. It's packaged using thewebpack.browser.config.js
, optimized for web browsers. The output files will have their content hash added to their names to enable long term caching in the browser.
Update the serverless.yaml
with your project name and additional resources you might need.
The frontend as well as the server-side code running on AWS Lambda share a common application configuration. Currently it is used for injecting the application name from the public/manifest.json
as well as setting the public host names. You can extend the configuration by adding your own variables to src/server/config.tsx
. They will become available in both your backend and frontend code via the useConfig
hook:
import useConfig from "../components/useConfig";
export default function MyComponent() {
const config = useConfig();
return (
// ...
)
}
I would recommend to create a separate Serverless service that provides the frontend with an API and protect it via Amazon Cognito, a custom Authorizer or even just an API Key. Mixing React with pure backend API functions is possible and perfectly fine, but in my experience it quickly becomes a hassle and you need to take special care not to leak anything to the browser that's not supposed to be known there.
The goal of this boilerplate is to offer a minimal setup that can be used as a basis for pretty much all your React needs. A lot of people love Redux, rely on React Router or need other external modules. I have intentionally left these out of the boilerplate code but it should be trivial to add them, following the standard documentation steps.
If you are interested in integrating with React Router, checkout out the Added React Router example configuration Pull Request.
Similar to the statement above, I have decided against integrating with a specific framework. The boilerplate uses plain and simple CSS and integrating another system should be easy enough.
This project constist of very little JavaScript and porting it to Flow or TypeScript is relatively straight forward. However, as pretty much all my own projects are written in TypeScript I'm providing a separate TypeScript branch. It contains everything necessary to run a TypeScript based React application.
To keep this repository lightweight no ESLint rules are included. There are many different plugins and people tend to prefer different coding styles. The existing code should be easily adaptable to any style you personally prefer. I recommend using Prettier to format your code automatically and a default configuration is already part of this repo, defined in package.json
. In addition, I recommend adding ESLint and Husky to your project to ensure your coding guidelines are followed.
To add ESLint with my preferred settings, follow these teps:
npm install --save-dev eslint-plugin-react-hooks eslint eslint-plugin-filenames eslint-plugin-prettier eslint-plugin-promise eslint-plugin-react husky import-sort-style-module lint-staged prettier-plugin-import-sort prettier-plugin-package
Update your package.json
and add the following sections somewhere at the end. Husky will automatically compile and lint your changes before every new commit to ensure all new files adhere to the defined standard.
/* package.json */
{
...
"husky": {
"hooks": {
"pre-commit": "tsc --noEmit && lint-staged"
}
},
"importSort": {
".js, .jsx, .ts, .tsx": {
"style": "module",
"parser": "typescript"
}
},
"lint-staged": {
"src/**/*.{js,jsx,ts,tsx}": ["npx prettier --write", "eslint --fix"]
}
}
Create a new .eslintrc.yml
file in your project root folder and add the following content. Adjust the rules to your linking:
# .eslintrc.yml
env:
node: true
browser: true
jest: true
plugins:
- "@typescript-eslint"
- filenames
- prettier
- promise
- react
- react-hooks
extends:
- eslint:recommended
- plugin:react/recommended
- plugin:promise/recommended
- plugin:@typescript-eslint/eslint-recommended
- plugin:@typescript-eslint/recommended
- plugin:@typescript-eslint/recommended-requiring-type-checking
parser: "@typescript-eslint/parser"
parserOptions:
project: ./tsconfig.json
settings:
react:
version: detect
rules:
"@typescript-eslint/ban-types": 0 # to allow "{}" as a type
"@typescript-eslint/camelcase": 0 #deprecated
"@typescript-eslint/explicit-function-return-type": 0 # allow implicit return types
"@typescript-eslint/explicit-module-boundary-types": 0
"@typescript-eslint/interface-name-prefix": 0 # interfaces prefixed with "I" are perfectly fine
"@typescript-eslint/no-empty-function": 0
"@typescript-eslint/no-explicit-any": 0
"@typescript-eslint/no-floating-promises": error
"@typescript-eslint/no-inferrable-types": 0
"@typescript-eslint/no-unused-vars": [error, { vars: all, ignoreRestSiblings: true }]
"@typescript-eslint/no-useless-constructor": error
"@typescript-eslint/no-var-requires": 0 # allow `require()`
"@typescript-eslint/require-await": 0
filenames/match-regex: 0
filenames/match-exported: error
filenames/no-index: 0
import/order: 0 # we use prettier import sorting by module
no-console: 0
no-restricted-imports:
- error
- paths:
- name: moment
message: Use date-fns instead!
- name: bluebird
message: Use native Promises and async/await instead!
no-unused-expressions: 0 # use @typescript-eslint/no-unused-expressions instead
no-unused-vars: 0 # use @typescript-eslint/no-unused-vars instead
no-useless-constructor: 0 # use @typescript-eslint/no-useless-constructor instead
prettier/prettier: error
react-hooks/exhaustive-deps: error
react-hooks/rules-of-hooks: error
overrides:
- files: ["*.js"]
rules:
"@typescript-eslint/no-unsafe-assignment": 0 # avoid linter errors with .js files
"@typescript-eslint/no-unsafe-call": 0
"@typescript-eslint/no-unsafe-member-access": 0
"@typescript-eslint/no-unsafe-return": 0
"@typescript-eslint/no-unused-vars": 0
That's it! You should have a fully working ESLint setup in your project now.
This is how I format my code in most of my projects. Please update the configuration to your likings. I have my own reasons why I enable certain settings and why others are disabled. But going through all of them here would make much sense. Instead, please simply update it to your personal preferences.
You can test the setup locally. However, due to the nature of this setup you will need to deploy the stack at least once to get everything prepared properly for local execution. Once the stack is deployed, no access to AWS is needed other than read-only to the stack itself as well as its dependencies. This allows developers to write and test code even if not everyone has full deployment access.
The whole application can be deployed with a single command:
npx sls deploy
For local testing run the following command and open your web browser at http://localhost:3000/dev (with dev
being the stage
name configured in your serverless.yml
). Static content such as images will be served via the Webpack DevServer running on http://localhost:8080. Note that the app has to be deployed first before you will be able to run locally.
npm start
Testing is set up as well, using Jest and will execute all *.test.ts
and *.test.tsx
ffiles in the src/
directory:
npm test
And finally to remove all AWS resources again run:
npx sls remove
This will delete all resources but the distribution S3 bucket. As it still contains the bundles you will have to delete it manually for now.