DEV Community

Cover image for Step by step React configuration from setup to deployment
Nathan Sebastian
Nathan Sebastian

Posted on • Edited on

Step by step React configuration from setup to deployment

You can download this guide as a PDF here

So you've been using create-react-app a.k.a CRA for a while now. It's great and you can get straight to coding. But when do you need to eject from create-react-app and start configuring your own React application? There will be a time when we have to let go of the safety check and start venturing out on our own.

This guide will cover the most simple React configuration that I personally used for almost all of my React projects. By the end of this tutorial we will have our own personal boilerplate and learn some configurations from it.

Table of Contents

Why create your own configuration?

There are certain reasons that make creating your own React configuration make sense. Obviously, you are good with React and you want to learn how to use tools like webpack and Babel on your own. These build tools are powerful, and if you have some extra time, it's always good to learn about them. Developers are naturally curious people, so if you feel you'd like to know how things work and which part does what, then let me help you with it.

Furthermore, hiding React configuration by create-react-app is meant for developers starting to learn React, as configuration should not stand in the way of getting started. But when things get serious, of course you need more tools to integrate in your project. Think about:

  • Adding webpack loaders for less, sass
  • Doing server side rendering
  • Using new ES versions
  • Adding MobX and Redux
  • Making your own configuration just for learning sake

If you look around the Internet, there are some hacks to get around CRA limitations like create-react-app rewired. But really, why not just learn React configuration on your own? I will help you get there. Step by step.

Now that you're convinced to learn some configuration, let's start by initializing a React project from scratch.

Open up the command line or Git bash and create a new directory

mkdir react-config-tutorial && cd react-config-tutorial
Enter fullscreen mode Exit fullscreen mode

Initialize NPM project by running:

npm init -y
Enter fullscreen mode Exit fullscreen mode

Now install react

npm install react react-dom
Enter fullscreen mode Exit fullscreen mode

Also, you can view the source code on GitHub while reading this tutorial for explanations about the setting.

Configuring webpack 4

Our first stop will be the webpack. It's a very popular and powerful tool for configuring not only React, but almost all front-end projects. The core function of webpack is that it takes a bunch of JavaScript files we write in our project and turns them into a single, minified file, so that it will be quick to serve. Starting from webpack 4, we aren't required to write a configuration file at all to use it, but in this tutorial we will write one so that we can understand it better.

First, let's do some installation

npm install --save-dev webpack webpack-dev-server webpack-cli
Enter fullscreen mode Exit fullscreen mode

This will install:

  • webpack module — which include all core webpack functionality
  • webpack-dev-server — this development server automatically rerun webpack when our file is changed
  • webpack-cli — enable running webpack from the command line

Let's try to run webpack by adding the following script to package.json

"scripts": {
  "start": "webpack-dev-server --mode development",
},
Enter fullscreen mode Exit fullscreen mode

Now create an index.html file in your root project with the following content:

<!DOCTYPE html>
<html>
  <head>
    <title>My React Configuration Setup</title>
  </head>
  <body>
    <div id="app"></div>
    <script src="./dist/bundle.js"></script>
  </body>
</html>
Enter fullscreen mode Exit fullscreen mode

Create a new directory named src and inside it, create a new index.js file

mkdir src && cd src && touch index.js
Enter fullscreen mode Exit fullscreen mode

Then write a React component into the file:

import React from "react";
import ReactDOM from "react-dom";

class Welcome extends React.Component {
  render() {
    return <h1>Hello World from React boilerplate</h1>;
  }
}

ReactDOM.render(<Welcome />, document.getElementById('root'));
Enter fullscreen mode Exit fullscreen mode

Run the webpack by using npm run start ... And an error will be triggered.

You may need an appropriate loader to handle this file type
Enter fullscreen mode Exit fullscreen mode

Configuring Babel 7

The React component we wrote above used the class syntax, which is a feature of ES6. Webpack needs Babel to process ES6 into ES5 syntaxes in order for this class to work.

Let's install Babel into our project

