Skip to content

Commit d93137b

Browse files
jasnelltargos
authored andcommitted
workers: fix spawning from preload scripts
Fix spawning nested worker threads from preload scripts and warn about doing so. Signed-off-by: James M Snell <jasnell@gmail.com> Fixes: #36531 PR-URL: #37481 Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Matteo Collina <matteo.collina@gmail.com>
1 parent cbfc6b1 commit d93137b

File tree

4 files changed

+35
-4
lines changed

4 files changed

+35
-4
lines changed

doc/api/worker_threads.md

+11
Original file line numberDiff line numberDiff line change
@@ -1127,6 +1127,17 @@ Calling `unref()` on a worker allows the thread to exit if this is the only
11271127
active handle in the event system. If the worker is already `unref()`ed calling
11281128
`unref()` again has no effect.
11291129

1130+
## Notes
1131+
1132+
### Launching worker threads from preload scripts
1133+
1134+
Take care when launching worker threads from preload scripts (scripts loaded
1135+
and run using the `-r` command line flag). Unless the `execArgv` option is
1136+
explicitly set, new Worker threads automatically inherit the command line flags
1137+
from the running process and will preload the same preload scripts as the main
1138+
thread. If the preload script unconditionally launches a worker thread, every
1139+
thread spawned will spawn another until the application crashes.
1140+
11301141
[Addons worker support]: addons.md#addons_worker_support
11311142
[ECMAScript module loader]: esm.md#esm_data_imports
11321143
[HTML structured clone algorithm]: https://developer.mozilla.org/en-US/docs/Web/API/Web_Workers_API/Structured_clone_algorithm

lib/internal/main/worker_thread.js

+5-4
Original file line numberDiff line numberDiff line change
@@ -121,10 +121,6 @@ port.on('message', (message) => {
121121
initializeCJSLoader();
122122
initializeESMLoader();
123123

124-
const CJSLoader = require('internal/modules/cjs/loader');
125-
assert(!CJSLoader.hasLoadedAnyUserCJSModule);
126-
loadPreloadModules();
127-
initializeFrozenIntrinsics();
128124
if (argv !== undefined) {
129125
ArrayPrototypePushApply(process.argv, argv);
130126
}
@@ -147,6 +143,11 @@ port.on('message', (message) => {
147143
};
148144
workerIo.sharedCwdCounter = cwdCounter;
149145

146+
const CJSLoader = require('internal/modules/cjs/loader');
147+
assert(!CJSLoader.hasLoadedAnyUserCJSModule);
148+
loadPreloadModules();
149+
initializeFrozenIntrinsics();
150+
150151
if (!hasStdin)
151152
process.stdin.push(null);
152153

test/fixtures/worker-preload.js

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
const {
2+
Worker,
3+
workerData,
4+
threadId
5+
} = require('worker_threads');
6+
7+
if (threadId < 2) {
8+
new Worker('1 + 1', { eval: true });
9+
}

test/parallel/test-preload-worker.js

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
'use strict';
2+
3+
const common = require('../common');
4+
const fixtures = require('../common/fixtures');
5+
const worker = fixtures.path('worker-preload.js');
6+
const { exec } = require('child_process');
7+
const kNodeBinary = process.argv[0];
8+
9+
10+
exec(`"${kNodeBinary}" -r "${worker}" -pe "1+1"`, common.mustSucceed());

0 commit comments

Comments
 (0)