@@ -183,87 +183,89 @@ unspecified.
183
183
184
184
### Package Entry Points
185
185
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:
202
216
203
217
<!-- eslint-skip -->
204
218
``` js
205
- // ./node_modules/es-module-package/package.json
206
219
{
207
- " type " : " module " ,
208
- " main " : " ./src/index .js"
220
+ " main " : " ./main.js " ,
221
+ " exports " : " ./main .js"
209
222
}
210
223
```
211
224
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 ` .
214
229
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 ` .
218
235
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
223
237
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:
239
241
240
242
<!-- eslint-skip -->
241
243
``` js
242
- // ./node_modules/es-module-package/package.json
243
244
{
245
+ " main" : " ./main.js" ,
244
246
" exports" : {
247
+ " ." : " ./main.js" ,
245
248
" ./submodule" : " ./src/submodule.js"
246
249
}
247
250
}
248
251
```
249
252
253
+ Now only the defined subpath in ` "exports" ` can be imported by a
254
+ consumer:
255
+
250
256
``` js
251
257
import submodule from ' es-module-package/submodule' ;
252
258
// Loads ./node_modules/es-module-package/src/submodule.js
253
259
```
254
260
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:
257
262
258
263
``` js
259
264
import submodule from ' es-module-package/private-module.js' ;
260
- // Throws ERR_MODULE_NOT_FOUND
265
+ // Throws ERR_PACKAGE_PATH_NOT_EXPORTED
261
266
```
262
267
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:
267
269
268
270
<!-- eslint-skip -->
269
271
``` js
@@ -275,20 +277,23 @@ Folders can also be mapped with package exports:
275
277
}
276
278
```
277
279
280
+ With the above, all modules within the ` ./src/features/ ` folder
281
+ are exposed deeply to ` import ` and ` require ` :
282
+
278
283
``` js
279
284
import feature from ' es-module-package/features/x.js' ;
280
285
// Loads ./node_modules/es-module-package/src/features/x.js
281
286
```
282
287
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.
286
292
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
289
294
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 :
292
297
293
298
<!-- eslint-skip -->
294
299
``` js
@@ -299,143 +304,127 @@ in order to be forwards-compatible with possible fallback workflows in future:
299
304
}
300
305
```
301
306
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
303
308
instead as the fallback, as if it were the only target.
304
309
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.
307
314
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.
310
317
311
318
<!-- eslint-skip -->
312
319
``` js
313
320
{
314
- " main" : " ./main-legacy.cjs" ,
315
321
" exports" : {
316
- " ." : " ./main-modern.cjs "
322
+ " ." : " ./main.js "
317
323
}
318
324
}
319
325
```
320
326
327
+ can be written:
328
+
329
+ <!-- eslint-skip -->
330
+ ``` js
331
+ {
332
+ " exports" : " ./main.js"
333
+ }
334
+ ```
335
+
321
336
#### Conditional Exports
322
337
323
338
Conditional exports provide a way to map to different paths depending on
324
339
certain conditions. They are supported for both CommonJS and ES module imports.
325
340
326
341
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:
328
343
329
344
<!-- eslint-skip -->
330
345
``` js
331
- // ./node_modules/pkg/ package.json
346
+ // package.json
332
347
{
333
- " type" : " module" ,
334
- " main" : " ./index.js" ,
348
+ " main" : " ./main-require.cjs" ,
335
349
" 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"
341
354
}
342
355
```
343
356
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:
346
358
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.
351
359
* ` "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.
356
362
* ` "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._
357
369
358
370
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._
365
373
366
374
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.
370
378
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] [ ] .
372
382
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:
378
384
379
385
<!-- eslint-skip -->
380
386
``` js
381
387
{
388
+ " main" : " ./main.js" ,
382
389
" exports" : {
383
- " ." : " ./main.js"
390
+ " ." : " ./main.js" ,
391
+ " ./feature" : {
392
+ " browser" : " ./feature-browser.js" ,
393
+ " default" : " ./feature.js"
394
+ }
384
395
}
385
396
}
386
397
```
387
398
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.
396
402
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
400
404
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.
412
406
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:
414
409
415
410
<!-- eslint-skip -->
416
411
``` js
417
412
{
413
+ " main" : " ./main.js" ,
418
414
" 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
+ }
421
420
}
422
421
}
423
422
```
424
423
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.
439
428
440
429
#### Self-referencing a package using its name
441
430
@@ -567,8 +556,8 @@ CommonJS entry point for `require`.
567
556
" type" : " module" ,
568
557
" main" : " ./index.cjs" ,
569
558
" exports" : {
570
- " require " : " ./index.cjs " ,
571
- " import " : " ./wrapper.mjs "
559
+ " import " : " ./wrapper.mjs " ,
560
+ " require " : " ./index.cjs "
572
561
}
573
562
}
574
563
```
@@ -912,8 +901,8 @@ can either be an URL-style relative path like `'./file.mjs'` or a package name
912
901
like ` 'fs'` .
913
902
914
903
Like in CommonJS , files within packages can be accessed by appending a path to
915
- the package name ; unless the package ’s ` 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 package ’s ` package.json` contains an ` "exports"`
905
+ field , in which case files within packages need to be accessed via the path
917
906
defined in ` "exports"` .
918
907
919
908
` ` ` js
@@ -1637,7 +1626,7 @@ The resolver can throw the following errors:
1637
1626
1638
1627
**PACKAGE_EXPORTS_TARGET_RESOLVE**(_packageURL_, _target_, _subpath_, _env_)
1639
1628
1640
- > 1.If _target_ is a String, then
1629
+ > 1. If _target_ is a String, then
1641
1630
> 1. If _target_ does not start with _"./"_ or contains any _"node_modules"_
1642
1631
> segments including _"node_modules"_ percent-encoding, throw an
1643
1632
> _Invalid Package Target_ error.
@@ -1738,10 +1727,8 @@ success!
1738
1727
[ECMAScript-modules implementation]: https://github.com/nodejs/modules/blob/master/doc/plan-for-new-modules-implementation.md
1739
1728
[ES Module Integration Proposal for Web Assembly]: https://github.com/webassembly/esm-integration
1740
1729
[Node.js EP for ES Modules]: https://github.com/nodejs/node-eps/blob/master/002-es-modules.md
1741
- [Package Exports]: #esm_package_exports
1742
1730
[Terminology]: #esm_terminology
1743
1731
[WHATWG JSON modules specification]: https://html.spec.whatwg.org/#creating-a-json-module-script
1744
- [`"exports"` field]: #esm_package_exports
1745
1732
[`data:` URLs]: https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/Data_URIs
1746
1733
[`esm`]: https://github.com/standard-things/esm#readme
1747
1734
[`export`]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/export
@@ -1756,6 +1743,7 @@ success!
1756
1743
[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
1757
1744
[special scheme]: https://url.spec.whatwg.org/#special-scheme
1758
1745
[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
1759
1747
[transpiler loader example]: #esm_transpiler_loader
1760
1748
[6.1.7 Array Index]: https://tc39.es/ecma262/#integer-index
1761
1749
[Top-Level Await]: https://github.com/tc39/proposal-top-level-await
0 commit comments