Skip to content

Commit 155a6c3

Browse files
guybedfordGeoffreyBooth
authored andcommittedApr 3, 2020
doc: update conditional exports recommendations
Backport-PR-URL: #32610 Co-Authored-By: Geoffrey Booth <GeoffreyBooth@users.noreply.github.com> PR-URL: #32098 Reviewed-By: Geoffrey Booth <webmaster@geoffreybooth.com>
1 parent 45d05e1 commit 155a6c3

File tree

2 files changed

+135
-147
lines changed

2 files changed

+135
-147
lines changed
 

‎doc/api/errors.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -2476,7 +2476,7 @@ such as `process.stdout.on('data')`.
24762476
[crypto digest algorithm]: crypto.html#crypto_crypto_gethashes
24772477
[domains]: domain.html
24782478
[event emitter-based]: events.html#events_class_eventemitter
2479-
[exports]: esm.html#esm_package_exports
2479+
[exports]: esm.html#esm_package_entry_points
24802480
[file descriptors]: https://en.wikipedia.org/wiki/File_descriptor
24812481
[policy]: policy.html
24822482
[stream-based]: stream.html

‎doc/api/esm.md

+134-146
Original file line numberDiff line numberDiff line change
@@ -183,87 +183,89 @@ unspecified.
183183

184184
### Package Entry Points
185185

186-
There are two fields that can define entry points for a package: `"main"` and
187-
`"exports"`. The `"main"` field is supported in all versions of Node.js, but its
188-
capabilities are limited: it only defines the main entry point of the package.
189-
The `"exports"` field, part of [Package Exports][], provides an alternative to
190-
`"main"` where the package main entry point can be defined while also
191-
encapsulating the package, preventing any other entry points besides those
192-
defined in `"exports"`. If package entry points are defined in both `"main"` and
193-
`"exports"`, the latter takes precedence in versions of Node.js that support
194-
`"exports"`. [Conditional Exports][] can also be used within `"exports"` to
195-
define different package entry points per environment.
196-
197-
#### <code>package.json</code> <code>"main"</code>
198-
199-
The `package.json` `"main"` field defines the entry point for a package,
200-
whether the package is included into CommonJS via `require` or into an ES
201-
module via `import`.
186+
In a package’s `package.json` file, two fields can define entry points for a
187+
package: `"main"` and `"exports"`. The `"main"` field is supported in all
188+
versions of Node.js, but its capabilities are limited: it only defines the main
189+
entry point of the package.
190+
191+
The `"exports"` field provides an alternative to `"main"` where the package
192+
main entry point can be defined while also encapsulating the package, preventing
193+
any other entry points besides those defined in `"exports"`. If package entry
194+
points are defined in both `"main"` and `"exports"`, the latter takes precedence
195+
in versions of Node.js that support `"exports"`. [Conditional Exports][] can
196+
also be used within `"exports"` to define different package entry points per
197+
environment, including whether the package is referenced via `require` or via
198+
`import`.
199+
200+
If both `"exports"` and `"main"` are defined, the `"exports"` field takes
201+
precedence over `"main"`.
202+
203+
Both `"main"` and `"exports"` entry points are not specific to ES modules or
204+
CommonJS; `"main"` will be overridden by `"exports"` in a `require` so it is
205+
not a CommonJS fallback.
206+
207+
This is important with regard to `require`, since `require` of ES module files
208+
throws an error in all versions of Node.js. To create a package that works both
209+
in modern Node.js via `import` and `require` and also legacy Node.js versions,
210+
see [the dual CommonJS/ES module packages section][].
211+
212+
#### Main Entry Point Export
213+
214+
To set the main entry point for a package, it is advisable to define both
215+
`"exports"` and `"main"` in the package’s `package.json` file:
202216

203217
<!-- eslint-skip -->
204218
```js
205-
// ./node_modules/es-module-package/package.json
206219
{
207-
"type": "module",
208-
"main": "./src/index.js"
220+
"main": "./main.js",
221+
"exports": "./main.js"
209222
}
210223
```
211224

212-
```js
213-
// ./my-app.mjs
225+
The benefit of doing this is that when using the `"exports"` field all
226+
subpaths of the package will no longer be available to importers under
227+
`require('pkg/subpath.js')`, and instead they will get a new error,
228+
`ERR_PACKAGE_PATH_NOT_EXPORTED`.
214229

215-
import { something } from 'es-module-package';
216-
// Loads from ./node_modules/es-module-package/src/index.js
217-
```
230+
This encapsulation of exports provides more reliable guarantees
231+
about package interfaces for tools and when handling semver upgrades for a
232+
package. It is not a strong encapsulation since a direct require of any
233+
absolute subpath of the package such as
234+
`require('/path/to/node_modules/pkg/subpath.js')` will still load `subpath.js`.
218235

