ConsumeSharedPlugin.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Tobias Koppers @sokra
  4. */
  5. "use strict";
  6. const ModuleNotFoundError = require("../ModuleNotFoundError");
  7. const RuntimeGlobals = require("../RuntimeGlobals");
  8. const WebpackError = require("../WebpackError");
  9. const { parseOptions } = require("../container/options");
  10. const LazySet = require("../util/LazySet");
  11. const createSchemaValidation = require("../util/create-schema-validation");
  12. const { parseRange } = require("../util/semver");
  13. const ConsumeSharedFallbackDependency = require("./ConsumeSharedFallbackDependency");
  14. const ConsumeSharedModule = require("./ConsumeSharedModule");
  15. const ConsumeSharedRuntimeModule = require("./ConsumeSharedRuntimeModule");
  16. const ProvideForSharedDependency = require("./ProvideForSharedDependency");
  17. const { resolveMatchedConfigs } = require("./resolveMatchedConfigs");
  18. const {
  19. getDescriptionFile,
  20. getRequiredVersionFromDescriptionFile,
  21. isRequiredVersion
  22. } = require("./utils");
  23. /** @typedef {import("enhanced-resolve").ResolveContext} ResolveContext */
  24. /** @typedef {import("../../declarations/plugins/sharing/ConsumeSharedPlugin").ConsumeSharedPluginOptions} ConsumeSharedPluginOptions */
  25. /** @typedef {import("../Compiler")} Compiler */
  26. /** @typedef {import("../Compilation").FileSystemDependencies} FileSystemDependencies */
  27. /** @typedef {import("../ResolverFactory").ResolveOptionsWithDependencyType} ResolveOptionsWithDependencyType */
  28. /** @typedef {import("../util/semver").SemVerRange} SemVerRange */
  29. /** @typedef {import("./ConsumeSharedModule").ConsumeOptions} ConsumeOptions */
  30. /** @typedef {import("./utils").DescriptionFile} DescriptionFile */
  31. const validate = createSchemaValidation(
  32. require("../../schemas/plugins/sharing/ConsumeSharedPlugin.check"),
  33. () => require("../../schemas/plugins/sharing/ConsumeSharedPlugin.json"),
  34. {
  35. name: "Consume Shared Plugin",
  36. baseDataPath: "options"
  37. }
  38. );
  39. /** @type {ResolveOptionsWithDependencyType} */
  40. const RESOLVE_OPTIONS = { dependencyType: "esm" };
  41. const PLUGIN_NAME = "ConsumeSharedPlugin";
  42. class ConsumeSharedPlugin {
  43. /**
  44. * @param {ConsumeSharedPluginOptions} options options
  45. */
  46. constructor(options) {
  47. if (typeof options !== "string") {
  48. validate(options);
  49. }
  50. /** @type {[string, ConsumeOptions][]} */
  51. this._consumes = parseOptions(
  52. options.consumes,
  53. (item, key) => {
  54. if (Array.isArray(item)) throw new Error("Unexpected array in options");
  55. /** @type {ConsumeOptions} */
  56. const result =
  57. item === key || !isRequiredVersion(item)
  58. ? // item is a request/key
  59. {
  60. import: key,
  61. shareScope: options.shareScope || "default",
  62. shareKey: key,
  63. requiredVersion: undefined,
  64. packageName: undefined,
  65. strictVersion: false,
  66. singleton: false,
  67. eager: false
  68. }
  69. : // key is a request/key
  70. // item is a version
  71. {
  72. import: key,
  73. shareScope: options.shareScope || "default",
  74. shareKey: key,
  75. requiredVersion: parseRange(item),
  76. strictVersion: true,
  77. packageName: undefined,
  78. singleton: false,
  79. eager: false
  80. };
  81. return result;
  82. },
  83. (item, key) => ({
  84. import: item.import === false ? undefined : item.import || key,
  85. shareScope: item.shareScope || options.shareScope || "default",
  86. shareKey: item.shareKey || key,
  87. requiredVersion:
  88. typeof item.requiredVersion === "string"
  89. ? parseRange(item.requiredVersion)
  90. : item.requiredVersion,
  91. strictVersion:
  92. typeof item.strictVersion === "boolean"
  93. ? item.strictVersion
  94. : item.import !== false && !item.singleton,
  95. packageName: item.packageName,
  96. singleton: Boolean(item.singleton),
  97. eager: Boolean(item.eager)
  98. })
  99. );
  100. }
  101. /**
  102. * Apply the plugin
  103. * @param {Compiler} compiler the compiler instance
  104. * @returns {void}
  105. */
  106. apply(compiler) {
  107. compiler.hooks.thisCompilation.tap(
  108. PLUGIN_NAME,
  109. (compilation, { normalModuleFactory }) => {
  110. compilation.dependencyFactories.set(
  111. ConsumeSharedFallbackDependency,
  112. normalModuleFactory
  113. );
  114. /** @typedef {Map<string, ConsumeOptions>} Consumes */
  115. /** @type {Consumes} */
  116. let unresolvedConsumes;
  117. /** @type {Consumes} */
  118. let resolvedConsumes;
  119. /** @type {Consumes} */
  120. let prefixedConsumes;
  121. const promise = resolveMatchedConfigs(compilation, this._consumes).then(
  122. ({ resolved, unresolved, prefixed }) => {
  123. resolvedConsumes = resolved;
  124. unresolvedConsumes = unresolved;
  125. prefixedConsumes = prefixed;
  126. }
  127. );
  128. const resolver = compilation.resolverFactory.get(
  129. "normal",
  130. RESOLVE_OPTIONS
  131. );
  132. /**
  133. * @param {string} context issuer directory
  134. * @param {string} request request
  135. * @param {ConsumeOptions} config options
  136. * @returns {Promise<ConsumeSharedModule>} create module
  137. */
  138. const createConsumeSharedModule = (context, request, config) => {
  139. /**
  140. * @param {string} details details
  141. */
  142. const requiredVersionWarning = (details) => {
  143. const error = new WebpackError(
  144. `No required version specified and unable to automatically determine one. ${details}`
  145. );
  146. error.file = `shared module ${request}`;
  147. compilation.warnings.push(error);
  148. };
  149. const directFallback =
  150. config.import &&
  151. /^(?:\.\.?(?:\/|$)|\/|[A-Z]:|\\\\)/i.test(config.import);
  152. return Promise.all([
  153. new Promise(
  154. /**
  155. * @param {(value?: string) => void} resolve resolve
  156. */
  157. (resolve) => {
  158. if (!config.import) {
  159. resolve();
  160. return;
  161. }
  162. /** @type {ResolveContext & { fileDependencies: FileSystemDependencies, contextDependencies: FileSystemDependencies, missingDependencies: FileSystemDependencies }} */
  163. const resolveContext = {
  164. fileDependencies: new LazySet(),
  165. contextDependencies: new LazySet(),
  166. missingDependencies: new LazySet()
  167. };
  168. resolver.resolve(
  169. {},
  170. directFallback ? compiler.context : context,
  171. config.import,
  172. resolveContext,
  173. (err, result) => {
  174. compilation.contextDependencies.addAll(
  175. resolveContext.contextDependencies
  176. );
  177. compilation.fileDependencies.addAll(
  178. resolveContext.fileDependencies
  179. );
  180. compilation.missingDependencies.addAll(
  181. resolveContext.missingDependencies
  182. );
  183. if (err) {
  184. compilation.errors.push(
  185. new ModuleNotFoundError(null, err, {
  186. name: `resolving fallback for shared module ${request}`
  187. })
  188. );
  189. return resolve();
  190. }
  191. resolve(/** @type {string} */ (result));
  192. }
  193. );
  194. }
  195. ),
  196. new Promise(
  197. /**
  198. * @param {(value?: SemVerRange) => void} resolve resolve
  199. */
  200. (resolve) => {
  201. if (config.requiredVersion !== undefined) {
  202. resolve(/** @type {SemVerRange} */ (config.requiredVersion));
  203. return;
  204. }
  205. let packageName = config.packageName;
  206. if (packageName === undefined) {
  207. if (/^(?:\/|[A-Z]:|\\\\)/i.test(request)) {
  208. // For relative or absolute requests we don't automatically use a packageName.
  209. // If wished one can specify one with the packageName option.
  210. resolve();
  211. return;
  212. }
  213. const match = /^(?:@[^\\/]+[\\/])?[^\\/]+/.exec(request);
  214. if (!match) {
  215. requiredVersionWarning(
  216. "Unable to extract the package name from request."
  217. );
  218. resolve();
  219. return;
  220. }
  221. packageName = match[0];
  222. }
  223. getDescriptionFile(
  224. compilation.inputFileSystem,
  225. context,
  226. ["package.json"],
  227. (err, result, checkedDescriptionFilePaths) => {
  228. if (err) {
  229. requiredVersionWarning(
  230. `Unable to read description file: ${err}`
  231. );
  232. return resolve();
  233. }
  234. const { data } =
  235. /** @type {DescriptionFile} */
  236. (result || {});
  237. if (!data) {
  238. if (checkedDescriptionFilePaths) {
  239. requiredVersionWarning(
  240. [
  241. `Unable to find required version for "${packageName}" in description file/s`,
  242. checkedDescriptionFilePaths.join("\n"),
  243. "It need to be in dependencies, devDependencies or peerDependencies."
  244. ].join("\n")
  245. );
  246. } else {
  247. requiredVersionWarning(
  248. `Unable to find description file in ${context}.`
  249. );
  250. }
  251. return resolve();
  252. }
  253. if (data.name === packageName) {
  254. // Package self-referencing
  255. return resolve();
  256. }
  257. const requiredVersion =
  258. getRequiredVersionFromDescriptionFile(data, packageName);
  259. if (requiredVersion) {
  260. return resolve(parseRange(requiredVersion));
  261. }
  262. resolve();
  263. },
  264. (result) => {
  265. if (!result) return false;
  266. const maybeRequiredVersion =
  267. getRequiredVersionFromDescriptionFile(
  268. result.data,
  269. packageName
  270. );
  271. return (
  272. result.data.name === packageName ||
  273. typeof maybeRequiredVersion === "string"
  274. );
  275. }
  276. );
  277. }
  278. )
  279. ]).then(
  280. ([importResolved, requiredVersion]) =>
  281. new ConsumeSharedModule(
  282. directFallback ? compiler.context : context,
  283. {
  284. ...config,
  285. importResolved,
  286. import: importResolved ? config.import : undefined,
  287. requiredVersion
  288. }
  289. )
  290. );
  291. };
  292. normalModuleFactory.hooks.factorize.tapPromise(
  293. PLUGIN_NAME,
  294. ({ context, request, dependencies }) =>
  295. // wait for resolving to be complete
  296. promise.then(() => {
  297. if (
  298. dependencies[0] instanceof ConsumeSharedFallbackDependency ||
  299. dependencies[0] instanceof ProvideForSharedDependency
  300. ) {
  301. return;
  302. }
  303. const match = unresolvedConsumes.get(request);
  304. if (match !== undefined) {
  305. return createConsumeSharedModule(context, request, match);
  306. }
  307. for (const [prefix, options] of prefixedConsumes) {
  308. if (request.startsWith(prefix)) {
  309. const remainder = request.slice(prefix.length);
  310. return createConsumeSharedModule(context, request, {
  311. ...options,
  312. import: options.import
  313. ? options.import + remainder
  314. : undefined,
  315. shareKey: options.shareKey + remainder
  316. });
  317. }
  318. }
  319. })
  320. );
  321. normalModuleFactory.hooks.createModule.tapPromise(
  322. PLUGIN_NAME,
  323. ({ resource }, { context, dependencies }) => {
  324. if (
  325. dependencies[0] instanceof ConsumeSharedFallbackDependency ||
  326. dependencies[0] instanceof ProvideForSharedDependency
  327. ) {
  328. return Promise.resolve();
  329. }
  330. const options = resolvedConsumes.get(
  331. /** @type {string} */ (resource)
  332. );
  333. if (options !== undefined) {
  334. return createConsumeSharedModule(
  335. context,
  336. /** @type {string} */ (resource),
  337. options
  338. );
  339. }
  340. return Promise.resolve();
  341. }
  342. );
  343. compilation.hooks.additionalTreeRuntimeRequirements.tap(
  344. PLUGIN_NAME,
  345. (chunk, set) => {
  346. set.add(RuntimeGlobals.module);
  347. set.add(RuntimeGlobals.moduleCache);
  348. set.add(RuntimeGlobals.moduleFactoriesAddOnly);
  349. set.add(RuntimeGlobals.shareScopeMap);
  350. set.add(RuntimeGlobals.initializeSharing);
  351. set.add(RuntimeGlobals.hasOwnProperty);
  352. compilation.addRuntimeModule(
  353. chunk,
  354. new ConsumeSharedRuntimeModule(set)
  355. );
  356. }
  357. );
  358. }
  359. );
  360. }
  361. }
  362. module.exports = ConsumeSharedPlugin;