You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
When the "exports" field is defined, all subpaths of the package are encapsulated and no longer available to importers. For example, require('pkg/subpath.js') throws an ERR_PACKAGE_PATH_NOT_EXPORTED error.
I've experienced this in a project where I wanted to import our design tokens.
Steps to reproduce
You can follow these steps to create a barebone test case project:
Create a directory: mkdir cloudfour-export-issue
Enter that directory: cd cloudfour-export-issue
Initialize a new npm project: npm init -y
Install our patterns: npm i -D @cloudfour/patterns
Create a script that attempts to require tokens:printf "const tokens = require(\"@cloudfour/patterns/src/compiled/tokens/json/tokens.json\");" > index.js
Run script: node index
Here is an example of the error you will likely see:
node:internal/modules/cjs/loader:499
throw e;
^
Error [ERR_PACKAGE_PATH_NOT_EXPORTED]: Package subpath './src/compiled/tokens/json/tokens.json' is not defined by "exports" in /PATH/TO/cloudfour-export-issue/node_modules/@cloudfour/patterns/package.json
at new NodeError (node:internal/errors:387:5)
at throwExportsNotFound (node:internal/modules/esm/resolve:464:9)
at packageExportsResolve (node:internal/modules/esm/resolve:748:3)
at resolveExports (node:internal/modules/cjs/loader:493:36)
at Function.Module._findPath (node:internal/modules/cjs/loader:533:31)
at Function.Module._resolveFilename (node:internal/modules/cjs/loader:942:27)
at Function.Module._load (node:internal/modules/cjs/loader:804:27)
at Module.require (node:internal/modules/cjs/loader:1028:19)
at require (node:internal/modules/cjs/helpers:102:18)
at Object.<anonymous> (/Users/tylersticka/Repos/exports-test/index.js:1:16) {
code: 'ERR_PACKAGE_PATH_NOT_EXPORTED'
}
The text was updated successfully, but these errors were encountered:
I dug into this a bit, but I'm putting it back because it requires a bit more discussion (and, frankly, a check-in with @calebeby about the right path forward).
My (admittedly shallow) understanding is that exports is the new standard, aiming to replace main and module, and has the capability of conditional exports. However, it also introduces the new side effect that @tylersticka mentioned.
My gut says the right answer is to remove exports in favor of our main and module lines, unless exports is giving us some advantage or feature that I'm missing.
A quick survey of how some other pattern libraries handle this:
You can use glob-like patterns in export maps. It looks like the documentation says that in their glob syntax * matches subpaths as well so in their example an export map like this:
main is standardized in node, but not for importing of esm files from esm files (that was always nonstandard and only implemented by bundlers). module, jsnext:main, es2015, and es2017 are all nonstandard, just implemented by some bundlers and tools.
I think we should stick to export maps and add the globs as needed.
Overview
We currently define an
exports
field in ourpackage.json
, with additional files (such as compiled tokens) available via other fields.But according to the Node docs, this prevents other files from being imported:
I've experienced this in a project where I wanted to import our design tokens.
Steps to reproduce
You can follow these steps to create a barebone test case project:
mkdir cloudfour-export-issue
cd cloudfour-export-issue
npm init -y
npm i -D @cloudfour/patterns
printf "const tokens = require(\"@cloudfour/patterns/src/compiled/tokens/json/tokens.json\");" > index.js
node index
Here is an example of the error you will likely see:
The text was updated successfully, but these errors were encountered: