@@ -18,7 +18,6 @@ const {
18
18
kIsExecuting,
19
19
kRequiredModuleSymbol,
20
20
} = require ( 'internal/modules/cjs/loader' ) ;
21
-
22
21
const { imported_cjs_symbol } = internalBinding ( 'symbols' ) ;
23
22
24
23
const assert = require ( 'internal/assert' ) ;
@@ -38,7 +37,15 @@ const {
38
37
forceDefaultLoader,
39
38
} = require ( 'internal/modules/esm/utils' ) ;
40
39
const { kImplicitTypeAttribute } = require ( 'internal/modules/esm/assert' ) ;
41
- const { ModuleWrap, kEvaluating, kEvaluated, kEvaluationPhase, kSourcePhase } = internalBinding ( 'module_wrap' ) ;
40
+ const {
41
+ ModuleWrap,
42
+ kEvaluated,
43
+ kEvaluating,
44
+ kEvaluationPhase,
45
+ kInstantiated,
46
+ kSourcePhase,
47
+ throwIfPromiseRejected,
48
+ } = internalBinding ( 'module_wrap' ) ;
42
49
const {
43
50
urlToFilename,
44
51
} = require ( 'internal/modules/helpers' ) ;
@@ -53,6 +60,10 @@ let defaultResolve, defaultLoad, defaultLoadSync, importMetaInitializer;
53
60
const { tracingChannel } = require ( 'diagnostics_channel' ) ;
54
61
const onImport = tracingChannel ( 'module.import' ) ;
55
62
63
+ let debug = require ( 'internal/util/debuglog' ) . debuglog ( 'esm' , ( fn ) => {
64
+ debug = fn ;
65
+ } ) ;
66
+
56
67
/**
57
68
* @typedef {import('./hooks.js').HooksProxy } HooksProxy
58
69
* @typedef {import('./module_job.js').ModuleJobBase } ModuleJobBase
@@ -86,6 +97,23 @@ function getTranslators() {
86
97
return translators ;
87
98
}
88
99
100
+ /**
101
+ * Generate message about potential race condition caused by requiring a cached module that has started
102
+ * async linking.
103
+ * @param {string } filename Filename of the module being required.
104
+ * @param {string|undefined } parentFilename Filename of the module calling require().
105
+ * @returns {string } Error message.
106
+ */
107
+ function getRaceMessage ( filename , parentFilename ) {
108
+ let raceMessage = `Cannot require() ES Module ${ filename } because it is not yet fully loaded. ` ;
109
+ raceMessage += 'This may be caused by a race condition if the module is simultaneously dynamically ' ;
110
+ raceMessage += 'import()-ed via Promise.all(). Try await-ing the import() sequentially in a loop instead.' ;
111
+ if ( parentFilename ) {
112
+ raceMessage += ` (from ${ parentFilename } )` ;
113
+ }
114
+ return raceMessage ;
115
+ }
116
+
89
117
/**
90
118
* @type {HooksProxy }
91
119
* Multiple loader instances exist for various, specific reasons (see code comments at site).
@@ -346,35 +374,53 @@ class ModuleLoader {
346
374
// evaluated at this point.
347
375
// TODO(joyeecheung): add something similar to CJS loader's requireStack to help
348
376
// debugging the the problematic links in the graph for import.
377
+ debug ( 'importSyncForRequire' , parent ?. filename , '->' , filename , job ) ;
349
378
if ( job !== undefined ) {
350
379
mod [ kRequiredModuleSymbol ] = job . module ;
351
380
const parentFilename = urlToFilename ( parent ?. filename ) ;
352
381
// TODO(node:55782): this race may stop to happen when the ESM resolution and loading become synchronous.
353
382
if ( ! job . module ) {
354
- let message = `Cannot require() ES Module ${ filename } because it is not yet fully loaded. ` ;
355
- message += 'This may be caused by a race condition if the module is simultaneously dynamically ' ;
356
- message += 'import()-ed via Promise.all(). Try await-ing the import() sequentially in a loop instead.' ;
357
- if ( parentFilename ) {
358
- message += ` (from ${ parentFilename } )` ;
359
- }
360
- assert ( job . module , message ) ;
383
+ assert . fail ( getRaceMessage ( filename , parentFilename ) ) ;
361
384
}
362
385
if ( job . module . async ) {
363
386
throw new ERR_REQUIRE_ASYNC_MODULE ( filename , parentFilename ) ;
364
387
}
365
- // job.module may be undefined if it's asynchronously loaded. Which means
366
- // there is likely a cycle.
367
- if ( job . module . getStatus ( ) !== kEvaluated ) {
368
- let message = `Cannot require() ES Module ${ filename } in a cycle.` ;
369
- if ( parentFilename ) {
370
- message += ` (from ${ parentFilename } )` ;
371
- }
372
- message += 'A cycle involving require(esm) is disallowed to maintain ' ;
373
- message += 'invariants madated by the ECMAScript specification' ;
374
- message += 'Try making at least part of the dependency in the graph lazily loaded.' ;
375
- throw new ERR_REQUIRE_CYCLE_MODULE ( message ) ;
388
+ const status = job . module . getStatus ( ) ;
389
+ debug ( 'Module status' , filename , status ) ;
390
+ if ( status === kEvaluated ) {
391
+ return { wrap : job . module , namespace : job . module . getNamespaceSync ( filename , parentFilename ) } ;
392
+ } else if ( status === kInstantiated ) {
393
+ // When it's an async job cached by another import request,
394
+ // which has finished linking but has not started its
395
+ // evaluation because the async run() task would be later
396
+ // in line. Then start the evaluation now with runSync(), which
397
+ // is guaranteed to finish by the time the other run() get to it,
398
+ // and the other task would just get the cached evaluation results,
399
+ // similar to what would happen when both are async.
400
+ mod [ kRequiredModuleSymbol ] = job . module ;
401
+ const { namespace } = job . runSync ( parent ) ;
402
+ return { wrap : job . module , namespace : namespace || job . module . getNamespace ( ) } ;
376
403
}
377
- return { wrap : job . module , namespace : job . module . getNamespaceSync ( filename , parentFilename ) } ;
404
+ // When the cached async job have already encountered a linking
405
+ // error that gets wrapped into a rejection, but is still later
406
+ // in line to throw on it, just unwrap and throw the linking error
407
+ // from require().
408
+ if ( job . instantiated ) {
409
+ throwIfPromiseRejected ( job . instantiated ) ;
410
+ }
411
+ if ( status !== kEvaluating ) {
412
+ assert . fail ( `Unexpected module status ${ status } . ` +
413
+ getRaceMessage ( filename , parentFilename ) ) ;
414
+ }
415
+ let message = `Cannot require() ES Module ${ filename } in a cycle.` ;
416
+ if ( parentFilename ) {
417
+ message += ` (from ${ parentFilename } )` ;
418
+ }
419
+ message += 'A cycle involving require(esm) is disallowed to maintain ' ;
420
+ message += 'invariants madated by the ECMAScript specification' ;
421
+ message += 'Try making at least part of the dependency in the graph lazily loaded.' ;
422
+ throw new ERR_REQUIRE_CYCLE_MODULE ( message ) ;
423
+
378
424
}
379
425
// TODO(joyeecheung): refactor this so that we pre-parse in C++ and hit the
380
426
// cache here, or use a carrier object to carry the compiled module script
0 commit comments