Usage:
npm install -g yo generator-webpack-react-redux-mocha-karma
yo webpack-react-redux-mocha-karma <appname>
and then
npm start
The solution is based on Webpack. The configuration file is split into 3:
- webpack.config.js - default set of configuration options
- webpack-test.config.js - overrides for tests
- webpack-production.config.js - overrides for production builds
Having those separated allows running commands from npm scripts in the proper context and with the appropriate build configuration.
React is used to represent application state.
Redux is used to maintain the application state.
Karma, MochaJS, Chai and Enzyme are used for testing.
Babel is used to transpile code from ES6 to ES5 (with es2015 and react presets)
ESLint is used to keep you organized in the wilderness of JavaScript capabilities.
The following commands are available:
Starts development server listening on port 3000 with hot reloading.
Starts webpack-dev-server in watch mode running tests. It is clever enough to know which modules have ties to which tests and when the sources change re-runs the dependant tests automatically.
Open http://localhost:3001 to run tests.
Runs all tests. It is intended for use in CI environment. The configured reporter here is junit to allow for easy integration with CI servers like Bamboo and Jenkins. Test results are stored in target
folder, for example target/tests/test-results.xml
. When running tests code coverage is analyzed and the results land in target/coverage/
in html, cobertura and text summary/details forms.
Executes static code analysis. Designed to be part of the CI environment. ESLint is hooked up as a preprocessor for all .js
files in the project when running npm start
so you'll see all the validation messages in the console (in the browser!) while developing your app.
Builds a production version of the application.
Builds static documentation of your project
Stats a server on port 3002 that serves the documentation. It watches for code changes and rebuilds the docs (good for when you work on your project's documentation)
Organization of the project is very similar in many aspects to that of a Java Maven project.
This means that all main project source files are inside of src/main
folder, all tests are located
underneath src/test
, all output goes to target
(not dist
as it usually is with Node projects).
There are 3 main entry points in the build:
src/main/index.js
- entry point for the applicationsrc/main/index.html
- entry point for the browsersrc/test/setup.js
- entry point for tests
All entry points undergo some sort of processing. index.js
and setup.js
are compiled using Babel ES6 transpiler. index.html
gets injected with the transpiled version of index.js
.
The Babel transpiler has been equipped with the obvious es2015
and react
presets. However, in the
interest of making things easier to read and write the react-html-attrs plugin has been added so that
class
and for
are not banned from JSX.
The CSS/Less loader has been configured so that styles are modules that can be imported into the sources and then used. The biggest advantage of this approach is human-readable code and scoped CSS (no more need for BEM or any other naming convention!). See src/main/components/Input.js & Input.less
for example. This blog post explains in details the behavior. Just note that with the addition of module
option for css-loader
all classes are local by default (postcss-local-scope not needed!!).
You can always create globally scoped classes. To do that define then in :global { ... }
scope. See src/main/components/App.less
for example.
A customized version of classnames-loader has been implemented. In the original version all you can do is concatenate the classes. In this extended version you can do that too but also you can access individual classes as if they were regular modules. See src/main/components/Input.js
to see it in action.
There are 3 special configuration options that are key for this project.
complexity
(as in cyclomatic code complexity) is set to be at most 5react/prop-types
is disabled (I don't care about those - yet)react/no-unknown-property
is tweaked because we're using react-html-attrs Babel plugin.
That additional configuration is done in .eslintrc
in the root folder and can be further extended as needed.
There are 2 things to know about the test environment:
- chai's
expect
, enzyme'smount
,shallow
andrender
methods as well as theReact
object are available globally - a preconfigured
mockStore
(redux-mock-store) is also available globally (seesrc/test/components/Title.test.js
)
The entire setup of test environment happens in src/test/setup.js
and can serve as extension point for
further enhancements.
ESLint configuration is tweaked in tests to reflect that setup. In particular global variables are made available for the linter.
The application's store is created in src/main/store.js
. In there you'll find a placeholder for middleware (none installed by default), a mixin for use with ReduxDevTools and a reducer created from exports from src/main/reducers.js
. I have choosen this approach to have a clear point where I add new reducers and a clear naming convention for that as well.
Reducers and actions are kept in the same file as per Dan Abramov's suggestion to keep things that change together close. Each file in the src/main/state
folder represents one entry in the root reducer and actions that modify it. I have choosen the approach to export reducer
by name and actions
by default so that importing them in src/main/reducers.js
is straightforward and plays nice with copy-pasted code. Actions on the other hand can have different names depending on where they are included (for example just actions
if that's all you care about or SomeActions
and OtherActions
if you'd like to include actions for more than just one part of the store).
One notable addition: instead of using a complex switch/case
statement I have opted to use a function that calls the appropriate subreducers based on the action provided. That functionality is provided by redux-action-reducer-mapper. That way the complexity of the inner-workings of the reducer is zero and no tests to cover that piece are necessary. That does not mean you should not tests your reducers! You just don't have to have a tests that checks if all paths through the swich/case
are covered. Had that not been the case the ESLint rule for max cyclomatic code complexity of 5 would have easily been violated.
All components in this basic setup are stateless and declared as pure functions. To connect them to Redux' store I used the react-redux package. Please see this and the next lessons on egghead.io. The only deviation from what's in that tutorial is that I didn't explicitly declare the mapStateToProps
and mapDispatchToProps
functions inlining them close to the call to connect
.
The project should allow for easy extension and tweaking. To achieve that all configuration points have extensive documentation in comments. For the sake of completeness here are the few honorable mentions:
- extending default build configuration to include additional types of files -
webpack.config.js/module/loaders
(useful when you'd like to load for example JSON files straight from the project's sources) - extending the environment for tests -
src/test/setup.js
(useful for adding new globally available identifiers) - making eslint like what you added in previous point -
src/test/.eslintrc - globals
- adding new actions and reducers - create a new file in
src/main/state
with the name of the new key in global state scope, export object with actions by default andreducer
as a named export. Add thereducer
named export tosrc/main/reducers.js
and use default export with actions as you see fit. - adding a new component - create a new file in
src/main/components
(try to keep it stateless!)
A word of caution: The current design of webpack configuration is based on the idea that the default configuration is geared 100% towards the development experience and other configuration types are derived from it. Deriving from a concrete version of the configuration poses a few challenges, like 'how do you remove a loader?', 'how do you change the preloaders configuration?'. To achieve those the ordering of elements in webpack.config.js
is not arbitrary. Therefore other configurations (like for example the production build) can take advantage of the indexation and access individual items in configuration arrays.
So whenever you want to add something to the configuration make sure you don't destroy the order of things defined in webpack.config.js