Introduction to webpack: Entry, Output, Loaders, and Plugins
For CSS-Tricks
Front-end development has shifted to a modular approach, improving the encapsulation and structure of codebases. Tooling became a critical part of any project, and right now there are a lot of possible choices.
webpack has gained popularity in the last years because of its power and scalability, but some developers found its configuration process confusing and hard to adopt.
We'll go step by step from an empty configuration file to a simple but complete setup to bundle a project.
This article assumes basic understanding of CommonJS notation and how modules work.
Concepts
Unlike most bundlers out there, the motivation behind webpack is to gather all your dependencies (not just code, but other assets as well) and generate a dependency graph.
At first, it might look strange to see a .js
file require a stylesheet, or a stylesheet retrieving an image modified as if it was a module, but these allow webpack to understand what is included in your bundle and helps you transform and optimize them.
Install
Let's first add the initial packages we are going to use:
npm install webpack webpack-dev-server --save-dev
Next we create a webpack.config.js
file in the root of our project and add two scripts to our package.json
files for both local development and production release.
"scripts": {
"start": "webpack-dev-server",
"build": "webpack"
}
Entry
There are many ways to specify our "entry point", which will be the root of our dependencies graph.
The easiest one is to pass a string:
var baseConfig = {
entry: './src/index.js'
};
We could also pass an object in case we need more than one entry in the future.
var baseConfig = {
entry: {
main: './src/index.js'
}
};
I recommend the last one since it will scale better as your project grows.
webpack commands will pick up the config file we've just created unless we indicate other action.
Output
The output in webpack is an object holding the path where our bundles and assets will go, as well as the name the entries will adopt.
var path = require('path');
var baseConfig = {
entry: {
main: './src/index.js'
},
output: {
filename: 'main.js',
path: path.resolve('./build')
}
};
// export configuration
module.exports = baseConfig;
If you're defining the entry with an object, rather than hardcoding the output filename with a string, you can do:
output: {
filename: '[name].js',
path: path.resolve('./build')
}
This way when new entries are added webpack will pick up their key to form the file name.
With just this small set of configurations, we are already able to run a server and develop locally with npm start
or npm run build
to bundle our code for release. By knowing the dependencies of the project, webpack-dev-server will watch them and reload the site when it detects one of them has changed.
Loaders
The goal of webpack is to handle all our dependencies.
// index.js file
import helpers from '/helpers/main.js';
// Hey webpack! I will need these styles:
import 'main.css';
What's that? Requiring a stylesheet in JavaScript? Yes! But bundlers are only prepared to handle JavaScript dependencies out-of-the-box. This is where "loaders" make their entrance.
Loaders provide an easy way to intercept our dependencies and preprocess them before they get bundled.
var baseConfig = {
// ...
module: {
rules: [
{
test: /* RegEx */,
use: [
{
loader: /* loader name */,
query: /* optional config object */
}
]
}
]
}
};
For loaders to work, we need a regular expression to identify the files we want to modify and a string or an array with the loaders we want to use.
Styles
To allow webpack to process our styles when required we are going to install css and style loaders.
npm install --save-dev css-loader style-loader
The css-loader will interpret styles as dependencies and the style-loader will automatically include a <style>
tag with them on the page when the bundle loads.
var baseConfig = {
entry: {
main: './src/index.js'
},
output: {
filename: '[name].js',
path: path.resolve('./build')
},
module: {
rules: [
{
test: /\.css$/,
use: [{ loader: 'style-loader' }, { loader: 'css-loader' }]
}
]
}
};
In this example, main.css
will go first through css-loader and then style-loader.
Preprocessors
Adding support for LESS or any other preprocessor is as simple as installing the corresponding loader and adding it to the rule.
rules: [
{
test: /\.less$/,
use: [
{ loader: 'style-loader' },
{ loader: 'css-loader' },
{ loader: 'less-loader' }
]
}
];
Transpiling
JavaScript can be transformed by loaders too. One example would be using a Babel loader to transpile our scripts.
rules: [
{
test: /\.js$/,
use: [{ loader: 'babel-loader' }]
}
];
Images
webpack has a great feature where it can detect url()
statements inside stylesheets and let loaders apply changes to the image file and the url itself.
// index.less file
@import 'less/vars';
body {
background-color: @background-color;
color: @text-color;
}
.logo {
background-image: url('./images/logo.svg');
}
By adding one rule, we could apply the file-loader to just copy the file or use the url-loader, the latest inlines the image as a base64 string unless it exceeds a byte limit, in which case it will replace the url statement with a relative path and copy the file to the output location for us.
{
test: /\.svg$/,
use: [
{
loader: 'url-loader',
query: { limit : 10000 }
}
]
}
Loaders can be configurable by passing a query
object with options, like here where we are configuring the loader to inline the file unless it exceeds 10Kb in size.
Managing our build process this way, we will only include the necessary resources instead of moving a hypothetical assets folder with tons of files that might or might not be used in our project.
If you use React or a similar library you can require the .svg
file in your component with the svg-inline-loader.
Plugins
webpack contains default behaviors to bundle most types of resources. When loaders are not enough, we can use plugins to modify or add capabilities to webpack.
For example, webpack by default includes our styles inside our bundle, but we can alter this by introducing a plugin.
Extracting Assets
A common use for a plugin is to extract the generated stylesheet and load it as we normally do using a <link>
tag.
var ExtractTextPlugin = require('extract-text-webpack-plugin');
var lessRules = {
use: [{ loader: 'css-loader' }, { loader: 'less-loader' }]
};
var baseConfig = {
// ...
module: {
rules: [
// ...
{ test: /\.less$/, use: ExtractTextPlugin.extract(lessRules) }
]
},
plugins: [new ExtractTextPlugin('main.css')]
};
Generate an index file
When building single-page applications we need a .html file to serve it.
The HtmlWebpackPlugin
automatically creates an index.html
file and adds script tags for each resulting bundle. It also supports templating syntax and is highly configurable.
var HTMLWebpackPlugin = require('html-webpack-plugin');
var baseConfig = {
// ...
plugins: [new HTMLWebpackPlugin()]
};
Building for Production
Define the Environment
A lot of libraries introduce warnings that are useful during development time but have no use in our production bundle and increase its size.
webpack comes with a built-in plugin to set global constants inside your bundle.
var ENV = process.env.NODE_ENV;
var baseConfig = {
// ...
plugins: [
new webpack.DefinePlugin({
'process.env.NODE_ENV': JSON.stringify(ENV)
})
]
};
We now need to specify the environment on our commands:
"scripts": {
"start": "NODE_ENV=development webpack-dev-server",
"build": "NODE_ENV=production webpack"
}
process.env.NODE_ENV
will be replaced by a string, allowing compressors to eliminate unreachable development code branches.
This is really useful to introduce warnings in your codebase for your team and they won't get to production.
if (process.env.NODE_ENV === 'development') {
console.warn('This warning will disappear on production build!');
}
Compressing
In production, we need to give users the fastest possible product. By minifying our code to remove unnecessary characters, this reduces the size of our bundle and improves loading times.
One of the most popular tools to do this is UglifyJS
, and webpack comes with a built-in plugin to pass our code through it.
// webpack.config.js file
var ENV = process.env.NODE_ENV;
var baseConfig = {
// ...
plugins: []
};
if (ENV === 'production') {
baseConfig.plugins.push(new webpack.optimize.UglifyJsPlugin());
}
Wrap-up
webpack config files are incredibly useful, and the complexity of the file will depend on your needs. Take care to organize them well as they can become harder to tame as your project grows.
In this article, we started with a blank config file and ended up with a base setup that would allow you to develop locally and release production code. There's more to explore in webpack, but these key parts and concepts can help you become more familiar with it.
If you want to go deeper, I recommend webpack official documentation which has been updated and improved for its second big release.
Do you want me to speak at your conference or write for your publication?
Click here to contact me for collaborations.