You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
171 lines
4.6 KiB
171 lines
4.6 KiB
/* |
|
MIT License http://www.opensource.org/licenses/mit-license.php |
|
Author Ivan Kopeykin @vankop |
|
*/ |
|
|
|
"use strict"; |
|
|
|
const path = require("path"); |
|
const DescriptionFileUtils = require("./DescriptionFileUtils"); |
|
const forEachBail = require("./forEachBail"); |
|
const { processImportsField } = require("./util/entrypoints"); |
|
const { parseIdentifier } = require("./util/identifier"); |
|
|
|
/** @typedef {import("./Resolver")} Resolver */ |
|
/** @typedef {import("./Resolver").ResolveStepHook} ResolveStepHook */ |
|
/** @typedef {import("./util/entrypoints").FieldProcessor} FieldProcessor */ |
|
/** @typedef {import("./util/entrypoints").ImportsField} ImportsField */ |
|
|
|
const dotCode = ".".charCodeAt(0); |
|
|
|
module.exports = class ImportsFieldPlugin { |
|
/** |
|
* @param {string | ResolveStepHook} source source |
|
* @param {Set<string>} conditionNames condition names |
|
* @param {string | string[]} fieldNamePath name path |
|
* @param {string | ResolveStepHook} targetFile target file |
|
* @param {string | ResolveStepHook} targetPackage target package |
|
*/ |
|
constructor( |
|
source, |
|
conditionNames, |
|
fieldNamePath, |
|
targetFile, |
|
targetPackage |
|
) { |
|
this.source = source; |
|
this.targetFile = targetFile; |
|
this.targetPackage = targetPackage; |
|
this.conditionNames = conditionNames; |
|
this.fieldName = fieldNamePath; |
|
/** @type {WeakMap<any, FieldProcessor>} */ |
|
this.fieldProcessorCache = new WeakMap(); |
|
} |
|
|
|
/** |
|
* @param {Resolver} resolver the resolver |
|
* @returns {void} |
|
*/ |
|
apply(resolver) { |
|
const targetFile = resolver.ensureHook(this.targetFile); |
|
const targetPackage = resolver.ensureHook(this.targetPackage); |
|
|
|
resolver |
|
.getHook(this.source) |
|
.tapAsync("ImportsFieldPlugin", (request, resolveContext, callback) => { |
|
// When there is no description file, abort |
|
if (!request.descriptionFilePath || request.request === undefined) { |
|
return callback(); |
|
} |
|
|
|
const remainingRequest = |
|
request.request + request.query + request.fragment; |
|
/** @type {ImportsField|null} */ |
|
const importsField = DescriptionFileUtils.getField( |
|
request.descriptionFileData, |
|
this.fieldName |
|
); |
|
if (!importsField) return callback(); |
|
|
|
if (request.directory) { |
|
return callback( |
|
new Error( |
|
`Resolving to directories is not possible with the imports field (request was ${remainingRequest}/)` |
|
) |
|
); |
|
} |
|
|
|
let paths; |
|
|
|
try { |
|
// We attach the cache to the description file instead of the importsField value |
|
// because we use a WeakMap and the importsField could be a string too. |
|
// Description file is always an object when exports field can be accessed. |
|
let fieldProcessor = this.fieldProcessorCache.get( |
|
request.descriptionFileData |
|
); |
|
if (fieldProcessor === undefined) { |
|
fieldProcessor = processImportsField(importsField); |
|
this.fieldProcessorCache.set( |
|
request.descriptionFileData, |
|
fieldProcessor |
|
); |
|
} |
|
paths = fieldProcessor(remainingRequest, this.conditionNames); |
|
} catch (err) { |
|
if (resolveContext.log) { |
|
resolveContext.log( |
|
`Imports field in ${request.descriptionFilePath} can't be processed: ${err}` |
|
); |
|
} |
|
return callback(err); |
|
} |
|
|
|
if (paths.length === 0) { |
|
return callback( |
|
new Error( |
|
`Package import ${remainingRequest} is not imported from package ${request.descriptionFileRoot} (see imports field in ${request.descriptionFilePath})` |
|
) |
|
); |
|
} |
|
|
|
forEachBail( |
|
paths, |
|
(p, callback) => { |
|
const parsedIdentifier = parseIdentifier(p); |
|
|
|
if (!parsedIdentifier) return callback(); |
|
|
|
const [path_, query, fragment] = parsedIdentifier; |
|
|
|
switch (path_.charCodeAt(0)) { |
|
// should be relative |
|
case dotCode: { |
|
const obj = { |
|
...request, |
|
request: undefined, |
|
path: path.join( |
|
/** @type {string} */ (request.descriptionFileRoot), |
|
path_ |
|
), |
|
relativePath: path_, |
|
query, |
|
fragment |
|
}; |
|
|
|
resolver.doResolve( |
|
targetFile, |
|
obj, |
|
"using imports field: " + p, |
|
resolveContext, |
|
callback |
|
); |
|
break; |
|
} |
|
|
|
// package resolving |
|
default: { |
|
const obj = { |
|
...request, |
|
request: path_, |
|
relativePath: path_, |
|
fullySpecified: true, |
|
query, |
|
fragment |
|
}; |
|
|
|
resolver.doResolve( |
|
targetPackage, |
|
obj, |
|
"using imports field: " + p, |
|
resolveContext, |
|
callback |
|
); |
|
} |
|
} |
|
}, |
|
(err, result) => callback(err, result || null) |
|
); |
|
}); |
|
} |
|
};
|
|
|