Build Rollup UMD bundle for CommonJS


Whereas Webpack is great for building apps, Rollup is great for building libraries.

Rollup supports ES modules out of the box. However, to support CommonJS, the following plugins are required:

Contents

Table of Contents

Prerequisites

Setup

Install rollup:

npm install rollup

Create the entry file index.js:

echo "export default 'Hello, world!'" > index.js

Initialize package.json:

npm init --yes

Add the build script to package.json:

{
  "scripts": {
    "build": "rollup index.js --file dist/bundle.js"
  }
}

The script compiles index.js to dist/bundle.js.

Run the build script:

npm run build

To find that you got the error:

Error: You must specify "output.format", which can be one of "amd", "cjs", "system", "esm", "iife" or "umd".

Output Format

Rollup expects an output format. Here are the general guidelines:

See table for summary:

Environment Output Format
Browser iife
Node.js cjs
Browser + Node.js umd

To generate a UMD bundle with MyModuleName as the export name:

npx rollup index.js --file dist/bundle.js --format umd --name 'MyModuleName'

Load with script

Load the module in a browser using a <script> tag:

<!-- script.html -->
<script src="dist/bundle.js"></script>
<script>
  console.log(window.MyModuleName);
</script>

Load with AMD

Load the module in a browser using AMD (Asynchronous Module Definition):

<!-- amd.html -->
<script src="https://cdnjs.cloudflare.com/ajax/libs/require.js/2.3.6/require.min.js"></script>
<script>
  window.requirejs(['dist/bundle'], function (MyModuleName) {
    console.log(MyModuleName);
  });
</script>

Load with CommonJS

Load the module in Node.js using CommonJS:

node
> const MyModuleName = require('./dist/bundle');
> console.log(MyModuleName);
Hello, world!

Config

Instead of passing the options via the CLI (command-line interface), you can specify them in the configuration file rollup.config.js.

Create the file rollup.config.js:

touch rollup.config.js

Add the config:

// rollup.config.js
const config = {
  input: 'index.js',
  output: {
    file: 'dist/bundle.js',
    format: 'umd',
    name: 'MyModuleName',
  },
};

export default config;

Update the build script in package.json to use the config file:

{
  "scripts": {
    "build": "rollup --config"
  }
}

To give the config file a name other than the default, you’ll need to specify the custom file location:

npx rollup --config my.config.js

Plugins

CommonJS

To use CommonJS syntax, install @rollup/plugin-commonjs:

npm install @rollup/plugin-commonjs

Add the plugin to the rollup config:

 // rollup.config.js
+import commonjs from '@rollup/plugin-commonjs';
 
 const config = {
   input: 'index.js',
   output: {
     file: 'dist/bundle.js',
     format: 'umd',
     name: 'MyModuleName',
   },
+  plugins: [commonjs()],
 };
 
 export default config;

Now refactor index.js:

// index.js
module.exports = 'Hello, world!';

After rebuilding the bundle, loading it with script, AMD, or CommonJS should continue to work:

npm run build && open script.html && open amd.html && node -p "require('./dist/bundle')"
Hello, world!

Node Resolve

Let’s say you want to use lodash in index.js:

npm install lodash
// index.js
var template = require('lodash/template');
var compiled = template('Hello, <%= name %>!');
module.exports = compiled({ name: 'world' });

In order for rollup to locate 3rd party modules in node_modules, you need to install @rollup/plugin-node-resolve:

npm install @rollup/plugin-node-resolve

Add the plugin to the rollup config:

 // rollup.config.js
 import commonjs from '@rollup/plugin-commonjs';
+import resolve from '@rollup/plugin-node-resolve';
 
 const config = {
   input: 'index.js',
   output: {
     file: 'dist/bundle.js',
     format: 'umd',
     name: 'MyModuleName',
   },
-  plugins: [commonjs()],
+  plugins: [commonjs(), resolve()],
 };
 
 export default config;

Build the bundle to verify it still works:

npm run build && node -p "require('./dist/bundle')"
Hello, world!

If you want the module resolution to respect the “browser” field in package.json, you can set the option resolve({ browser: true }).

Terser

To minify your bundle with rollup v2, use terser.

Install rollup-plugin-terser:

npm install rollup-plugin-terser

Add the plugin to the rollup config:

 // rollup.config.js
 import commonjs from '@rollup/plugin-commonjs';
 import import resolve from '@rollup/plugin-node-resolve';
+import { terser } from 'rollup-plugin-terser';
 
 const config = {
   input: 'index.js',
   output: {
     format: 'umd',
     name: 'MyModuleName',
   },
-  plugins: [commonjs(), resolve()],
+  plugins: [commonjs(), resolve(), terser()],
 };
 
 export default config;

Use an environment variable to determine whether you want to build a minified or unminified bundle:

 // rollup.config.js
 import commonjs from '@rollup/plugin-commonjs';
 import import resolve from '@rollup/plugin-node-resolve';
 import { terser } from 'rollup-plugin-terser';
 
 const config = {
   input: 'index.js',
   output: {
     format: 'umd',
     name: 'MyModuleName',
   },
-  plugins: [commonjs(), resolve(), terser()],
+  plugins: [commonjs(), resolve()],
 };
 
+if (process.env.NODE_ENV === 'production') {
+  config.plugins.push(terser());
+}
+
 export default config;

Set the environment variable before you run the build command:

NODE_ENV=production npm run build

Alternatively, this can be done in the package.json build script:

{
  "scripts": {
    "build": "npm run build:min && npm run build:unmin",
    "build:min": "NODE_ENV=production rollup --config",
    "build:unmin": "NODE_ENV=production rollup --config --file dist/bundle.min.js"
  }
}

Uglify

To minify the bundle with rollup v1, use UglifyJS.

Install rollup-plugin-uglify:

npm install rollup-plugin-uglify

Add the plugin to the rollup config:

 // rollup.config.js
 import commonjs from '@rollup/plugin-commonjs';
 import import resolve from '@rollup/plugin-node-resolve';
+import { uglify } from 'rollup-plugin-uglify';
 
 const config = {
   input: 'index.js',
   output: {
     format: 'umd',
     name: 'MyModuleName',
   },
-  plugins: [commonjs(), resolve()],
+  plugins: [commonjs(), resolve(), uglify()],
 };
 
 export default config;

Now you can build an uglified bundle:

npm run build && node -p "require('./dist/bundle')"
Hello, world!

JSON

To import JSON files, install @rollup/plugin-json:

npm install @rollup/plugin-json

Add the plugin to the rollup config similarly to how it was done for the other plugins:

 // rollup.config.js
 import commonjs from '@rollup/plugin-commonjs';
 import import resolve from '@rollup/plugin-node-resolve';
 import { terser } from 'rollup-plugin-terser';
+import json from '@rollup/plugin-json';
 
 const config = {
   input: 'index.js',
   output: {
     format: 'umd',
     name: 'MyModuleName',
   },
-  plugins: [commonjs(), resolve(), terser()],
+  plugins: [commonjs(), resolve(), terser(), json()],
 };
 
 export default config;

Resources

To learn more about Rollup, check out the official documentation.



Please support this site and join our Discord!