-
Notifications
You must be signed in to change notification settings - Fork 30.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
module: Conditional exports #29978
module: Conditional exports #29978
Conversation
This seems like a reasonable solution to a problem I think we should actually move away from having at all. Let me explain. ESM is relatively new in Browsers and in Node.js. To date, I know of close to zero modules that will load “natively” (without a compiler) in Node.js’ ESM or in the Browser. All of the modules currently written to ESM syntax are in fact not native ESM and require a compiler in order to be loaded in the Browser and virtually all of those modules are written for the browser and not Node.js. In other words, to date, JS developers have almost exclusively used ESM syntax to write modules for compilers and not directly for browsers. But the goal of ESM is not to provide a new syntax for compilers. The goal, particularly in the Browser, has been to enable truly dynamic loading of native JS modules for the Browser. As Node.js develops native support for ESM this presents an opportunity. If done correctly, ESM in Node.js could lead us towards “universal JS modules” that work in Browsers and in Node.js without any compiler (or even a local install step). To their credit, the people who have been working on ESM have been moving in this direction (removing globals not accessible to the browser, for instance). The solution in this PR, designed for use by compilers, pushes in the other direction. Solving any problem with package.json means you have not solved it in a way that will work for native ESM in the Browser. Yes, this is limiting. Yes, this is actually kind of painful right now because it’s difficult to write modules that work everywhere. But, if we fix these issues with features that move away from truly universal modules we will make this transition even worse, and could end up in this position indefinitely without a native solution. I think a reasonable question to ask of any feature related to ESM in Node.js is “how does this work without a compiler or local install when loading the same code in the Browser?” That’s a question that Node.js has never had to ask before, but it’s a necessary one if we want to reach a universal module format. |
@mikeal It's a valid point though whether we should be branching on entry points (as in this PR), or relying entirely on code splitting and conditional code-based branching. But this point makes the opposite argument to what you are making - that if we want to avoid compilers, we should be using import map generation tools that can handle conditional branching of exports resolution, in order to avoid needing to compile and optimize module code branches for different types of environment conditions. That is, this feature enables import map generators to support universal mapping generation without compilation, as the environment optimizations can already be done by package authors at publish time. |
I am really not sure what the use case is here. I don't understand the base premise of why a package could or should have multiple sets of exports. That seems like a very confusing idea to introduce to packages & I'd like to understand more clearly both what for & why it is necessary? |
This is the problem, you just used the word “generation.” Who does the generation? How does my static universal module generate anything? This assumes an install or compile step on a system that is then deployed in some way. That workflow is actually relatively new in web development and native ESM gives us an opportunity to support an older and simpler workflow in which people consume JS code from the internet using a URL and publish code statically to URLs, none of which requires a local system in which things are installed and then deployed. Whether a compiler/generator runs at install, publish, or load time, it’s still required in the workflow if such a step is necessary. I appreciate that this feature is in line with what can be supported in import maps, but I’m actually skeptical that this sort of up-front generation of an import map is going to be sufficient. This is probably a much longer side thread that I don’t want to have derail this one, but if you consider what you can do with a backend or service worker for dynamic module loading the case for generated static mapping is unclear given how much less you can do with it. I think import maps are a nice feature to have, particularly in the case that someone is writing them by hand rather than generating them, but I’ve not been convinced that they solve the range of concerns people seem to think they solve. |
If you look at the number of different "main" fields people use in package.json files, that pretty much explains the use case. The With Node.js supporting |
IMO, the fact that we only have one “module” field in package.json for ESM modules is a feature, not a bug. It means we’re defining a single entry point for universal ESM modules. I may be in the minority on this one though. |
@rektide every package with multiple files already has multiple exports. @mikeal similarly, only providing a single entry point for ESM, while CJS has infinite, would be a problem. Also, as has been said, the "browser" field has for years provided a mapping of N files for bundlers, in its object form. As for "without a compiler" - browsers will never have all the capabilities node has, and so there will never be a guarantee that any node module will work in a browser without transformation. While it's certainly possible to write a node module that can work in a browser without changes, not just for ESM but also for CJS (it's just harder), imo it's not ever going to be an achievable goal in the abstract. |
My position is the exact opposite, that infinite entry points is exactly what keeps us from universal modules ;) |
This comment has been minimized.
This comment has been minimized.
@ljharb you’re making it sound as though adding features don’t have consequences. They certainly do. The ecosystem we have today is a direct result of how the module system, in tandem with npm, was engineered. The fact that two dependencies can load different versions of a sub-dependency is a feature, one that isn’t available in most languages, and which had the effect of creating an explosively large ecosystem and creating a much more complicated de-duplication task for front end bundlers. I don’t think it’s productive to say “you can still do this other thing even though we have features that incentivize you not to solve problems that way.” If universal modules are a goal, which I believe they are, the features in Node.js need to point towards that and not away from it. |
@mikeal and @ljharb and @rektide we have a separate issue at nodejs/modules#401 for general discussion of this proposal. Do you mind taking your conversation there? And we can keep the thread on this PR specific to discussion of the code in the PR. |
This comment has been minimized.
This comment has been minimized.
Moved my comment to nodejs/modules#401 (comment). |
42303c6
to
d99ede3
Compare
I've updated the The benefit of this inversion for CommonJS / ES module branching is that it allows us to support a flag to enable or disable this feature, something like The nice thing is such a flag would not break application code by design, whereas a flagged Considering a flag that does not cause breaking behaviour either way gives us more options for the middle ground in our disagreements on this hazard. |
I've pushed a commit to this PR to allow the following shorthand exports configuration: {
"exports": {
"module": "./index-module.mjs",
"default": "./index-other.cjs"
}
} which is effectively sugar for the dot main in exports, based on the simple rule that the first key string does not start with a dot. While it could be seen as a little magical, I think it's important to have a very simple syntax for the base main case, which is really the main consideration for us in unflagging here. Feedback further welcome. |
Do we need to invert the flag to support |
(reposting from #30051 (comment)) i think it would be better for the ecosystem to just not support this for the time being. the current way this is exposed to people writing modules is incredibly complex and confusing, and having code like this littered around is definitely not an improvement to codebases. yes, we lose the ability to ship packages that are both supported on old node and have named exports on new node, but we gain pure simplicity. |
I think users would prefer the capabilities to the simplicity. Any users who want simplicity can just not use conditional exports. |
@GeoffreyBooth it creates complexity for those consuming the module too, and they didn't opt into it. it creates complexity for tools which now have to parse our weird package.json configs, and they didn't opt into it. it creates complexity for package managers, which may need to parse that information in order to better optimize packages on disk, and they didn't opt into it. |
Package authors will publish dual packages no matter what we tell them, whether using conditional exports or Most of the issues related to the hazard via conditional exports are still present even in a |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Two suggestions that are docs-only.
Co-Authored-By: Jan Olaf Krems <jan.krems@gmail.com>
Here's the CI error that keeps coming up just on that distribution -
there is nothing here associated with this PR, and this is the only env failure, so I'm going to go ahead with landing. |
PR-URL: #29978 Reviewed-By: Jan Krems <jan.krems@gmail.com> Reviewed-By: Myles Borins <myles.borins@gmail.com>
PR-URL: #29978 Reviewed-By: Jan Krems <jan.krems@gmail.com> Reviewed-By: Myles Borins <myles.borins@gmail.com>
Landed in 2367474. |
PR-URL: #29978 Reviewed-By: Jan Krems <jan.krems@gmail.com> Reviewed-By: Myles Borins <myles.borins@gmail.com>
PR-URL: #29978 Reviewed-By: Jan Krems <jan.krems@gmail.com> Reviewed-By: Myles Borins <myles.borins@gmail.com>
PR-URL: #29978 Reviewed-By: Jan Krems <jan.krems@gmail.com> Reviewed-By: Myles Borins <myles.borins@gmail.com>
PR-URL: #29978 Reviewed-By: Jan Krems <jan.krems@gmail.com> Reviewed-By: Myles Borins <myles.borins@gmail.com>
PR-URL: #29978 Reviewed-By: Jan Krems <jan.krems@gmail.com> Reviewed-By: Myles Borins <myles.borins@gmail.com>
PR-URL: #29978 Reviewed-By: Jan Krems <jan.krems@gmail.com> Reviewed-By: Myles Borins <myles.borins@gmail.com>
@guybedford It's still not clear from the docs how to use this in the runtime: Let's say we have
How can I set Fwiw, I can't use the built-in |
@desmap IIRC right now setting custom conditions may require using loader hooks (e.g. adding |
This PR implements a way to define conditional
"exports"
in the package.json, allowing for exports mapping to different values depending on certain conditions that are constants of the resolver in the environment. The proposal is described in more detail in the included documentation at at https://github.com/nodejs/node/compare/master...guybedford:exports-conditions?diff=unified&expand=1&short_path=8e67f40#user-content-conditional-exports, and follows the initial proposal from https://github.com/jkrems/proposal-pkg-exports#2-conditional-mapping.Currently there are two major driving use cases for this - supporting defining
"exports"
targets that can allow a custom browser mapping without leaving it to userland to work out how to update the "browser" field or a new custom field to handle this (and dealing with compatibility issues such as for example that the browser field behaves differently to exports semantics in mapping internal specifiers as well as external), and also in supporting packages which can provide a different module file when loading viaimport
orrequire()
, which was one of the points raised as a remaining request for the modules implementation by @ljharb at the last meeting.The
"require"
condition can be thought of as a way to "opt-in" to the instancing hazard which we've worked to carefully ensure is never enabled by default. The documentation also carefully mentions this instancing hazard enabled by this condition and strongly suggests ways to avoid it, including proposing a new wrapper module approach to achieve named exports support for CommonJS that is enabled by this feature and avoids the hazard. Both the"require"
condition and"node"
condition are flagged behind a--experimental-conditional-exports
flag.In addition to the above, other possible use cases of this feature include runtime-specific conditions, to support e.g.
"electron"
and"react-native"
specific entry points in exports, or even conditions such as"production"
-specific mapping, without any of these needing to be natively supported by Node.js itself.This builds on work that has previously been explored through various userland proposals from discussion with RollupJS and Parcel core teams.
The basic approach is to allow
"exports"
target values to be objects whose keys are condition names, and values are the target to apply for that condition, where conditions apply in a priority order. See the conditional exports docs for the example of browser mapping, and the dual publishing section for the example of defining named exports for an export with a wrapper module technique through this proposal.Feedback very welcome, this still needs to be discussed at the modules meeting before any consideration of merging, but the more eyes on it the better.
Checklist
make -j4 test
(UNIX), orvcbuild test
(Windows) passes