Skip to content
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

Core: Fix the exports setup to make bundlers work with ESM & CommonJS #5429

Merged
merged 14 commits into from
Mar 11, 2024

Conversation

mgol
Copy link
Member

@mgol mgol commented Mar 2, 2024

Summary

We cannot pass a single file via the module condition as then
require( "jquery" ) will not return jQuery but instead the module object with
default, $ & jQuery as keys. Instead:

  1. For Node.js, detected via the node condition:
    1. Expose a regular CommonJS version to require
    2. Expose a tiny wrapper over CommonJS to import
  2. For bundlers, detected via the module condition:
    1. Expose a regular ESM version to import
    2. Expose a tiny wrapper over ESM to require
  3. If neither Node.js nor bundlers are detected (no node or module
    conditions`):
    1. Expose a regular CommonJS version to require
    2. Expose a regular ESM version to import

The reasons for such definitions are as follows:

  1. In Node.js, one can synchronously import from a CommonJS file inside of
    an ESM one but not vice-versa. To use an ESM file in a CommonJS one,
    a dynamic import is required and that forces asynchronicity.
  2. In some bundlers CommonJS is not necessarily enabled - e.g. in Rollup without
    the CommonJS plugin. Therefore, the ESM version needs to be pure ESM.
    However, bundlers allow synchronously calling require on an ESM file. This
    is possible since bundlers merge the files before they are passed to
    the browser to execute and the final bundles no longer contain async import
    code.
  3. Bare ESM & CommonJS versions are provided to non-Node non-bundler
    environments where we cannot assume interoperability between ESM & CommonJS
    is supported.
  4. Bare versions cannot be supplied to Node or bundlers as projects using both
    ESM & CommonJS to fetch jQuery would result in duplicate jQuery instances,
    leading to increased JS size and disjoint data storage.

In addition to the above changes, the script condition has been dropped. Only
Webpack documents this condition and it's not clear when exactly it's triggered.
Adding support for a new condition can be added later without a breaking change;
removing is not so easy.

The production & development conditions have been removed as well. They were
not really applied correctly; we'd need to provide both of them to each current
leaf which would double the size of the definition for the . & ./slim entry
points. In jQuery, the only difference between development & production builds
is minification; there are no logic changes so we can pass unminified versions
to all the tooling, expecting minification down the line.

For ./factory & ./factory-slim entry points, only the ESM version is now
supported. This was done to simplify the setup for this niche use case.

Tests with Rollup (both pure ESM & with the CommonJS plugin) & Webpack have
been added.

I added an explainer document to the wiki at https://github.com/jquery/jquery/wiki/jQuery-4-exports-explainer that should help understand the whole exports definition. It can be helpful especially if we need to make more changes.

Fixes gh-5416

Checklist

@mgol mgol added Core Needs review Discuss in Meeting Reserved for Issues and PRs that anyone would like to discuss in the weekly meeting. labels Mar 2, 2024
@mgol mgol added this to the 4.0.0 milestone Mar 2, 2024
@mgol mgol self-assigned this Mar 2, 2024
@mgol mgol marked this pull request as draft March 2, 2024 22:54
@mgol
Copy link
Member Author

mgol commented Mar 2, 2024

CI is failing because pretest now de facto depends on build:all. This means core tests now require a build. I need to think how to approach this.

@mgol mgol force-pushed the bundler-require-fixes branch from d3a28fa to 7145c00 Compare March 2, 2024 23:09
Copy link
Member

@timmywil timmywil left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I know it's not ready for review yet, but I went ahead and reviewed anyway.

@mgol mgol force-pushed the bundler-require-fixes branch 3 times, most recently from 86e62a3 to 1e3ce12 Compare March 9, 2024 22:36
@mgol mgol marked this pull request as ready for review March 9, 2024 22:38
@mgol mgol requested a review from timmywil March 9, 2024 22:39
@mgol
Copy link
Member Author

mgol commented Mar 9, 2024

@timmywil the PR is ready for review now.

@timmywil
Copy link
Member

Very nice! I only have suggestions on the npm scripts

@mgol mgol force-pushed the bundler-require-fixes branch from 1e3ce12 to edd36c6 Compare March 10, 2024 23:21
@mgol
Copy link
Member Author

mgol commented Mar 10, 2024

@timmywil I haven't addressed your feedback yet but I pushed 3 new commits that I had in progress (the first one is titled "Docs: Mention that the CommonJS module doesn't expose named exports"). The first one is a small docs update, the other two are adding more bundler/Node tests for our exports setup - for the slim build & both factory ones.

I contemplated simplifying the factory definitions by only exposing the ESM versions but I ultimately decided it's too breaking. People depending on factory behavior can update to our new entry points but they may not be able to switch to ESM. Funny enough, I hit it myself here as our Promises/A+ tests adapter uses the factory build but it's a CommonJS file and needs to expose the API synchronously...

The factory setup is still way simpler because there is no default export in ESM and the CommonJS version exports a similar object instead of just the API - and this means adapters are not needed, we can just expose the CommonJS version to Node and the ESM one to bundlers.

@mgol
Copy link
Member Author

mgol commented Mar 11, 2024

@timmywil I addressed your feedback now, please re-review.

@mgol mgol force-pushed the bundler-require-fixes branch from c8d1468 to cb875a2 Compare March 11, 2024 12:22
@mgol mgol requested a review from timmywil March 11, 2024 12:30
Copy link
Member

@timmywil timmywil left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well done!

@mgol
Copy link
Member Author

mgol commented Mar 11, 2024

@timmywil FYI, I added an explainer document to the wiki at https://github.com/jquery/jquery/wiki/jQuery-4-exports-explainer that should help understand the whole exports definition. It can be helpful especially if we need to make more changes.

@timmywil timmywil removed Needs review Discuss in Meeting Reserved for Issues and PRs that anyone would like to discuss in the weekly meeting. labels Mar 11, 2024
mgol added 2 commits March 12, 2024 00:29
Also, tweak the Node wrappers & tests a bit without changing
the core of the logic.
mgol added 12 commits March 12, 2024 00:29
The module `node:assert` is deprecated.
We cannot pass a single file via the `module` condition as then
`require( "jquery" )` will not return jQuery but instead the module object with
`default`, `$` & `jQuery` as keys. Instead:

1. For Node.js, detected via the `node` condition:
    1. Expose a regular CommonJS version to `require`
    2. Expose a tiny wrapper over CommonJS to `import`
2. For bundlers, detected via the `module` condition:
    1. Expose a regular ESM version to `import`
    2. Expose a tiny wrapper over ESM to `require`
3. If neither Node.js nor bundlers are detected (no `node` or `module`
   conditions`):
    1. Expose a regular CommonJS version to `require`
    2. Expose a regular ESM version to `import`