npm install --save-dev @babel/core @babel/preset-env \
@babel/preset-react babel-loader
Enter fullscreen mode Exit fullscreen mode

Why we need these packages?

  • @babel/core is the main dependency that include babel transform script
  • @babel/preset-env is the default Babel preset used to transform ES6+ into valid ES5 code. Optionally configure browser polyfills automatically
  • @babel/preset-react is used for transforming JSX and React class syntax into valid JavaScript code
  • babel-loader is a webpack loader that hook Babel into webpack. We will run Babel from webpack with this package

To hook Babel into our webpack, we need to create a webpack configuration file. Let's write a webpack.config.js file:

module.exports = {
  entry: './src/index.js',
  output: {
    path: __dirname + '/dist',
    publicPath: '/',
    filename: 'bundle.js'
  },
  devServer: {
    contentBase: './',
    publicPath: '/dist/'
  },
  module: {
    rules: [
      {
        test: /\.(js|jsx)$/,
        exclude: /node_modules/,
        use: ['babel-loader']
      }
    ]
  },
};
Enter fullscreen mode Exit fullscreen mode

This webpack config is basically saying that the entry point of our application is from index.js, so pull everything that's needed by that file, then put the output of the bundling process into the dist directory, named bundle.js. Oh, if we're running on webpack-dev-server, then Tell the server to serve content from contentBase config, which is the same directory this config is in. Also place the bundle.js in the dist/ directory, please. For all .js or .jsx files, use babel-loader to transpile all of them.

In order to use Babel presets, create a new .babelrc file

touch .babelrc
Enter fullscreen mode Exit fullscreen mode

Write the following content:

{
  "presets": [
    "@babel/preset-env",
    "@babel/preset-react"
  ]
}
Enter fullscreen mode Exit fullscreen mode

Now run npm run start again. This time it will work.

Adding Prettier

To further speed up development, let's make our code formatter using Prettier. Install the dependency locally and use the --save-exact argument since Prettier introduce stylistic changes in patch releases.

npm install --save-dev --save-exact prettier
Enter fullscreen mode Exit fullscreen mode

Now we need to write the .prettierrc configuration file:

{
  "semi": true,
  "singleQuote": true,
  "trailingComma": "es5"
}
Enter fullscreen mode Exit fullscreen mode

The rules means that we want to add semicolon for the end of every statement, use a single quote whenever appropriate and put trailing commas for multi-line ES5 code like object or array.

You can run Prettier from the command line with:

npx prettier --write "src/**/*.js"
Enter fullscreen mode Exit fullscreen mode

Or add a new script to our package.json file:

"scripts": {
  "test": "echo \"Error: no test specified\" && exit 1",
  "start": "webpack-dev-server --open --mode development",
  "format": "prettier --write \"src/**/*.js\""
},
Enter fullscreen mode Exit fullscreen mode

Now we can run Prettier using npm run format

Additionally, if you're using VSCode for development, you can install Prettier extension and run it every time you save your changes by adding this setting:

"editor.formatOnSave": false
Enter fullscreen mode Exit fullscreen mode

Adding source map for better error logs

Since webpack bundles the code, source maps are mandatory to get a reference to the original file that raised an error. For example, if you bundle three source files (a.js, b.js, and c.js) into one bundle (bundle.js) and one of the source files contains an error, the stack trace will simply point to bundle.js. This is problematic as you probably want to know exactly if it's a, b, or c file that is causing an error.

You can tell webpack to generate source maps using the devtool property of the configuration:

module.exports = {
  devtool: 'inline-source-map',
  // ... the rest of the config

};
Enter fullscreen mode Exit fullscreen mode

Although it will cause slower build, it has no effect on production. Sourcemaps are only downloaded if you open the browser DevTools.

Setting up ESLint

Linter is a program that checks our code for any error or warning that can cause bugs. JavaScript's linter, ESLint, is a very flexible linting program that can be configured in many ways.

But before we get ahead, let's install ESLint into our project

npm --save-dev install eslint eslint-loader babel-eslint eslint-config-react eslint-plugin-react
Enter fullscreen mode Exit fullscreen mode

