ExportsFieldPlugin.js 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206
  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 { processExportsField } = require("./util/entrypoints");
  9. const { parseIdentifier } = require("./util/identifier");
  10. const {
  11. deprecatedInvalidSegmentRegEx,
  12. invalidSegmentRegEx,
  13. } = require("./util/path");
  14. /** @typedef {import("./Resolver")} Resolver */
  15. /** @typedef {import("./Resolver").JsonObject} JsonObject */
  16. /** @typedef {import("./Resolver").ResolveRequest} ResolveRequest */
  17. /** @typedef {import("./Resolver").ResolveStepHook} ResolveStepHook */
  18. /** @typedef {import("./util/entrypoints").ExportsField} ExportsField */
  19. /** @typedef {import("./util/entrypoints").FieldProcessor} FieldProcessor */
  20. module.exports = class ExportsFieldPlugin {
  21. /**
  22. * @param {string | ResolveStepHook} source source
  23. * @param {Set<string>} conditionNames condition names
  24. * @param {string | string[]} fieldNamePath name path
  25. * @param {string | ResolveStepHook} target target
  26. */
  27. constructor(source, conditionNames, fieldNamePath, target) {
  28. this.source = source;
  29. this.target = target;
  30. this.conditionNames = conditionNames;
  31. this.fieldName = fieldNamePath;
  32. /** @type {WeakMap<JsonObject, FieldProcessor>} */
  33. this.fieldProcessorCache = new WeakMap();
  34. }
  35. /**
  36. * @param {Resolver} resolver the resolver
  37. * @returns {void}
  38. */
  39. apply(resolver) {
  40. const target = resolver.ensureHook(this.target);
  41. resolver
  42. .getHook(this.source)
  43. .tapAsync("ExportsFieldPlugin", (request, resolveContext, callback) => {
  44. // When there is no description file, abort
  45. if (!request.descriptionFilePath) return callback();
  46. if (
  47. // When the description file is inherited from parent, abort
  48. // (There is no description file inside of this package)
  49. request.relativePath !== "." ||
  50. request.request === undefined
  51. ) {
  52. return callback();
  53. }
  54. const remainingRequest =
  55. request.query || request.fragment
  56. ? (request.request === "." ? "./" : request.request) +
  57. request.query +
  58. request.fragment
  59. : request.request;
  60. const exportsField =
  61. /** @type {ExportsField|null|undefined} */
  62. (
  63. DescriptionFileUtils.getField(
  64. /** @type {JsonObject} */ (request.descriptionFileData),
  65. this.fieldName,
  66. )
  67. );
  68. if (!exportsField) return callback();
  69. if (request.directory) {
  70. return callback(
  71. new Error(
  72. `Resolving to directories is not possible with the exports field (request was ${remainingRequest}/)`,
  73. ),
  74. );
  75. }
  76. /** @type {string[]} */
  77. let paths;
  78. /** @type {string | null} */
  79. let usedField;
  80. try {
  81. // We attach the cache to the description file instead of the exportsField value
  82. // because we use a WeakMap and the exportsField could be a string too.
  83. // Description file is always an object when exports field can be accessed.
  84. let fieldProcessor = this.fieldProcessorCache.get(
  85. /** @type {JsonObject} */ (request.descriptionFileData),
  86. );
  87. if (fieldProcessor === undefined) {
  88. fieldProcessor = processExportsField(exportsField);
  89. this.fieldProcessorCache.set(
  90. /** @type {JsonObject} */ (request.descriptionFileData),
  91. fieldProcessor,
  92. );
  93. }
  94. [paths, usedField] = fieldProcessor(
  95. remainingRequest,
  96. this.conditionNames,
  97. );
  98. } catch (/** @type {unknown} */ err) {
  99. if (resolveContext.log) {
  100. resolveContext.log(
  101. `Exports field in ${request.descriptionFilePath} can't be processed: ${err}`,
  102. );
  103. }
  104. return callback(/** @type {Error} */ (err));
  105. }
  106. if (paths.length === 0) {
  107. const conditions = [...this.conditionNames];
  108. const conditionsStr =
  109. conditions.length === 1
  110. ? `the condition "${conditions[0]}"`
  111. : `the conditions ${JSON.stringify(conditions)}`;
  112. return callback(
  113. new Error(
  114. `"${remainingRequest}" is not exported under ${conditionsStr} from package ${request.descriptionFileRoot} (see exports field in ${request.descriptionFilePath})`,
  115. ),
  116. );
  117. }
  118. forEachBail(
  119. paths,
  120. /**
  121. * @param {string} path path
  122. * @param {(err?: null|Error, result?: null|ResolveRequest) => void} callback callback
  123. * @param {number} i index
  124. * @returns {void}
  125. */
  126. (path, callback, i) => {
  127. const parsedIdentifier = parseIdentifier(path);
  128. if (!parsedIdentifier) return callback();
  129. const [relativePath, query, fragment] = parsedIdentifier;
  130. if (relativePath.length === 0 || !relativePath.startsWith("./")) {
  131. if (paths.length === i) {
  132. return callback(
  133. new Error(
  134. `Invalid "exports" target "${path}" defined for "${usedField}" in the package config ${request.descriptionFilePath}, targets must start with "./"`,
  135. ),
  136. );
  137. }
  138. return callback();
  139. }
  140. if (
  141. invalidSegmentRegEx.exec(relativePath.slice(2)) !== null &&
  142. deprecatedInvalidSegmentRegEx.test(relativePath.slice(2))
  143. ) {
  144. if (paths.length === i) {
  145. return callback(
  146. new Error(
  147. `Invalid "exports" target "${path}" defined for "${usedField}" in the package config ${request.descriptionFilePath}, targets must start with "./"`,
  148. ),
  149. );
  150. }
  151. return callback();
  152. }
  153. /** @type {ResolveRequest} */
  154. const obj = {
  155. ...request,
  156. request: undefined,
  157. path: resolver.join(
  158. /** @type {string} */ (request.descriptionFileRoot),
  159. relativePath,
  160. ),
  161. relativePath,
  162. query,
  163. fragment,
  164. };
  165. resolver.doResolve(
  166. target,
  167. obj,
  168. `using exports field: ${path}`,
  169. resolveContext,
  170. (err, result) => {
  171. if (err) return callback(err);
  172. // Don't allow to continue - https://github.com/webpack/enhanced-resolve/issues/400
  173. if (result === undefined) return callback(null, null);
  174. callback(null, result);
  175. },
  176. );
  177. },
  178. /**
  179. * @param {(null | Error)=} err error
  180. * @param {(null | ResolveRequest)=} result result
  181. * @returns {void}
  182. */
  183. (err, result) => callback(err, result || null),
  184. );
  185. });
  186. }
  187. };