The reasons for such definitions are as follows:
1. In Node.js, one can synchronously import from a CommonJS file inside of
   an ESM one but not vice-versa. To use an ESM file in a CommonJS one,
   a dynamic import is required and that forces asynchronicity.
2. In some bundlers CommonJS is not necessarily enabled - e.g. in Rollup without
   the CommonJS plugin. Therefore, the ESM version needs to be pure ESM.
   However, bundlers allow synchronously calling `require` on an ESM file. This
   is possible since bundlers merge the files before they are passed to
   the browser to execute and the final bundles no longer contain async import
   code.
3. Bare ESM & CommonJS versions are provided to non-Node non-bundler
   environments where we cannot assume interoperability between ESM & CommonJS
   is supported.
4. Bare versions cannot be supplied to Node or bundlers as projects using both
   ESM & CommonJS to fetch jQuery would result in duplicate jQuery instances,
   leading to increased JS size and disjoint data storage.

In addition to the above changes, the `script` condition has been dropped. Only
Webpack documents this condition and it's not clear when exactly it's triggered.
Adding support for a new condition can be added later without a breaking change;
removing is not so easy.

The `production` & `development` conditions have been removed as well. They were
not really applied correctly; we'd need to provide both of them to each current
leaf which would double the size of the definition for the `.` & `./slim` entry
points. In jQuery, the only difference between development & production builds
is minification; there are no logic changes so we can pass unminified versions
to all the tooling, expecting minification down the line.

Fixes jquerygh-5416
After this commit:
1. Node.js always gets the CommonJS version
2. Bundlers always get the ESM version
3. Other tools take the ESM version when using `import` and the CommonJS when
   using `require`.

The complexity is lower than for the `.` & `./slim` entry points because there's
no default export to handle so node/bundler wrapper files are not necessary.
@mgol mgol force-pushed the bundler-require-fixes branch from cb875a2 to e8c6046 Compare March 11, 2024 23:29
@mgol mgol merged commit 60f11b5 into jquery:main Mar 11, 2024
13 checks passed
@mgol mgol deleted the bundler-require-fixes branch March 11, 2024 23:39
Mti-isf

This comment was marked as spam.

@jquery jquery deleted a comment from Mti-isf Aug 5, 2024
@jquery jquery deleted a comment from Mti-isf Aug 5, 2024
mgol added a commit to mgol/jquery-migrate that referenced this pull request Mar 31, 2025
The Migrate setup is simpler than Core as it doesn't have the slim or factory
versions, but the core ideas are similar.

Ref jquery/jquery#5255
Ref jquery/jquery#5429
mgol added a commit to mgol/jquery-migrate that referenced this pull request Mar 31, 2025
The Migrate setup is simpler than Core as it doesn't have the slim or factory
versions, but the core ideas are similar.

Ref jquery/jquery#5255
Ref jquery/jquery#5429
mgol added a commit to mgol/jquery-migrate that referenced this pull request Mar 31, 2025
The Migrate setup is simpler than Core as it doesn't have the slim or factory
versions, but the core ideas are similar.

Fixes jquerygh-524
Ref jquery/jquery#5255
Ref jquery/jquery#5429
mgol added a commit to mgol/jquery-migrate that referenced this pull request Apr 1, 2025
The Migrate setup is simpler than Core as it doesn't have the slim or factory
versions, but the core ideas are similar.

Fixes jquerygh-524
Ref jquery/jquery#5255
Ref jquery/jquery#5429
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Development

Successfully merging this pull request may close these issues.

require( "jquery" ) returns a module object when used with Webpack & jQuery 4.0.0-beta
3 participants