eslint is the core dependency for all functionalities, while eslint-loader enables us to hook eslint into webpack. Now since React used ES6+ syntax, we will add babel-eslint — a parser that enables eslint to lint all valid ES6+ codes.

eslint-config-react and eslint-plugin-react are both used to enable ESLint to use pre-made rules.

Since we already have webpack, we only have to modify the config slightly:

module.exports = {

  // modify the module
  module: {
    rules: [
      {
        test: /\.(js|jsx)$/,
        exclude: /node_modules/,
        use: ['babel-loader', 'eslint-loader'] // include eslint-loader
      }
    ]
  },
};
Enter fullscreen mode Exit fullscreen mode

Then create an eslint config file named .eslintrc with this content:

{
  "parser": "babel-eslint",
  "extends": "react",
  "env": {
    "browser": true,
    "node": true
  },
  "settings": {
    "react": {
      "version": "detect"
    }
  }
}
Enter fullscreen mode Exit fullscreen mode

The config is basically saying, "Hey ESLint, please parse the code using babel-eslint before you check it, and when you're checking it, please check if all the rules from our React rules config is passed. take global variables from the environment of browser and node. Oh, and if it's React code, take the version from the module itself. That way the user won't have to specify version manually."

Rather than specifying our own rules manually, we simply extends react rules which was made available by eslint-config-react eslint-plugin-react

I found errors! What do I do?

Unfortunately the only way to really figure out how to fix ESLint errors are by looking at the documentation for rules. There's a quick way to fix ESLint errors by using eslint --fix, and it's actually good for a quick fix. Let's add a script on our package.json file:

"scripts": {
  "test": "echo \"Error: no test specified\" && exit 1",
  "start": "webpack-dev-server --mode development",
  "format": "prettier --write \"src/**/*.js\"",
  "eslint-fix": "eslint --fix \"src/**/*.js\"", // the eslint script
  "build": "webpack --mode production"
},
Enter fullscreen mode Exit fullscreen mode

Then run it with npm run eslint-fix. Don't worry if you're still fuzzy about ESLint for now. You will learn more about ESLint as you use it.

Adding CSS LESS processor

In order to add LESS processor into our React application, we will require both less and loader packages from webpack

npm install --save-dev less less-loader css-loader style-loader
Enter fullscreen mode Exit fullscreen mode

less-loader will compile our less file into css, whil css-loader will resolve css syntax like import or url(). The style-loader will get our compiled css and load it up into <style> tag in our bundle. This is great for development because it lets us to update our style on the fly, without needing to refresh browser.

Now let's add some css file create a new style directory in src/style

cd src && mkdir style && touch header.less && touch main.less
Enter fullscreen mode Exit fullscreen mode

header.less content:

.header {
  background-color: #3d3d;
}
Enter fullscreen mode Exit fullscreen mode

main.less content:

@import "header.less";

@color: #f5adad;

body {
  background-color: @color;
}
Enter fullscreen mode Exit fullscreen mode

Now import our main.less file from index.js:

import "./style/main.less";
Enter fullscreen mode Exit fullscreen mode

Then update our webpack configuration module property:

module: {
    rules: [
      {
        test: /\.(js|jsx)$/,
        exclude: /node_modules/,
        use: ['babel-loader', 'eslint-loader']
      },
      { 
        test: /\.less$/,
        use: [ 
          'style-loader',
          'css-loader', 
          'less-loader',
        ],
      },
    ]
  },
Enter fullscreen mode Exit fullscreen mode

Run the start script and we're good to go!

Deploying React app to Netlify

All applications need to be deployed for its last step, and for React application, deployment is very easy.

First, let's change the build output and development contentBase from dist to build in our Webpack config.

