A somewhat opinionated, hot-reloading boilerplate for React + Redux, Express, and SQL.
__ __ _ _
\ \ / /__| | ___ ___ _ __ ___ ___ | |_ ___
\ \ /\ / / _ \ |/ __/ _ \| '_ ` _ \ / _ \ | __/ _ \
\ V V / __/ | (_| (_) | | | | | | __/ | || (_) |
\_/\_/ \___|_|\___\___/|_| |_| |_|\___| \__\___/
____ _ _ _ ___
| __ ) ___ (_) | ___ _ __ _ __ | |( _ )
| _ \ / _ \| | |/ _ \ '__| '_ \| |/ _ \
| |_) | (_) | | | __/ | | |_) | | (_) |
|____/ \___/|_|_|\___|_| | .__/|_|\___/
|_|
First, create a file in the root of the directory called .env
. In here, you should have the
following structure, at a minimum. Put any other environment variables you like in there, too.
DB_PRODUCTION=
DB_TESTING=postgres://test:[email protected]/project_test
DB_DEVELOPMENT=postgres://user:[email protected]/project
Of course, change those DB urls to whatever they need to be on your machine.
Optional environment variables are:
APPLICATION_NAME=<your application name> <-- defaults to package.json -> name
PORT=<port> <-- defaults to 3000
API_VERSION=<number> <-- defaults to 1
Next, run yarn
to install dependencies.
Do yourself a favour and get redux-dev-tools for Chrome, by the amazing Dan Abramov. This project is set up to support it in development mode. Have a look at it when you're playing around with the app. See what happens when you drag the slider back and forth.
You also will probably want React Developer Tools by Facebook. It shows you the Chrome Elements layout, with your React elements. Also shows you the props, state and store for every component, and gives you access to them via the Chrome console.
This project has a yarn.lock
for deterministic package installation, but it's completely
optional.
There are many scripts included.
npm run serve:dev
- serves the express server, and starts webpack.npm run serve:dash
- same asserve:dev
, but useswebpack-dashboard
for a nice user interface.npm run lint
- runseslint
on the entire codebase, using airbnb's config.npm test
- runs the tests. All files under./test
, ending in.test.js
are run.npm report
- compiles the test coverage, generated bynyc
during testing, into HTML for easy viewing
The build commands are still a work-in-progress, but have recently gotten a little better. They are:
build
- runs clean and build on the client and serverclean
- removes all compiled filesbuild:client
- compiles the client-side code and stores it in./static/bundle.js
build:server
- compiles the code inside of./src
into./out
And finally,
npm start
- runsforever
on the compiled./src/index.js
in production mode.
The server follows a fairly straightforward layout. I don't tend to follow the Rails paradigm of storing model controls in the models, but rather inside the RESTful API that will control it.
The way that the models are structured lend themselves to this. All your Sequelize models will
be attached to the DB, which can be used by import db from './models'
from wherever you need it
. See the ./models/User.js
file for an example, and it's use inside of ./routes/user.js
.
All routes are required by default into the core router, under /api/v1/<file_name>
by default.
Change the version by setting the API_VERSION
environment variable. I would recommend against
modifying the index. It is set up to serve the Webpack bundle, as well and the index.
You should only need to add route files to ./server/routes
and models to ./server/models
.
If you want a route that isn't attached to /api/v1/<route_name>
, simply name the route file
__<route_name>.js
and it will be attached at /<route_name>
.
The client contains all the React code.
All actions are stored inside of ./client/actions
. I store action constants in here, so
they're namespaced and easier to keep track of. This is standard redux flow though, so check out
their docs for more.
The reducers are all stored inside of ./client/reducers
. Similar to the routes and models,
all reducers are required automatically and combined. This way, sections of the state are
automatically split by Redux's combineReducers
function. Additionally, I use immutable.js
to ensure that state changes are reflected, and easy to reason about. The project contains
redux-immutable
to allow all parts of the state to be Immutable objects.
CSS Modules with PostCSS is included. There are some great resources resources included by default, and you can find more at PostCSS.
./client/components/App.jsx
is a pure, stateless component, but you can do what you like to it.
Currently, it hot reloads, and all sub-components of it should also do it. The console will warn
you if you break hot reloading.
At present, both sections of the project hot-reload. The main motivation behind hot-reloading the server was so that the hot-reloading of the client was not broken by using something like nodemon.
The server-side hot-reloading is a tad hacky. Essentially, what it does, it to soft-require, or
require without assigning to a variable, all the routes (and subsequently all their sub-modules).
When there is a change to any file inside of ./server
, the node require cache is invalidated,
so on the next request, all the modules are re-required, executing the new code.
This project, and most definitely this README is a work in progress. I update it after almost every project I start from it, so I wouldn't go staking your life's work on it. Almost every update is completely breaking.
It is a good place to start if you want to see how a project is set up. It is also good if you want to see how you can incorporate hot-reloading into your existing project.