Spaces:
Sleeping
Sleeping
import { dirname, resolve, extname, normalize, sep } from 'path'; | |
import builtinList from 'builtin-modules'; | |
import deepFreeze from 'deep-freeze'; | |
import deepMerge from 'deepmerge'; | |
import isModule from 'is-module'; | |
import fs, { realpathSync } from 'fs'; | |
import { promisify } from 'util'; | |
import { createFilter } from '@rollup/pluginutils'; | |
import resolveModule from 'resolve'; | |
function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { | |
try { | |
var info = gen[key](arg); | |
var value = info.value; | |
} catch (error) { | |
reject(error); | |
return; | |
} | |
if (info.done) { | |
resolve(value); | |
} else { | |
Promise.resolve(value).then(_next, _throw); | |
} | |
} | |
function _asyncToGenerator(fn) { | |
return function () { | |
var self = this, | |
args = arguments; | |
return new Promise(function (resolve, reject) { | |
var gen = fn.apply(self, args); | |
function _next(value) { | |
asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); | |
} | |
function _throw(err) { | |
asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); | |
} | |
_next(undefined); | |
}); | |
}; | |
} | |
function _slicedToArray(arr, i) { | |
return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); | |
} | |
function _arrayWithHoles(arr) { | |
if (Array.isArray(arr)) return arr; | |
} | |
function _iterableToArrayLimit(arr, i) { | |
if (typeof Symbol === "undefined" || !(Symbol.iterator in Object(arr))) return; | |
var _arr = []; | |
var _n = true; | |
var _d = false; | |
var _e = undefined; | |
try { | |
for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { | |
_arr.push(_s.value); | |
if (i && _arr.length === i) break; | |
} | |
} catch (err) { | |
_d = true; | |
_e = err; | |
} finally { | |
try { | |
if (!_n && _i["return"] != null) _i["return"](); | |
} finally { | |
if (_d) throw _e; | |
} | |
} | |
return _arr; | |
} | |
function _unsupportedIterableToArray(o, minLen) { | |
if (!o) return; | |
if (typeof o === "string") return _arrayLikeToArray(o, minLen); | |
var n = Object.prototype.toString.call(o).slice(8, -1); | |
if (n === "Object" && o.constructor) n = o.constructor.name; | |
if (n === "Map" || n === "Set") return Array.from(n); | |
if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); | |
} | |
function _arrayLikeToArray(arr, len) { | |
if (len == null || len > arr.length) len = arr.length; | |
for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i]; | |
return arr2; | |
} | |
function _nonIterableRest() { | |
throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); | |
} | |
function _createForOfIteratorHelper(o) { | |
if (typeof Symbol === "undefined" || o[Symbol.iterator] == null) { | |
if (Array.isArray(o) || (o = _unsupportedIterableToArray(o))) { | |
var i = 0; | |
var F = function () {}; | |
return { | |
s: F, | |
n: function () { | |
if (i >= o.length) return { | |
done: true | |
}; | |
return { | |
done: false, | |
value: o[i++] | |
}; | |
}, | |
e: function (e) { | |
throw e; | |
}, | |
f: F | |
}; | |
} | |
throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); | |
} | |
var it, | |
normalCompletion = true, | |
didErr = false, | |
err; | |
return { | |
s: function () { | |
it = o[Symbol.iterator](); | |
}, | |
n: function () { | |
var step = it.next(); | |
normalCompletion = step.done; | |
return step; | |
}, | |
e: function (e) { | |
didErr = true; | |
err = e; | |
}, | |
f: function () { | |
try { | |
if (!normalCompletion && it.return != null) it.return(); | |
} finally { | |
if (didErr) throw err; | |
} | |
} | |
}; | |
} | |
const exists = promisify(fs.exists); | |
const readFile = promisify(fs.readFile); | |
const realpath = promisify(fs.realpath); | |
const stat = promisify(fs.stat); | |
const onError = error => { | |
if (error.code === 'ENOENT') { | |
return false; | |
} | |
throw error; | |
}; | |
const makeCache = fn => { | |
const cache = new Map(); | |
const wrapped = /*#__PURE__*/function () { | |
var _ref = _asyncToGenerator(function* (param, done) { | |
if (cache.has(param) === false) { | |
cache.set(param, fn(param).catch(err => { | |
cache.delete(param); | |
throw err; | |
})); | |
} | |
try { | |
const result = cache.get(param); | |
const value = yield result; | |
return done(null, value); | |
} catch (error) { | |
return done(error); | |
} | |
}); | |
return function wrapped(_x, _x2) { | |
return _ref.apply(this, arguments); | |
}; | |
}(); | |
wrapped.clear = () => cache.clear(); | |
return wrapped; | |
}; | |
const isDirCached = makeCache( /*#__PURE__*/function () { | |
var _ref2 = _asyncToGenerator(function* (file) { | |
try { | |
const stats = yield stat(file); | |
return stats.isDirectory(); | |
} catch (error) { | |
return onError(error); | |
} | |
}); | |
return function (_x3) { | |
return _ref2.apply(this, arguments); | |
}; | |
}()); | |
const isFileCached = makeCache( /*#__PURE__*/function () { | |
var _ref3 = _asyncToGenerator(function* (file) { | |
try { | |
const stats = yield stat(file); | |
return stats.isFile(); | |
} catch (error) { | |
return onError(error); | |
} | |
}); | |
return function (_x4) { | |
return _ref3.apply(this, arguments); | |
}; | |
}()); | |
const readCachedFile = makeCache(readFile); | |
const resolveId = promisify(resolveModule); // returns the imported package name for bare module imports | |
function getPackageName(id) { | |
if (id.startsWith('.') || id.startsWith('/')) { | |
return null; | |
} | |
const split = id.split('/'); // @my-scope/my-package/foo.js -> @my-scope/my-package | |
// @my-scope/my-package -> @my-scope/my-package | |
if (split[0][0] === '@') { | |
return `${split[0]}/${split[1]}`; | |
} // my-package/foo.js -> my-package | |
// my-package -> my-package | |
return split[0]; | |
} | |
function getMainFields(options) { | |
let mainFields; | |
if (options.mainFields) { | |
mainFields = options.mainFields; | |
} else { | |
mainFields = ['module', 'main']; | |
} | |
if (options.browser && mainFields.indexOf('browser') === -1) { | |
return ['browser'].concat(mainFields); | |
} | |
if (!mainFields.length) { | |
throw new Error('Please ensure at least one `mainFields` value is specified'); | |
} | |
return mainFields; | |
} | |
function getPackageInfo(options) { | |
const cache = options.cache, | |
extensions = options.extensions, | |
pkg = options.pkg, | |
mainFields = options.mainFields, | |
preserveSymlinks = options.preserveSymlinks, | |
useBrowserOverrides = options.useBrowserOverrides; | |
let pkgPath = options.pkgPath; | |
if (cache.has(pkgPath)) { | |
return cache.get(pkgPath); | |
} // browserify/resolve doesn't realpath paths returned in its packageFilter callback | |
if (!preserveSymlinks) { | |
pkgPath = realpathSync(pkgPath); | |
} | |
const pkgRoot = dirname(pkgPath); | |
const packageInfo = { | |
// copy as we are about to munge the `main` field of `pkg`. | |
packageJson: Object.assign({}, pkg), | |
// path to package.json file | |
packageJsonPath: pkgPath, | |
// directory containing the package.json | |
root: pkgRoot, | |
// which main field was used during resolution of this module (main, module, or browser) | |
resolvedMainField: 'main', | |
// whether the browser map was used to resolve the entry point to this module | |
browserMappedMain: false, | |
// the entry point of the module with respect to the selected main field and any | |
// relevant browser mappings. | |
resolvedEntryPoint: '' | |
}; | |
let overriddenMain = false; | |
for (let i = 0; i < mainFields.length; i++) { | |
const field = mainFields[i]; | |
if (typeof pkg[field] === 'string') { | |
pkg.main = pkg[field]; | |
packageInfo.resolvedMainField = field; | |
overriddenMain = true; | |
break; | |
} | |
} | |
const internalPackageInfo = { | |
cachedPkg: pkg, | |
hasModuleSideEffects: () => null, | |
hasPackageEntry: overriddenMain !== false || mainFields.indexOf('main') !== -1, | |
packageBrowserField: useBrowserOverrides && typeof pkg.browser === 'object' && Object.keys(pkg.browser).reduce((browser, key) => { | |
let resolved = pkg.browser[key]; | |
if (resolved && resolved[0] === '.') { | |
resolved = resolve(pkgRoot, resolved); | |
} | |
/* eslint-disable no-param-reassign */ | |
browser[key] = resolved; | |
if (key[0] === '.') { | |
const absoluteKey = resolve(pkgRoot, key); | |
browser[absoluteKey] = resolved; | |
if (!extname(key)) { | |
extensions.reduce((subBrowser, ext) => { | |
subBrowser[absoluteKey + ext] = subBrowser[key]; | |
return subBrowser; | |
}, browser); | |
} | |
} | |
return browser; | |
}, {}), | |
packageInfo | |
}; | |
const browserMap = internalPackageInfo.packageBrowserField; | |
if (useBrowserOverrides && typeof pkg.browser === 'object' && // eslint-disable-next-line no-prototype-builtins | |
browserMap.hasOwnProperty(pkg.main)) { | |
packageInfo.resolvedEntryPoint = browserMap[pkg.main]; | |
packageInfo.browserMappedMain = true; | |
} else { | |
// index.node is technically a valid default entrypoint as well... | |
packageInfo.resolvedEntryPoint = resolve(pkgRoot, pkg.main || 'index.js'); | |
packageInfo.browserMappedMain = false; | |
} | |
const packageSideEffects = pkg.sideEffects; | |
if (typeof packageSideEffects === 'boolean') { | |
internalPackageInfo.hasModuleSideEffects = () => packageSideEffects; | |
} else if (Array.isArray(packageSideEffects)) { | |
internalPackageInfo.hasModuleSideEffects = createFilter(packageSideEffects, null, { | |
resolve: pkgRoot | |
}); | |
} | |
cache.set(pkgPath, internalPackageInfo); | |
return internalPackageInfo; | |
} | |
function normalizeInput(input) { | |
if (Array.isArray(input)) { | |
return input; | |
} else if (typeof input === 'object') { | |
return Object.values(input); | |
} // otherwise it's a string | |
return [input]; | |
} // Resolve module specifiers in order. Promise resolves to the first module that resolves | |
// successfully, or the error that resulted from the last attempted module resolution. | |
function resolveImportSpecifiers(importSpecifierList, resolveOptions) { | |
let promise = Promise.resolve(); | |
for (let i = 0; i < importSpecifierList.length; i++) { | |
promise = promise.then(value => { | |
// if we've already resolved to something, just return it. | |
if (value) { | |
return value; | |
} | |
return resolveId(importSpecifierList[i], resolveOptions).then(result => { | |
if (!resolveOptions.preserveSymlinks) { | |
result = realpathSync(result); | |
} | |
return result; | |
}); | |
}); | |
if (i < importSpecifierList.length - 1) { | |
// swallow MODULE_NOT_FOUND errors from all but the last resolution | |
promise = promise.catch(error => { | |
if (error.code !== 'MODULE_NOT_FOUND') { | |
throw error; | |
} | |
}); | |
} | |
} | |
return promise; | |
} | |
const builtins = new Set(builtinList); | |
const ES6_BROWSER_EMPTY = '\0node-resolve:empty.js'; | |
const nullFn = () => null; | |
const defaults = { | |
customResolveOptions: {}, | |
dedupe: [], | |
// It's important that .mjs is listed before .js so that Rollup will interpret npm modules | |
// which deploy both ESM .mjs and CommonJS .js files as ESM. | |
extensions: ['.mjs', '.js', '.json', '.node'], | |
resolveOnly: [] | |
}; | |
const DEFAULTS = deepFreeze(deepMerge({}, defaults)); | |
function nodeResolve(opts = {}) { | |
const options = Object.assign({}, defaults, opts); | |
const customResolveOptions = options.customResolveOptions, | |
extensions = options.extensions, | |
jail = options.jail; | |
const warnings = []; | |
const packageInfoCache = new Map(); | |
const idToPackageInfo = new Map(); | |
const mainFields = getMainFields(options); | |
const useBrowserOverrides = mainFields.indexOf('browser') !== -1; | |
const isPreferBuiltinsSet = options.preferBuiltins === true || options.preferBuiltins === false; | |
const preferBuiltins = isPreferBuiltinsSet ? options.preferBuiltins : true; | |
const rootDir = options.rootDir || process.cwd(); | |
let dedupe = options.dedupe; | |
let rollupOptions; | |
if (options.only) { | |
warnings.push('node-resolve: The `only` options is deprecated, please use `resolveOnly`'); | |
options.resolveOnly = options.only; | |
} | |
if (typeof dedupe !== 'function') { | |
dedupe = importee => options.dedupe.includes(importee) || options.dedupe.includes(getPackageName(importee)); | |
} | |
const resolveOnly = options.resolveOnly.map(pattern => { | |
if (pattern instanceof RegExp) { | |
return pattern; | |
} | |
const normalized = pattern.replace(/[\\^$*+?.()|[\]{}]/g, '\\$&'); | |
return new RegExp(`^${normalized}$`); | |
}); | |
const browserMapCache = new Map(); | |
let preserveSymlinks; | |
return { | |
name: 'node-resolve', | |
buildStart(options) { | |
rollupOptions = options; | |
var _iterator = _createForOfIteratorHelper(warnings), | |
_step; | |
try { | |
for (_iterator.s(); !(_step = _iterator.n()).done;) { | |
const warning = _step.value; | |
this.warn(warning); | |
} | |
} catch (err) { | |
_iterator.e(err); | |
} finally { | |
_iterator.f(); | |
} | |
preserveSymlinks = options.preserveSymlinks; | |
}, | |
generateBundle() { | |
readCachedFile.clear(); | |
isFileCached.clear(); | |
isDirCached.clear(); | |
}, | |
resolveId(importee, importer) { | |
var _this = this; | |
return _asyncToGenerator(function* () { | |
if (importee === ES6_BROWSER_EMPTY) { | |
return importee; | |
} // ignore IDs with null character, these belong to other plugins | |
if (/\0/.test(importee)) return null; // strip hash and query params from import | |
const _importee$split = importee.split('#'), | |
_importee$split2 = _slicedToArray(_importee$split, 2), | |
withoutHash = _importee$split2[0], | |
hash = _importee$split2[1]; | |
const _withoutHash$split = withoutHash.split('?'), | |
_withoutHash$split2 = _slicedToArray(_withoutHash$split, 2), | |
importPath = _withoutHash$split2[0], | |
params = _withoutHash$split2[1]; | |
const importSuffix = `${params ? `?${params}` : ''}${hash ? `#${hash}` : ''}`; | |
importee = importPath; | |
const basedir = !importer || dedupe(importee) ? rootDir : dirname(importer); // https://github.com/defunctzombie/package-browser-field-spec | |
const browser = browserMapCache.get(importer); | |
if (useBrowserOverrides && browser) { | |
const resolvedImportee = resolve(basedir, importee); | |
if (browser[importee] === false || browser[resolvedImportee] === false) { | |
return ES6_BROWSER_EMPTY; | |
} | |
const browserImportee = browser[importee] || browser[resolvedImportee] || browser[`${resolvedImportee}.js`] || browser[`${resolvedImportee}.json`]; | |
if (browserImportee) { | |
importee = browserImportee; | |
} | |
} | |
const parts = importee.split(/[/\\]/); | |
let id = parts.shift(); | |
let isRelativeImport = false; | |
if (id[0] === '@' && parts.length > 0) { | |
// scoped packages | |
id += `/${parts.shift()}`; | |
} else if (id[0] === '.') { | |
// an import relative to the parent dir of the importer | |
id = resolve(basedir, importee); | |
isRelativeImport = true; | |
} | |
if (!isRelativeImport && resolveOnly.length && !resolveOnly.some(pattern => pattern.test(id))) { | |
if (normalizeInput(rollupOptions.input).includes(importee)) { | |
return null; | |
} | |
return false; | |
} | |
let hasModuleSideEffects = nullFn; | |
let hasPackageEntry = true; | |
let packageBrowserField = false; | |
let packageInfo; | |
const filter = (pkg, pkgPath) => { | |
const info = getPackageInfo({ | |
cache: packageInfoCache, | |
extensions, | |
pkg, | |
pkgPath, | |
mainFields, | |
preserveSymlinks, | |
useBrowserOverrides | |
}); | |
packageInfo = info.packageInfo; | |
hasModuleSideEffects = info.hasModuleSideEffects; | |
hasPackageEntry = info.hasPackageEntry; | |
packageBrowserField = info.packageBrowserField; | |
return info.cachedPkg; | |
}; | |
let resolveOptions = { | |
basedir, | |
packageFilter: filter, | |
readFile: readCachedFile, | |
isFile: isFileCached, | |
isDirectory: isDirCached, | |
extensions | |
}; | |
if (preserveSymlinks !== undefined) { | |
resolveOptions.preserveSymlinks = preserveSymlinks; | |
} | |
const importSpecifierList = []; | |
if (importer === undefined && !importee[0].match(/^\.?\.?\//)) { | |
// For module graph roots (i.e. when importer is undefined), we | |
// need to handle 'path fragments` like `foo/bar` that are commonly | |
// found in rollup config files. If importee doesn't look like a | |
// relative or absolute path, we make it relative and attempt to | |
// resolve it. If we don't find anything, we try resolving it as we | |
// got it. | |
importSpecifierList.push(`./${importee}`); | |
} | |
const importeeIsBuiltin = builtins.has(importee); | |
if (importeeIsBuiltin && (!preferBuiltins || !isPreferBuiltinsSet)) { | |
// The `resolve` library will not resolve packages with the same | |
// name as a node built-in module. If we're resolving something | |
// that's a builtin, and we don't prefer to find built-ins, we | |
// first try to look up a local module with that name. If we don't | |
// find anything, we resolve the builtin which just returns back | |
// the built-in's name. | |
importSpecifierList.push(`${importee}/`); | |
} // TypeScript files may import '.js' to refer to either '.ts' or '.tsx' | |
if (importer && importee.endsWith('.js')) { | |
for (var _i = 0, _arr = ['.ts', '.tsx']; _i < _arr.length; _i++) { | |
const ext = _arr[_i]; | |
if (importer.endsWith(ext) && extensions.includes(ext)) { | |
importSpecifierList.push(importee.replace(/.js$/, ext)); | |
} | |
} | |
} | |
importSpecifierList.push(importee); | |
resolveOptions = Object.assign(resolveOptions, customResolveOptions); | |
try { | |
let resolved = yield resolveImportSpecifiers(importSpecifierList, resolveOptions); | |
if (resolved && packageBrowserField) { | |
if (Object.prototype.hasOwnProperty.call(packageBrowserField, resolved)) { | |
if (!packageBrowserField[resolved]) { | |
browserMapCache.set(resolved, packageBrowserField); | |
return ES6_BROWSER_EMPTY; | |
} | |
resolved = packageBrowserField[resolved]; | |
} | |
browserMapCache.set(resolved, packageBrowserField); | |
} | |
if (hasPackageEntry && !preserveSymlinks && resolved) { | |
const fileExists = yield exists(resolved); | |
if (fileExists) { | |
resolved = yield realpath(resolved); | |
} | |
} | |
idToPackageInfo.set(resolved, packageInfo); | |
if (hasPackageEntry) { | |
if (builtins.has(resolved) && preferBuiltins && isPreferBuiltinsSet) { | |
return null; | |
} else if (importeeIsBuiltin && preferBuiltins) { | |
if (!isPreferBuiltinsSet) { | |
_this.warn(`preferring built-in module '${importee}' over local alternative at '${resolved}', pass 'preferBuiltins: false' to disable this behavior or 'preferBuiltins: true' to disable this warning`); | |
} | |
return null; | |
} else if (jail && resolved.indexOf(normalize(jail.trim(sep))) !== 0) { | |
return null; | |
} | |
} | |
if (resolved && options.modulesOnly) { | |
const code = yield readFile(resolved, 'utf-8'); | |
if (isModule(code)) { | |
return { | |
id: `${resolved}${importSuffix}`, | |
moduleSideEffects: hasModuleSideEffects(resolved) | |
}; | |
} | |
return null; | |
} | |
const result = { | |
id: `${resolved}${importSuffix}`, | |
moduleSideEffects: hasModuleSideEffects(resolved) | |
}; | |
return result; | |
} catch (error) { | |
return null; | |
} | |
})(); | |
}, | |
load(importee) { | |
if (importee === ES6_BROWSER_EMPTY) { | |
return 'export default {};'; | |
} | |
return null; | |
}, | |
getPackageInfoForId(id) { | |
return idToPackageInfo.get(id); | |
} | |
}; | |
} | |
export default nodeResolve; | |
export { DEFAULTS, nodeResolve }; | |