219-
An attempt to `require` the above `es-module-package` would attempt to load
220-
`./node_modules/es-module-package/src/index.js` as CommonJS, which would throw
221-
an error as Node.js would not be able to parse the `export` statement in
222-
CommonJS.
236+
#### Subpath Exports
223237

224-
As with `import` statements, for ES module usage the value of `"main"` must be
225-
a full path including extension: `"./index.mjs"`, not `"./index"`.
226-
227-
If the `package.json` `"type"` field is omitted, a `.js` file in `"main"` will
228-
be interpreted as CommonJS.
229-
230-
The `"main"` field can point to exactly one file, regardless of whether the
231-
package is referenced via `require` (in a CommonJS context) or `import` (in an
232-
ES module context).
233-
234-
#### Package Exports
235-
236-
By default, all subpaths from a package can be imported (`import 'pkg/x.js'`).
237-
Custom subpath aliasing and encapsulation can be provided through the
238-
`"exports"` field.
238+
When using the `"exports"` field, custom subpaths can be defined along
239+
with the main entry point by treating the main entry point as the
240+
`"."` subpath:
239241

240242
<!-- eslint-skip -->
241243
```js
242-
// ./node_modules/es-module-package/package.json
243244
{
245+
"main": "./main.js",
244246
"exports": {
247+
".": "./main.js",
245248
"./submodule": "./src/submodule.js"
246249
}
247250
}
248251
```
249252

253+
Now only the defined subpath in `"exports"` can be imported by a
254+
consumer:
255+
250256
```js
251257
import submodule from 'es-module-package/submodule';
252258
// Loads ./node_modules/es-module-package/src/submodule.js
253259
```
254260

255-
In addition to defining an alias, subpaths not defined by `"exports"` will
256-
throw when an attempt is made to import them:
261+
While other subpaths will error:
257262

258263
```js
259264
import submodule from 'es-module-package/private-module.js';
260-
// Throws ERR_MODULE_NOT_FOUND
265+
// Throws ERR_PACKAGE_PATH_NOT_EXPORTED
261266
```
262267

263-
> Note: this is not a strong encapsulation as any private modules can still be
264-
> loaded by absolute paths.
265-
266-
Folders can also be mapped with package exports:
268+
Entire folders can also be mapped with package exports:
267269

268270
<!-- eslint-skip -->
269271
```js
@@ -275,20 +277,23 @@ Folders can also be mapped with package exports:
275277
}
276278
```
277279

280+
With the above, all modules within the `./src/features/` folder
281+
are exposed deeply to `import` and `require`:
282+
278283
```js
279284
import feature from 'es-module-package/features/x.js';
280285
// Loads ./node_modules/es-module-package/src/features/x.js
281286
```
282287

283-
If a package has no exports, setting `"exports": false` can be used instead of
284-
`"exports": {}` to indicate the package does not intend for submodules to be
285-
exposed.
288+
When using folder mappings, ensure that you do want to expose every
289+
module inside the subfolder. Any modules which are not public
290+
should be moved to another folder to retain the encapsulation
291+
benefits of exports.
286292

287-
Any invalid exports entries will be ignored. This includes exports not
288-
starting with `"./"` or a missing trailing `"/"` for directory exports.
293+
#### Package Exports Fallbacks
289294

290-
Array fallback support is provided for exports, similarly to import maps
291-
in order to be forwards-compatible with possible fallback workflows in future:
295+
For possible new specifier support in future, array fallbacks are
296+
supported for all invalid specifiers:
292297

293298
<!-- eslint-skip -->
294299
```js
@@ -299,143 +304,127 @@ in order to be forwards-compatible with possible fallback workflows in future:
299304
}
300305
```
301306

302-
Since `"not:valid"` is not a supported target, `"./submodule.js"` is used
307+
Since `"not:valid"` is not a valid specifier, `"./submodule.js"` is used
303308
instead as the fallback, as if it were the only target.
304309

305-
Defining a `"."` export will define the main entry point for the package,
306-
and will always take precedence over the `"main"` field in the `package.json`.
310+
#### Exports Sugar
311+
312+
If the `"."` export is the only export, the `"exports"` field provides sugar
313+
for this case being the direct `"exports"` field value.
307314

308-
This allows defining a different entry point for Node.js versions that support
309-
ECMAScript modules and versions that don't, for example:
315+
If the `"."` export has a fallback array or string value, then the `"exports"`
316+
field can be set to this value directly.
310317

