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

esm, loader: move to own thread #43658

Closed
1 task
Tracked by #2
JakobJingleheimer opened this issue Jul 2, 2022 · 27 comments · Fixed by #44710
Closed
1 task
Tracked by #2

esm, loader: move to own thread #43658

JakobJingleheimer opened this issue Jul 2, 2022 · 27 comments · Fixed by #44710
Assignees
Labels
esm Issues and PRs related to the ECMAScript Modules implementation. loaders Issues and PRs related to ES module loaders

Comments

@JakobJingleheimer
Copy link
Member

JakobJingleheimer commented Jul 2, 2022

What is the problem this feature will solve?

  • limit contamination
  • facilitate synchronous import.meta.resolve()

For the initial implementation, the same loaders thread will be used for all user-land threads. A subsequent enhancement may add a configuration option to the Worker constructor to spawn its own dedicated loaders thread.

What is the feature you are proposing to solve the problem?

Move loaders off-thread

What alternatives have you considered?

No response


Per direction from TSC

Following in the vein of babel: babel/babel#14025
And https://github.com/bmeck/using-node-workshop/tree/main/steps/6_async_and_blocking


  • notify package authors of change (please comment to be added if not already included—sorry I know only so many)
    • esmock
    • jest?
    • mocha (Gil Tayar)
    • ts-node (Andrew Bradley)
    • yarn (Maël Nison)
@JakobJingleheimer JakobJingleheimer added the feature request Issues that request new features to be added to Node.js. label Jul 2, 2022
@JakobJingleheimer JakobJingleheimer self-assigned this Jul 2, 2022
@JakobJingleheimer JakobJingleheimer added esm Issues and PRs related to the ECMAScript Modules implementation. loaders Issues and PRs related to ES module loaders labels Jul 2, 2022
@JakobJingleheimer
Copy link
Member Author

@bmeck @MylesBorins could either of you speak to the impetus for this?

@JakobJingleheimer JakobJingleheimer removed the feature request Issues that request new features to be added to Node.js. label Jul 2, 2022
@aduh95
Copy link
Contributor

aduh95 commented Jul 5, 2022

Related: #31229

@GeoffreyBooth
Copy link
Member

@bmeck @MylesBorins could either of you speak to the impetus for this?

Pros and cons here: nodejs/modules#351 (comment)

@cspotcode
Copy link

Can add to the list of package authors:
esmock

I wonder, does it make sense for the loaders team to have a thread somewhere which can be subscribed to for notifications of breaking changes? All relevant discussion can happen elsewhere, the thread would be like an RSS feed. Package maintainers can opt-in to notifications of breaking or potentially exciting/disruptive changes by subscribing to that thread.

Might scale better than us hoping we know a comprehensive list of all loaders.

@JakobJingleheimer
Copy link
Member Author

Since Node.js doesn't maintain anything of the sort, that sounds like a better alternative to "surprise!". It's not full-proof, though.

@JakobJingleheimer
Copy link
Member Author

We have a working Proof of Concept: https://github.com/JakobJingleheimer/worker-atomics

@JakobJingleheimer JakobJingleheimer changed the title esm, loaders: move to own thread esm, loader: move to own thread Sep 18, 2022
@loynoir

This comment was marked as off-topic.

@VulgarnyKarlson

This comment was marked as off-topic.

@JakobJingleheimer
Copy link
Member Author

JakobJingleheimer commented Oct 19, 2022

Hi! Any estimated date for done this thread ?

This thread blocking #43772 which needed for

https://github.com/yarnpkg/berry/discussions/4044#discussioncomment-2740697

Asking cause looks like it's long time thread ( currently 4 months )

Hiya! Please look right above your post to find the active WIP PR link (in the fancy GitHub callout), which was updated recently.

@JakobJingleheimer
Copy link
Member Author

I was chatting with Anna earlier today, and she mentioned the CPU and memory cost of the 2nd thread is non-trivial—effectively doubling node's basic footprint.

Is that something we've already considered? Do we know of any way to mitigate that?

@bmeck
Copy link
Member

bmeck commented Oct 21, 2022 via email

@mcollina
Copy link
Member

mcollina commented Oct 23, 2022

The way this issue is framed implies that import loaders are moved off thread. In other terms, only users deliberately opting in are using a different thread. However, this is not the direction the work went and all ESM loading logic is being moved. I spoke to quite a few people over time about this and this was not brought up before today.

This is a significant blocker for me as Node.js is often used in environment that are constrained by memory. At the bare minimum we should investigate:

  1. how much more memory is needed?
  2. how much more latency this will add?
  3. can we avoid it, i.e. only move user-provided code off thread?
  4. will this memory cost disappear or will the thread be kept around?

@mcollina mcollina added the tsc-agenda Issues and PRs to discuss during the meetings of the TSC. label Oct 23, 2022
@GeoffreyBooth
Copy link
Member

I agree that we should find answers to those questions. I would hold off on adding it to the TSC agenda until we have those answers.

@JakobJingleheimer
Copy link
Member Author

JakobJingleheimer commented Oct 23, 2022

Edit: I believe it was always our intention to get those numbers before landing.


  1. how much more memory is needed?

