Skip to content
This repository was archived by the owner on Apr 16, 2020. It is now read-only.

Commit bf69a15

Browse files
guybedfordMylesBorins
authored andcommitted
esm: minimal resolver spec and implementation refinements
PR URL: #12
1 parent 22f6054 commit bf69a15

9 files changed

+118
-169
lines changed

doc/api/errors.md

+3-10
Original file line numberDiff line numberDiff line change
@@ -1414,26 +1414,19 @@ a `dynamicInstantiate` hook.
14141414
A `MessagePort` was found in the object passed to a `postMessage()` call,
14151415
but not provided in the `transferList` for that call.
14161416

1417-
<a id="ERR_MISSING_MODULE"></a>
1418-
### ERR_MISSING_MODULE
1419-
1420-
> Stability: 1 - Experimental
1421-
1422-
An [ES6 module][] could not be resolved.
1423-
14241417
<a id="ERR_MISSING_PLATFORM_FOR_WORKER"></a>
14251418
### ERR_MISSING_PLATFORM_FOR_WORKER
14261419

14271420
The V8 platform used by this instance of Node.js does not support creating
14281421
Workers. This is caused by lack of embedder support for Workers. In particular,
14291422
this error will not occur with standard builds of Node.js.
14301423

1431-
<a id="ERR_MODULE_RESOLUTION_LEGACY"></a>
1432-
### ERR_MODULE_RESOLUTION_LEGACY
1424+
<a id="ERR_MODULE_NOT_FOUND"></a>
1425+
### ERR_MODULE_NOT_FOUND
14331426

14341427
> Stability: 1 - Experimental
14351428
1436-
A failure occurred resolving imports in an [ES6 module][].
1429+
An [ESM module][] could not be resolved.
14371430

14381431
<a id="ERR_MULTIPLE_CALLBACK"></a>
14391432
### ERR_MULTIPLE_CALLBACK

doc/api/esm.md

+46-4
Original file line numberDiff line numberDiff line change
@@ -63,10 +63,6 @@ ESM must have the `.mjs` extension.
6363

6464
You must provide a file extension when using the `import` keyword.
6565

66-
### No importing directories
67-
68-
There is no support for importing directories.
69-
7066
### No NODE_PATH
7167

