ImportsFieldPlugin.js 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Ivan Kopeykin @vankop
  4. */
  5. "use strict";
  6. const DescriptionFileUtils = require("./DescriptionFileUtils");
  7. const forEachBail = require("./forEachBail");
  8. const { processImportsField } = require("./util/entrypoints");
  9. const { parseIdentifier } = require("./util/identifier");
  10. const { invalidSegmentRegEx } = require("./util/path");
  11. /** @typedef {import("./Resolver")} Resolver */
  12. /** @typedef {import("./Resolver").JsonObject} JsonObject */
  13. /** @typedef {import("./Resolver").ResolveRequest} ResolveRequest */
  14. /** @typedef {import("./Resolver").ResolveStepHook} ResolveStepHook */
  15. /** @typedef {import("./util/entrypoints").FieldProcessor} FieldProcessor */
  16. /** @typedef {import("./util/entrypoints").ImportsField} ImportsField */
  17. const dotCode = ".".charCodeAt(0);
  18. module.exports = class ImportsFieldPlugin {
  19. /**
  20. * @param {string | ResolveStepHook} source source
  21. * @param {Set<string>} conditionNames condition names
  22. * @param {string | string[]} fieldNamePath name path
  23. * @param {string | ResolveStepHook} targetFile target file
  24. * @param {string | ResolveStepHook} targetPackage target package
  25. */
  26. constructor(
  27. source,
  28. conditionNames,
  29. fieldNamePath,
  30. targetFile,
  31. targetPackage,
  32. ) {
  33. this.source = source;
  34. this.targetFile = targetFile;
  35. this.targetPackage = targetPackage;
  36. this.conditionNames = conditionNames;
  37. this.fieldName = fieldNamePath;
  38. // `null` is cached for description files that have no imports field,
  39. // so subsequent resolves against the same package.json skip the
  40. // `DescriptionFileUtils.getField` walk entirely.
  41. /** @type {WeakMap<JsonObject, FieldProcessor | null>} */
  42. this._fieldProcessorCache = new WeakMap();
  43. }
  44. /**
  45. * @param {Resolver} resolver the resolver
  46. * @returns {void}
  47. */
  48. apply(resolver) {
  49. const targetFile = resolver.ensureHook(this.targetFile);
  50. const targetPackage = resolver.ensureHook(this.targetPackage);
  51. resolver
  52. .getHook(this.source)
  53. .tapAsync("ImportsFieldPlugin", (request, resolveContext, callback) => {
  54. // When there is no description file, abort
  55. if (!request.descriptionFileData || request.request === undefined) {
  56. return callback();
  57. }
  58. const { descriptionFileData } = request;
  59. // Skip the concat when there's nothing to append — the common
  60. // case has empty query/fragment, so this avoids an allocation
  61. // per resolve. Mirrors the pattern in ExportsFieldPlugin.
  62. const remainingRequest =
  63. request.query || request.fragment
  64. ? request.request + request.query + request.fragment
  65. : request.request;
  66. /** @type {string[]} */
  67. let paths;
  68. /** @type {string | null} */
  69. let usedField;
  70. try {
  71. // Look up the cached processor first. On a cache hit we
  72. // avoid re-walking the description file for the imports
  73. // field — and `null` is cached for description files that
  74. // have no imports field at all, so those skip the read
  75. // entirely. `processImportsField` can throw on a
  76. // malformed `imports` map, so building the processor must
  77. // stay inside this try/catch.
  78. let fieldProcessor =
  79. this._fieldProcessorCache.get(descriptionFileData);
  80. if (
  81. fieldProcessor === undefined &&
  82. !this._fieldProcessorCache.has(descriptionFileData)
  83. ) {
  84. const importsField =
  85. /** @type {ImportsField | null | undefined} */
  86. (
  87. DescriptionFileUtils.getField(
  88. descriptionFileData,
  89. this.fieldName,
  90. )
  91. );
  92. fieldProcessor = importsField
  93. ? processImportsField(importsField)
  94. : null;
  95. this._fieldProcessorCache.set(descriptionFileData, fieldProcessor);
  96. }
  97. if (!fieldProcessor) return callback();
  98. if (request.directory) {
  99. return callback(
  100. new Error(
  101. `Resolving to directories is not possible with the imports field (request was ${remainingRequest}/)`,
  102. ),
  103. );
  104. }
  105. [paths, usedField] = fieldProcessor(
  106. remainingRequest,
  107. this.conditionNames,
  108. );
  109. } catch (/** @type {unknown} */ err) {
  110. if (resolveContext.log) {
  111. resolveContext.log(
  112. `Imports field in ${request.descriptionFilePath} can't be processed: ${err}`,
  113. );
  114. }
  115. return callback(/** @type {Error} */ (err));
  116. }
  117. if (paths.length === 0) {
  118. return callback(
  119. new Error(
  120. `Package import ${remainingRequest} is not imported from package ${request.descriptionFileRoot} (see imports field in ${request.descriptionFilePath})`,
  121. ),
  122. );
  123. }
  124. forEachBail(
  125. paths,
  126. /**
  127. * @param {string} path path
  128. * @param {(err?: null | Error, result?: null | ResolveRequest) => void} callback callback
  129. * @param {number} i index
  130. * @returns {void}
  131. */
  132. (path, callback, i) => {
  133. const parsedIdentifier = parseIdentifier(path);
  134. if (!parsedIdentifier) return callback();
  135. const [path_, query, fragment] = parsedIdentifier;
  136. switch (path_.charCodeAt(0)) {
  137. // should be relative
  138. case dotCode: {
  139. const withoutDotSlash = path_.slice(2);
  140. if (invalidSegmentRegEx.test(withoutDotSlash)) {
  141. if (paths.length === i) {
  142. return callback(
  143. new Error(
  144. `Invalid "imports" target "${path}" defined for "${usedField}" in the package config ${request.descriptionFilePath}, targets must start with "./"`,
  145. ),
  146. );
  147. }
  148. return callback();
  149. }
  150. /** @type {ResolveRequest} */
  151. const obj = {
  152. ...request,
  153. request: undefined,
  154. path: resolver.join(
  155. /** @type {string} */ (request.descriptionFileRoot),
  156. path_,
  157. ),
  158. relativePath: path_,
  159. query,
  160. fragment,
  161. };
  162. resolver.doResolve(
  163. targetFile,
  164. obj,
  165. `using imports field: ${path}`,
  166. resolveContext,
  167. (err, result) => {
  168. if (err) return callback(err);
  169. // Don't allow to continue - https://github.com/webpack/enhanced-resolve/issues/400
  170. if (result === undefined) return callback(null, null);
  171. callback(null, result);
  172. },
  173. );
  174. break;
  175. }
  176. // package resolving
  177. default: {
  178. /** @type {ResolveRequest} */
  179. const obj = {
  180. ...request,
  181. request: path_,
  182. relativePath: path_,
  183. fullySpecified: true,
  184. query,
  185. fragment,
  186. };
  187. resolver.doResolve(
  188. targetPackage,
  189. obj,
  190. `using imports field: ${path}`,
  191. resolveContext,
  192. (err, result) => {
  193. if (err) return callback(err);
  194. // Don't allow to continue - https://github.com/webpack/enhanced-resolve/issues/400
  195. if (result === undefined) return callback(null, null);
  196. callback(null, result);
  197. },
  198. );
  199. }
  200. }
  201. },
  202. /**
  203. * @param {null | Error=} err error
  204. * @param {null | ResolveRequest=} result result
  205. * @returns {void}
  206. */
  207. (err, result) => callback(err, result || null),
  208. );
  209. });
  210. }
  211. };