From beb57376eaea706869ccd42ad09910ff5dc4be96 Mon Sep 17 00:00:00 2001 From: Sergey Chernyshev Date: Thu, 25 Apr 2024 01:36:10 +0200 Subject: [PATCH] bootstrap: lazy load non-essential modules It turns out that even with startup snapshots, there is a non-trivial overhead for loading internal modules. This patch makes the loading of the non-essential modules lazy again. Caveat: we have to make some of the globals lazily-loaded too, so the WPT runner is updated to test what the state of the global scope is after the globals are accessed (and replaced with the loaded value). PR-URL: https://github.com/nodejs/node/pull/45659 Backport-PR-URL: https://github.com/nodejs/node/pull/46425 Reviewed-By: Matteo Collina Reviewed-By: Yagiz Nizipli Reviewed-By: Daeyeon Jeong Reviewed-By: Jacob Smith Reviewed-By: James M Snell Reviewed-By: Rafael Gonzaga Reviewed-By: Antoine du Hamel Reviewed-By: Minwoo Jung Reviewed-By: Tierney Cyren --- graal-nodejs/lib/buffer.js | 24 +-- graal-nodejs/lib/fs.js | 26 ++- graal-nodejs/lib/internal/async_hooks.js | 3 +- .../lib/internal/bootstrap/browser.js | 162 +++++++----------- graal-nodejs/lib/internal/bootstrap/node.js | 30 ++-- .../bootstrap/switches/is_main_thread.js | 8 +- .../lib/internal/console/constructor.js | 8 +- graal-nodejs/lib/internal/dns/utils.js | 10 +- graal-nodejs/lib/internal/errors.js | 2 +- .../lib/internal/modules/cjs/loader.js | 2 +- .../lib/internal/modules/esm/get_format.js | 2 +- graal-nodejs/lib/internal/modules/esm/load.js | 4 +- .../lib/internal/modules/esm/loader.js | 33 ++-- .../lib/internal/modules/esm/module_map.js | 3 +- .../internal/modules/esm/package_config.js | 2 +- .../lib/internal/modules/esm/translators.js | 6 +- graal-nodejs/lib/internal/modules/run_main.js | 6 +- .../lib/internal/process/per_thread.js | 7 +- .../lib/internal/process/pre_execution.js | 31 ++-- .../internal/source_map/source_map_cache.js | 2 +- graal-nodejs/lib/internal/util.js | 87 ++++++++++ graal-nodejs/lib/util.js | 27 ++- graal-nodejs/test/common/wpt.js | 26 +++ .../test/parallel/test-bootstrap-modules.js | 132 ++------------ graal-nodejs/test/wpt/test-streams.js | 1 + 25 files changed, 314 insertions(+), 330 deletions(-) diff --git a/graal-nodejs/lib/buffer.js b/graal-nodejs/lib/buffer.js index 13a737e2b06..0216dd4dd32 100644 --- a/graal-nodejs/lib/buffer.js +++ b/graal-nodejs/lib/buffer.js @@ -86,6 +86,7 @@ const { lazyDOMException, normalizeEncoding, kIsEncodingSymbol, + defineLazyProperties, } = require('internal/util'); const { isAnyArrayBuffer, @@ -129,15 +130,6 @@ const { createUnsafeBuffer, } = require('internal/buffer'); -const { - Blob, - resolveObjectURL, -} = require('internal/blob'); - -const { - File, -} = require('internal/file'); - FastBuffer.prototype.constructor = Buffer; Buffer.prototype = FastBuffer.prototype; addBufferPrototypeMethods(Buffer.prototype); @@ -1385,9 +1377,6 @@ function isAscii(input) { } module.exports = { - Blob, - File, - resolveObjectURL, Buffer, SlowBuffer, transcode, @@ -1416,3 +1405,14 @@ ObjectDefineProperties(module.exports, { set(val) { INSPECT_MAX_BYTES = val; }, }, }); + +defineLazyProperties( + module.exports, + 'internal/blob', + ['Blob', 'resolveObjectURL'], +); +defineLazyProperties( + module.exports, + 'internal/file', + ['File'], +); diff --git a/graal-nodejs/lib/fs.js b/graal-nodejs/lib/fs.js index db3c202bfa6..8b6af16e5e5 100644 --- a/graal-nodejs/lib/fs.js +++ b/graal-nodejs/lib/fs.js @@ -87,6 +87,7 @@ const { custom: kCustomPromisifiedSymbol, }, SideEffectFreeRegExpPrototypeExec, + defineLazyProperties, } = require('internal/util'); const { constants: { @@ -125,11 +126,6 @@ const { validatePrimitiveStringAfterArrayBufferView, warnOnNonPortableTemplate, } = require('internal/fs/utils'); -const { - Dir, - opendir, - opendirSync, -} = require('internal/fs/dir'); const { CHAR_FORWARD_SLASH, CHAR_BACKWARD_SLASH, @@ -146,9 +142,6 @@ const { validateString, } = require('internal/validators'); -const watchers = require('internal/fs/watchers'); -const ReadFileContext = require('internal/fs/read_file_context'); - let truncateWarn = true; let fs; @@ -392,6 +385,7 @@ function checkAborted(signal, callback) { function readFile(path, options, callback) { callback = maybeCallback(callback || options); options = getOptions(options, { flag: 'r' }); + const ReadFileContext = require('internal/fs/read_file_context'); const context = new ReadFileContext(callback, options.encoding); context.isUserFd = isFd(path); // File descriptor ownership @@ -2422,12 +2416,13 @@ function watch(filename, options, listener) { if (options.recursive === undefined) options.recursive = false; if (options.recursive && !(isOSX || isWindows)) throw new ERR_FEATURE_UNAVAILABLE_ON_PLATFORM('watch recursively'); + + const watchers = require('internal/fs/watchers'); const watcher = new watchers.FSWatcher(); watcher[watchers.kFSWatchStart](filename, options.persistent, options.recursive, options.encoding); - if (listener) { watcher.addListener('change', listener); } @@ -2486,7 +2481,7 @@ function watchFile(filename, options, listener) { validateFunction(listener, 'listener'); stat = statWatchers.get(filename); - + const watchers = require('internal/fs/watchers'); if (stat === undefined) { stat = new watchers.StatWatcher(options.bigint); stat[watchers.kFSStatWatcherStart](filename, @@ -2512,7 +2507,7 @@ function unwatchFile(filename, listener) { const stat = statWatchers.get(filename); if (stat === undefined) return; - + const watchers = require('internal/fs/watchers'); if (typeof listener === 'function') { const beforeListenerCount = stat.listenerCount('change'); stat.removeListener('change', listener); @@ -3116,8 +3111,6 @@ module.exports = fs = { mkdtempSync, open, openSync, - opendir, - opendirSync, readdir, readdirSync, read, @@ -3157,7 +3150,6 @@ module.exports = fs = { writeSync, writev, writevSync, - Dir, Dirent, Stats, @@ -3203,6 +3195,12 @@ module.exports = fs = { _toUnixTimestamp: toUnixTimestamp, }; +defineLazyProperties( + fs, + 'internal/fs/dir', + ['Dir', 'opendir', 'opendirSync'], +); + ObjectDefineProperties(fs, { F_OK: { __proto__: null, enumerable: true, value: F_OK || 0 }, R_OK: { __proto__: null, enumerable: true, value: R_OK || 0 }, diff --git a/graal-nodejs/lib/internal/async_hooks.js b/graal-nodejs/lib/internal/async_hooks.js index 9294773fe41..ee9094b184f 100644 --- a/graal-nodejs/lib/internal/async_hooks.js +++ b/graal-nodejs/lib/internal/async_hooks.js @@ -8,8 +8,6 @@ const { Symbol, } = primordials; -const promiseHooks = require('internal/promise_hooks'); - const async_wrap = internalBinding('async_wrap'); const { setCallbackTrampoline } = async_wrap; /* async_hook_fields is a Uint32Array wrapping the uint32_t array of @@ -382,6 +380,7 @@ function updatePromiseHookMode() { initHook = destroyTracking; } if (stopPromiseHook) stopPromiseHook(); + const promiseHooks = require('internal/promise_hooks'); stopPromiseHook = promiseHooks.createHook({ init: initHook, before: promiseBeforeHook, diff --git a/graal-nodejs/lib/internal/bootstrap/browser.js b/graal-nodejs/lib/internal/bootstrap/browser.js index 5be4dd61764..93ae26f77a1 100644 --- a/graal-nodejs/lib/internal/bootstrap/browser.js +++ b/graal-nodejs/lib/internal/bootstrap/browser.js @@ -9,6 +9,9 @@ const { defineOperation, exposeInterface, lazyDOMExceptionClass, + defineLazyProperties, + defineReplaceableLazyAttribute, + exposeLazyInterfaces, } = require('internal/util'); const config = internalBinding('config'); @@ -28,37 +31,6 @@ exposeGetterAndSetter(globalThis, exposeInterface(globalThis, 'DOMException', value); }); -const { - TextEncoder, - TextDecoder, -} = require('internal/encoding'); -// https://encoding.spec.whatwg.org/#textencoder -exposeInterface(globalThis, 'TextEncoder', TextEncoder); -// https://encoding.spec.whatwg.org/#textdecoder -exposeInterface(globalThis, 'TextDecoder', TextDecoder); - -const { - AbortController, - AbortSignal, -} = require('internal/abort_controller'); -exposeInterface(globalThis, 'AbortController', AbortController); -exposeInterface(globalThis, 'AbortSignal', AbortSignal); - -const { - EventTarget, - Event, -} = require('internal/event_target'); -exposeInterface(globalThis, 'EventTarget', EventTarget); -exposeInterface(globalThis, 'Event', Event); -const { - MessageChannel, - MessagePort, - MessageEvent, -} = require('internal/worker/io'); -exposeInterface(globalThis, 'MessageChannel', MessageChannel); -exposeInterface(globalThis, 'MessagePort', MessagePort); -exposeInterface(globalThis, 'MessageEvent', MessageEvent); - // https://html.spec.whatwg.org/multipage/webappapis.html#windoworworkerglobalscope const timers = require('timers'); defineOperation(globalThis, 'clearInterval', timers.clearInterval); @@ -66,18 +38,32 @@ defineOperation(globalThis, 'clearTimeout', timers.clearTimeout); defineOperation(globalThis, 'setInterval', timers.setInterval); defineOperation(globalThis, 'setTimeout', timers.setTimeout); -const buffer = require('buffer'); -defineOperation(globalThis, 'atob', buffer.atob); -defineOperation(globalThis, 'btoa', buffer.btoa); - +// Lazy ones. +exposeLazyInterfaces(globalThis, 'internal/abort_controller', [ + 'AbortController', 'AbortSignal', +]); +exposeLazyInterfaces(globalThis, 'internal/event_target', [ + 'EventTarget', 'Event', +]); +exposeLazyInterfaces(globalThis, 'internal/worker/io', [ + 'MessageChannel', 'MessagePort', 'MessageEvent', +]); +defineLazyProperties(globalThis, 'buffer', ['atob', 'btoa']); // https://www.w3.org/TR/FileAPI/#dfn-Blob -exposeInterface(globalThis, 'Blob', buffer.Blob); - +exposeLazyInterfaces(globalThis, 'internal/blob', ['Blob']); // https://www.w3.org/TR/hr-time-2/#the-performance-attribute -const perf_hooks = require('perf_hooks'); -exposeInterface(globalThis, 'Performance', perf_hooks.Performance); -defineReplacableAttribute(globalThis, 'performance', - perf_hooks.performance); + +exposeLazyInterfaces(globalThis, 'perf_hooks', [ + 'Performance', +]); + +defineReplaceableLazyAttribute(globalThis, 'perf_hooks', ['performance']); + +// https://encoding.spec.whatwg.org/#textencoder +// https://encoding.spec.whatwg.org/#textdecoder +exposeLazyInterfaces(globalThis, + 'internal/encoding', + ['TextEncoder', 'TextDecoder']); function createGlobalConsole() { const consoleFromNode = @@ -115,67 +101,43 @@ function exposeGetterAndSetter(target, name, getter, setter = undefined) { }); } -// https://heycam.github.io/webidl/#Replaceable -function defineReplacableAttribute(target, name, value) { - ObjectDefineProperty(target, name, { - __proto__: null, - writable: true, - enumerable: true, - configurable: true, - value, - }); -} - // Web Streams API -const { - TransformStream, - TransformStreamDefaultController, -} = require('internal/webstreams/transformstream'); - -const { - WritableStream, - WritableStreamDefaultController, - WritableStreamDefaultWriter, -} = require('internal/webstreams/writablestream'); +exposeLazyInterfaces( + globalThis, + 'internal/webstreams/transformstream', + ['TransformStream', 'TransformStreamDefaultController']); -const { - ReadableStream, - ReadableStreamDefaultReader, - ReadableStreamBYOBReader, - ReadableStreamBYOBRequest, - ReadableByteStreamController, - ReadableStreamDefaultController, -} = require('internal/webstreams/readablestream'); +exposeLazyInterfaces( + globalThis, + 'internal/webstreams/writablestream', + ['WritableStream', 'WritableStreamDefaultController', 'WritableStreamDefaultWriter']); -const { - ByteLengthQueuingStrategy, - CountQueuingStrategy, -} = require('internal/webstreams/queuingstrategies'); +exposeLazyInterfaces( + globalThis, + 'internal/webstreams/readablestream', + [ + 'ReadableStream', 'ReadableStreamDefaultReader', + 'ReadableStreamBYOBReader', 'ReadableStreamBYOBRequest', + 'ReadableByteStreamController', 'ReadableStreamDefaultController', + ]); + +exposeLazyInterfaces( + globalThis, + 'internal/webstreams/queuingstrategies', + [ + 'ByteLengthQueuingStrategy', 'CountQueuingStrategy', + ]); -const { - TextEncoderStream, - TextDecoderStream, -} = require('internal/webstreams/encoding'); +exposeLazyInterfaces( + globalThis, + 'internal/webstreams/encoding', + [ + 'TextEncoderStream', 'TextDecoderStream', + ]); -const { - CompressionStream, - DecompressionStream, -} = require('internal/webstreams/compression'); - -exposeInterface(globalThis, 'ReadableStream', ReadableStream); -exposeInterface(globalThis, 'ReadableStreamDefaultReader', ReadableStreamDefaultReader); -exposeInterface(globalThis, 'ReadableStreamBYOBReader', ReadableStreamBYOBReader); -exposeInterface(globalThis, 'ReadableStreamBYOBRequest', ReadableStreamBYOBRequest); -exposeInterface(globalThis, 'ReadableByteStreamController', ReadableByteStreamController); -exposeInterface(globalThis, 'ReadableStreamDefaultController', ReadableStreamDefaultController); -exposeInterface(globalThis, 'TransformStream', TransformStream); -exposeInterface(globalThis, 'TransformStreamDefaultController', TransformStreamDefaultController); -exposeInterface(globalThis, 'WritableStream', WritableStream); -exposeInterface(globalThis, 'WritableStreamDefaultWriter', WritableStreamDefaultWriter); -exposeInterface(globalThis, 'WritableStreamDefaultController', WritableStreamDefaultController); -exposeInterface(globalThis, 'ByteLengthQueuingStrategy', ByteLengthQueuingStrategy); -exposeInterface(globalThis, 'CountQueuingStrategy', CountQueuingStrategy); -exposeInterface(globalThis, 'TextEncoderStream', TextEncoderStream); -exposeInterface(globalThis, 'TextDecoderStream', TextDecoderStream); -exposeInterface(globalThis, 'CompressionStream', CompressionStream); -exposeInterface(globalThis, 'DecompressionStream', DecompressionStream); +exposeLazyInterfaces( + globalThis, + 'internal/webstreams/compression', + [ + 'CompressionStream', 'DecompressionStream', + ]); diff --git a/graal-nodejs/lib/internal/bootstrap/node.js b/graal-nodejs/lib/internal/bootstrap/node.js index 92bfb0ee020..74afb7fdedb 100644 --- a/graal-nodejs/lib/internal/bootstrap/node.js +++ b/graal-nodejs/lib/internal/bootstrap/node.js @@ -71,7 +71,8 @@ const internalTimers = require('internal/timers'); const { defineOperation, deprecate, - exposeInterface, + defineLazyProperties, + exposeLazyInterfaces, } = require('internal/util'); const { privateSymbols: { @@ -241,20 +242,23 @@ const { } = require('internal/process/task_queues'); // Non-standard extensions: -const { BroadcastChannel } = require('internal/worker/io'); -exposeInterface(globalThis, 'BroadcastChannel', BroadcastChannel); - defineOperation(globalThis, 'queueMicrotask', queueMicrotask); const timers = require('timers'); defineOperation(globalThis, 'clearImmediate', timers.clearImmediate); defineOperation(globalThis, 'setImmediate', timers.setImmediate); -const { - structuredClone, -} = require('internal/structured_clone'); -defineOperation(globalThis, 'structuredClone', structuredClone); +defineLazyProperties( + globalThis, + 'internal/structured_clone', + ['structuredClone'], +); +exposeLazyInterfaces( + globalThis, + 'internal/worker/io', + ['BroadcastChannel'], +); // Set the per-Environment callback that will be called // when the TrackingTraceStateObserver updates trace state. // Note that when NODE_USE_V8_PLATFORM is true, the observer is @@ -366,16 +370,6 @@ process.emitWarning = emitWarning; // Note: only after this point are the timers effective } -// Preload modules so that they are included in the builtin snapshot. -require('fs'); -require('v8'); -require('vm'); -require('url'); -require('internal/options'); -if (config.hasOpenSSL) { - require('crypto'); -} - function setupPrepareStackTrace() { const { setEnhanceStackForFatalException, diff --git a/graal-nodejs/lib/internal/bootstrap/switches/is_main_thread.js b/graal-nodejs/lib/internal/bootstrap/switches/is_main_thread.js index b2f40d9c9b2..ace7dc1c904 100644 --- a/graal-nodejs/lib/internal/bootstrap/switches/is_main_thread.js +++ b/graal-nodejs/lib/internal/bootstrap/switches/is_main_thread.js @@ -3,9 +3,11 @@ const { ObjectDefineProperty } = primordials; const rawMethods = internalBinding('process_methods'); const { - addSerializeCallback, - isBuildingSnapshot, -} = require('v8').startupSnapshot; + namespace: { + addSerializeCallback, + isBuildingSnapshot, + }, +} = require('internal/v8/startup_snapshot'); // TODO(joyeecheung): deprecate and remove these underscore methods process._debugProcess = rawMethods._debugProcess; process._debugEnd = rawMethods._debugEnd; diff --git a/graal-nodejs/lib/internal/console/constructor.js b/graal-nodejs/lib/internal/console/constructor.js index 21e74f336c4..cbc9cd8d3a5 100644 --- a/graal-nodejs/lib/internal/console/constructor.js +++ b/graal-nodejs/lib/internal/console/constructor.js @@ -693,9 +693,11 @@ function initializeGlobalConsole(globalConsole) { globalConsole[kBindProperties](true, 'auto'); const { - addSerializeCallback, - isBuildingSnapshot, - } = require('v8').startupSnapshot; + namespace: { + addSerializeCallback, + isBuildingSnapshot, + }, + } = require('internal/v8/startup_snapshot'); if (!internalBinding('config').hasInspector || !isBuildingSnapshot()) { return; diff --git a/graal-nodejs/lib/internal/dns/utils.js b/graal-nodejs/lib/internal/dns/utils.js index 56b2b3930b2..10395ecb896 100644 --- a/graal-nodejs/lib/internal/dns/utils.js +++ b/graal-nodejs/lib/internal/dns/utils.js @@ -37,10 +37,12 @@ const { } = errors.codes; const { - addSerializeCallback, - addDeserializeCallback, - isBuildingSnapshot, -} = require('v8').startupSnapshot; + namespace: { + addSerializeCallback, + addDeserializeCallback, + isBuildingSnapshot, + }, +} = require('internal/v8/startup_snapshot'); function validateTimeout(options) { const { timeout = -1 } = { ...options }; diff --git a/graal-nodejs/lib/internal/errors.js b/graal-nodejs/lib/internal/errors.js index fbe04a1e096..7bc7998f918 100644 --- a/graal-nodejs/lib/internal/errors.js +++ b/graal-nodejs/lib/internal/errors.js @@ -205,7 +205,7 @@ function lazyBuffer() { function isErrorStackTraceLimitWritable() { // Do no touch Error.stackTraceLimit as V8 would attempt to install // it again during deserialization. - if (require('v8').startupSnapshot.isBuildingSnapshot()) { + if (require('internal/v8/startup_snapshot').namespace.isBuildingSnapshot()) { return false; } diff --git a/graal-nodejs/lib/internal/modules/cjs/loader.js b/graal-nodejs/lib/internal/modules/cjs/loader.js index 88bb870a8fd..cc0e17cb488 100644 --- a/graal-nodejs/lib/internal/modules/cjs/loader.js +++ b/graal-nodejs/lib/internal/modules/cjs/loader.js @@ -95,7 +95,6 @@ const internalFS = require('internal/fs/utils'); const path = require('path'); const { sep } = path; const { internalModuleStat } = internalBinding('fs'); -const packageJsonReader = require('internal/modules/package_json_reader'); const { safeGetenv } = internalBinding('credentials'); const { privateSymbols: { @@ -367,6 +366,7 @@ function readPackage(requestPath) { const existing = packageJsonCache.get(jsonPath); if (existing !== undefined) return existing; + const packageJsonReader = require('internal/modules/package_json_reader'); const result = packageJsonReader.read(jsonPath); const json = result.containsKeys === false ? '{}' : result.string; if (json === undefined) { diff --git a/graal-nodejs/lib/internal/modules/esm/get_format.js b/graal-nodejs/lib/internal/modules/esm/get_format.js index 219ef03a212..e9da560396d 100644 --- a/graal-nodejs/lib/internal/modules/esm/get_format.js +++ b/graal-nodejs/lib/internal/modules/esm/get_format.js @@ -8,7 +8,6 @@ const { } = primordials; const { basename, extname, relative } = require('path'); const { getOptionValue } = require('internal/options'); -const { fetchModule } = require('internal/modules/esm/fetch_module'); const { extensionFormatMap, getLegacyExtensionFormat, @@ -89,6 +88,7 @@ function getFileProtocolModuleFormat(url, context, ignoreErrors) { */ function getHttpProtocolModuleFormat(url, context) { if (experimentalNetworkImports) { + const { fetchModule } = require('internal/modules/esm/fetch_module'); return PromisePrototypeThen( PromiseResolve(fetchModule(url, context)), (entry) => { diff --git a/graal-nodejs/lib/internal/modules/esm/load.js b/graal-nodejs/lib/internal/modules/esm/load.js index 71a9f8da0b4..d71a81a4edc 100644 --- a/graal-nodejs/lib/internal/modules/esm/load.js +++ b/graal-nodejs/lib/internal/modules/esm/load.js @@ -9,7 +9,6 @@ const { const { defaultGetFormat } = require('internal/modules/esm/get_format'); const { validateAssertions } = require('internal/modules/esm/assert'); const { getOptionValue } = require('internal/options'); -const { fetchModule } = require('internal/modules/esm/fetch_module'); // Do not eagerly grab .manifest, it may be in TDZ const policy = getOptionValue('--experimental-policy') ? @@ -20,7 +19,6 @@ const experimentalNetworkImports = const { Buffer: { from: BufferFrom } } = require('buffer'); -const { readFile: readFileAsync } = require('internal/fs/promises').exports; const { URL } = require('internal/url'); const { ERR_INVALID_URL, @@ -39,6 +37,7 @@ async function getSource(url, context) { let responseURL = href; let source; if (protocol === 'file:') { + const { readFile: readFileAsync } = require('internal/fs/promises').exports; source = await readFileAsync(url); } else if (protocol === 'data:') { const match = RegExpPrototypeExec(DATA_URL_PATTERN, url.pathname); @@ -51,6 +50,7 @@ async function getSource(url, context) { protocol === 'https:' || protocol === 'http:' )) { + const { fetchModule } = require('internal/modules/esm/fetch_module'); const res = await fetchModule(url, context); source = await res.body; responseURL = res.resolvedHREF; diff --git a/graal-nodejs/lib/internal/modules/esm/loader.js b/graal-nodejs/lib/internal/modules/esm/loader.js index 32b34ee8478..64d07fb1468 100644 --- a/graal-nodejs/lib/internal/modules/esm/loader.js +++ b/graal-nodejs/lib/internal/modules/esm/loader.js @@ -20,7 +20,6 @@ const { StringPrototypeToUpperCase, globalThis, } = primordials; -const { MessageChannel } = require('internal/worker/io'); const { ERR_LOADER_CHAIN_INCOMPLETE, @@ -41,19 +40,20 @@ const { validateObject, validateString, } = require('internal/validators'); -const ModuleMap = require('internal/modules/esm/module_map'); -const ModuleJob = require('internal/modules/esm/module_job'); +function newModuleMap() { + const ModuleMap = require('internal/modules/esm/module_map'); + return new ModuleMap(); +} const { defaultResolve, DEFAULT_CONDITIONS, } = require('internal/modules/esm/resolve'); -const { - initializeImportMeta, -} = require('internal/modules/esm/initialize_import_meta'); -const { defaultLoad } = require('internal/modules/esm/load'); -const { translators } = require( - 'internal/modules/esm/translators'); + +function getTranslators() { + const { translators } = require('internal/modules/esm/translators'); + return translators; +} const { getOptionValue } = require('internal/options'); /** @@ -178,6 +178,7 @@ function nextHookFactory(chain, meta, { validateArgs, validateOutput }) { * Currently, this is a singleton -- there is only one used for loading * the main module and everything in its dependency graph. */ + class ESMLoader { #hooks = { /** @@ -194,7 +195,7 @@ class ESMLoader { */ load: [ { - fn: defaultLoad, + fn: require('internal/modules/esm/load').defaultLoad, url: 'node:internal/modules/esm/load', }, ], @@ -212,7 +213,7 @@ class ESMLoader { ], }; - #importMetaInitializer = initializeImportMeta; + #importMetaInitializer = require('internal/modules/esm/initialize_import_meta').initializeImportMeta; /** * Map of already-loaded CJS modules to use @@ -227,12 +228,12 @@ class ESMLoader { /** * Registry of loaded modules, akin to `require.cache` */ - moduleMap = new ModuleMap(); + moduleMap = newModuleMap(); /** * Methods which translate input code or other information into ES modules */ - translators = translators; + translators = getTranslators(); constructor() { if (getOptionValue('--experimental-loader').length > 0) { @@ -384,6 +385,7 @@ class ESMLoader { return module; }; + const ModuleJob = require('internal/modules/esm/module_job'); const job = new ModuleJob( this, url, undefined, evalInstance, false, false); this.moduleMap.set(url, undefined, job); @@ -459,7 +461,7 @@ class ESMLoader { importAssertions, }); - const translator = translators.get(finalFormat); + const translator = getTranslators().get(finalFormat); if (!translator) { throw new ERR_UNKNOWN_MODULE_FORMAT(finalFormat, responseURL); @@ -476,7 +478,7 @@ class ESMLoader { if (process.env.WATCH_REPORT_DEPENDENCIES && process.send) { process.send({ 'watch:import': [url] }); } - + const ModuleJob = require('internal/modules/esm/module_job'); const job = new ModuleJob( this, url, @@ -683,6 +685,7 @@ class ESMLoader { preload() { for (let i = this.#hooks.globalPreload.length - 1; i >= 0; i--) { + const { MessageChannel } = require('internal/worker/io'); const channel = new MessageChannel(); const { port1: insidePreload, diff --git a/graal-nodejs/lib/internal/modules/esm/module_map.js b/graal-nodejs/lib/internal/modules/esm/module_map.js index df02ebd7085..7280f052fef 100644 --- a/graal-nodejs/lib/internal/modules/esm/module_map.js +++ b/graal-nodejs/lib/internal/modules/esm/module_map.js @@ -1,6 +1,5 @@ 'use strict'; -const ModuleJob = require('internal/modules/esm/module_job'); const { kImplicitAssertType } = require('internal/modules/esm/assert'); const { ObjectCreate, @@ -23,6 +22,8 @@ class ModuleMap extends SafeMap { set(url, type = kImplicitAssertType, job) { validateString(url, 'url'); validateString(type, 'type'); + + const ModuleJob = require('internal/modules/esm/module_job'); if (job instanceof ModuleJob !== true && typeof job !== 'function') { throw new ERR_INVALID_ARG_TYPE('job', 'ModuleJob', job); diff --git a/graal-nodejs/lib/internal/modules/esm/package_config.js b/graal-nodejs/lib/internal/modules/esm/package_config.js index 1327bf1470d..dc3c37f6042 100644 --- a/graal-nodejs/lib/internal/modules/esm/package_config.js +++ b/graal-nodejs/lib/internal/modules/esm/package_config.js @@ -11,7 +11,6 @@ const { ERR_INVALID_PACKAGE_CONFIG, } = require('internal/errors').codes; -const packageJsonReader = require('internal/modules/package_json_reader'); const { filterOwnProperties } = require('internal/util'); @@ -42,6 +41,7 @@ function getPackageConfig(path, specifier, base) { if (existing !== undefined) { return existing; } + const packageJsonReader = require('internal/modules/package_json_reader'); const source = packageJsonReader.read(path).string; if (source === undefined) { const packageConfig = { diff --git a/graal-nodejs/lib/internal/modules/esm/translators.js b/graal-nodejs/lib/internal/modules/esm/translators.js index a425749e82a..f42674f21f2 100644 --- a/graal-nodejs/lib/internal/modules/esm/translators.js +++ b/graal-nodejs/lib/internal/modules/esm/translators.js @@ -36,8 +36,6 @@ const { cjsParseCache, } = require('internal/modules/cjs/loader'); const internalURLModule = require('internal/url'); -const createDynamicModule = require( - 'internal/modules/esm/create_dynamic_module'); const { fileURLToPath, URL } = require('url'); let debug = require('internal/util/debuglog').debuglog('esm', (fn) => { debug = fn; @@ -52,7 +50,6 @@ const moduleWrap = internalBinding('module_wrap'); const { ModuleWrap } = moduleWrap; const asyncESM = require('internal/process/esm_loader'); const { emitWarningSync } = require('internal/process/warning'); -const { TextDecoder } = require('internal/encoding'); let cjsParse; async function initCJSParse() { @@ -94,6 +91,7 @@ function assertBufferSource(body, allowString, hookName) { function stringify(body) { if (typeof body === 'string') return body; assertBufferSource(body, false, 'transformSource'); + const { TextDecoder } = require('internal/encoding'); DECODER = DECODER === null ? new TextDecoder() : DECODER; return DECODER.decode(body); } @@ -341,6 +339,8 @@ translators.set('wasm', async function(url, source) { ArrayPrototypeMap(WebAssembly.Module.exports(compiled), ({ name }) => name); + const createDynamicModule = require( + 'internal/modules/esm/create_dynamic_module'); return createDynamicModule(imports, exports, url, (reflect) => { const { exports } = new WebAssembly.Instance(compiled, reflect.imports); for (const expt of ObjectKeys(exports)) diff --git a/graal-nodejs/lib/internal/modules/run_main.js b/graal-nodejs/lib/internal/modules/run_main.js index daaa153516c..e4b971d0c4e 100644 --- a/graal-nodejs/lib/internal/modules/run_main.js +++ b/graal-nodejs/lib/internal/modules/run_main.js @@ -8,9 +8,6 @@ const CJSLoader = require('internal/modules/cjs/loader'); const { Module, toRealPath, readPackageScope } = CJSLoader; const { getOptionValue } = require('internal/options'); const path = require('path'); -const { - handleProcessExit, -} = require('internal/modules/esm/handle_process_exit'); function resolveMainPath(main) { // Note extension resolution for the main entry point can be deprecated in a @@ -65,6 +62,9 @@ function runMainESM(mainPath) { } async function handleMainPromise(promise) { + const { + handleProcessExit, + } = require('internal/modules/esm/handle_process_exit'); process.on('exit', handleProcessExit); try { return await promise; diff --git a/graal-nodejs/lib/internal/process/per_thread.js b/graal-nodejs/lib/internal/process/per_thread.js index d3f9ff479cd..22362e420b2 100644 --- a/graal-nodejs/lib/internal/process/per_thread.js +++ b/graal-nodejs/lib/internal/process/per_thread.js @@ -49,10 +49,6 @@ const { } = require('internal/validators'); const constants = internalBinding('constants').os.signals; -const { - handleProcessExit, -} = require('internal/modules/esm/handle_process_exit'); - const kInternal = Symbol('internal properties'); function assert(x, msg) { @@ -182,6 +178,9 @@ function wrapProcessMethods(binding) { memoryUsage.rss = rss; function exit(code) { + const { + handleProcessExit, + } = require('internal/modules/esm/handle_process_exit'); process.off('exit', handleProcessExit); if (code || code === 0) diff --git a/graal-nodejs/lib/internal/process/pre_execution.js b/graal-nodejs/lib/internal/process/pre_execution.js index 3b44634b9d5..086750369f8 100644 --- a/graal-nodejs/lib/internal/process/pre_execution.js +++ b/graal-nodejs/lib/internal/process/pre_execution.js @@ -4,7 +4,6 @@ const { NumberParseInt, ObjectDefineProperties, ObjectDefineProperty, - ObjectGetOwnPropertyDescriptor, SafeMap, SafeWeakMap, StringPrototypeStartsWith, @@ -23,6 +22,8 @@ const { reconnectZeroFillToggle } = require('internal/buffer'); const { defineOperation, exposeInterface, + exposeLazyInterfaces, + defineReplaceableLazyAttribute, setupCoverageHooks, } = require('internal/util'); @@ -30,11 +31,12 @@ const { ERR_MANIFEST_ASSERT_INTEGRITY, } = require('internal/errors').codes; const assert = require('internal/assert'); - const { - addSerializeCallback, - isBuildingSnapshot, -} = require('v8').startupSnapshot; + namespace: { + addSerializeCallback, + isBuildingSnapshot, + }, +} = require('internal/v8/startup_snapshot'); function prepareMainThreadExecution(expandArgv1 = false, initializeModules = true) { prepareExecution({ @@ -286,19 +288,14 @@ function setupWebCrypto() { return; } - let webcrypto; - ObjectDefineProperty(globalThis, 'crypto', - { __proto__: null, ...ObjectGetOwnPropertyDescriptor({ - get crypto() { - webcrypto ??= require('internal/crypto/webcrypto'); - return webcrypto.crypto; - }, - }, 'crypto') }); if (internalBinding('config').hasOpenSSL) { - webcrypto ??= require('internal/crypto/webcrypto'); - exposeInterface(globalThis, 'Crypto', webcrypto.Crypto); - exposeInterface(globalThis, 'CryptoKey', webcrypto.CryptoKey); - exposeInterface(globalThis, 'SubtleCrypto', webcrypto.SubtleCrypto); + defineReplaceableLazyAttribute( + globalThis, 'internal/crypto/webcrypto', ['crypto'], false, + ); + exposeLazyInterfaces( + globalThis, 'internal/crypto/webcrypto', + ['Crypto', 'CryptoKey', 'SubtleCrypto'], + ); } } diff --git a/graal-nodejs/lib/internal/source_map/source_map_cache.js b/graal-nodejs/lib/internal/source_map/source_map_cache.js index f51f97d4420..f8367608b13 100644 --- a/graal-nodejs/lib/internal/source_map/source_map_cache.js +++ b/graal-nodejs/lib/internal/source_map/source_map_cache.js @@ -23,7 +23,6 @@ const { Buffer } = require('buffer'); let debug = require('internal/util/debuglog').debuglog('source_map', (fn) => { debug = fn; }); -const fs = require('fs'); const { getOptionValue } = require('internal/options'); const { IterableWeakMap } = require('internal/util/iterable_weak_map'); const { @@ -222,6 +221,7 @@ function lineLengths(content) { function sourceMapFromFile(mapURL) { try { + const fs = require('fs'); const content = fs.readFileSync(fileURLToPath(mapURL), 'utf8'); const data = JSONParse(content); return sourcesToAbsolute(mapURL, data); diff --git a/graal-nodejs/lib/internal/util.js b/graal-nodejs/lib/internal/util.js index 2a366f34540..4d72783617d 100644 --- a/graal-nodejs/lib/internal/util.js +++ b/graal-nodejs/lib/internal/util.js @@ -509,6 +509,90 @@ function exposeInterface(target, name, interfaceObject) { }); } +function defineLazyProperties(target, id, keys, enumerable = true) { + const descriptors = { __proto__: null }; + let mod; + for (let i = 0; i < keys.length; i++) { + const key = keys[i]; + let lazyLoadedValue; + function set(value) { + ObjectDefineProperty(target, key, { + __proto__: null, + writable: true, + value, + }); + } + ObjectDefineProperty(set, 'name', { + __proto__: null, + value: `set ${key}`, + }); + function get() { + mod ??= require(id); + if (lazyLoadedValue === undefined) { + lazyLoadedValue = mod[key]; + set(lazyLoadedValue); + } + return lazyLoadedValue; + } + ObjectDefineProperty(get, 'name', { + __proto__: null, + value: `get ${key}`, + }); + descriptors[key] = { + __proto__: null, + configurable: true, + enumerable, + get, + set, + }; + } + ObjectDefineProperties(target, descriptors); +} + +function defineReplaceableLazyAttribute(target, id, keys, writable = true) { + let mod; + for (let i = 0; i < keys.length; i++) { + const key = keys[i]; + let value; + let setterCalled = false; + + function get() { + if (setterCalled) { + return value; + } + mod ??= require(id); + value ??= mod[key]; + return value; + } + + ObjectDefineProperty(get, 'name', { + __proto__: null, + value: `get ${key}`, + }); + + function set(val) { + setterCalled = true; + value = val; + } + ObjectDefineProperty(set, 'name', { + __proto__: null, + value: `set ${key}`, + }); + + ObjectDefineProperty(target, key, { + __proto__: null, + enumerable: true, + configurable: true, + get, + set: writable ? set : undefined, + }); + } +} + +function exposeLazyInterfaces(target, id, keys) { + defineLazyProperties(target, id, keys, false); +} + let _DOMException; const lazyDOMExceptionClass = () => { _DOMException ??= internalBinding('messaging').DOMException; @@ -636,9 +720,12 @@ module.exports = { createDeferredPromise, decorateErrorStack, defineOperation, + defineLazyProperties, + defineReplaceableLazyAttribute, deprecate, emitExperimentalWarning, exposeInterface, + exposeLazyInterfaces, filterDuplicateStrings, filterOwnProperties, getConstructorOf, diff --git a/graal-nodejs/lib/util.js b/graal-nodejs/lib/util.js index f71089c6b4e..7fb7994e653 100644 --- a/graal-nodejs/lib/util.js +++ b/graal-nodejs/lib/util.js @@ -62,13 +62,10 @@ const { stripVTControlCharacters, } = require('internal/util/inspect'); const { debuglog } = require('internal/util/debuglog'); -const { parseArgs } = require('internal/util/parse_args/parse_args'); const { validateFunction, validateNumber, } = require('internal/validators'); -const { TextDecoder, TextEncoder } = require('internal/encoding'); -const { MIMEType, MIMEParams } = require('internal/mime'); const { isBuffer } = require('buffer').Buffer; const types = require('internal/util/types'); @@ -78,6 +75,7 @@ const { getSystemErrorName: internalErrorName, promisify, toUSVString, + defineLazyProperties, } = require('internal/util'); let abortController; @@ -385,14 +383,9 @@ module.exports = { isFunction, isPrimitive, log, - MIMEType, - MIMEParams, - parseArgs, promisify, stripVTControlCharacters, toUSVString, - TextDecoder, - TextEncoder, get transferableAbortSignal() { return lazyAbortController().transferableAbortSignal; }, @@ -404,3 +397,21 @@ module.exports = { }, types, }; + +defineLazyProperties( + module.exports, + 'internal/util/parse_args/parse_args', + ['parseArgs'], +); + +defineLazyProperties( + module.exports, + 'internal/encoding', + ['TextDecoder', 'TextEncoder'], +); + +defineLazyProperties( + module.exports, + 'internal/mime', + ['MIMEType', 'MIMEParams'], +); diff --git a/graal-nodejs/test/common/wpt.js b/graal-nodejs/test/common/wpt.js index 048850a5be3..f33911ea784 100644 --- a/graal-nodejs/test/common/wpt.js +++ b/graal-nodejs/test/common/wpt.js @@ -487,6 +487,7 @@ class WPTRunner { this.globalThisInitScripts.push( `global.Window = Object.getPrototypeOf(globalThis).constructor; self.GLOBAL.isWorker = () => false;`); + this.loadLazyGlobals(); break; } @@ -500,6 +501,31 @@ class WPTRunner { } } + loadLazyGlobals() { + const lazyProperties = [ + 'Performance', 'PerformanceEntry', 'PerformanceMark', 'PerformanceMeasure', + 'PerformanceObserver', 'PerformanceObserverEntryList', 'PerformanceResourceTiming', + 'Blob', 'atob', 'btoa', + 'MessageChannel', 'MessagePort', 'MessageEvent', + 'EventTarget', 'Event', + 'AbortController', 'AbortSignal', + 'performance', + 'TransformStream', 'TransformStreamDefaultController', + 'WritableStream', 'WritableStreamDefaultController', 'WritableStreamDefaultWriter', + 'ReadableStream', 'ReadableStreamDefaultReader', + 'ReadableStreamBYOBReader', 'ReadableStreamBYOBRequest', + 'ReadableByteStreamController', 'ReadableStreamDefaultController', + 'ByteLengthQueuingStrategy', 'CountQueuingStrategy', + 'TextEncoderStream', 'TextDecoderStream', + 'CompressionStream', 'DecompressionStream', + ]; + if (Boolean(process.versions.openssl) && !process.env.NODE_SKIP_CRYPTO) { + lazyProperties.push('crypto'); + } + const script = lazyProperties.map((name) => `globalThis.${name};`).join('\n'); + this.globalThisInitScripts.push(script); + } + // TODO(joyeecheung): work with the upstream to port more tests in .html // to .js. async runJsTests() { diff --git a/graal-nodejs/test/parallel/test-bootstrap-modules.js b/graal-nodejs/test/parallel/test-bootstrap-modules.js index 396d5673c19..2b98ffa8c19 100644 --- a/graal-nodejs/test/parallel/test-bootstrap-modules.js +++ b/graal-nodejs/test/parallel/test-bootstrap-modules.js @@ -10,7 +10,6 @@ const assert = require('assert'); const expectedModules = new Set([ 'Internal Binding async_wrap', - 'Internal Binding block_list', 'Internal Binding buffer', 'Internal Binding builtins', 'Internal Binding config', @@ -18,74 +17,48 @@ const expectedModules = new Set([ 'Internal Binding contextify', 'Internal Binding credentials', 'Internal Binding errors', - 'Internal Binding fs_dir', - 'Internal Binding fs_event_wrap', 'Internal Binding fs', - 'Internal Binding heap_utils', 'Internal Binding mksnapshot', 'Internal Binding messaging', 'Internal Binding module_wrap', 'Internal Binding options', 'Internal Binding performance', - 'Internal Binding pipe_wrap', 'Internal Binding process_methods', 'Internal Binding report', - 'Internal Binding serdes', - 'Internal Binding stream_wrap', 'Internal Binding string_decoder', 'Internal Binding symbols', 'Internal Binding task_queue', - 'Internal Binding tcp_wrap', 'Internal Binding timers', 'Internal Binding trace_events', 'Internal Binding types', 'Internal Binding url', 'Internal Binding util', - 'Internal Binding uv', - 'Internal Binding v8', 'Internal Binding wasm_web_api', 'Internal Binding worker', 'NativeModule buffer', 'NativeModule events', 'NativeModule fs', - 'NativeModule internal/abort_controller', 'NativeModule internal/assert', 'NativeModule internal/async_hooks', - 'NativeModule internal/blocklist', 'NativeModule internal/buffer', 'NativeModule internal/console/constructor', 'NativeModule internal/console/global', 'NativeModule internal/constants', - 'NativeModule internal/dtrace', 'NativeModule internal/dns/utils', - 'NativeModule internal/encoding', 'NativeModule internal/errors', - 'NativeModule internal/event_target', 'NativeModule internal/fixed_queue', - 'NativeModule internal/fs/dir', - 'NativeModule internal/fs/promises', - 'NativeModule internal/fs/read_file_context', - 'NativeModule internal/fs/rimraf', 'NativeModule internal/fs/utils', - 'NativeModule internal/fs/watchers', 'NativeModule internal/graal/buffer', - 'NativeModule internal/heap_utils', - 'NativeModule internal/histogram', 'NativeModule internal/idna', 'NativeModule internal/linkedlist', - 'NativeModule internal/mime', 'NativeModule internal/modules/cjs/helpers', 'NativeModule internal/modules/cjs/loader', 'NativeModule internal/modules/esm/assert', - 'NativeModule internal/modules/esm/create_dynamic_module', - 'NativeModule internal/modules/esm/fetch_module', 'NativeModule internal/modules/esm/formats', 'NativeModule internal/modules/esm/get_format', - 'NativeModule internal/modules/esm/handle_process_exit', 'NativeModule internal/modules/esm/initialize_import_meta', 'NativeModule internal/modules/esm/load', 'NativeModule internal/modules/esm/loader', - 'NativeModule internal/modules/esm/module_job', 'NativeModule internal/modules/esm/module_map', 'NativeModule internal/modules/esm/package_config', 'NativeModule internal/modules/esm/resolve', @@ -94,15 +67,6 @@ const expectedModules = new Set([ 'NativeModule internal/modules/run_main', 'NativeModule internal/net', 'NativeModule internal/options', - 'NativeModule internal/perf/event_loop_delay', - 'NativeModule internal/perf/event_loop_utilization', - 'NativeModule internal/perf/nodetiming', - 'NativeModule internal/perf/observe', - 'NativeModule internal/perf/performance_entry', - 'NativeModule internal/perf/performance', - 'NativeModule internal/perf/timerify', - 'NativeModule internal/perf/usertiming', - 'NativeModule internal/perf/resource_timing', 'NativeModule internal/perf/utils', 'NativeModule internal/priority_queue', 'NativeModule internal/process/esm_loader', @@ -114,70 +78,26 @@ const expectedModules = new Set([ 'NativeModule internal/process/signal', 'NativeModule internal/process/task_queues', 'NativeModule internal/process/warning', - 'NativeModule internal/promise_hooks', 'NativeModule internal/querystring', - 'NativeModule internal/readline/callbacks', - 'NativeModule internal/readline/interface', - 'NativeModule internal/readline/utils', - 'NativeModule internal/socketaddress', 'NativeModule internal/source_map/source_map_cache', - 'NativeModule internal/stream_base_commons', - 'NativeModule internal/streams/add-abort-signal', - 'NativeModule internal/streams/buffer_list', - 'NativeModule internal/streams/compose', - 'NativeModule internal/streams/destroy', - 'NativeModule internal/streams/duplex', - 'NativeModule internal/streams/end-of-stream', - 'NativeModule internal/streams/from', - 'NativeModule internal/streams/legacy', - 'NativeModule internal/streams/operators', - 'NativeModule internal/streams/passthrough', - 'NativeModule internal/streams/pipeline', - 'NativeModule internal/streams/readable', - 'NativeModule internal/streams/state', - 'NativeModule internal/streams/transform', - 'NativeModule internal/streams/utils', - 'NativeModule internal/streams/writable', - 'NativeModule internal/structured_clone', 'NativeModule internal/timers', 'NativeModule internal/url', 'NativeModule internal/util', 'NativeModule internal/util/debuglog', 'NativeModule internal/util/inspect', 'NativeModule internal/util/iterable_weak_map', - 'NativeModule internal/util/parse_args/utils', - 'NativeModule internal/util/parse_args/parse_args', 'NativeModule internal/util/types', 'NativeModule internal/validators', 'NativeModule internal/vm', 'NativeModule internal/vm/module', 'NativeModule internal/wasm_web_api', - 'NativeModule internal/webidl', - 'NativeModule internal/webstreams/adapters', - 'NativeModule internal/webstreams/compression', - 'NativeModule internal/webstreams/encoding', - 'NativeModule internal/webstreams/queuingstrategies', - 'NativeModule internal/webstreams/readablestream', - 'NativeModule internal/webstreams/transformstream', - 'NativeModule internal/webstreams/util', - 'NativeModule internal/webstreams/writablestream', - 'NativeModule internal/worker/io', 'NativeModule internal/worker/js_transferable', 'Internal Binding blob', - 'NativeModule internal/blob', - 'NativeModule internal/file', 'NativeModule async_hooks', - 'NativeModule net', 'NativeModule path', - 'NativeModule perf_hooks', 'NativeModule querystring', - 'NativeModule stream', - 'NativeModule stream/promises', - 'NativeModule string_decoder', 'NativeModule timers', 'NativeModule url', - 'NativeModule util', - 'NativeModule v8', 'NativeModule internal/v8/startup_snapshot', 'NativeModule vm', ]); @@ -188,23 +108,33 @@ if (!common.isMainThread) { 'Internal Binding performance', 'Internal Binding symbols', 'Internal Binding worker', - 'NativeModule internal/streams/duplex', - 'NativeModule internal/streams/passthrough', - 'NativeModule internal/streams/readable', - 'NativeModule internal/streams/transform', - 'NativeModule internal/streams/writable', + 'NativeModule internal/abort_controller', 'NativeModule internal/error_serdes', + 'NativeModule internal/event_target', 'NativeModule internal/process/worker_thread_only', + 'NativeModule internal/streams/add-abort-signal', 'NativeModule internal/streams/buffer_list', + 'NativeModule internal/streams/compose', 'NativeModule internal/streams/destroy', + 'NativeModule internal/streams/duplex', 'NativeModule internal/streams/end-of-stream', + 'NativeModule internal/streams/from', 'NativeModule internal/streams/legacy', + 'NativeModule internal/streams/operators', + 'NativeModule internal/streams/passthrough', 'NativeModule internal/streams/pipeline', + 'NativeModule internal/streams/readable', 'NativeModule internal/streams/state', + 'NativeModule internal/streams/transform', + 'NativeModule internal/streams/utils', + 'NativeModule internal/streams/writable', 'NativeModule internal/worker', 'NativeModule internal/worker/io', - 'NativeModule stream', 'NativeModule worker_threads', + 'NativeModule stream', + 'NativeModule stream/promises', + 'NativeModule string_decoder', + 'NativeModule util', ].forEach(expectedModules.add.bind(expectedModules)); } @@ -218,36 +148,6 @@ if (process.features.inspector) { expectedModules.add('Internal Binding inspector'); expectedModules.add('NativeModule internal/inspector_async_hook'); expectedModules.add('NativeModule internal/util/inspector'); - expectedModules.add('Internal Binding profiler'); -} - -if (process.env.NODE_V8_COVERAGE) { - expectedModules.add('Internal Binding profiler'); -} - -if (common.hasCrypto) { - expectedModules.add('Internal Binding crypto'); - expectedModules.add('NativeModule crypto'); - expectedModules.add('NativeModule internal/crypto/certificate'); - expectedModules.add('NativeModule internal/crypto/cipher'); - expectedModules.add('NativeModule internal/crypto/diffiehellman'); - expectedModules.add('NativeModule internal/crypto/hash'); - expectedModules.add('NativeModule internal/crypto/hashnames'); - expectedModules.add('NativeModule internal/crypto/hkdf'); - expectedModules.add('NativeModule internal/crypto/keygen'); - expectedModules.add('NativeModule internal/crypto/keys'); - expectedModules.add('NativeModule internal/crypto/pbkdf2'); - expectedModules.add('NativeModule internal/crypto/random'); - expectedModules.add('NativeModule internal/crypto/scrypt'); - expectedModules.add('NativeModule internal/crypto/sig'); - expectedModules.add('NativeModule internal/crypto/util'); - expectedModules.add('NativeModule internal/crypto/x509'); - expectedModules.add('NativeModule internal/streams/lazy_transform'); -} - -const { internalBinding } = require('internal/test/binding'); -if (internalBinding('config').hasDtrace) { - expectedModules.add('Internal Binding dtrace'); } const difference = (setA, setB) => { diff --git a/graal-nodejs/test/wpt/test-streams.js b/graal-nodejs/test/wpt/test-streams.js index 6aa435303ab..faf906efb2e 100644 --- a/graal-nodejs/test/wpt/test-streams.js +++ b/graal-nodejs/test/wpt/test-streams.js @@ -5,6 +5,7 @@ const { WPTRunner } = require('../common/wpt'); const runner = new WPTRunner('streams'); // Set a script that will be executed in the worker before running the tests. +runner.pretendGlobalThisAs('Window'); runner.setInitScript(` // Simulate global postMessage for enqueue-with-detached-buffer.window.js function postMessage(value, origin, transferList) {