bugl
bugl
HomeLearnPatternsPathsSearch
HomeLearnPatternsPathsSearch

Loading lesson path

Learn/Node.js/Module Basics
Node.js•Module Basics

Node.js ES Modules

Flash cards

Review the key moves

1/4
Core idea

What is the main idea behind Node.js ES Modules?

Lesson checks

Practice each idea before moving on

Short Mimo-style checks built from this lesson's code, terms, and sequence.

1Quick choice

Which statement best captures the main point of this lesson?

2Fill blank

Complete the missing token from the example code.

// ___.js (CommonJS)
3Order

Put the learning moves in the order that makes the concept easiest to apply.

Enabling ES Modules
CommonJS vs ES Modules
Introduction to ES Modules

Introduction to ES Modules

ES Modules (ESM) is the official standard format for packaging JavaScript code for reuse.

It was introduced in ES6 (ES2015) and is now supported in Node.js.

Prior to ES Modules, Node.js exclusively used the CommonJS module format (require/exports).

Now developers can choose between CommonJS and ES Modules based on their project needs.

ES Modules provides a more structured and statically analyzable way to work with modules compared to CommonJS, with benefits like tree-shaking for smaller builds.

CommonJS vs ES Modules

Here's how CommonJS and ES Modules differ:

FeatureCommonJSES Modules
File Extension.js (default).mjs (or .js with proper config)
Import Syntaxrequire()import
Export Syntaxmodule.exports / exportsexport / export default
Import TimingDynamic (runtime)Static (parsed before execution)
Top-level AwaitNot supportedSupported
File URL in ImportsNot requiredRequired for local files

Example: CommonJS Module

// math.js (CommonJS)
function add(a, b) {
 return a + b;
}
function subtract(a, b) {
 return a - b;
}
module.exports = {
 add,
 subtract
};
// app.js (CommonJS)
const math = require('./math');
console.log(math.add(5, 3)); // 8

Example: ES Module

// math.mjs (ES Module)
export function add(a, b) {
 return a + b;
}
export function subtract(a, b) {
 return a - b;
}
// app.mjs (ES Module)
import { add, subtract } from './math.mjs';
console.log(add(5, 3)); // 8

Enabling ES Modules

There are several ways to enable ES Modules in Node.js:

Using the .mjs File Extension

The simplest way is to use the .mjs extension for your files.

Node.js will automatically treat these files as ES Modules.

Setting "type": "module" in package.json

To use ES Modules with regular .js files, add the following to your package.json:

{
 "name": "my-package",
 "version": "1.0.0",
 "type": "module"
}

With this setting, all .js files in your project will be treated as ES Modules.

Using the --input-type=module Flag

For scripts run directly with the node command, you can specify the module system:

node --input-type=module script.js

Note

If you're working with a codebase that primarily uses CommonJS but you want to use ES Modules in one file, using the .mjs extension is the most explicit and least error-prone approach.

Import and Export Syntax

ES Modules provide more flexible ways to import and export code compared to CommonJS.

Export Syntax

// Multiple named exports
export function sayHello() {
 console.log('Hello');
}
export function sayGoodbye() {
 console.log('Goodbye');
}
// Alternative: export list at the end
function add(a, b) {
 return a + b;
}
function subtract(a, b) {
 return a - b;
}
export { add, subtract };
// Only one default export per module
export default function() {
 console.log('I am the default export');
}
// Or with a named function/class/object
function mainFunction() {
 return 'Main functionality';
}
export default mainFunction;
// Combining default and named exports
export const VERSION = '1.0.0';
function main() {
 console.log('Main function');
}
export { main as default }; // Alternative way to set default

Import Syntax

// Import specific named exports
import { sayHello, sayGoodbye } from './greetings.mjs';
sayHello(); // Hello
// Rename imports to avoid naming conflicts
import { add as sum, subtract as minus } from './math.mjs';
console.log(sum(5, 3)); // 8
// Import all named exports as an object
import * as math from './math.mjs';
console.log(math.add(7, 4)); // 11
// Import the default export
import mainFunction from './main.mjs';
mainFunction();
// You can name the default import anything you want
import anyNameYouWant from './main.mjs';
anyNameYouWant();
// Import both default and named exports
import main, { VERSION } from './main.mjs';
console.log(VERSION); // 1.0.0
main(); // Main function

Dynamic Imports

ES Modules support dynamic imports, allowing you to load modules conditionally or on-demand.

Example: Dynamic Imports

// app.mjs
async function loadModule(moduleName) {
 try {
 // Dynamic import returns a promise
 const module = await import(`./${moduleName}.mjs`);
 return module;
 } catch (error) {
 console.error(`Failed to load ${moduleName}:`, error);
}
}
// Load a module based on a condition
const moduleName = process.env.NODE_ENV === 'production' ? 'prod' : 'dev';
loadModule(moduleName).then(module => {
 module.default(); // Call the default export
});
// Or with simpler await syntax
(async () => {
 const mathModule = await import('./math.mjs');
 console.log(mathModule.add(10, 5)); // 15
})();

Use Case: Dynamic imports are great for code-splitting, lazy-loading modules, or conditionally loading modules based on runtime conditions.

Top-level Await

Unlike CommonJS, ES Modules support top-level await, allowing you to use await outside of async functions at the module level.

Example: Top-level Await

// data-loader.mjs
// This would cause an error in CommonJS or in a script
// But works at the top level in an ES Module
console.log('Loading data...');
// Top-level await - the module's execution pauses here
const response = await fetch('https://jsonplaceholder.typicode.com/todos/1');
const data = await response.json();
console.log('Data loaded!');
export { data };
// When another module imports this one, it will only get the exports
// after all the top-level await operations have completed

Top-level await is especially useful for

  • Loading configuration from files or remote sources
  • Connecting to databases before exporting functionality
  • Conditional imports or module initialization

Best Practices

When working with ES Modules in Node.js, follow these best practices:

Be Clear About File Extensions

Always include file extensions in your import statements for local files:

// Good
import { someFunction } from './utils.mjs';
// Bad - might not work depending on configuration
import { someFunction } from './utils';

Use Directory Indexes Properly

For directory imports, create index.mjs files:

// utils/index.mjs
export * from './string-utils.mjs';
export * from './number-utils.mjs';
// app.mjs
import { formatString, add } from './utils/index.mjs';

Choose the Right Export Style

Use named exports for multiple functions/values, and default exports for main functionality:

// For libraries with many utilities, use named exports
export function validate() { /* ... */ }
export function format() { /* ... */ }
// For components or classes that are the primary export
export default class UserService { /* ... */ }

Handle the Transition from CommonJS

When working with a codebase that mixes CommonJS and ES Modules:

  • ES Modules can import from CommonJS modules using default import
  • CommonJS can require() ES Modules only with dynamic import()
  • Use the compatibility helpers in the Node.js 'module' package for interoperability
// Importing CommonJS module from ESM
import fs from 'fs'; // The default import is module.exports
// Importing ESM from CommonJS (Node.js 12+)
// In a CommonJS module:
(async () => {
 const { default: myEsmModule } = await import('./my-esm-module.mjs');
})();

Dual Package Hazard

For npm packages that support both module systems, use the "exports" field in package.json to specify different entry points:

{
 "name": "my-package",
 "exports": {
 ".": {
 "import": "./index.mjs",
 "require": "./index.cjs"
 }
 }
}

Node.js Support: ES Modules are fully supported in Node.js since v12, with better support in v14+.

For older versions, you may need a transpiler like Babel.

Previous

Node.js Modules

Next

Node.js NPM