diff --git a/doc/api/errors.md b/doc/api/errors.md index 88d982fd195394..c5041c796be011 100644 --- a/doc/api/errors.md +++ b/doc/api/errors.md @@ -1957,6 +1957,11 @@ An attempt was made to load a module with an unknown or unsupported format. An invalid or unknown process signal was passed to an API expecting a valid signal (such as [`subprocess.kill()`][]). + +### `ERR_UNSUPPORTED_ESM_URL_SCHEME` + +`import` with URL schemes other than `file` and `data` is unsupported. + ### `ERR_V8BREAKITERATOR` diff --git a/lib/internal/errors.js b/lib/internal/errors.js index a0f3f26d5d34be..a0b7ee38cf1963 100644 --- a/lib/internal/errors.js +++ b/lib/internal/errors.js @@ -1335,6 +1335,8 @@ E('ERR_UNKNOWN_FILE_EXTENSION', TypeError); E('ERR_UNKNOWN_MODULE_FORMAT', 'Unknown module format: %s', RangeError); E('ERR_UNKNOWN_SIGNAL', 'Unknown signal: %s', TypeError); +E('ERR_UNSUPPORTED_ESM_URL_SCHEME', 'Only file and data URLs are supported ' + + 'by the default ESM loader', Error); E('ERR_V8BREAKITERATOR', 'Full ICU data not installed. See https://github.com/nodejs/node/wiki/Intl', diff --git a/lib/internal/modules/esm/default_resolve.js b/lib/internal/modules/esm/default_resolve.js index 86e682d28e329f..749f6861cd3ee6 100644 --- a/lib/internal/modules/esm/default_resolve.js +++ b/lib/internal/modules/esm/default_resolve.js @@ -21,7 +21,8 @@ const { resolve: moduleWrapResolve, getPackageType } = internalBinding('module_wrap'); const { URL, pathToFileURL, fileURLToPath } = require('internal/url'); const { ERR_INPUT_TYPE_NOT_ALLOWED, - ERR_UNKNOWN_FILE_EXTENSION } = require('internal/errors').codes; + ERR_UNKNOWN_FILE_EXTENSION, + ERR_UNSUPPORTED_ESM_URL_SCHEME } = require('internal/errors').codes; const realpathCache = new SafeMap(); @@ -52,8 +53,9 @@ if (experimentalJsonModules) extensionFormatMap['.json'] = legacyExtensionFormatMap['.json'] = 'json'; function resolve(specifier, parentURL) { + let parsed; try { - const parsed = new URL(specifier); + parsed = new URL(specifier); if (parsed.protocol === 'data:') { const [ , mime ] = /^([^/]+\/[^;,]+)(?:[^,]*?)(;base64)?,/.exec(parsed.pathname) || [ null, null, null ]; const format = ({ @@ -68,6 +70,8 @@ function resolve(specifier, parentURL) { }; } } catch {} + if (parsed && parsed.protocol !== 'file:' && parsed.protocol !== 'data:') + throw new ERR_UNSUPPORTED_ESM_URL_SCHEME(); if (NativeModule.canBeRequiredByUsers(specifier)) { return { url: specifier, diff --git a/test/es-module/test-esm-dynamic-import.js b/test/es-module/test-esm-dynamic-import.js index 1fc22f1d98934a..e60309fe886cf9 100644 --- a/test/es-module/test-esm-dynamic-import.js +++ b/test/es-module/test-esm-dynamic-import.js @@ -15,8 +15,8 @@ function expectErrorProperty(result, propertyKey, value) { })); } -function expectMissingModuleError(result) { - expectErrorProperty(result, 'code', 'ERR_MODULE_NOT_FOUND'); +function expectModuleError(result, err) { + expectErrorProperty(result, 'code', err); } function expectOkNamespace(result) { @@ -56,10 +56,10 @@ function expectFsNamespace(result) { expectFsNamespace(eval('import("fs")')); expectFsNamespace(eval('import("fs")')); - expectMissingModuleError(import('./not-an-existing-module.mjs')); - // TODO(jkrems): Right now this doesn't hit a protocol error because the - // module resolution step already rejects it. These arguably should be - // protocol errors. - expectMissingModuleError(import('node:fs')); - expectMissingModuleError(import('http://example.com/foo.js')); + expectModuleError(import('./not-an-existing-module.mjs'), + 'ERR_MODULE_NOT_FOUND'); + expectModuleError(import('node:fs'), + 'ERR_UNSUPPORTED_ESM_URL_SCHEME'); + expectModuleError(import('http://example.com/foo.js'), + 'ERR_UNSUPPORTED_ESM_URL_SCHEME'); })();