311318
<!-- eslint-skip -->
312319
```js
313320
{
314-
"main": "./main-legacy.cjs",
315321
"exports": {
316-
".": "./main-modern.cjs"
322+
".": "./main.js"
317323
}
318324
}
319325
```
320326

327+
can be written:
328+
329+
<!-- eslint-skip -->
330+
```js
331+
{
332+
"exports": "./main.js"
333+
}
334+
```
335+
321336
#### Conditional Exports
322337

323338
Conditional exports provide a way to map to different paths depending on
324339
certain conditions. They are supported for both CommonJS and ES module imports.
325340

326341
For example, a package that wants to provide different ES module exports for
327-
Node.js and the browser can be written:
342+
`require()` and `import` can be written:
328343

329344
<!-- eslint-skip -->
330345
```js
331-
// ./node_modules/pkg/package.json
346+
// package.json
332347
{
333-
"type": "module",
334-
"main": "./index.js",
348+
"main": "./main-require.cjs",
335349
"exports": {
336-
"./feature": {
337-
"import": "./feature-default.js",
338-
"browser": "./feature-browser.js"
339-
}
340-
}
350+
"import": "./main-module.js",
351+
"require": "./main-require.cjs"
352+
},
353+
"type": "module"
341354
}
342355
```
343356

344-
When resolving the `"."` export, if no matching target is found, the `"main"`
345-
will be used as the final fallback.
357+
Node.js supports the following conditions:
346358

347-
The conditions supported in Node.js condition matching:
348-
349-
* `"default"` - the generic fallback that will always match. Can be a CommonJS
350-
or ES module file.
351359
* `"import"` - matched when the package is loaded via `import` or
352-
`import()`. Can be any module format, this field does not set the type
353-
interpretation.
354-
* `"node"` - matched for any Node.js environment. Can be a CommonJS or ES
355-
module file.
360+
`import()`. Can reference either an ES module or CommonJS file, as both
361+
`import` and `import()` can load either ES module or CommonJS sources.
356362
* `"require"` - matched when the package is loaded via `require()`.
363+
As `require()` only supports CommonJS, the referenced file must be CommonJS.
364+
* `"node"` - matched for any Node.js environment. Can be a CommonJS or ES
365+
module file. _This condition should always come after `"import"` or
366+
`"require"`._
367+
* `"default"` - the generic fallback that will always match. Can be a CommonJS
368+
or ES module file. _This condition should always come last._
357369

358370
Condition matching is applied in object order from first to last within the
359-
`"exports"` object.
360-
361-
Using the `"require"` condition it is possible to define a package that will
362-
have a different exported value for CommonJS and ES modules, which can be a
363-
hazard in that it can result in having two separate instances of the same
364-
package in use in an application, which can cause a number of bugs.
371+
`"exports"` object. _The general rule is that conditions should be used
372+
from most specific to least specific in object order._
365373

366374
Other conditions such as `"browser"`, `"electron"`, `"deno"`, `"react-native"`,
367-
etc. could be defined in other runtimes or tools. Condition names must not start
368-
with `"."` or be numbers. Further restrictions, definitions or guidance on
369-
condition names may be provided in future.
375+
etc. are ignored by Node.js but may be used by other runtimes or tools.
376+
Further restrictions, definitions or guidance on condition names may be
377+
provided in the future.
370378

371-
#### Exports Sugar
379+
Using the `"import"` and `"require"` conditions can lead to some hazards,
380+
which are explained further in
381+
[the dual CommonJS/ES module packages section][].
372382

373-
If the `"."` export is the only export, the `"exports"` field provides sugar
374-
for this case being the direct `"exports"` field value.
375-
376-
If the `"."` export has a fallback array or string value, then the `"exports"`
377-
field can be set to this value directly.
383+
Conditional exports can also be extended to exports subpaths, for example:
378384

379385
<!-- eslint-skip -->
380386
```js
381387
{
388+
"main": "./main.js",
382389
"exports": {
383-
".": "./main.js"
390+
".": "./main.js",
391+
"./feature": {
392+
"browser": "./feature-browser.js",
393+
"default": "./feature.js"
394+
}
384395
}
385396
}
386397
```
387398

388-
can be written:
389-
390-
<!-- eslint-skip -->
391-
```js
392-
{
393-
"exports": "./main.js"
394-
}
395-
```
399+
Defines a package where `require('pkg/feature')` and `import 'pkg/feature'`
400+
could provide different implementations between the browser and Node.js,
401+
given third-party tool support for a `"browser"` condition.
396402

397-
When using [Conditional Exports][], the rule is that all keys in the object
398-
mapping must not start with a `"."` otherwise they would be indistinguishable
399-
from exports subpaths.
403+
#### Nested conditions
400404

