esmod-pmb
v0.1.15
Published
Use esm (ES modules, import, export for Node v6) with less boilerplate.
Readme
esmod-pmb
Use esm (ES modules, import, export for Node v6) with less boilerplate.
⚠ DEPRECATED ⚠
As of 2026-04-15, this module is deprecated. I released a compatibility layer for Node.js v24 that you can use to make old software continue to work, but you should really upgrade your code base to instead embrace the native ESM support of Node.js v24 or later.
What you'll have to update:
- Clarify your
importstatements:- Use full filenames with explicit filename extension.
- If you're already using
eslint-config-nodejs-pmbas your eslint rules, it's eay: The rules package ships a scriptutil/massfix_missing_import_fext.shthat will auto-fix probably all occurrences. Make sure your project repo is clean before you run it, then from the top of your project repo, run the fix script by full path.
- If you're already using
- If you import JSON files, declare the type, e.g.
import pkgInfo from './package.json' with { type: 'json' };
- Use full filenames with explicit filename extension.
- Delete all bridge modules.
- In all places where you had mentioned a bridge module
(probably in your
package.json), write the.mjsfile instead. - In all places where you called
nodemjs(probably in yourpackage.json), call Node.js directly instead. - Properly declare that your project now requires Node.js v24 or later
(probably in your
package.jsonand docs).
How to mute the deprecation warning
If you don't want to deal with the deprecation warning today, you can mute it
by setting environment variable ESMODPMB_IGNORE_DEPRECATION_UNTIL
to a six-digit date (yymmdd).
The next few sections are kept for historians. They explain why and how this module was useful for ancient Node.js versions.
Usage
Let's assume you have an ES module like usage.mjs:
import dfOnly from './default-export-only';
import * as named from './named-exports-only';
export default dfOnly;
export const { foo, answer } = named;… and you want to use it from old node's CommonJS realm.
The esm module can do it:
module.exports = require('esm')(module)('./usage.mjs')But that's still a bit too much boilerplate for my taste.
Don't waste your time micro-managing the filename
and whether you should add .default to the above formula!
There's a much easier way:
Make a bridge module with almost the same name,
except it ends with .node.js instead of .mjs
(thus here, usage.node.js):
require('esmod-pmb')(module);It should work out of the box:
$ nodejs -p "require('./usage.node.js')"
{ foo: [Getter], answer: [Getter], default: [Getter] }To see values instead of getters, copy them to another object:
$ nodejs -p "Object.assign({}, require('./usage.node.js'))"
{ foo: 23,
answer: 42,
default: { isDefaultExport: true, bar: 5 } }For modules that have a default export and no named exports,
like default-export-only.mjs:
export default { isDefaultExport: true, bar: 5 };… your bridge module will export that as the top level:
$ nodejs -p "require('./default-export-only.node.js')"
{ isDefaultExport: true, bar: 5 }
$ nodejs -p "require('esm')(module)('./default-export-only.mjs')"
{ default: [Getter] }API
// standard:
require('esmod-pmb')(module);
// custom options:
require('esmod-pmb')(module, { preferDefaultExport: false });
// detailed:
var bridgeBuilder = require('esmod-pmb');
var opt = { reexport: false };
var esmRequire = bridgeBuilder(module, opt);
var yourEsModule = esmRequire('./yourmodule.mjs');This module exports one function:
esmRqr = bridgeBuilder(baseModule[, opt])
Returns an ESM-capable require-like function obtained from esm.
The optional options object opt can be used to modify the default
options.
If you need custom options for something you consider "normal"/"usual", please file an issue so we can try to provide better defaults. After all, the purpose of this module is to reduce boilerplate.
That said, bridgeBuilder uses the same options object as esm,
so you can use all of its options, and some additional ones:
reexport: (bool, default: true) WhetherbridgeBuildershall do its main magic:- guess the name of your ES module,
- import it,
- (skippable) resolve the imported bindings
- and "re-export" them, i.e. assign the imported values to
baseModule.exports.
- Set to
falseto opt-out of the main magic, but still get anesmRqrwith the combined options based on this module's defaults.
Options used only when reexport is enabled:
stripSuffixes: (regexp) andaddSuffix(string, default:".mjs") To guess the ES module filename, thebridgeBuilderremoves the part matched bystripSuffixesand appendsaddSuffix.resolveImportedValues: (bool, default:true) Whether to resolve (copy) the imported values. Set tofalseif you prefer getter functions. Sane modules usually shouldn't mutate their exports, so usually resolving them makes life easier by reducing the amount of magic involved.preferDefaultExport: Determines which of the ESM exports shall be re-export into CommonJS land.false= all of them. The re-export will be a dictionary (object). If the ES module only has a default export, the dictionary will contain only one key,default.true= only the default export. If the ES module doesn't have a default export, the re-export will beundefined.1(as a number; also the default) = automatic. The re-export will be a dictionary with all exports, except in case there is only one single export and its name isdefault; in that case, that default export will be re-exported directly.
Known issues
- Needs more/better tests and docs.
License
ISC
