| Leo Repp | 58b9f11 | 2021-11-22 11:57:47 +0100 | [diff] [blame^] | 1 | var isCore = require('is-core-module'); |
| 2 | var fs = require('fs'); |
| 3 | var path = require('path'); |
| 4 | var caller = require('./caller'); |
| 5 | var nodeModulesPaths = require('./node-modules-paths'); |
| 6 | var normalizeOptions = require('./normalize-options'); |
| 7 | |
| 8 | var realpathFS = fs.realpathSync && typeof fs.realpathSync.native === 'function' ? fs.realpathSync.native : fs.realpathSync; |
| 9 | |
| 10 | var 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 | |
| 20 | var 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 | |
| 30 | var 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 | |
| 41 | var maybeRealpathSync = function maybeRealpathSync(realpathSync, x, opts) { |
| 42 | if (opts && opts.preserveSymlinks === false) { |
| 43 | return realpathSync(x); |
| 44 | } |
| 45 | return x; |
| 46 | }; |
| 47 | |
| 48 | var 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 | |
| 56 | module.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 | }; |