blob: d5308c926e498b3a78cd1617bfe4c27ca3607239 [file] [log] [blame]
Leo Repp58b9f112021-11-22 11:57:47 +01001var isCore = require('is-core-module');
2var fs = require('fs');
3var path = require('path');
4var caller = require('./caller');
5var nodeModulesPaths = require('./node-modules-paths');
6var normalizeOptions = require('./normalize-options');
7
8var realpathFS = fs.realpathSync && typeof fs.realpathSync.native === 'function' ? fs.realpathSync.native : fs.realpathSync;
9
10var defaultIsFile = function isFile(file) {
11 try {
12 var stat = fs.statSync(file);
13 } catch (e) {
14 if (e && (e.code === 'ENOENT' || e.code === 'ENOTDIR')) return false;
15 throw e;
16 }
17 return stat.isFile() || stat.isFIFO();
18};
19
20var defaultIsDir = function isDirectory(dir) {
21 try {
22 var stat = fs.statSync(dir);
23 } catch (e) {
24 if (e && (e.code === 'ENOENT' || e.code === 'ENOTDIR')) return false;
25 throw e;
26 }
27 return stat.isDirectory();
28};
29
30var defaultRealpathSync = function realpathSync(x) {
31 try {
32 return realpathFS(x);
33 } catch (realpathErr) {
34 if (realpathErr.code !== 'ENOENT') {
35 throw realpathErr;
36 }
37 }
38 return x;
39};
40
41var maybeRealpathSync = function maybeRealpathSync(realpathSync, x, opts) {
42 if (opts && opts.preserveSymlinks === false) {
43 return realpathSync(x);
44 }
45 return x;
46};
47
48var getPackageCandidates = function getPackageCandidates(x, start, opts) {
49 var dirs = nodeModulesPaths(start, opts, x);
50 for (var i = 0; i < dirs.length; i++) {
51 dirs[i] = path.join(dirs[i], x);
52 }
53 return dirs;
54};
55
56module.exports = function resolveSync(x, options) {
57 if (typeof x !== 'string') {
58 throw new TypeError('Path must be a string.');
59 }
60 var opts = normalizeOptions(x, options);
61
62 var isFile = opts.isFile || defaultIsFile;
63 var readFileSync = opts.readFileSync || fs.readFileSync;
64 var isDirectory = opts.isDirectory || defaultIsDir;
65 var realpathSync = opts.realpathSync || defaultRealpathSync;
66 var packageIterator = opts.packageIterator;
67
68 var extensions = opts.extensions || ['.js'];
69 var includeCoreModules = opts.includeCoreModules !== false;
70 var basedir = opts.basedir || path.dirname(caller());
71 var parent = opts.filename || basedir;
72
73 opts.paths = opts.paths || [];
74
75 // ensure that `basedir` is an absolute path at this point, resolving against the process' current working directory
76 var absoluteStart = maybeRealpathSync(realpathSync, path.resolve(basedir), opts);
77
78 if ((/^(?:\.\.?(?:\/|$)|\/|([A-Za-z]:)?[/\\])/).test(x)) {
79 var res = path.resolve(absoluteStart, x);
80 if (x === '.' || x === '..' || x.slice(-1) === '/') res += '/';
81 var m = loadAsFileSync(res) || loadAsDirectorySync(res);
82 if (m) return maybeRealpathSync(realpathSync, m, opts);
83 } else if (includeCoreModules && isCore(x)) {
84 return x;
85 } else {
86 var n = loadNodeModulesSync(x, absoluteStart);
87 if (n) return maybeRealpathSync(realpathSync, n, opts);
88 }
89
90 var err = new Error("Cannot find module '" + x + "' from '" + parent + "'");
91 err.code = 'MODULE_NOT_FOUND';
92 throw err;
93
94 function loadAsFileSync(x) {
95 var pkg = loadpkg(path.dirname(x));
96
97 if (pkg && pkg.dir && pkg.pkg && opts.pathFilter) {
98 var rfile = path.relative(pkg.dir, x);
99 var r = opts.pathFilter(pkg.pkg, x, rfile);
100 if (r) {
101 x = path.resolve(pkg.dir, r); // eslint-disable-line no-param-reassign
102 }
103 }
104
105 if (isFile(x)) {
106 return x;
107 }
108
109 for (var i = 0; i < extensions.length; i++) {
110 var file = x + extensions[i];
111 if (isFile(file)) {
112 return file;
113 }
114 }
115 }
116
117 function loadpkg(dir) {
118 if (dir === '' || dir === '/') return;
119 if (process.platform === 'win32' && (/^\w:[/\\]*$/).test(dir)) {
120 return;
121 }
122 if ((/[/\\]node_modules[/\\]*$/).test(dir)) return;
123
124 var pkgfile = path.join(maybeRealpathSync(realpathSync, dir, opts), 'package.json');
125
126 if (!isFile(pkgfile)) {
127 return loadpkg(path.dirname(dir));
128 }
129
130 var body = readFileSync(pkgfile);
131
132 try {
133 var pkg = JSON.parse(body);
134 } catch (jsonErr) {}
135
136 if (pkg && opts.packageFilter) {
137 // v2 will pass pkgfile
138 pkg = opts.packageFilter(pkg, /*pkgfile,*/ dir); // eslint-disable-line spaced-comment
139 }
140
141 return { pkg: pkg, dir: dir };
142 }
143
144 function loadAsDirectorySync(x) {
145 var pkgfile = path.join(maybeRealpathSync(realpathSync, x, opts), '/package.json');
146 if (isFile(pkgfile)) {
147 try {
148 var body = readFileSync(pkgfile, 'UTF8');
149 var pkg = JSON.parse(body);
150 } catch (e) {}
151
152 if (pkg && opts.packageFilter) {
153 // v2 will pass pkgfile
154 pkg = opts.packageFilter(pkg, /*pkgfile,*/ x); // eslint-disable-line spaced-comment
155 }
156
157 if (pkg && pkg.main) {
158 if (typeof pkg.main !== 'string') {
159 var mainError = new TypeError('package “' + pkg.name + '” `main` must be a string');
160 mainError.code = 'INVALID_PACKAGE_MAIN';
161 throw mainError;
162 }
163 if (pkg.main === '.' || pkg.main === './') {
164 pkg.main = 'index';
165 }
166 try {
167 var m = loadAsFileSync(path.resolve(x, pkg.main));
168 if (m) return m;
169 var n = loadAsDirectorySync(path.resolve(x, pkg.main));
170 if (n) return n;
171 } catch (e) {}
172 }
173 }
174
175 return loadAsFileSync(path.join(x, '/index'));
176 }
177
178 function loadNodeModulesSync(x, start) {
179 var thunk = function () { return getPackageCandidates(x, start, opts); };
180 var dirs = packageIterator ? packageIterator(x, start, thunk, opts) : thunk();
181
182 for (var i = 0; i < dirs.length; i++) {
183 var dir = dirs[i];
184 if (isDirectory(path.dirname(dir))) {
185 var m = loadAsFileSync(dir);
186 if (m) return m;
187 var n = loadAsDirectorySync(dir);
188 if (n) return n;
189 }
190 }
191 }
192};