401-
<!-- eslint-skip -->
402-
```js
403-
{
404-
"exports": {
405-
".": {
406-
"import": "./main.js",
407-
"require": "./main.cjs"
408-
}
409-
}
410-
}
411-
```
405+
In addition to direct mappings, Node.js also supports nested condition objects.
412406

413-
can be written:
407+
For example, to define a package that only has dual mode entry points for
408+
use in Node.js but not the browser:
414409

415410
<!-- eslint-skip -->
416411
```js
417412
{
413+
"main": "./main.js",
418414
"exports": {
419-
"import": "./main.js",
420-
"require": "./main.cjs"
415+
"browser": "./feature-browser.mjs",
416+
"node": {
417+
"import": "./feature-node.mjs",
418+
"require": "./feature-node.cjs"
419+
}
421420
}
422421
}
423422
```
424423

425-
If writing any exports value that mixes up these two forms, an error will be
426-
thrown:
427-
428-
<!-- eslint-skip -->
429-
```js
430-
{
431-
// Throws on resolution!
432-
"exports": {
433-
"./feature": "./lib/feature.js",
434-
"import": "./main.js",
435-
"require": "./main.cjs"
436-
}
437-
}
438-
```
424+
Conditions continue to be matched in order as with flat conditions. If
425+
a nested conditional does not have any mapping it will continue checking
426+
the remaining conditions of the parent condition. In this way nested
427+
conditions behave analogously to nested JavaScript `if` statements.
439428

440429
#### Self-referencing a package using its name
441430

@@ -567,8 +556,8 @@ CommonJS entry point for `require`.
567556
"type": "module",
568557
"main": "./index.cjs",
569558
"exports": {
570-
"require": "./index.cjs",
571-
"import": "./wrapper.mjs"
559+
"import": "./wrapper.mjs",
560+
"require": "./index.cjs"
572561
}
573562
}
574563
```
@@ -912,8 +901,8 @@ can either be an URL-style relative path like `'./file.mjs'` or a package name
912901
like `'fs'`.
913902

914903
Like in CommonJS, files within packages can be accessed by appending a path to
915-
the package name; unless the packages `package.json` contains an [`"exports"`
916-
field][], in which case files within packages need to be accessed via the path
904+
the package name; unless the packages `package.json` contains an `"exports"`
905+
field, in which case files within packages need to be accessed via the path
917906
defined in `"exports"`.
918907

919908
```js
@@ -1637,7 +1626,7 @@ The resolver can throw the following errors:
16371626
16381627
**PACKAGE_EXPORTS_TARGET_RESOLVE**(_packageURL_, _target_, _subpath_, _env_)
16391628
1640-
> 1.If _target_ is a String, then
1629+
> 1. If _target_ is a String, then
16411630
> 1. If _target_ does not start with _"./"_ or contains any _"node_modules"_
16421631
> segments including _"node_modules"_ percent-encoding, throw an
16431632
> _Invalid Package Target_ error.
@@ -1738,10 +1727,8 @@ success!
17381727
[ECMAScript-modules implementation]: https://github.com/nodejs/modules/blob/master/doc/plan-for-new-modules-implementation.md
17391728
[ES Module Integration Proposal for Web Assembly]: https://github.com/webassembly/esm-integration
17401729
[Node.js EP for ES Modules]: https://github.com/nodejs/node-eps/blob/master/002-es-modules.md
1741-
[Package Exports]: #esm_package_exports
17421730
[Terminology]: #esm_terminology
17431731
[WHATWG JSON modules specification]: https://html.spec.whatwg.org/#creating-a-json-module-script
1744-
[`"exports"` field]: #esm_package_exports
17451732
[`data:` URLs]: https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/Data_URIs
17461733
[`esm`]: https://github.com/standard-things/esm#readme
17471734
[`export`]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/export
@@ -1756,6 +1743,7 @@ success!
17561743
[import an ES or CommonJS module for its side effects only]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/import#Import_a_module_for_its_side_effects_only
17571744
[special scheme]: https://url.spec.whatwg.org/#special-scheme
17581745
[the official standard format]: https://tc39.github.io/ecma262/#sec-modules
1746+
[the dual CommonJS/ES module packages section]: #esm_dual_commonjs_es_module_packages
17591747
[transpiler loader example]: #esm_transpiler_loader
17601748
[6.1.7 Array Index]: https://tc39.es/ecma262/#integer-index
17611749
[Top-Level Await]: https://github.com/tc39/proposal-top-level-await

0 commit comments

Comments
 (0)
Please sign in to comment.