RequireContextPlugin.js 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Tobias Koppers @sokra
  4. */
  5. "use strict";
  6. const {
  7. JAVASCRIPT_MODULE_TYPE_AUTO,
  8. JAVASCRIPT_MODULE_TYPE_DYNAMIC
  9. } = require("../ModuleTypeConstants");
  10. const { cachedSetProperty } = require("../util/cleverMerge");
  11. const ContextElementDependency = require("./ContextElementDependency");
  12. const RequireContextDependency = require("./RequireContextDependency");
  13. const RequireContextDependencyParserPlugin = require("./RequireContextDependencyParserPlugin");
  14. /** @typedef {import("../../declarations/WebpackOptions").JavascriptParserOptions} JavascriptParserOptions */
  15. /** @typedef {import("../../declarations/WebpackOptions").ResolveOptions} ResolveOptions */
  16. /** @typedef {import("../Compiler")} Compiler */
  17. /** @typedef {import("../javascript/JavascriptParser")} Parser */
  18. /** @type {ResolveOptions} */
  19. const EMPTY_RESOLVE_OPTIONS = {};
  20. const PLUGIN_NAME = "RequireContextPlugin";
  21. class RequireContextPlugin {
  22. /**
  23. * Apply the plugin
  24. * @param {Compiler} compiler the compiler instance
  25. * @returns {void}
  26. */
  27. apply(compiler) {
  28. compiler.hooks.compilation.tap(
  29. PLUGIN_NAME,
  30. (compilation, { contextModuleFactory, normalModuleFactory }) => {
  31. compilation.dependencyFactories.set(
  32. RequireContextDependency,
  33. contextModuleFactory
  34. );
  35. compilation.dependencyTemplates.set(
  36. RequireContextDependency,
  37. new RequireContextDependency.Template()
  38. );
  39. compilation.dependencyFactories.set(
  40. ContextElementDependency,
  41. normalModuleFactory
  42. );
  43. /**
  44. * @param {Parser} parser parser parser
  45. * @param {JavascriptParserOptions} parserOptions parserOptions
  46. * @returns {void}
  47. */
  48. const handler = (parser, parserOptions) => {
  49. if (
  50. parserOptions.requireContext !== undefined &&
  51. !parserOptions.requireContext
  52. ) {
  53. return;
  54. }
  55. new RequireContextDependencyParserPlugin().apply(parser);
  56. };
  57. normalModuleFactory.hooks.parser
  58. .for(JAVASCRIPT_MODULE_TYPE_AUTO)
  59. .tap(PLUGIN_NAME, handler);
  60. normalModuleFactory.hooks.parser
  61. .for(JAVASCRIPT_MODULE_TYPE_DYNAMIC)
  62. .tap(PLUGIN_NAME, handler);
  63. contextModuleFactory.hooks.alternativeRequests.tap(
  64. PLUGIN_NAME,
  65. (items, options) => {
  66. if (items.length === 0) return items;
  67. const finalResolveOptions = compiler.resolverFactory.get(
  68. "normal",
  69. cachedSetProperty(
  70. options.resolveOptions || EMPTY_RESOLVE_OPTIONS,
  71. "dependencyType",
  72. /** @type {string} */
  73. (options.category)
  74. )
  75. ).options;
  76. /** @type {{ context: string, request: string }[]} */
  77. let newItems;
  78. if (!finalResolveOptions.fullySpecified) {
  79. newItems = [];
  80. for (const item of items) {
  81. const { request, context } = item;
  82. for (const ext of finalResolveOptions.extensions) {
  83. if (request.endsWith(ext)) {
  84. newItems.push({
  85. context,
  86. request: request.slice(0, -ext.length)
  87. });
  88. }
  89. }
  90. if (!finalResolveOptions.enforceExtension) {
  91. newItems.push(item);
  92. }
  93. }
  94. items = newItems;
  95. newItems = [];
  96. for (const obj of items) {
  97. const { request, context } = obj;
  98. for (const mainFile of finalResolveOptions.mainFiles) {
  99. if (request.endsWith(`/${mainFile}`)) {
  100. newItems.push({
  101. context,
  102. request: request.slice(0, -mainFile.length)
  103. });
  104. newItems.push({
  105. context,
  106. request: request.slice(0, -mainFile.length - 1)
  107. });
  108. }
  109. }
  110. newItems.push(obj);
  111. }
  112. items = newItems;
  113. }
  114. newItems = [];
  115. for (const item of items) {
  116. let hideOriginal = false;
  117. for (const modulesItems of finalResolveOptions.modules) {
  118. if (Array.isArray(modulesItems)) {
  119. for (const dir of modulesItems) {
  120. if (item.request.startsWith(`./${dir}/`)) {
  121. newItems.push({
  122. context: item.context,
  123. request: item.request.slice(dir.length + 3)
  124. });
  125. hideOriginal = true;
  126. }
  127. }
  128. } else {
  129. const dir = modulesItems.replace(/\\/g, "/");
  130. const fullPath =
  131. item.context.replace(/\\/g, "/") + item.request.slice(1);
  132. if (fullPath.startsWith(dir)) {
  133. newItems.push({
  134. context: item.context,
  135. request: fullPath.slice(dir.length + 1)
  136. });
  137. }
  138. }
  139. }
  140. if (!hideOriginal) {
  141. newItems.push(item);
  142. }
  143. }
  144. return newItems;
  145. }
  146. );
  147. }
  148. );
  149. }
  150. }
  151. module.exports = RequireContextPlugin;