1 | const { isAbsolute, resolve, sep } = require('path');
|
2 |
|
3 |
|
4 |
|
5 |
|
6 | module.exports = function resolveDependency (specifier, parent, job) {
|
7 | let resolved;
|
8 | if (isAbsolute(specifier) || specifier === '.' || specifier === '..' || specifier.startsWith('./') || specifier.startsWith('../'))
|
9 | resolved = resolvePath(resolve(parent, '..', specifier), parent, job);
|
10 | else
|
11 | resolved = resolvePackage(specifier, parent, job);
|
12 | if (resolved.startsWith('node:')) return resolved;
|
13 | return job.realpath(resolved, parent);
|
14 | };
|
15 |
|
16 | function resolvePath (path, parent, job) {
|
17 | return resolveFile(path, parent, job) || resolveDir(path, parent, job) || notFound(path, parent);
|
18 | }
|
19 |
|
20 | function resolveFile (path, parent, job) {
|
21 | path = job.realpath(path, parent);
|
22 | if (path.endsWith(sep)) return;
|
23 | if (job.isFile(path)) return path;
|
24 | if (job.ts && path.startsWith(job.base) && path.substr(job.base.length).indexOf(sep + 'node_modules' + sep) === -1 && job.isFile(path + '.ts')) return path + '.ts';
|
25 | if (job.ts && path.startsWith(job.base) && path.substr(job.base.length).indexOf(sep + 'node_modules' + sep) === -1 && job.isFile(path + '.tsx')) return path + '.tsx';
|
26 | if (job.isFile(path + '.js')) return path + '.js';
|
27 | if (job.isFile(path + '.json')) return path + '.json';
|
28 | if (job.isFile(path + '.node')) return path + '.node';
|
29 | }
|
30 |
|
31 | function resolveDir (path, parent, job) {
|
32 | if (!job.isDir(path)) return;
|
33 | const realPjsonPath = job.realpath(path + sep + 'package.json', parent);
|
34 | const pjsonSource = job.readFile(realPjsonPath);
|
35 | if (pjsonSource) {
|
36 | try {
|
37 | var pjson = JSON.parse(pjsonSource);
|
38 | }
|
39 | catch (e) {}
|
40 | if (pjson && typeof pjson.main === 'string') {
|
41 | const resolved = resolveFile(resolve(path, pjson.main), parent, job) || resolveFile(resolve(path, pjson.main, 'index'), parent, job);
|
42 | if (resolved) {
|
43 | job.emitFile(realPjsonPath, 'resolve', parent);
|
44 | return resolved;
|
45 | }
|
46 | }
|
47 | }
|
48 | return resolveFile(resolve(path, 'index'), parent, job);
|
49 | }
|
50 |
|
51 | function notFound (specifier, parent) {
|
52 | const e = new Error("Cannot find module '" + specifier + "' loaded from " + parent);
|
53 | e.code = 'MODULE_NOT_FOUND';
|
54 | throw e;
|
55 | }
|
56 |
|
57 | const nodeBuiltins = new Set([...require("repl")._builtinLibs, "constants", "module", "timers", "console", "_stream_writable", "_stream_readable", "_stream_duplex", "process", "sys"]);
|
58 |
|
59 | function resolvePackage (name, parent, job) {
|
60 | let packageParent = parent;
|
61 | if (nodeBuiltins.has(name)) return 'node:' + name;
|
62 | let separatorIndex;
|
63 | const rootSeparatorIndex = packageParent.indexOf(sep);
|
64 | while ((separatorIndex = packageParent.lastIndexOf(sep)) > rootSeparatorIndex) {
|
65 | packageParent = packageParent.substr(0, separatorIndex);
|
66 | const nodeModulesDir = packageParent + sep + 'node_modules';
|
67 | const stat = job.stat(nodeModulesDir);
|
68 | if (!stat || !stat.isDirectory()) continue;
|
69 | const resolved = resolveFile(nodeModulesDir + sep + name, parent, job) || resolveDir(nodeModulesDir + sep + name, parent, job);
|
70 | if (resolved) return resolved;
|
71 | }
|
72 | if (Object.hasOwnProperty.call(job.paths, name)) {
|
73 | return job.paths[name];
|
74 | }
|
75 | for (const path of Object.keys(job.paths)) {
|
76 | if (path.endsWith('/') && name.startsWith(path)) {
|
77 | const pathTarget = job.paths[path] + name.slice(path.length);
|
78 | return resolveFile(pathTarget, parent, job) || resolveDir(pathTarget, parent, job);
|
79 | }
|
80 | }
|
81 | notFound(name, parent);
|
82 | }
|