|
import path, { dirname, resolve, extname, normalize, sep } from 'path'; |
|
import isBuiltinModule from 'is-builtin-module'; |
|
import deepMerge from 'deepmerge'; |
|
import isModule from 'is-module'; |
|
import fs, { realpathSync } from 'fs'; |
|
import { promisify } from 'util'; |
|
import { pathToFileURL, fileURLToPath } from 'url'; |
|
import resolve$1 from 'resolve'; |
|
import { createFilter } from '@rollup/pluginutils'; |
|
|
|
var version = "15.2.3"; |
|
var peerDependencies = { |
|
rollup: "^2.78.0||^3.0.0||^4.0.0" |
|
}; |
|
|
|
promisify(fs.access); |
|
const readFile$1 = promisify(fs.readFile); |
|
const realpath = promisify(fs.realpath); |
|
const stat = promisify(fs.stat); |
|
async function fileExists(filePath) { |
|
try { |
|
const res = await stat(filePath); |
|
return res.isFile(); |
|
} |
|
catch { |
|
return false; |
|
} |
|
} |
|
async function resolveSymlink(path) { |
|
return (await fileExists(path)) ? realpath(path) : path; |
|
} |
|
|
|
const onError = (error) => { |
|
if (error.code === 'ENOENT') { |
|
return false; |
|
} |
|
throw error; |
|
}; |
|
|
|
const makeCache = (fn) => { |
|
const cache = new Map(); |
|
const wrapped = async (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 = await result; |
|
return done(null, value); |
|
} catch (error) { |
|
return done(error); |
|
} |
|
}; |
|
|
|
wrapped.clear = () => cache.clear(); |
|
|
|
return wrapped; |
|
}; |
|
|
|
const isDirCached = makeCache(async (file) => { |
|
try { |
|
const stats = await stat(file); |
|
return stats.isDirectory(); |
|
} catch (error) { |
|
return onError(error); |
|
} |
|
}); |
|
|
|
const isFileCached = makeCache(async (file) => { |
|
try { |
|
const stats = await stat(file); |
|
return stats.isFile(); |
|
} catch (error) { |
|
return onError(error); |
|
} |
|
}); |
|
|
|
const readCachedFile = makeCache(readFile$1); |
|
|
|
function handleDeprecatedOptions(opts) { |
|
const warnings = []; |
|
|
|
if (opts.customResolveOptions) { |
|
const { customResolveOptions } = opts; |
|
if (customResolveOptions.moduleDirectory) { |
|
|
|
opts.moduleDirectories = Array.isArray(customResolveOptions.moduleDirectory) |
|
? customResolveOptions.moduleDirectory |
|
: [customResolveOptions.moduleDirectory]; |
|
|
|
warnings.push( |
|
'node-resolve: The `customResolveOptions.moduleDirectory` option has been deprecated. Use `moduleDirectories`, which must be an array.' |
|
); |
|
} |
|
|
|
if (customResolveOptions.preserveSymlinks) { |
|
throw new Error( |
|
'node-resolve: `customResolveOptions.preserveSymlinks` is no longer an option. We now always use the rollup `preserveSymlinks` option.' |
|
); |
|
} |
|
|
|
[ |
|
'basedir', |
|
'package', |
|
'extensions', |
|
'includeCoreModules', |
|
'readFile', |
|
'isFile', |
|
'isDirectory', |
|
'realpath', |
|
'packageFilter', |
|
'pathFilter', |
|
'paths', |
|
'packageIterator' |
|
].forEach((resolveOption) => { |
|
if (customResolveOptions[resolveOption]) { |
|
throw new Error( |
|
`node-resolve: \`customResolveOptions.${resolveOption}\` is no longer an option. If you need this, please open an issue.` |
|
); |
|
} |
|
}); |
|
} |
|
|
|
return { warnings }; |
|
} |
|
|
|
|
|
function getPackageName(id) { |
|
if (id.startsWith('.') || id.startsWith('/')) { |
|
return null; |
|
} |
|
|
|
const split = id.split('/'); |
|
|
|
|
|
|
|
if (split[0][0] === '@') { |
|
return `${split[0]}/${split[1]}`; |
|
} |
|
|
|
|
|
|
|
return split[0]; |
|
} |
|
|
|
function getMainFields(options) { |
|
let mainFields; |
|
if (options.mainFields) { |
|
({ mainFields } = options); |
|
} 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, |
|
extensions, |
|
pkg, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
rootDir, |
|
ignoreSideEffectsForRoot |
|
} = options; |
|
let { pkgPath } = options; |
|
|
|
if (cache.has(pkgPath)) { |
|
return cache.get(pkgPath); |
|
} |
|
|
|
|
|
if (!preserveSymlinks) { |
|
pkgPath = realpathSync(pkgPath); |
|
} |
|
|
|
const pkgRoot = dirname(pkgPath); |
|
|
|
const packageInfo = { |
|
|
|
packageJson: { ...pkg }, |
|
|
|
|
|
packageJsonPath: pkgPath, |
|
|
|
|
|
root: pkgRoot, |
|
|
|
|
|
resolvedMainField: 'main', |
|
|
|
|
|
browserMappedMain: false, |
|
|
|
|
|
|
|
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); |
|
} |
|
|
|
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' && |
|
|
|
browserMap.hasOwnProperty(pkg.main) |
|
) { |
|
packageInfo.resolvedEntryPoint = browserMap[pkg.main]; |
|
packageInfo.browserMappedMain = true; |
|
} else { |
|
|
|
packageInfo.resolvedEntryPoint = resolve(pkgRoot, pkg.main || 'index.js'); |
|
packageInfo.browserMappedMain = false; |
|
} |
|
|
|
if (!ignoreSideEffectsForRoot || rootDir !== pkgRoot) { |
|
const packageSideEffects = pkg.sideEffects; |
|
if (typeof packageSideEffects === 'boolean') { |
|
internalPackageInfo.hasModuleSideEffects = () => packageSideEffects; |
|
} else if (Array.isArray(packageSideEffects)) { |
|
const finalPackageSideEffects = packageSideEffects.map((sideEffect) => { |
|
|
|
|
|
|
|
|
|
if (sideEffect.includes('/')) { |
|
return sideEffect; |
|
} |
|
return `**/${sideEffect}`; |
|
}); |
|
internalPackageInfo.hasModuleSideEffects = createFilter(finalPackageSideEffects, 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); |
|
} |
|
|
|
|
|
return [input]; |
|
} |
|
|
|
|
|
function isModuleDir(current, moduleDirs) { |
|
return moduleDirs.some((dir) => current.endsWith(dir)); |
|
} |
|
async function findPackageJson(base, moduleDirs) { |
|
const { root } = path.parse(base); |
|
let current = base; |
|
while (current !== root && !isModuleDir(current, moduleDirs)) { |
|
const pkgJsonPath = path.join(current, 'package.json'); |
|
if (await fileExists(pkgJsonPath)) { |
|
const pkgJsonString = fs.readFileSync(pkgJsonPath, 'utf-8'); |
|
return { pkgJson: JSON.parse(pkgJsonString), pkgPath: current, pkgJsonPath }; |
|
} |
|
current = path.resolve(current, '..'); |
|
} |
|
return null; |
|
} |
|
function isUrl(str) { |
|
try { |
|
return !!new URL(str); |
|
} |
|
catch (_) { |
|
return false; |
|
} |
|
} |
|
|
|
|
|
|
|
function isConditions(exports) { |
|
return typeof exports === 'object' && Object.keys(exports).every((k) => !k.startsWith('.')); |
|
} |
|
|
|
|
|
|
|
function isMappings(exports) { |
|
return typeof exports === 'object' && !isConditions(exports); |
|
} |
|
|
|
|
|
|
|
function isMixedExports(exports) { |
|
const keys = Object.keys(exports); |
|
return keys.some((k) => k.startsWith('.')) && keys.some((k) => !k.startsWith('.')); |
|
} |
|
function createBaseErrorMsg(importSpecifier, importer) { |
|
return `Could not resolve import "${importSpecifier}" in ${importer}`; |
|
} |
|
function createErrorMsg(context, reason, isImports) { |
|
const { importSpecifier, importer, pkgJsonPath } = context; |
|
const base = createBaseErrorMsg(importSpecifier, importer); |
|
const field = isImports ? 'imports' : 'exports'; |
|
return `${base} using ${field} defined in ${pkgJsonPath}.${reason ? ` ${reason}` : ''}`; |
|
} |
|
class ResolveError extends Error { |
|
} |
|
class InvalidConfigurationError extends ResolveError { |
|
constructor(context, reason) { |
|
super(createErrorMsg(context, `Invalid "exports" field. ${reason}`)); |
|
} |
|
} |
|
class InvalidModuleSpecifierError extends ResolveError { |
|
constructor(context, isImports, reason) { |
|
super(createErrorMsg(context, reason, isImports)); |
|
} |
|
} |
|
class InvalidPackageTargetError extends ResolveError { |
|
constructor(context, reason) { |
|
super(createErrorMsg(context, reason)); |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
function includesInvalidSegments(pathSegments, moduleDirs) { |
|
const invalidSegments = ['', '.', '..', ...moduleDirs]; |
|
|
|
return pathSegments.some((v) => invalidSegments.includes(v) || invalidSegments.includes(decodeURI(v))); |
|
} |
|
async function resolvePackageTarget(context, { target, patternMatch, isImports }) { |
|
|
|
if (typeof target === 'string') { |
|
|
|
if (!target.startsWith('./')) { |
|
|
|
if (!isImports || ['/', '../'].some((p) => target.startsWith(p)) || isUrl(target)) { |
|
|
|
throw new InvalidPackageTargetError(context, `Invalid mapping: "${target}".`); |
|
} |
|
|
|
if (typeof patternMatch === 'string') { |
|
|
|
const result = await context.resolveId(target.replace(/\*/g, patternMatch), context.pkgURL.href); |
|
return result ? pathToFileURL(result.location).href : null; |
|
} |
|
|
|
const result = await context.resolveId(target, context.pkgURL.href); |
|
return result ? pathToFileURL(result.location).href : null; |
|
} |
|
|
|
|
|
if (context.allowExportsFolderMapping) { |
|
target = target.replace(/\/$/, '/*'); |
|
} |
|
|
|
{ |
|
const pathSegments = target.split(/\/|\\/); |
|
|
|
const firstDot = pathSegments.indexOf('.'); |
|
firstDot !== -1 && pathSegments.slice(firstDot); |
|
if (firstDot !== -1 && |
|
firstDot < pathSegments.length - 1 && |
|
includesInvalidSegments(pathSegments.slice(firstDot + 1), context.moduleDirs)) { |
|
throw new InvalidPackageTargetError(context, `Invalid mapping: "${target}".`); |
|
} |
|
} |
|
|
|
const resolvedTarget = new URL(target, context.pkgURL); |
|
|
|
if (!resolvedTarget.href.startsWith(context.pkgURL.href)) { |
|
throw new InvalidPackageTargetError(context, `Resolved to ${resolvedTarget.href} which is outside package ${context.pkgURL.href}`); |
|
} |
|
|
|
if (!patternMatch) { |
|
|
|
return resolvedTarget; |
|
} |
|
|
|
if (includesInvalidSegments(patternMatch.split(/\/|\\/), context.moduleDirs)) { |
|
|
|
throw new InvalidModuleSpecifierError(context); |
|
} |
|
|
|
return resolvedTarget.href.replace(/\*/g, patternMatch); |
|
} |
|
|
|
if (Array.isArray(target)) { |
|
|
|
if (target.length === 0) { |
|
return null; |
|
} |
|
let lastError = null; |
|
|
|
for (const item of target) { |
|
|
|
|
|
try { |
|
const resolved = await resolvePackageTarget(context, { |
|
target: item, |
|
patternMatch, |
|
isImports |
|
}); |
|
|
|
|
|
if (resolved !== undefined) { |
|
return resolved; |
|
} |
|
} |
|
catch (error) { |
|
if (!(error instanceof InvalidPackageTargetError)) { |
|
throw error; |
|
} |
|
else { |
|
lastError = error; |
|
} |
|
} |
|
} |
|
|
|
if (lastError) { |
|
throw lastError; |
|
} |
|
return null; |
|
} |
|
|
|
if (target && typeof target === 'object') { |
|
|
|
for (const [key, value] of Object.entries(target)) { |
|
|
|
|
|
|
|
if (key === 'default' || context.conditions.includes(key)) { |
|
|
|
|
|
const resolved = await resolvePackageTarget(context, { |
|
target: value, |
|
patternMatch, |
|
isImports |
|
}); |
|
|
|
|
|
if (resolved !== undefined) { |
|
return resolved; |
|
} |
|
} |
|
} |
|
|
|
return undefined; |
|
} |
|
|
|
if (target === null) { |
|
return null; |
|
} |
|
|
|
throw new InvalidPackageTargetError(context, `Invalid exports field.`); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
function nodePatternKeyCompare(keyA, keyB) { |
|
|
|
const baseLengthA = keyA.includes('*') ? keyA.indexOf('*') + 1 : keyA.length; |
|
|
|
const baseLengthB = keyB.includes('*') ? keyB.indexOf('*') + 1 : keyB.length; |
|
|
|
const rval = baseLengthB - baseLengthA; |
|
if (rval !== 0) |
|
return rval; |
|
|
|
if (!keyA.includes('*')) |
|
return 1; |
|
|
|
if (!keyB.includes('*')) |
|
return -1; |
|
|
|
|
|
|
|
return keyB.length - keyA.length; |
|
} |
|
async function resolvePackageImportsExports(context, { matchKey, matchObj, isImports }) { |
|
|
|
if (!matchKey.includes('*') && matchKey in matchObj) { |
|
|
|
const target = matchObj[matchKey]; |
|
|
|
const resolved = await resolvePackageTarget(context, { target, patternMatch: '', isImports }); |
|
return resolved; |
|
} |
|
|
|
const expansionKeys = Object.keys(matchObj) |
|
|
|
.filter((k) => k.endsWith('/') || k.includes('*')) |
|
|
|
.sort(nodePatternKeyCompare); |
|
|
|
for (const expansionKey of expansionKeys) { |
|
const indexOfAsterisk = expansionKey.indexOf('*'); |
|
|
|
const patternBase = indexOfAsterisk === -1 ? expansionKey : expansionKey.substring(0, indexOfAsterisk); |
|
|
|
if (matchKey.startsWith(patternBase) && matchKey !== patternBase) { |
|
|
|
const patternTrailer = indexOfAsterisk !== -1 ? expansionKey.substring(indexOfAsterisk + 1) : ''; |
|
|
|
if (patternTrailer.length === 0 || |
|
|
|
(matchKey.endsWith(patternTrailer) && matchKey.length >= expansionKey.length)) { |
|
|
|
const target = matchObj[expansionKey]; |
|
|
|
|
|
const patternMatch = matchKey.substring(patternBase.length, matchKey.length - patternTrailer.length); |
|
|
|
const resolved = await resolvePackageTarget(context, { |
|
target, |
|
patternMatch, |
|
isImports |
|
}); |
|
return resolved; |
|
} |
|
} |
|
} |
|
throw new InvalidModuleSpecifierError(context, isImports); |
|
} |
|
|
|
|
|
|
|
|
|
async function resolvePackageExports(context, subpath, exports) { |
|
|
|
if (isMixedExports(exports)) { |
|
|
|
throw new InvalidConfigurationError(context, 'All keys must either start with ./, or without one.'); |
|
} |
|
|
|
if (subpath === '.') { |
|
|
|
let mainExport; |
|
|
|
if (typeof exports === 'string' || Array.isArray(exports) || isConditions(exports)) { |
|
|
|
mainExport = exports; |
|
|
|
} |
|
else if (isMappings(exports)) { |
|
|
|
mainExport = exports['.']; |
|
} |
|
|
|
if (mainExport) { |
|
|
|
const resolved = await resolvePackageTarget(context, { |
|
target: mainExport, |
|
patternMatch: '', |
|
isImports: false |
|
}); |
|
|
|
if (resolved) { |
|
return resolved; |
|
} |
|
} |
|
|
|
} |
|
else if (isMappings(exports)) { |
|
|
|
const resolvedMatch = await resolvePackageImportsExports(context, { |
|
matchKey: subpath, |
|
matchObj: exports, |
|
isImports: false |
|
}); |
|
|
|
if (resolvedMatch) { |
|
return resolvedMatch; |
|
} |
|
} |
|
|
|
throw new InvalidModuleSpecifierError(context); |
|
} |
|
|
|
async function resolvePackageImports({ importSpecifier, importer, moduleDirs, conditions, resolveId }) { |
|
const result = await findPackageJson(importer, moduleDirs); |
|
if (!result) { |
|
throw new Error(`${createBaseErrorMsg(importSpecifier, importer)}. Could not find a parent package.json.`); |
|
} |
|
const { pkgPath, pkgJsonPath, pkgJson } = result; |
|
const pkgURL = pathToFileURL(`${pkgPath}/`); |
|
const context = { |
|
importer, |
|
importSpecifier, |
|
moduleDirs, |
|
pkgURL, |
|
pkgJsonPath, |
|
conditions, |
|
resolveId |
|
}; |
|
|
|
if (!importSpecifier.startsWith('#')) { |
|
throw new InvalidModuleSpecifierError(context, true, 'Invalid import specifier.'); |
|
} |
|
|
|
if (importSpecifier === '#' || importSpecifier.startsWith('#/')) { |
|
|
|
throw new InvalidModuleSpecifierError(context, true, 'Invalid import specifier.'); |
|
} |
|
const { imports } = pkgJson; |
|
if (!imports) { |
|
throw new InvalidModuleSpecifierError(context, true); |
|
} |
|
|
|
|
|
return resolvePackageImportsExports(context, { |
|
matchKey: importSpecifier, |
|
matchObj: imports, |
|
isImports: true |
|
}); |
|
} |
|
|
|
const resolveImportPath = promisify(resolve$1); |
|
const readFile = promisify(fs.readFile); |
|
|
|
async function getPackageJson(importer, pkgName, resolveOptions, moduleDirectories) { |
|
if (importer) { |
|
const selfPackageJsonResult = await findPackageJson(importer, moduleDirectories); |
|
if (selfPackageJsonResult && selfPackageJsonResult.pkgJson.name === pkgName) { |
|
|
|
return selfPackageJsonResult; |
|
} |
|
} |
|
|
|
try { |
|
const pkgJsonPath = await resolveImportPath(`${pkgName}/package.json`, resolveOptions); |
|
const pkgJson = JSON.parse(await readFile(pkgJsonPath, 'utf-8')); |
|
return { pkgJsonPath, pkgJson, pkgPath: dirname(pkgJsonPath) }; |
|
} catch (_) { |
|
return null; |
|
} |
|
} |
|
|
|
async function resolveIdClassic({ |
|
importSpecifier, |
|
packageInfoCache, |
|
extensions, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
baseDir, |
|
moduleDirectories, |
|
modulePaths, |
|
rootDir, |
|
ignoreSideEffectsForRoot |
|
}) { |
|
let hasModuleSideEffects = () => null; |
|
let hasPackageEntry = true; |
|
let packageBrowserField = false; |
|
let packageInfo; |
|
|
|
const filter = (pkg, pkgPath) => { |
|
const info = getPackageInfo({ |
|
cache: packageInfoCache, |
|
extensions, |
|
pkg, |
|
pkgPath, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
rootDir, |
|
ignoreSideEffectsForRoot |
|
}); |
|
|
|
({ packageInfo, hasModuleSideEffects, hasPackageEntry, packageBrowserField } = info); |
|
|
|
return info.cachedPkg; |
|
}; |
|
|
|
const resolveOptions = { |
|
basedir: baseDir, |
|
readFile: readCachedFile, |
|
isFile: isFileCached, |
|
isDirectory: isDirCached, |
|
extensions, |
|
includeCoreModules: false, |
|
moduleDirectory: moduleDirectories, |
|
paths: modulePaths, |
|
preserveSymlinks, |
|
packageFilter: filter |
|
}; |
|
|
|
let location; |
|
try { |
|
location = await resolveImportPath(importSpecifier, resolveOptions); |
|
} catch (error) { |
|
if (error.code !== 'MODULE_NOT_FOUND') { |
|
throw error; |
|
} |
|
return null; |
|
} |
|
|
|
return { |
|
location: preserveSymlinks ? location : await resolveSymlink(location), |
|
hasModuleSideEffects, |
|
hasPackageEntry, |
|
packageBrowserField, |
|
packageInfo |
|
}; |
|
} |
|
|
|
async function resolveWithExportMap({ |
|
importer, |
|
importSpecifier, |
|
exportConditions, |
|
packageInfoCache, |
|
extensions, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
baseDir, |
|
moduleDirectories, |
|
modulePaths, |
|
rootDir, |
|
ignoreSideEffectsForRoot, |
|
allowExportsFolderMapping |
|
}) { |
|
if (importSpecifier.startsWith('#')) { |
|
|
|
const resolveResult = await resolvePackageImports({ |
|
importSpecifier, |
|
importer, |
|
moduleDirs: moduleDirectories, |
|
conditions: exportConditions, |
|
resolveId(id ) { |
|
return resolveIdClassic({ |
|
importSpecifier: id, |
|
packageInfoCache, |
|
extensions, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
baseDir, |
|
moduleDirectories, |
|
modulePaths |
|
}); |
|
} |
|
}); |
|
|
|
const location = fileURLToPath(resolveResult); |
|
return { |
|
location: preserveSymlinks ? location : await resolveSymlink(location), |
|
hasModuleSideEffects: () => null, |
|
hasPackageEntry: true, |
|
packageBrowserField: false, |
|
|
|
packageInfo: undefined |
|
}; |
|
} |
|
|
|
const pkgName = getPackageName(importSpecifier); |
|
if (pkgName) { |
|
|
|
let hasModuleSideEffects = () => null; |
|
let hasPackageEntry = true; |
|
let packageBrowserField = false; |
|
let packageInfo; |
|
|
|
const filter = (pkg, pkgPath) => { |
|
const info = getPackageInfo({ |
|
cache: packageInfoCache, |
|
extensions, |
|
pkg, |
|
pkgPath, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
rootDir, |
|
ignoreSideEffectsForRoot |
|
}); |
|
|
|
({ packageInfo, hasModuleSideEffects, hasPackageEntry, packageBrowserField } = info); |
|
|
|
return info.cachedPkg; |
|
}; |
|
|
|
const resolveOptions = { |
|
basedir: baseDir, |
|
readFile: readCachedFile, |
|
isFile: isFileCached, |
|
isDirectory: isDirCached, |
|
extensions, |
|
includeCoreModules: false, |
|
moduleDirectory: moduleDirectories, |
|
paths: modulePaths, |
|
preserveSymlinks, |
|
packageFilter: filter |
|
}; |
|
|
|
const result = await getPackageJson(importer, pkgName, resolveOptions, moduleDirectories); |
|
|
|
if (result && result.pkgJson.exports) { |
|
const { pkgJson, pkgJsonPath } = result; |
|
const subpath = |
|
pkgName === importSpecifier ? '.' : `.${importSpecifier.substring(pkgName.length)}`; |
|
const pkgDr = pkgJsonPath.replace('package.json', ''); |
|
const pkgURL = pathToFileURL(pkgDr); |
|
|
|
const context = { |
|
importer, |
|
importSpecifier, |
|
moduleDirs: moduleDirectories, |
|
pkgURL, |
|
pkgJsonPath, |
|
allowExportsFolderMapping, |
|
conditions: exportConditions |
|
}; |
|
const resolvedPackageExport = await resolvePackageExports(context, subpath, pkgJson.exports); |
|
const location = fileURLToPath(resolvedPackageExport); |
|
if (location) { |
|
return { |
|
location: preserveSymlinks ? location : await resolveSymlink(location), |
|
hasModuleSideEffects, |
|
hasPackageEntry, |
|
packageBrowserField, |
|
packageInfo |
|
}; |
|
} |
|
} |
|
} |
|
|
|
return null; |
|
} |
|
|
|
async function resolveWithClassic({ |
|
importer, |
|
importSpecifierList, |
|
exportConditions, |
|
warn, |
|
packageInfoCache, |
|
extensions, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
baseDir, |
|
moduleDirectories, |
|
modulePaths, |
|
rootDir, |
|
ignoreSideEffectsForRoot |
|
}) { |
|
for (let i = 0; i < importSpecifierList.length; i++) { |
|
|
|
const result = await resolveIdClassic({ |
|
importer, |
|
importSpecifier: importSpecifierList[i], |
|
exportConditions, |
|
warn, |
|
packageInfoCache, |
|
extensions, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
baseDir, |
|
moduleDirectories, |
|
modulePaths, |
|
rootDir, |
|
ignoreSideEffectsForRoot |
|
}); |
|
|
|
if (result) { |
|
return result; |
|
} |
|
} |
|
|
|
return null; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
async function resolveImportSpecifiers({ |
|
importer, |
|
importSpecifierList, |
|
exportConditions, |
|
warn, |
|
packageInfoCache, |
|
extensions, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
baseDir, |
|
moduleDirectories, |
|
modulePaths, |
|
rootDir, |
|
ignoreSideEffectsForRoot, |
|
allowExportsFolderMapping |
|
}) { |
|
try { |
|
const exportMapRes = await resolveWithExportMap({ |
|
importer, |
|
importSpecifier: importSpecifierList[0], |
|
exportConditions, |
|
packageInfoCache, |
|
extensions, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
baseDir, |
|
moduleDirectories, |
|
modulePaths, |
|
rootDir, |
|
ignoreSideEffectsForRoot, |
|
allowExportsFolderMapping |
|
}); |
|
if (exportMapRes) return exportMapRes; |
|
} catch (error) { |
|
if (error instanceof ResolveError) { |
|
warn(error); |
|
return null; |
|
} |
|
throw error; |
|
} |
|
|
|
|
|
return resolveWithClassic({ |
|
importer, |
|
importSpecifierList, |
|
exportConditions, |
|
warn, |
|
packageInfoCache, |
|
extensions, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
baseDir, |
|
moduleDirectories, |
|
modulePaths, |
|
rootDir, |
|
ignoreSideEffectsForRoot |
|
}); |
|
} |
|
|
|
const versionRegexp = /\^(\d+\.\d+\.\d+)/g; |
|
|
|
function validateVersion(actualVersion, peerDependencyVersion) { |
|
let minMajor = Infinity; |
|
let minMinor = Infinity; |
|
let minPatch = Infinity; |
|
let foundVersion; |
|
|
|
while ((foundVersion = versionRegexp.exec(peerDependencyVersion))) { |
|
const [foundMajor, foundMinor, foundPatch] = foundVersion[1].split('.').map(Number); |
|
if (foundMajor < minMajor) { |
|
minMajor = foundMajor; |
|
minMinor = foundMinor; |
|
minPatch = foundPatch; |
|
} |
|
} |
|
if (!actualVersion) { |
|
throw new Error( |
|
`Insufficient Rollup version: "@rollup/plugin-node-resolve" requires at least rollup@${minMajor}.${minMinor}.${minPatch}.` |
|
); |
|
} |
|
const [major, minor, patch] = actualVersion.split('.').map(Number); |
|
if ( |
|
major < minMajor || |
|
(major === minMajor && (minor < minMinor || (minor === minMinor && patch < minPatch))) |
|
) { |
|
throw new Error( |
|
`Insufficient rollup version: "@rollup/plugin-node-resolve" requires at least rollup@${minMajor}.${minMinor}.${minPatch} but found rollup@${actualVersion}.` |
|
); |
|
} |
|
} |
|
|
|
|
|
|
|
const ES6_BROWSER_EMPTY = '\0node-resolve:empty.js'; |
|
const deepFreeze = (object) => { |
|
Object.freeze(object); |
|
|
|
for (const value of Object.values(object)) { |
|
if (typeof value === 'object' && !Object.isFrozen(value)) { |
|
deepFreeze(value); |
|
} |
|
} |
|
|
|
return object; |
|
}; |
|
|
|
const baseConditions = ['default', 'module']; |
|
const baseConditionsEsm = [...baseConditions, 'import']; |
|
const baseConditionsCjs = [...baseConditions, 'require']; |
|
const defaults = { |
|
dedupe: [], |
|
|
|
|
|
extensions: ['.mjs', '.js', '.json', '.node'], |
|
resolveOnly: [], |
|
moduleDirectories: ['node_modules'], |
|
modulePaths: [], |
|
ignoreSideEffectsForRoot: false, |
|
|
|
allowExportsFolderMapping: true |
|
}; |
|
const DEFAULTS = deepFreeze(deepMerge({}, defaults)); |
|
|
|
function nodeResolve(opts = {}) { |
|
const { warnings } = handleDeprecatedOptions(opts); |
|
|
|
const options = { ...defaults, ...opts }; |
|
const { extensions, jail, moduleDirectories, modulePaths, ignoreSideEffectsForRoot } = options; |
|
const conditionsEsm = [...baseConditionsEsm, ...(options.exportConditions || [])]; |
|
const conditionsCjs = [...baseConditionsCjs, ...(options.exportConditions || [])]; |
|
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 = resolve(options.rootDir || process.cwd()); |
|
let { dedupe } = options; |
|
let rollupOptions; |
|
|
|
if (moduleDirectories.some((name) => name.includes('/'))) { |
|
throw new Error( |
|
'`moduleDirectories` option must only contain directory names. If you want to load modules from somewhere not supported by the default module resolution algorithm, see `modulePaths`.' |
|
); |
|
} |
|
|
|
if (typeof dedupe !== 'function') { |
|
dedupe = (importee) => |
|
options.dedupe.includes(importee) || options.dedupe.includes(getPackageName(importee)); |
|
} |
|
|
|
|
|
const allowPatterns = (patterns) => { |
|
const regexPatterns = patterns.map((pattern) => { |
|
if (pattern instanceof RegExp) { |
|
return pattern; |
|
} |
|
const normalized = pattern.replace(/[\\^$*+?.()|[\]{}]/g, '\\$&'); |
|
return new RegExp(`^${normalized}$`); |
|
}); |
|
return (id) => !regexPatterns.length || regexPatterns.some((pattern) => pattern.test(id)); |
|
}; |
|
|
|
const resolveOnly = |
|
typeof options.resolveOnly === 'function' |
|
? options.resolveOnly |
|
: allowPatterns(options.resolveOnly); |
|
|
|
const browserMapCache = new Map(); |
|
let preserveSymlinks; |
|
|
|
const resolveLikeNode = async (context, importee, importer, custom) => { |
|
|
|
const [importPath, params] = importee.split('?'); |
|
const importSuffix = `${params ? `?${params}` : ''}`; |
|
importee = importPath; |
|
|
|
const baseDir = !importer || dedupe(importee) ? rootDir : dirname(importer); |
|
|
|
|
|
const browser = browserMapCache.get(importer); |
|
if (useBrowserOverrides && browser) { |
|
const resolvedImportee = resolve(baseDir, importee); |
|
if (browser[importee] === false || browser[resolvedImportee] === false) { |
|
return { id: ES6_BROWSER_EMPTY }; |
|
} |
|
const browserImportee = |
|
(importee[0] !== '.' && 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) { |
|
|
|
id += `/${parts.shift()}`; |
|
} else if (id[0] === '.') { |
|
|
|
id = resolve(baseDir, importee); |
|
isRelativeImport = true; |
|
} |
|
|
|
|
|
if (!isRelativeImport && !resolveOnly(id)) { |
|
if (normalizeInput(rollupOptions.input).includes(importee)) { |
|
return null; |
|
} |
|
return false; |
|
} |
|
|
|
const importSpecifierList = [importee]; |
|
|
|
if (importer === undefined && !importee[0].match(/^\.?\.?\//)) { |
|
|
|
|
|
|
|
|
|
|
|
importSpecifierList.push(`./${importee}`); |
|
} |
|
|
|
|
|
|
|
if (importer && /\.(ts|mts|cts|tsx)$/.test(importer)) { |
|
for (const [importeeExt, resolvedExt] of [ |
|
['.js', '.ts'], |
|
['.js', '.tsx'], |
|
['.jsx', '.tsx'], |
|
['.mjs', '.mts'], |
|
['.cjs', '.cts'] |
|
]) { |
|
if (importee.endsWith(importeeExt) && extensions.includes(resolvedExt)) { |
|
importSpecifierList.push(importee.slice(0, -importeeExt.length) + resolvedExt); |
|
} |
|
} |
|
} |
|
|
|
const warn = (...args) => context.warn(...args); |
|
const isRequire = custom && custom['node-resolve'] && custom['node-resolve'].isRequire; |
|
const exportConditions = isRequire ? conditionsCjs : conditionsEsm; |
|
|
|
if (useBrowserOverrides && !exportConditions.includes('browser')) |
|
exportConditions.push('browser'); |
|
|
|
const resolvedWithoutBuiltins = await resolveImportSpecifiers({ |
|
importer, |
|
importSpecifierList, |
|
exportConditions, |
|
warn, |
|
packageInfoCache, |
|
extensions, |
|
mainFields, |
|
preserveSymlinks, |
|
useBrowserOverrides, |
|
baseDir, |
|
moduleDirectories, |
|
modulePaths, |
|
rootDir, |
|
ignoreSideEffectsForRoot, |
|
allowExportsFolderMapping: options.allowExportsFolderMapping |
|
}); |
|
|
|
const importeeIsBuiltin = isBuiltinModule(importee); |
|
const resolved = |
|
importeeIsBuiltin && preferBuiltins |
|
? { |
|
packageInfo: undefined, |
|
hasModuleSideEffects: () => null, |
|
hasPackageEntry: true, |
|
packageBrowserField: false |
|
} |
|
: resolvedWithoutBuiltins; |
|
if (!resolved) { |
|
return null; |
|
} |
|
|
|
const { packageInfo, hasModuleSideEffects, hasPackageEntry, packageBrowserField } = resolved; |
|
let { location } = resolved; |
|
if (packageBrowserField) { |
|
if (Object.prototype.hasOwnProperty.call(packageBrowserField, location)) { |
|
if (!packageBrowserField[location]) { |
|
browserMapCache.set(location, packageBrowserField); |
|
return { id: ES6_BROWSER_EMPTY }; |
|
} |
|
location = packageBrowserField[location]; |
|
} |
|
browserMapCache.set(location, packageBrowserField); |
|
} |
|
|
|
if (hasPackageEntry && !preserveSymlinks) { |
|
const exists = await fileExists(location); |
|
if (exists) { |
|
location = await realpath(location); |
|
} |
|
} |
|
|
|
idToPackageInfo.set(location, packageInfo); |
|
|
|
if (hasPackageEntry) { |
|
if (importeeIsBuiltin && preferBuiltins) { |
|
if (!isPreferBuiltinsSet && resolvedWithoutBuiltins && resolved !== importee) { |
|
context.warn( |
|
`preferring built-in module '${importee}' over local alternative at '${resolvedWithoutBuiltins.location}', pass 'preferBuiltins: false' to disable this behavior or 'preferBuiltins: true' to disable this warning` |
|
); |
|
} |
|
return false; |
|
} else if (jail && location.indexOf(normalize(jail.trim(sep))) !== 0) { |
|
return null; |
|
} |
|
} |
|
|
|
if (options.modulesOnly && (await fileExists(location))) { |
|
const code = await readFile$1(location, 'utf-8'); |
|
if (isModule(code)) { |
|
return { |
|
id: `${location}${importSuffix}`, |
|
moduleSideEffects: hasModuleSideEffects(location) |
|
}; |
|
} |
|
return null; |
|
} |
|
return { |
|
id: `${location}${importSuffix}`, |
|
moduleSideEffects: hasModuleSideEffects(location) |
|
}; |
|
}; |
|
|
|
return { |
|
name: 'node-resolve', |
|
|
|
version, |
|
|
|
buildStart(buildOptions) { |
|
validateVersion(this.meta.rollupVersion, peerDependencies.rollup); |
|
rollupOptions = buildOptions; |
|
|
|
for (const warning of warnings) { |
|
this.warn(warning); |
|
} |
|
|
|
({ preserveSymlinks } = buildOptions); |
|
}, |
|
|
|
generateBundle() { |
|
readCachedFile.clear(); |
|
isFileCached.clear(); |
|
isDirCached.clear(); |
|
}, |
|
|
|
resolveId: { |
|
order: 'post', |
|
async handler(importee, importer, resolveOptions) { |
|
if (importee === ES6_BROWSER_EMPTY) { |
|
return importee; |
|
} |
|
|
|
if (/\0/.test(importee)) return null; |
|
|
|
const { custom = {} } = resolveOptions; |
|
const { 'node-resolve': { resolved: alreadyResolved } = {} } = custom; |
|
if (alreadyResolved) { |
|
return alreadyResolved; |
|
} |
|
|
|
if (/\0/.test(importer)) { |
|
importer = undefined; |
|
} |
|
|
|
const resolved = await resolveLikeNode(this, importee, importer, custom); |
|
if (resolved) { |
|
|
|
|
|
|
|
|
|
|
|
const resolvedResolved = await this.resolve(resolved.id, importer, { |
|
...resolveOptions, |
|
skipSelf: false, |
|
custom: { ...custom, 'node-resolve': { ...custom['node-resolve'], resolved, importee } } |
|
}); |
|
if (resolvedResolved) { |
|
|
|
if (resolvedResolved.external) { |
|
return false; |
|
} |
|
|
|
|
|
if (resolvedResolved.id !== resolved.id) { |
|
return resolvedResolved; |
|
} |
|
|
|
return { ...resolved, meta: resolvedResolved.meta }; |
|
} |
|
} |
|
return resolved; |
|
} |
|
}, |
|
|
|
load(importee) { |
|
if (importee === ES6_BROWSER_EMPTY) { |
|
return 'export default {};'; |
|
} |
|
return null; |
|
}, |
|
|
|
getPackageInfoForId(id) { |
|
return idToPackageInfo.get(id); |
|
} |
|
}; |
|
} |
|
|
|
export { DEFAULTS, nodeResolve as default, nodeResolve }; |
|
|
|
|