module.exports = {
  entry: './src/index.js',
  output: {
    path: path.resolve(__dirname, 'build'), // change this
    publicPath: '/',
    filename: 'bundle.js'
  },
  devServer: {
    contentBase: "./build",
  },
  //...
Enter fullscreen mode Exit fullscreen mode

Now let's install a new Webpack plugin named HtmlWebpackPlugin

npm install html-webpack-plugin -D
Enter fullscreen mode Exit fullscreen mode

This plugin will generate index.html file in the same directory where our bundle.js is created by Webpack. In this case, the build directory.

Why do we need this plugin? Because Netlify requires a single directory to be made the root directory, so we can't use index.html in our root directory using Netlify. You need to update your webpack config to look like this:

const path = require('path');
const HtmlWebpackPlugin = require('html-webpack-plugin');

module.exports = {
  entry: //... 
  output: {
    //...
  },
  devServer: {
    contentBase: "./build",
  },
  module: {
    //...
  },
  plugins: [
    new HtmlWebpackPlugin({
      template:  path.resolve('./index.html'),
    }),
  ]
};
Enter fullscreen mode Exit fullscreen mode

And please remove the script tag from your index.html:

<!DOCTYPE html>
<html>
  <head>
    <title>My React Configuration Setup</title>
  </head>
  <body>
    <div id="root"></div>
  </body>
</html>
Enter fullscreen mode Exit fullscreen mode

Now you can test the config with npm run build command. Once it's done, push your boilerplate into a GitHub repo. It's time to deploy our application!

Let's register a Netlify account.

If you haven't heard of Netlify before, it's an amazing static site hosting that provides all the tools you need to deploy a static site for free. What's a static site? It's a website created from a collection of static HTML pages, without any backend. Our React boilerplate as it is now counts as a static site, because we have no backend configured and its just HTML and JavaScript.

After sign up, select new site from Git and Choose GitHub as your Git provider

Netlify provider

You need to grant permissions for Netlify, and then select your React boilerplate repo.

Netlify select boilerplate

Now you need to enter the build command and publishing directory. As you can see, this is why we need HtmlWebpackPlugin, because we need to server everything from one directory only. Rather than manually updating our root index.html file for changes, we just generate it using the plugin.

Netlify select boilerplate

Make sure you have the same command as the screenshot above, or your app might not run.

Netlify select boilerplate

Once the deploys status turns to published (number 2 above), you can go to the random site name Netlify has assigned for your application (number 1).

Your React application is deployed. Awesome!

Conclusion

You've just created your very own React project boilerplate and deploy it live to Netlify. Congratulations! Granted, I didn't go very deep on webpack configurations, because this boilerplate is meant to be a generic starter. In some cases where we need advanced features like server side rendering, we need to tweak the configuration again. But don't worry! You've come this far, which means you already understand what webpack, Babel, Prettier and ESLint do. Webpack has many powerful loaders that can help you with many cases you'll frequently counter when building a web application. Don't worry if you're still uncomfortable with configurations. You'll get used to it as you practice more.

I'll make more tutorials on React such as using React router and React helmet in the future.. so if you're interested for that, you can follow me here on DEV or subscribe to my newsletter to get notified when new guides are released. You can also have this configuration as a PDF for free! 🎁

Thank you for reading! Hope you learned something new :)

Top comments (3)

Collapse
 
elliottribner profile image
Elliott Ribner

Nathan, Thanks for the article. You have one small error that will cause an error in the browser console. in the js file you have: ReactDOM.render(<Welcome />, document.getElementById('root'));, however, in the HTML file no element has the ID root, instead you have used the id of "app". The above line of code should be switched to `ReactDOM.render(<Welcome />, document.getElementById('app'));

Collapse
 
servatj profile image
Josep • Edited

Great post Nathan, after creating .babelrc you mention that we can run our react app, by running npm run start, but we need to create first the dist folder by building it with

  "scripts": {
    "build-dev": "webpack -d --mode development",

Am I right?

I hope this will helps other people that got stuck at this point like me. I appreciate a lot your work and don't hesitate to let me know if I'm missing something.

Collapse
 
j0ckde profile image
Josef Molz

Thank you very much for that good guide!

You wrote that you should change "editor.formatOnSave" : " false" if you got the prettier extension for VSCode installed.
Shouldn't it be set to true?

Thanks in advance!