7268
`NODE_PATH` is not part of resolving `import` specifiers. Please use symlinks
@@ -146,6 +142,52 @@ fs.readFileSync = () => Buffer.from('Hello, ESM');
146142
fs.readFileSync === readFileSync;
147143
```
148144
145+
## Resolution Algorithm
146+
147+
### Features
148+
149+
The resolver has the following properties:
150+
151+
* FileURL-based resolution as is used by ES modules
152+
* Support for builtin module loading
153+
* Relative and absolute URL resolution
154+
* No default extensions
155+
* No folder mains
156+
* Bare specifier package resolution lookup through node_modules
157+
158+
### Resolver Algorithm
159+
160+
The algorithm to resolve an ES module specifier is provided through
161+
_ESM_RESOLVE_:
162+
163+
**ESM_RESOLVE**(_specifier_, _parentURL_)
164+
> 1. Let _resolvedURL_ be _undefined_.
165+
> 1. If _specifier_ is a valid URL then,
166+
> 1. Set _resolvedURL_ to the result of parsing and reserializing
167+
> _specifier_ as a URL.
168+
> 1. Otherwise, if _specifier_ starts with _"/"_, _"./"_ or _"../"_ then,
169+
> 1. Set _resolvedURL_ to the URL resolution of _specifier_ relative to
170+
> _parentURL_.
171+
> 1. Otherwise,
172+
> 1. Note: _specifier_ is now a bare specifier.
173+
> 1. Set _resolvedURL_ the result of
174+
> **PACKAGE_RESOLVE**(_specifier_, _parentURL_).
175+
> 1. If the file at _resolvedURL_ does not exist then,
176+
> 1. Throw a _Module Not Found_ error.
177+
> 1. Return _resolvedURL_.
178+
179+
**PACKAGE_RESOLVE**(_packageSpecifier_, _parentURL_)
180+
> 1. Assert: _packageSpecifier_ is a bare specifier.
181+
> 1. If _packageSpecifier_ is a Node.js builtin module then,
182+
> 1. Return the string _"node:"_ concatenated with _packageSpecifier_.
183+
> 1. While _parentURL_ contains a non-empty _pathname_,
184+
> 1. Set _parentURL_ to the parent folder URL of _parentURL_.
185+
> 1. Let _packageURL_ be the URL resolution of the string concatenation of
186+
> _parentURL_, _"/node_modules/"_ and _"packageSpecifier"_.
187+
> 1. If the file at _packageURL_ exists then,
188+
> 1. Return _packageURL_.
189+
> 1. Throw a _Module Not Found_ error.
190+
149191
[Node.js EP for ES Modules]: https://github.com/nodejs/node-eps/blob/master/002-es-modules.md
150192
[`module.createRequireFromPath()`]: modules.html#modules_module_createrequirefrompath_filename
151193
[ESM Minimal Kernel]: https://github.com/nodejs/modules/blob/master/doc/plan-for-new-modules-implementation.md

lib/internal/errors.js

+10-6
Original file line numberDiff line numberDiff line change
@@ -842,11 +842,13 @@ E('ERR_MISSING_ARGS',
842842
E('ERR_MISSING_DYNAMIC_INSTANTIATE_HOOK',
843843
'The ES Module loader may not return a format of \'dynamic\' when no ' +
844844
'dynamicInstantiate function was provided', Error);
845-
E('ERR_MISSING_MODULE', 'Cannot find module %s', Error);
846-
E('ERR_MODULE_RESOLUTION_LEGACY',
847-
'%s not found by import in %s.' +
848-
' Legacy behavior in require() would have found it at %s',
849-
Error);
845+
E('ERR_MODULE_NOT_FOUND', (module, base, legacyResolution) => {
846+
let msg = `Cannot find module '${module}' imported from ${base}.`;
847+
if (legacyResolution)
848+
msg += ' Legacy behavior in require() would have found it at ' +
849+
legacyResolution;
850+
return msg;
851+
}, Error);
850852
E('ERR_MULTIPLE_CALLBACK', 'Callback called multiple times', Error);
851853
E('ERR_NAPI_CONS_FUNCTION', 'Constructor must be a function', TypeError);
852854
E('ERR_NAPI_INVALID_DATAVIEW_ARGS',
@@ -940,11 +942,13 @@ E('ERR_UNHANDLED_ERROR',
940942
if (err === undefined) return msg;
941943
return `${msg} (${err})`;
942944
}, Error);
945+
// This should probably be a `TypeError`.
943946
E('ERR_UNKNOWN_CREDENTIAL', '%s identifier does not exist: %s', Error);
944947
E('ERR_UNKNOWN_ENCODING', 'Unknown encoding: %s', TypeError);
945948

946949
// This should probably be a `TypeError`.
947-
E('ERR_UNKNOWN_FILE_EXTENSION', 'Unknown file extension: %s', Error);
950+
E('ERR_UNKNOWN_FILE_EXTENSION', 'Unknown file extension: \'%s\' imported ' +
951+
'from %s', Error);
948952
E('ERR_UNKNOWN_MODULE_FORMAT', 'Unknown module format: %s', RangeError);
949953
E('ERR_UNKNOWN_SIGNAL', 'Unknown signal: %s', TypeError);
950954

lib/internal/modules/esm/default_resolve.js

+6-21
Original file line numberDiff line numberDiff line change
@@ -10,21 +10,15 @@ const { getOptionValue } = require('internal/options');
1010
const preserveSymlinks = getOptionValue('--preserve-symlinks');
1111
const preserveSymlinksMain = getOptionValue('--preserve-symlinks-main');
1212
const {
13-
ERR_MISSING_MODULE,
14-
ERR_MODULE_RESOLUTION_LEGACY,
13+
ERR_MODULE_NOT_FOUND,
1514
ERR_UNKNOWN_FILE_EXTENSION
1615
} = require('internal/errors').codes;
1716
const { resolve: moduleWrapResolve } = internalBinding('module_wrap');
18-
const StringStartsWith = Function.call.bind(String.prototype.startsWith);
1917
const { pathToFileURL, fileURLToPath } = require('internal/url');
2018

2119
const realpathCache = new Map();
2220

2321
function search(target, base) {
24-
if (base === undefined) {
25-
// We cannot search without a base.
26-
throw new ERR_MISSING_MODULE(target);
27-
}
2822
try {
2923
return moduleWrapResolve(target, base);
3024
} catch (e) {
@@ -36,7 +30,7 @@ function search(target, base) {
3630
tmpMod.paths = CJSmodule._nodeModulePaths(
3731
new URL('./', questionedBase).pathname);
3832
const found = CJSmodule._resolveFilename(target, tmpMod);
39-
error = new ERR_MODULE_RESOLUTION_LEGACY(target, base, found);
33+
error = new ERR_MODULE_NOT_FOUND(target, fileURLToPath(base), found);
4034
} catch {
4135
// ignore
4236
}
@@ -57,16 +51,8 @@ function resolve(specifier, parentURL) {
5751
};
5852
}
5953

60-
let url;
61-
try {
62-
url = search(specifier,
63-
parentURL || pathToFileURL(`${process.cwd()}/`).href);
64-
} catch (e) {
65-
if (typeof e.message === 'string' &&
66-
StringStartsWith(e.message, 'Cannot find module'))
67-
e.code = 'MODULE_NOT_FOUND';
68-
throw e;
69-
}
54+
let url = search(specifier,
55+
parentURL || pathToFileURL(`${process.cwd()}/`).href);
7056

7157
const isMain = parentURL === undefined;
7258

@@ -87,12 +73,11 @@ function resolve(specifier, parentURL) {
8773
if (isMain)
8874
format = 'cjs';
8975
else
90-
throw new ERR_UNKNOWN_FILE_EXTENSION(url.pathname);
76+
throw new ERR_UNKNOWN_FILE_EXTENSION(url.pathname,
77+
fileURLToPath(parentURL));
9178
}
9279

9380
return { url: `${url}`, format };
9481
}
9582

9683
module.exports = resolve;
97-
// exported for tests
98-
module.exports.search = search;

src/module_wrap.cc

+50-108
Original file line numberDiff line numberDiff line change
@@ -467,104 +467,37 @@ std::string ReadFile(uv_file file) {
467467
return contents;
468468
}
469469

470-
enum CheckFileOptions {
471-
LEAVE_OPEN_AFTER_CHECK,
472-
CLOSE_AFTER_CHECK
470+
enum DescriptorType {
471+
NONE,
472+
FILE,
473+
DIRECTORY
473474
};
474475

475-
Maybe<uv_file> CheckFile(const std::string& path,
476-
CheckFileOptions opt = CLOSE_AFTER_CHECK) {
476+
DescriptorType CheckDescriptor(const std::string& path) {
477477
uv_fs_t fs_req;
478-
if (path.empty()) {
479-
return Nothing<uv_file>();
480-
}
481-
482-
uv_file fd = uv_fs_open(nullptr, &fs_req, path.c_str(), O_RDONLY, 0, nullptr);
483-
uv_fs_req_cleanup(&fs_req);
484-
485-
if (fd < 0) {
486-
return Nothing<uv_file>();
487-
}
488-
489-
uv_fs_fstat(nullptr, &fs_req, fd, nullptr);
490-
uint64_t is_directory = fs_req.statbuf.st_mode & S_IFDIR;
491-
uv_fs_req_cleanup(&fs_req);
492-
493-
if (is_directory) {
494-
CHECK_EQ(0, uv_fs_close(nullptr, &fs_req, fd, nullptr));
495-
uv_fs_req_cleanup(&fs_req);
496-
return Nothing<uv_file>();
497-
}
498-
499-
if (opt == CLOSE_AFTER_CHECK) {
500-
CHECK_EQ(0, uv_fs_close(nullptr, &fs_req, fd, nullptr));
478+
int rc = uv_fs_stat(nullptr, &fs_req, path.c_str(), nullptr);
479+
if (rc == 0) {
480+
uint64_t is_directory = fs_req.statbuf.st_mode & S_IFDIR;
501481
uv_fs_req_cleanup(&fs_req);
482+
return is_directory ? DIRECTORY : FILE;
502483
}
503-
504-
return Just(fd);
505-
}
506-
507-
enum ResolveExtensionsOptions {
508-
TRY_EXACT_NAME,
509-
ONLY_VIA_EXTENSIONS
510-
};
511-
512-
inline bool ResolvesToFile(const URL& search) {
513-
std::string filePath = search.ToFilePath();
514-
Maybe<uv_file> check = CheckFile(filePath);
515-
return !check.IsNothing();
516-
}
517-
518-
template <ResolveExtensionsOptions options>
519-
Maybe<URL> ResolveExtensions(const URL& search) {
520-
if (options == TRY_EXACT_NAME) {
521-
std::string filePath = search.ToFilePath();
522-
Maybe<uv_file> check = CheckFile(filePath);
523-
if (!check.IsNothing()) {
524-
return Just(search);
525-
}
526-
}
527-
528-
for (const char* extension : EXTENSIONS) {
529-
URL guess(search.path() + extension, &search);
530-
Maybe<uv_file> check = CheckFile(guess.ToFilePath());
531-
if (!check.IsNothing()) {
532-
return Just(guess);
533-
}
534-
}
535-
536-
return Nothing<URL>();
537-
}
538-
539-
inline Maybe<URL> ResolveIndex(const URL& search) {
540-
return ResolveExtensions<ONLY_VIA_EXTENSIONS>(URL("index", search));
484+
uv_fs_req_cleanup(&fs_req);
485+
return NONE;
541486
}
542487

543-
Maybe<URL> ResolveModule(Environment* env,
544-
const std::string& specifier,
545-
const URL& base) {
488+
Maybe<URL> PackageResolve(Environment* env,
489+
const std::string& specifier,
490+
const URL& base) {
546491
URL parent(".", base);
547-
URL dir("");
492+
std::string last_path;
548493
do {
549-
dir = parent;
550-
Maybe<URL> check =
551-
Resolve(env, "./node_modules/" + specifier, dir);
552-
if (!check.IsNothing()) {
553-
const size_t limit = specifier.find('/');
554-
const size_t spec_len =
555-
limit == std::string::npos ? specifier.length() :
556-
limit + 1;
557-
std::string chroot =
558-
dir.path() + "node_modules/" + specifier.substr(0, spec_len);
559-
if (check.FromJust().path().substr(0, chroot.length()) != chroot) {
560-
return Nothing<URL>();
561-
}
562-
return check;
563-
} else {
564-
// TODO(bmeck) PREVENT FALLTHROUGH
565-
}
566-
parent = URL("..", &dir);
567-
} while (parent.path() != dir.path());
494+
URL pkg_url("./node_modules/" + specifier, &parent);
495+
DescriptorType check = CheckDescriptor(pkg_url.ToFilePath());
496+
if (check == FILE) return Just(pkg_url);
497+
last_path = parent.path();
498+
parent = URL("..", &parent);
499+
// cross-platform root check
500+
} while (parent.path() != last_path);
568501
return Nothing<URL>();
569502
}
570503

@@ -573,26 +506,27 @@ Maybe<URL> ResolveModule(Environment* env,
573506
Maybe<URL> Resolve(Environment* env,
574507
const std::string& specifier,
575508
const URL& base) {
576-
URL pure_url(specifier);
577-
if (!(pure_url.flags() & URL_FLAGS_FAILED)) {
578-
// just check existence, without altering
579-
Maybe<uv_file> check = CheckFile(pure_url.ToFilePath());
580-
if (check.IsNothing()) {
581-
return Nothing<URL>();
509+
// Order swapped from spec for minor perf gain.
510+
// Ok since relative URLs cannot parse as URLs.
511+
URL resolved;
512+
if (ShouldBeTreatedAsRelativeOrAbsolutePath(specifier)) {
513+
resolved = URL(specifier, base);
514+
} else {
515+
URL pure_url(specifier);
516+
if (!(pure_url.flags() & URL_FLAGS_FAILED)) {
517+
resolved = pure_url;
518+
} else {
519+
return PackageResolve(env, specifier, base);
582520
}
583-
return Just(pure_url);
584521
}
585-
if (specifier.length() == 0) {
522+
DescriptorType check = CheckDescriptor(resolved.ToFilePath());
523+
if (check != FILE) {
524+
std::string msg = "Cannot find module '" + resolved.ToFilePath() +
525+
"' imported from " + base.ToFilePath();
526+
node::THROW_ERR_MODULE_NOT_FOUND(env, msg.c_str());
586527
return Nothing<URL>();
587528
}
588-
if (ShouldBeTreatedAsRelativeOrAbsolutePath(specifier)) {
589-
URL resolved(specifier, base);
590-
if (ResolvesToFile(resolved))
591-
return Just(resolved);
592-
return Nothing<URL>();
593-
} else {
594-
return ResolveModule(env, specifier, base);
595-
}
529+
return Just(resolved);
596530
}
597531

598532
void ModuleWrap::Resolve(const FunctionCallbackInfo<Value>& args) {
@@ -614,10 +548,18 @@ void ModuleWrap::Resolve(const FunctionCallbackInfo<Value>& args) {
614548
env, "second argument is not a URL string");
615549
}
616550

551+
TryCatchScope try_catch(env);
617552
Maybe<URL> result = node::loader::Resolve(env, specifier_std, url);
618-
if (result.IsNothing() || (result.FromJust().flags() & URL_FLAGS_FAILED)) {
619-
std::string msg = "Cannot find module " + specifier_std;
620-
return node::THROW_ERR_MISSING_MODULE(env, msg.c_str());
553+
if (try_catch.HasCaught()) {
554+
try_catch.ReThrow();
555+
return;
556+
} else if (result.IsNothing() ||
557+
(result.FromJust().flags() & URL_FLAGS_FAILED)) {
558+
std::string msg = "Cannot find module '" + specifier_std +
559+
"' imported from " + url.ToFilePath();
560+
node::THROW_ERR_MODULE_NOT_FOUND(env, msg.c_str());
561+
try_catch.ReThrow();
562+
return;
621563
}
622564

623565
MaybeLocal<Value> obj = result.FromJust().ToObject(env);

src/node_errors.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -63,8 +63,8 @@ void FatalException(const v8::FunctionCallbackInfo<v8::Value>& args);
6363
V(ERR_MEMORY_ALLOCATION_FAILED, Error) \
6464
V(ERR_MISSING_ARGS, TypeError) \
6565
V(ERR_MISSING_MESSAGE_PORT_IN_TRANSFER_LIST, TypeError) \
66-
V(ERR_MISSING_MODULE, Error) \
6766
V(ERR_MISSING_PLATFORM_FOR_WORKER, Error) \
67+
V(ERR_MODULE_NOT_FOUND, Error) \
6868
V(ERR_OUT_OF_RANGE, RangeError) \
6969
V(ERR_SCRIPT_EXECUTION_INTERRUPTED, Error) \
7070
V(ERR_SCRIPT_EXECUTION_TIMEOUT, Error) \

0 commit comments

Comments
 (0)