2
2
3
3
const {
4
4
ArrayPrototypePush,
5
+ ArrayPrototypePushApply,
5
6
FunctionPrototypeCall,
6
7
Int32Array,
7
8
ObjectAssign,
@@ -47,8 +48,10 @@ const {
47
48
validateObject,
48
49
validateString,
49
50
} = require ( 'internal/validators' ) ;
50
-
51
- const { kEmptyObject } = require ( 'internal/util' ) ;
51
+ const {
52
+ emitExperimentalWarning,
53
+ kEmptyObject,
54
+ } = require ( 'internal/util' ) ;
52
55
53
56
const {
54
57
defaultResolve,
@@ -83,6 +86,7 @@ let importMetaInitializer;
83
86
84
87
// [2] `validate...()`s throw the wrong error
85
88
89
+ let globalPreloadWarned = false ;
86
90
class Hooks {
87
91
#chains = {
88
92
/**
@@ -127,31 +131,43 @@ class Hooks {
127
131
* Import and register custom/user-defined module loader hook(s).
128
132
* @param {string } urlOrSpecifier
129
133
* @param {string } parentURL
134
+ * @param {any } [data] Arbitrary data to be passed from the custom
135
+ * loader (user-land) to the worker.
130
136
*/
131
- async register ( urlOrSpecifier , parentURL ) {
137
+ async register ( urlOrSpecifier , parentURL , data ) {
132
138
const moduleLoader = require ( 'internal/process/esm_loader' ) . esmLoader ;
133
139
const keyedExports = await moduleLoader . import (
134
140
urlOrSpecifier ,
135
141
parentURL ,
136
142
kEmptyObject ,
137
143
) ;
138
- this . addCustomLoader ( urlOrSpecifier , keyedExports ) ;
144
+ return this . addCustomLoader ( urlOrSpecifier , keyedExports , data ) ;
139
145
}
140
146
141
147
/**
142
148
* Collect custom/user-defined module loader hook(s).
143
149
* After all hooks have been collected, the global preload hook(s) must be initialized.
144
150
* @param {string } url Custom loader specifier
145
151
* @param {Record<string, unknown> } exports
152
+ * @param {any } [data] Arbitrary data to be passed from the custom loader (user-land)
153
+ * to the worker.
154
+ * @returns {any } The result of the loader's `initialize` hook, if provided.
146
155
*/
147
- addCustomLoader ( url , exports ) {
156
+ addCustomLoader ( url , exports , data ) {
148
157
const {
149
158
globalPreload,
159
+ initialize,
150
160
resolve,
151
161
load,
152
162
} = pluckHooks ( exports ) ;
153
163
154
- if ( globalPreload ) {
164
+ if ( globalPreload && ! initialize ) {
165
+ if ( globalPreloadWarned === false ) {
166
+ globalPreloadWarned = true ;
167
+ emitExperimentalWarning (
168
+ '`globalPreload` will be removed in a future version. Please use `initialize` instead.' ,
169
+ ) ;
170
+ }
155
171
ArrayPrototypePush ( this . #chains. globalPreload , { __proto__ : null , fn : globalPreload , url } ) ;
156
172
}
157
173
if ( resolve ) {
@@ -162,6 +178,7 @@ class Hooks {
162
178
const next = this . #chains. load [ this . #chains. load . length - 1 ] ;
163
179
ArrayPrototypePush ( this . #chains. load , { __proto__ : null , fn : load , url, next } ) ;
164
180
}
181
+ return initialize ?. ( data ) ;
165
182
}
166
183
167
184
/**
@@ -553,15 +570,30 @@ class HooksProxy {
553
570
}
554
571
}
555
572
556
- async makeAsyncRequest ( method , ...args ) {
573
+ /**
574
+ * Invoke a remote method asynchronously.
575
+ * @param {string } method Method to invoke
576
+ * @param {any[] } [transferList] Objects in `args` to be transferred
577
+ * @param {any[] } args Arguments to pass to `method`
578
+ * @returns {Promise<any> }
579
+ */
580
+ async makeAsyncRequest ( method , transferList , ...args ) {
557
581
this . waitForWorker ( ) ;
558
582
559
583
MessageChannel ??= require ( 'internal/worker/io' ) . MessageChannel ;
560
584
const asyncCommChannel = new MessageChannel ( ) ;
561
585
562
586
// Pass work to the worker.
563
- debug ( 'post async message to worker' , { method, args } ) ;
564
- this . #worker. postMessage ( { method, args, port : asyncCommChannel . port2 } , [ asyncCommChannel . port2 ] ) ;
587
+ debug ( 'post async message to worker' , { method, args, transferList } ) ;
588
+ const finalTransferList = [ asyncCommChannel . port2 ] ;
589
+ if ( transferList ) {
590
+ ArrayPrototypePushApply ( finalTransferList , transferList ) ;
591
+ }
592
+ this . #worker. postMessage ( {
593
+ __proto__ : null ,
594
+ method, args,
595
+ port : asyncCommChannel . port2 ,
596
+ } , finalTransferList ) ;
565
597
566
598
if ( this . #numberOfPendingAsyncResponses++ === 0 ) {
567
599
// On the next lines, the main thread will await a response from the worker thread that might
@@ -593,12 +625,19 @@ class HooksProxy {
593
625
return body ;
594
626
}
595
627
596
- makeSyncRequest ( method , ...args ) {
628
+ /**
629
+ * Invoke a remote method synchronously.
630
+ * @param {string } method Method to invoke
631
+ * @param {any[] } [transferList] Objects in `args` to be transferred
632
+ * @param {any[] } args Arguments to pass to `method`
633
+ * @returns {any }
634
+ */
635
+ makeSyncRequest ( method , transferList , ...args ) {
597
636
this . waitForWorker ( ) ;
598
637
599
638
// Pass work to the worker.
600
- debug ( 'post sync message to worker' , { method, args } ) ;
601
- this . #worker. postMessage ( { method, args } ) ;
639
+ debug ( 'post sync message to worker' , { method, args, transferList } ) ;
640
+ this . #worker. postMessage ( { __proto__ : null , method, args } , transferList ) ;
602
641
603
642
let response ;
604
643
do {
@@ -708,6 +747,7 @@ ObjectSetPrototypeOf(HooksProxy.prototype, null);
708
747
*/
709
748
function pluckHooks ( {
710
749
globalPreload,
750
+ initialize,
711
751
resolve,
712
752
load,
713
753
} ) {
@@ -723,6 +763,10 @@ function pluckHooks({
723
763
acceptedHooks . load = load ;
724
764
}
725
765
766
+ if ( initialize ) {
767
+ acceptedHooks . initialize = initialize ;
768
+ }
769
+
726
770
return acceptedHooks ;
727
771
}
728
772
0 commit comments