This does not seem to be a straightforward answer. I can't truly measure it until the implementation is runnable (currently blocked by a V8 error). Preliminary result data from our experiment via /usr/bin/time -l are:

metric baseline off-threaded
maximum resident set size 39174144 54509568
average shared memory size 0 0
average unshared data size 0 0
average unshared stack size 0 0
page reclaims 2581 3517
page faults 0 0
swaps 0 0
block input operations 0 0
block output operations 0 0
messages sent 0 0
messages received 0 0
signals received 0 0
voluntary context switches 1
involuntary context switches 67 134
instructions retired 280244491 550971942
cycles elapsed 101541186 203115216
peak memory footprint 15487232 28354496

This very basic rough comparison suggests additional memory consumption at peak of 12,867,264 (~83% increase).

  1. how much more latency this will add?

Preliminary results suggest there's a small initial cost when the worker initialises, but in terms of latency incurred for the work off-loaded, there's effectively none (nanoseconds).

  1. can we avoid it, i.e. only move user-provided code off thread?

I think this is contrary to the goals we're attempting to achieve with a separate thread (but maybe we only care about custom loaders).

  1. will this memory cost disappear or will the thread be kept around?

The thread gracefully terminates after the last call to "public" ESMLoader. However, in the tests done so far, that coincides with node itself gracefully terminating. We could specifically test keeping node around for a while after all the ESM / loader stuff to see if the thread terminates. I'm not sure how best to go about that: worker.once('exit') does not trigger (and also, I fear if it did work, it would prevent/undo the worker.unref(), leading to false results).

@GeoffreyBooth
Copy link
Member

There's also a potential benefit in moving non-custom loading off-thread as it would protect internals from prototype pollution (I think). That would argue that we should make this same refactor for CommonJS too.

@aduh95
Copy link
Contributor

aduh95 commented Oct 23, 2022

This is a significant blocker for me as Node.js is often used in environment that are constrained by memory. At the bare minimum we should investigate:

  1. how much more memory is needed?
  2. how much more latency this will add?
  3. can we avoid it, i.e. only move user-provided code off thread?
  4. will this memory cost disappear or will the thread be kept around?

@mcollina I'm not convinced we can answer those questions before we have a fully working implementation; without data, we can only make assumptions, and I wouldn't want us to draw a conclusion over possibly baseless assumptions.

@mcollina
Copy link
Member

Absolutely! I'm concerned about the addition to our startup memory footprint, as this matters for some of our usecases.

I'm flagging that this might be problematic and I was surprised because it was not mentioned in the main text of the issue.

A few more question:

  1. if a new Worker thread is spawned in the lifetime of the application, will this create another thread to load ESM?
  2. what about dynamic import? Will it need re-spawning the thread?

@JakobJingleheimer
Copy link
Member Author

A few more question:

  1. if a new Worker thread is spawned in the lifetime of the application, will this create another thread to load ESM?
  2. what about dynamic import? Will it need re-spawning the thread?

In both cases, (in the current design/implementation) only if the "loaders" worker is not around (otherwise it will be re-used).

@mcollina
Copy link
Member

In both cases, (in the current design/implementation) only if the "loaders" worker is not around (otherwise it will be re-used).

So there is only one loaders worker for all worker threads created by Node.js?

@targos
Copy link
Member

targos commented Oct 24, 2022

No, there's a separate loaders worker for each worker thread.

@Flarna
Copy link
Member

Flarna commented Oct 24, 2022

I assume it should be possible to configure loader hooks per worker thread therefore this may complicate this thread.

I'm also a bit skeptical to have a single loader thread for all workers as this seems to allow "leaking" data between workers which should be isolated. Also this single loader worker would be parallelism blocker.

@mhdawson
Copy link
Member

mhdawson commented Nov 1, 2022

The comment from @targos and @JakobJingleheimer seem contradictory?

@cspotcode
Copy link

There are 3 different threading models being considered, that I'm aware of.

@JakobJingleheimer
Copy link
Member Author

Sorry, I think #43658 (comment) is the intended behaviour (the current PR may not achieve that yet). The rationale being that workers are intended to be isolated, so their loaders' state should also be isolated/fresh.

@JakobJingleheimer
Copy link
Member Author

Quick update from our recent team meeting: We invited Gil Tayar (author/maintainer of several pertinent libraries) to discuss spawning a dedicated loaders thread per user-land thread, and he noted that will add enormous complexity to library authors (on top of the extra complexity on node's side). We decided for an initial implementation, it would be better to use a single loaders thread shared by all user-land threads and add caveats to the relevant sections of the docs. If there is sufficient appetite, we can subsequently add a configuration option (perhaps to the Worker constructor) to spawn dedicated loaders threads.

@mcollina
Copy link
Member

Was there any progress on not spawning any thread if no custom loaders are defined?

@JakobJingleheimer
Copy link
Member Author

Yep! nodejs/loaders#118 (comment) and I believe this should be logistically/technically possible.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
esm Issues and PRs related to the ECMAScript Modules implementation. loaders Issues and PRs related to ES module loaders
Projects
None yet
Development

Successfully merging a pull request may close this issue.