RuntimePlugin.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Tobias Koppers @sokra
  4. */
  5. "use strict";
  6. const RuntimeGlobals = require("./RuntimeGlobals");
  7. const RuntimeRequirementsDependency = require("./dependencies/RuntimeRequirementsDependency");
  8. const JavascriptModulesPlugin = require("./javascript/JavascriptModulesPlugin");
  9. const AsyncModuleRuntimeModule = require("./runtime/AsyncModuleRuntimeModule");
  10. const AutoPublicPathRuntimeModule = require("./runtime/AutoPublicPathRuntimeModule");
  11. const CompatGetDefaultExportRuntimeModule = require("./runtime/CompatGetDefaultExportRuntimeModule");
  12. const CompatRuntimeModule = require("./runtime/CompatRuntimeModule");
  13. const CreateFakeNamespaceObjectRuntimeModule = require("./runtime/CreateFakeNamespaceObjectRuntimeModule");
  14. const CreateScriptUrlRuntimeModule = require("./runtime/CreateScriptUrlRuntimeModule");
  15. const DefinePropertyGettersRuntimeModule = require("./runtime/DefinePropertyGettersRuntimeModule");
  16. const EnsureChunkRuntimeModule = require("./runtime/EnsureChunkRuntimeModule");
  17. const GetChunkFilenameRuntimeModule = require("./runtime/GetChunkFilenameRuntimeModule");
  18. const GetMainFilenameRuntimeModule = require("./runtime/GetMainFilenameRuntimeModule");
  19. const GlobalRuntimeModule = require("./runtime/GlobalRuntimeModule");
  20. const HasOwnPropertyRuntimeModule = require("./runtime/HasOwnPropertyRuntimeModule");
  21. const LoadScriptRuntimeModule = require("./runtime/LoadScriptRuntimeModule");
  22. const MakeNamespaceObjectRuntimeModule = require("./runtime/MakeNamespaceObjectRuntimeModule");
  23. const OnChunksLoadedRuntimeModule = require("./runtime/OnChunksLoadedRuntimeModule");
  24. const PublicPathRuntimeModule = require("./runtime/PublicPathRuntimeModule");
  25. const RelativeUrlRuntimeModule = require("./runtime/RelativeUrlRuntimeModule");
  26. const RuntimeIdRuntimeModule = require("./runtime/RuntimeIdRuntimeModule");
  27. const SystemContextRuntimeModule = require("./runtime/SystemContextRuntimeModule");
  28. const ShareRuntimeModule = require("./sharing/ShareRuntimeModule");
  29. const StringXor = require("./util/StringXor");
  30. /** @typedef {import("./Chunk")} Chunk */
  31. /** @typedef {import("./Compiler")} Compiler */
  32. /** @typedef {import("./Module")} Module */
  33. const GLOBALS_ON_REQUIRE = [
  34. RuntimeGlobals.chunkName,
  35. RuntimeGlobals.runtimeId,
  36. RuntimeGlobals.compatGetDefaultExport,
  37. RuntimeGlobals.createFakeNamespaceObject,
  38. RuntimeGlobals.createScriptUrl,
  39. RuntimeGlobals.definePropertyGetters,
  40. RuntimeGlobals.ensureChunk,
  41. RuntimeGlobals.entryModuleId,
  42. RuntimeGlobals.getFullHash,
  43. RuntimeGlobals.global,
  44. RuntimeGlobals.makeNamespaceObject,
  45. RuntimeGlobals.moduleCache,
  46. RuntimeGlobals.moduleFactories,
  47. RuntimeGlobals.moduleFactoriesAddOnly,
  48. RuntimeGlobals.interceptModuleExecution,
  49. RuntimeGlobals.publicPath,
  50. RuntimeGlobals.baseURI,
  51. RuntimeGlobals.relativeUrl,
  52. RuntimeGlobals.scriptNonce,
  53. RuntimeGlobals.uncaughtErrorHandler,
  54. RuntimeGlobals.asyncModule,
  55. RuntimeGlobals.wasmInstances,
  56. RuntimeGlobals.instantiateWasm,
  57. RuntimeGlobals.shareScopeMap,
  58. RuntimeGlobals.initializeSharing,
  59. RuntimeGlobals.loadScript,
  60. RuntimeGlobals.systemContext,
  61. RuntimeGlobals.onChunksLoaded
  62. ];
  63. const MODULE_DEPENDENCIES = {
  64. [RuntimeGlobals.moduleLoaded]: [RuntimeGlobals.module],
  65. [RuntimeGlobals.moduleId]: [RuntimeGlobals.module]
  66. };
  67. const TREE_DEPENDENCIES = {
  68. [RuntimeGlobals.definePropertyGetters]: [RuntimeGlobals.hasOwnProperty],
  69. [RuntimeGlobals.compatGetDefaultExport]: [
  70. RuntimeGlobals.definePropertyGetters
  71. ],
  72. [RuntimeGlobals.createFakeNamespaceObject]: [
  73. RuntimeGlobals.definePropertyGetters,
  74. RuntimeGlobals.makeNamespaceObject,
  75. RuntimeGlobals.require
  76. ],
  77. [RuntimeGlobals.initializeSharing]: [RuntimeGlobals.shareScopeMap],
  78. [RuntimeGlobals.shareScopeMap]: [RuntimeGlobals.hasOwnProperty]
  79. };
  80. class RuntimePlugin {
  81. /**
  82. * @param {Compiler} compiler the Compiler
  83. * @returns {void}
  84. */
  85. apply(compiler) {
  86. compiler.hooks.compilation.tap("RuntimePlugin", compilation => {
  87. compilation.dependencyTemplates.set(
  88. RuntimeRequirementsDependency,
  89. new RuntimeRequirementsDependency.Template()
  90. );
  91. for (const req of GLOBALS_ON_REQUIRE) {
  92. compilation.hooks.runtimeRequirementInModule
  93. .for(req)
  94. .tap("RuntimePlugin", (module, set) => {
  95. set.add(RuntimeGlobals.requireScope);
  96. });
  97. compilation.hooks.runtimeRequirementInTree
  98. .for(req)
  99. .tap("RuntimePlugin", (module, set) => {
  100. set.add(RuntimeGlobals.requireScope);
  101. });
  102. }
  103. for (const req of Object.keys(TREE_DEPENDENCIES)) {
  104. const deps = TREE_DEPENDENCIES[req];
  105. compilation.hooks.runtimeRequirementInTree
  106. .for(req)
  107. .tap("RuntimePlugin", (chunk, set) => {
  108. for (const dep of deps) set.add(dep);
  109. });
  110. }
  111. for (const req of Object.keys(MODULE_DEPENDENCIES)) {
  112. const deps = MODULE_DEPENDENCIES[req];
  113. compilation.hooks.runtimeRequirementInModule
  114. .for(req)
  115. .tap("RuntimePlugin", (chunk, set) => {
  116. for (const dep of deps) set.add(dep);
  117. });
  118. }
  119. compilation.hooks.runtimeRequirementInTree
  120. .for(RuntimeGlobals.definePropertyGetters)
  121. .tap("RuntimePlugin", chunk => {
  122. compilation.addRuntimeModule(
  123. chunk,
  124. new DefinePropertyGettersRuntimeModule()
  125. );
  126. return true;
  127. });
  128. compilation.hooks.runtimeRequirementInTree
  129. .for(RuntimeGlobals.makeNamespaceObject)
  130. .tap("RuntimePlugin", chunk => {
  131. compilation.addRuntimeModule(
  132. chunk,
  133. new MakeNamespaceObjectRuntimeModule()
  134. );
  135. return true;
  136. });
  137. compilation.hooks.runtimeRequirementInTree
  138. .for(RuntimeGlobals.createFakeNamespaceObject)
  139. .tap("RuntimePlugin", chunk => {
  140. compilation.addRuntimeModule(
  141. chunk,
  142. new CreateFakeNamespaceObjectRuntimeModule()
  143. );
  144. return true;
  145. });
  146. compilation.hooks.runtimeRequirementInTree
  147. .for(RuntimeGlobals.hasOwnProperty)
  148. .tap("RuntimePlugin", chunk => {
  149. compilation.addRuntimeModule(
  150. chunk,
  151. new HasOwnPropertyRuntimeModule()
  152. );
  153. return true;
  154. });
  155. compilation.hooks.runtimeRequirementInTree
  156. .for(RuntimeGlobals.compatGetDefaultExport)
  157. .tap("RuntimePlugin", chunk => {
  158. compilation.addRuntimeModule(
  159. chunk,
  160. new CompatGetDefaultExportRuntimeModule()
  161. );
  162. return true;
  163. });
  164. compilation.hooks.runtimeRequirementInTree
  165. .for(RuntimeGlobals.runtimeId)
  166. .tap("RuntimePlugin", chunk => {
  167. compilation.addRuntimeModule(chunk, new RuntimeIdRuntimeModule());
  168. return true;
  169. });
  170. compilation.hooks.runtimeRequirementInTree
  171. .for(RuntimeGlobals.publicPath)
  172. .tap("RuntimePlugin", (chunk, set) => {
  173. const { outputOptions } = compilation;
  174. const { publicPath: globalPublicPath, scriptType } = outputOptions;
  175. const entryOptions = chunk.getEntryOptions();
  176. const publicPath =
  177. entryOptions && entryOptions.publicPath !== undefined
  178. ? entryOptions.publicPath
  179. : globalPublicPath;
  180. if (publicPath === "auto") {
  181. const module = new AutoPublicPathRuntimeModule();
  182. if (scriptType !== "module") set.add(RuntimeGlobals.global);
  183. compilation.addRuntimeModule(chunk, module);
  184. } else {
  185. const module = new PublicPathRuntimeModule(publicPath);
  186. if (
  187. typeof publicPath !== "string" ||
  188. /\[(full)?hash\]/.test(publicPath)
  189. ) {
  190. module.fullHash = true;
  191. }
  192. compilation.addRuntimeModule(chunk, module);
  193. }
  194. return true;
  195. });
  196. compilation.hooks.runtimeRequirementInTree
  197. .for(RuntimeGlobals.global)
  198. .tap("RuntimePlugin", chunk => {
  199. compilation.addRuntimeModule(chunk, new GlobalRuntimeModule());
  200. return true;
  201. });
  202. compilation.hooks.runtimeRequirementInTree
  203. .for(RuntimeGlobals.asyncModule)
  204. .tap("RuntimePlugin", chunk => {
  205. compilation.addRuntimeModule(chunk, new AsyncModuleRuntimeModule());
  206. return true;
  207. });
  208. compilation.hooks.runtimeRequirementInTree
  209. .for(RuntimeGlobals.systemContext)
  210. .tap("RuntimePlugin", chunk => {
  211. const { outputOptions } = compilation;
  212. const { library: globalLibrary } = outputOptions;
  213. const entryOptions = chunk.getEntryOptions();
  214. const libraryType =
  215. entryOptions && entryOptions.library !== undefined
  216. ? entryOptions.library.type
  217. : globalLibrary.type;
  218. if (libraryType === "system") {
  219. compilation.addRuntimeModule(
  220. chunk,
  221. new SystemContextRuntimeModule()
  222. );
  223. }
  224. return true;
  225. });
  226. compilation.hooks.runtimeRequirementInTree
  227. .for(RuntimeGlobals.getChunkScriptFilename)
  228. .tap("RuntimePlugin", (chunk, set) => {
  229. if (
  230. typeof compilation.outputOptions.chunkFilename === "string" &&
  231. /\[(full)?hash(:\d+)?\]/.test(
  232. compilation.outputOptions.chunkFilename
  233. )
  234. ) {
  235. set.add(RuntimeGlobals.getFullHash);
  236. }
  237. compilation.addRuntimeModule(
  238. chunk,
  239. new GetChunkFilenameRuntimeModule(
  240. "javascript",
  241. "javascript",
  242. RuntimeGlobals.getChunkScriptFilename,
  243. chunk =>
  244. chunk.filenameTemplate ||
  245. (chunk.canBeInitial()
  246. ? compilation.outputOptions.filename
  247. : compilation.outputOptions.chunkFilename),
  248. false
  249. )
  250. );
  251. return true;
  252. });
  253. compilation.hooks.runtimeRequirementInTree
  254. .for(RuntimeGlobals.getChunkCssFilename)
  255. .tap("RuntimePlugin", (chunk, set) => {
  256. if (
  257. typeof compilation.outputOptions.cssChunkFilename === "string" &&
  258. /\[(full)?hash(:\d+)?\]/.test(
  259. compilation.outputOptions.cssChunkFilename
  260. )
  261. ) {
  262. set.add(RuntimeGlobals.getFullHash);
  263. }
  264. compilation.addRuntimeModule(
  265. chunk,
  266. new GetChunkFilenameRuntimeModule(
  267. "css",
  268. "css",
  269. RuntimeGlobals.getChunkCssFilename,
  270. chunk =>
  271. chunk.cssFilenameTemplate ||
  272. compilation.outputOptions.cssChunkFilename,
  273. false
  274. )
  275. );
  276. return true;
  277. });
  278. compilation.hooks.runtimeRequirementInTree
  279. .for(RuntimeGlobals.getChunkUpdateScriptFilename)
  280. .tap("RuntimePlugin", (chunk, set) => {
  281. if (
  282. /\[(full)?hash(:\d+)?\]/.test(
  283. compilation.outputOptions.hotUpdateChunkFilename
  284. )
  285. )
  286. set.add(RuntimeGlobals.getFullHash);
  287. compilation.addRuntimeModule(
  288. chunk,
  289. new GetChunkFilenameRuntimeModule(
  290. "javascript",
  291. "javascript update",
  292. RuntimeGlobals.getChunkUpdateScriptFilename,
  293. c => compilation.outputOptions.hotUpdateChunkFilename,
  294. true
  295. )
  296. );
  297. return true;
  298. });
  299. compilation.hooks.runtimeRequirementInTree
  300. .for(RuntimeGlobals.getUpdateManifestFilename)
  301. .tap("RuntimePlugin", (chunk, set) => {
  302. if (
  303. /\[(full)?hash(:\d+)?\]/.test(
  304. compilation.outputOptions.hotUpdateMainFilename
  305. )
  306. ) {
  307. set.add(RuntimeGlobals.getFullHash);
  308. }
  309. compilation.addRuntimeModule(
  310. chunk,
  311. new GetMainFilenameRuntimeModule(
  312. "update manifest",
  313. RuntimeGlobals.getUpdateManifestFilename,
  314. compilation.outputOptions.hotUpdateMainFilename
  315. )
  316. );
  317. return true;
  318. });
  319. compilation.hooks.runtimeRequirementInTree
  320. .for(RuntimeGlobals.ensureChunk)
  321. .tap("RuntimePlugin", (chunk, set) => {
  322. const hasAsyncChunks = chunk.hasAsyncChunks();
  323. if (hasAsyncChunks) {
  324. set.add(RuntimeGlobals.ensureChunkHandlers);
  325. }
  326. compilation.addRuntimeModule(
  327. chunk,
  328. new EnsureChunkRuntimeModule(set)
  329. );
  330. return true;
  331. });
  332. compilation.hooks.runtimeRequirementInTree
  333. .for(RuntimeGlobals.ensureChunkIncludeEntries)
  334. .tap("RuntimePlugin", (chunk, set) => {
  335. set.add(RuntimeGlobals.ensureChunkHandlers);
  336. });
  337. compilation.hooks.runtimeRequirementInTree
  338. .for(RuntimeGlobals.shareScopeMap)
  339. .tap("RuntimePlugin", (chunk, set) => {
  340. compilation.addRuntimeModule(chunk, new ShareRuntimeModule());
  341. return true;
  342. });
  343. compilation.hooks.runtimeRequirementInTree
  344. .for(RuntimeGlobals.loadScript)
  345. .tap("RuntimePlugin", (chunk, set) => {
  346. const withCreateScriptUrl = !!compilation.outputOptions.trustedTypes;
  347. if (withCreateScriptUrl) {
  348. set.add(RuntimeGlobals.createScriptUrl);
  349. }
  350. compilation.addRuntimeModule(
  351. chunk,
  352. new LoadScriptRuntimeModule(withCreateScriptUrl)
  353. );
  354. return true;
  355. });
  356. compilation.hooks.runtimeRequirementInTree
  357. .for(RuntimeGlobals.createScriptUrl)
  358. .tap("RuntimePlugin", (chunk, set) => {
  359. compilation.addRuntimeModule(
  360. chunk,
  361. new CreateScriptUrlRuntimeModule()
  362. );
  363. return true;
  364. });
  365. compilation.hooks.runtimeRequirementInTree
  366. .for(RuntimeGlobals.relativeUrl)
  367. .tap("RuntimePlugin", (chunk, set) => {
  368. compilation.addRuntimeModule(chunk, new RelativeUrlRuntimeModule());
  369. return true;
  370. });
  371. compilation.hooks.runtimeRequirementInTree
  372. .for(RuntimeGlobals.onChunksLoaded)
  373. .tap("RuntimePlugin", (chunk, set) => {
  374. compilation.addRuntimeModule(
  375. chunk,
  376. new OnChunksLoadedRuntimeModule()
  377. );
  378. return true;
  379. });
  380. // TODO webpack 6: remove CompatRuntimeModule
  381. compilation.hooks.additionalTreeRuntimeRequirements.tap(
  382. "RuntimePlugin",
  383. (chunk, set) => {
  384. const { mainTemplate } = compilation;
  385. if (
  386. mainTemplate.hooks.bootstrap.isUsed() ||
  387. mainTemplate.hooks.localVars.isUsed() ||
  388. mainTemplate.hooks.requireEnsure.isUsed() ||
  389. mainTemplate.hooks.requireExtensions.isUsed()
  390. ) {
  391. compilation.addRuntimeModule(chunk, new CompatRuntimeModule());
  392. }
  393. }
  394. );
  395. JavascriptModulesPlugin.getCompilationHooks(compilation).chunkHash.tap(
  396. "RuntimePlugin",
  397. (chunk, hash, { chunkGraph }) => {
  398. const xor = new StringXor();
  399. for (const m of chunkGraph.getChunkRuntimeModulesIterable(chunk)) {
  400. xor.add(chunkGraph.getModuleHash(m, chunk.runtime));
  401. }
  402. xor.updateHash(hash);
  403. }
  404. );
  405. });
  406. }
  407. }
  408. module.exports = RuntimePlugin;