Module.js 30 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Tobias Koppers @sokra
  4. */
  5. "use strict";
  6. const util = require("util");
  7. const ChunkGraph = require("./ChunkGraph");
  8. const DependenciesBlock = require("./DependenciesBlock");
  9. const ModuleGraph = require("./ModuleGraph");
  10. const RuntimeGlobals = require("./RuntimeGlobals");
  11. const { first } = require("./util/SetHelpers");
  12. const { compareChunksById } = require("./util/comparators");
  13. const makeSerializable = require("./util/makeSerializable");
  14. /** @typedef {import("webpack-sources").Source} Source */
  15. /** @typedef {import("../declarations/WebpackOptions").ResolveOptions} ResolveOptions */
  16. /** @typedef {import("../declarations/WebpackOptions").WebpackOptionsNormalized} WebpackOptions */
  17. /** @typedef {import("./Chunk")} Chunk */
  18. /** @typedef {import("./ChunkGroup")} ChunkGroup */
  19. /** @typedef {import("./CodeGenerationResults")} CodeGenerationResults */
  20. /** @typedef {import("./Compilation")} Compilation */
  21. /** @typedef {import("./ConcatenationScope")} ConcatenationScope */
  22. /** @typedef {import("./Dependency")} Dependency */
  23. /** @typedef {import("./Dependency").UpdateHashContext} UpdateHashContext */
  24. /** @typedef {import("./DependencyTemplates")} DependencyTemplates */
  25. /** @typedef {import("./ExportsInfo").UsageStateType} UsageStateType */
  26. /** @typedef {import("./FileSystemInfo")} FileSystemInfo */
  27. /** @typedef {import("./ModuleGraphConnection").ConnectionState} ConnectionState */
  28. /** @typedef {import("./NormalModuleFactory")} NormalModuleFactory */
  29. /** @typedef {import("./RequestShortener")} RequestShortener */
  30. /** @typedef {import("./ResolverFactory").ResolverWithOptions} ResolverWithOptions */
  31. /** @typedef {import("./RuntimeTemplate")} RuntimeTemplate */
  32. /** @typedef {import("./WebpackError")} WebpackError */
  33. /** @typedef {import("./util/Hash")} Hash */
  34. /** @template T @typedef {import("./util/LazySet")<T>} LazySet<T> */
  35. /** @template T @typedef {import("./util/SortableSet")<T>} SortableSet<T> */
  36. /** @typedef {import("./util/fs").InputFileSystem} InputFileSystem */
  37. /** @typedef {import("./util/runtime").RuntimeSpec} RuntimeSpec */
  38. /**
  39. * @typedef {Object} SourceContext
  40. * @property {DependencyTemplates} dependencyTemplates the dependency templates
  41. * @property {RuntimeTemplate} runtimeTemplate the runtime template
  42. * @property {ModuleGraph} moduleGraph the module graph
  43. * @property {ChunkGraph} chunkGraph the chunk graph
  44. * @property {RuntimeSpec} runtime the runtimes code should be generated for
  45. * @property {string=} type the type of source that should be generated
  46. */
  47. /**
  48. * @typedef {Object} CodeGenerationContext
  49. * @property {DependencyTemplates} dependencyTemplates the dependency templates
  50. * @property {RuntimeTemplate} runtimeTemplate the runtime template
  51. * @property {ModuleGraph} moduleGraph the module graph
  52. * @property {ChunkGraph} chunkGraph the chunk graph
  53. * @property {RuntimeSpec} runtime the runtimes code should be generated for
  54. * @property {ConcatenationScope=} concatenationScope when in concatenated module, information about other concatenated modules
  55. * @property {CodeGenerationResults} codeGenerationResults code generation results of other modules (need to have a codeGenerationDependency to use that)
  56. */
  57. /**
  58. * @typedef {Object} ConcatenationBailoutReasonContext
  59. * @property {ModuleGraph} moduleGraph the module graph
  60. * @property {ChunkGraph} chunkGraph the chunk graph
  61. */
  62. /**
  63. * @typedef {Object} CodeGenerationResult
  64. * @property {Map<string, Source>} sources the resulting sources for all source types
  65. * @property {Map<string, any>=} data the resulting data for all source types
  66. * @property {ReadonlySet<string>} runtimeRequirements the runtime requirements
  67. * @property {string=} hash a hash of the code generation result (will be automatically calculated from sources and runtimeRequirements if not provided)
  68. */
  69. /**
  70. * @typedef {Object} LibIdentOptions
  71. * @property {string} context absolute context path to which lib ident is relative to
  72. * @property {Object=} associatedObjectForCache object for caching
  73. */
  74. /**
  75. * @typedef {Object} KnownBuildMeta
  76. * @property {string=} moduleArgument
  77. * @property {string=} exportsArgument
  78. * @property {boolean=} strict
  79. * @property {string=} moduleConcatenationBailout
  80. * @property {("default" | "namespace" | "flagged" | "dynamic")=} exportsType
  81. * @property {(false | "redirect" | "redirect-warn")=} defaultObject
  82. * @property {boolean=} strictHarmonyModule
  83. * @property {boolean=} async
  84. * @property {boolean=} sideEffectFree
  85. */
  86. /**
  87. * @typedef {Object} NeedBuildContext
  88. * @property {Compilation} compilation
  89. * @property {FileSystemInfo} fileSystemInfo
  90. * @property {Map<string, string | Set<string>>} valueCacheVersions
  91. */
  92. /** @typedef {KnownBuildMeta & Record<string, any>} BuildMeta */
  93. const EMPTY_RESOLVE_OPTIONS = {};
  94. let debugId = 1000;
  95. const DEFAULT_TYPES_UNKNOWN = new Set(["unknown"]);
  96. const DEFAULT_TYPES_JS = new Set(["javascript"]);
  97. const deprecatedNeedRebuild = util.deprecate(
  98. (module, context) => {
  99. return module.needRebuild(
  100. context.fileSystemInfo.getDeprecatedFileTimestamps(),
  101. context.fileSystemInfo.getDeprecatedContextTimestamps()
  102. );
  103. },
  104. "Module.needRebuild is deprecated in favor of Module.needBuild",
  105. "DEP_WEBPACK_MODULE_NEED_REBUILD"
  106. );
  107. /** @typedef {(requestShortener: RequestShortener) => string} OptimizationBailoutFunction */
  108. class Module extends DependenciesBlock {
  109. /**
  110. * @param {string} type the module type
  111. * @param {string=} context an optional context
  112. * @param {string=} layer an optional layer in which the module is
  113. */
  114. constructor(type, context = null, layer = null) {
  115. super();
  116. /** @type {string} */
  117. this.type = type;
  118. /** @type {string | null} */
  119. this.context = context;
  120. /** @type {string | null} */
  121. this.layer = layer;
  122. /** @type {boolean} */
  123. this.needId = true;
  124. // Unique Id
  125. /** @type {number} */
  126. this.debugId = debugId++;
  127. // Info from Factory
  128. /** @type {ResolveOptions} */
  129. this.resolveOptions = EMPTY_RESOLVE_OPTIONS;
  130. /** @type {object | undefined} */
  131. this.factoryMeta = undefined;
  132. // TODO refactor this -> options object filled from Factory
  133. // TODO webpack 6: use an enum
  134. /** @type {boolean} */
  135. this.useSourceMap = false;
  136. /** @type {boolean} */
  137. this.useSimpleSourceMap = false;
  138. // Info from Build
  139. /** @type {WebpackError[] | undefined} */
  140. this._warnings = undefined;
  141. /** @type {WebpackError[] | undefined} */
  142. this._errors = undefined;
  143. /** @type {BuildMeta} */
  144. this.buildMeta = undefined;
  145. /** @type {Record<string, any>} */
  146. this.buildInfo = undefined;
  147. /** @type {Dependency[] | undefined} */
  148. this.presentationalDependencies = undefined;
  149. /** @type {Dependency[] | undefined} */
  150. this.codeGenerationDependencies = undefined;
  151. }
  152. // TODO remove in webpack 6
  153. // BACKWARD-COMPAT START
  154. get id() {
  155. return ChunkGraph.getChunkGraphForModule(
  156. this,
  157. "Module.id",
  158. "DEP_WEBPACK_MODULE_ID"
  159. ).getModuleId(this);
  160. }
  161. set id(value) {
  162. if (value === "") {
  163. this.needId = false;
  164. return;
  165. }
  166. ChunkGraph.getChunkGraphForModule(
  167. this,
  168. "Module.id",
  169. "DEP_WEBPACK_MODULE_ID"
  170. ).setModuleId(this, value);
  171. }
  172. /**
  173. * @returns {string} the hash of the module
  174. */
  175. get hash() {
  176. return ChunkGraph.getChunkGraphForModule(
  177. this,
  178. "Module.hash",
  179. "DEP_WEBPACK_MODULE_HASH"
  180. ).getModuleHash(this, undefined);
  181. }
  182. /**
  183. * @returns {string} the shortened hash of the module
  184. */
  185. get renderedHash() {
  186. return ChunkGraph.getChunkGraphForModule(
  187. this,
  188. "Module.renderedHash",
  189. "DEP_WEBPACK_MODULE_RENDERED_HASH"
  190. ).getRenderedModuleHash(this, undefined);
  191. }
  192. get profile() {
  193. return ModuleGraph.getModuleGraphForModule(
  194. this,
  195. "Module.profile",
  196. "DEP_WEBPACK_MODULE_PROFILE"
  197. ).getProfile(this);
  198. }
  199. set profile(value) {
  200. ModuleGraph.getModuleGraphForModule(
  201. this,
  202. "Module.profile",
  203. "DEP_WEBPACK_MODULE_PROFILE"
  204. ).setProfile(this, value);
  205. }
  206. get index() {
  207. return ModuleGraph.getModuleGraphForModule(
  208. this,
  209. "Module.index",
  210. "DEP_WEBPACK_MODULE_INDEX"
  211. ).getPreOrderIndex(this);
  212. }
  213. set index(value) {
  214. ModuleGraph.getModuleGraphForModule(
  215. this,
  216. "Module.index",
  217. "DEP_WEBPACK_MODULE_INDEX"
  218. ).setPreOrderIndex(this, value);
  219. }
  220. get index2() {
  221. return ModuleGraph.getModuleGraphForModule(
  222. this,
  223. "Module.index2",
  224. "DEP_WEBPACK_MODULE_INDEX2"
  225. ).getPostOrderIndex(this);
  226. }
  227. set index2(value) {
  228. ModuleGraph.getModuleGraphForModule(
  229. this,
  230. "Module.index2",
  231. "DEP_WEBPACK_MODULE_INDEX2"
  232. ).setPostOrderIndex(this, value);
  233. }
  234. get depth() {
  235. return ModuleGraph.getModuleGraphForModule(
  236. this,
  237. "Module.depth",
  238. "DEP_WEBPACK_MODULE_DEPTH"
  239. ).getDepth(this);
  240. }
  241. set depth(value) {
  242. ModuleGraph.getModuleGraphForModule(
  243. this,
  244. "Module.depth",
  245. "DEP_WEBPACK_MODULE_DEPTH"
  246. ).setDepth(this, value);
  247. }
  248. get issuer() {
  249. return ModuleGraph.getModuleGraphForModule(
  250. this,
  251. "Module.issuer",
  252. "DEP_WEBPACK_MODULE_ISSUER"
  253. ).getIssuer(this);
  254. }
  255. set issuer(value) {
  256. ModuleGraph.getModuleGraphForModule(
  257. this,
  258. "Module.issuer",
  259. "DEP_WEBPACK_MODULE_ISSUER"
  260. ).setIssuer(this, value);
  261. }
  262. get usedExports() {
  263. return ModuleGraph.getModuleGraphForModule(
  264. this,
  265. "Module.usedExports",
  266. "DEP_WEBPACK_MODULE_USED_EXPORTS"
  267. ).getUsedExports(this, undefined);
  268. }
  269. /**
  270. * @deprecated
  271. * @returns {(string | OptimizationBailoutFunction)[]} list
  272. */
  273. get optimizationBailout() {
  274. return ModuleGraph.getModuleGraphForModule(
  275. this,
  276. "Module.optimizationBailout",
  277. "DEP_WEBPACK_MODULE_OPTIMIZATION_BAILOUT"
  278. ).getOptimizationBailout(this);
  279. }
  280. get optional() {
  281. return this.isOptional(
  282. ModuleGraph.getModuleGraphForModule(
  283. this,
  284. "Module.optional",
  285. "DEP_WEBPACK_MODULE_OPTIONAL"
  286. )
  287. );
  288. }
  289. addChunk(chunk) {
  290. const chunkGraph = ChunkGraph.getChunkGraphForModule(
  291. this,
  292. "Module.addChunk",
  293. "DEP_WEBPACK_MODULE_ADD_CHUNK"
  294. );
  295. if (chunkGraph.isModuleInChunk(this, chunk)) return false;
  296. chunkGraph.connectChunkAndModule(chunk, this);
  297. return true;
  298. }
  299. removeChunk(chunk) {
  300. return ChunkGraph.getChunkGraphForModule(
  301. this,
  302. "Module.removeChunk",
  303. "DEP_WEBPACK_MODULE_REMOVE_CHUNK"
  304. ).disconnectChunkAndModule(chunk, this);
  305. }
  306. isInChunk(chunk) {
  307. return ChunkGraph.getChunkGraphForModule(
  308. this,
  309. "Module.isInChunk",
  310. "DEP_WEBPACK_MODULE_IS_IN_CHUNK"
  311. ).isModuleInChunk(this, chunk);
  312. }
  313. isEntryModule() {
  314. return ChunkGraph.getChunkGraphForModule(
  315. this,
  316. "Module.isEntryModule",
  317. "DEP_WEBPACK_MODULE_IS_ENTRY_MODULE"
  318. ).isEntryModule(this);
  319. }
  320. getChunks() {
  321. return ChunkGraph.getChunkGraphForModule(
  322. this,
  323. "Module.getChunks",
  324. "DEP_WEBPACK_MODULE_GET_CHUNKS"
  325. ).getModuleChunks(this);
  326. }
  327. getNumberOfChunks() {
  328. return ChunkGraph.getChunkGraphForModule(
  329. this,
  330. "Module.getNumberOfChunks",
  331. "DEP_WEBPACK_MODULE_GET_NUMBER_OF_CHUNKS"
  332. ).getNumberOfModuleChunks(this);
  333. }
  334. get chunksIterable() {
  335. return ChunkGraph.getChunkGraphForModule(
  336. this,
  337. "Module.chunksIterable",
  338. "DEP_WEBPACK_MODULE_CHUNKS_ITERABLE"
  339. ).getOrderedModuleChunksIterable(this, compareChunksById);
  340. }
  341. /**
  342. * @param {string} exportName a name of an export
  343. * @returns {boolean | null} true, if the export is provided why the module.
  344. * null, if it's unknown.
  345. * false, if it's not provided.
  346. */
  347. isProvided(exportName) {
  348. return ModuleGraph.getModuleGraphForModule(
  349. this,
  350. "Module.usedExports",
  351. "DEP_WEBPACK_MODULE_USED_EXPORTS"
  352. ).isExportProvided(this, exportName);
  353. }
  354. // BACKWARD-COMPAT END
  355. /**
  356. * @deprecated moved to .buildInfo.exportsArgument
  357. * @returns {string} name of the exports argument
  358. */
  359. get exportsArgument() {
  360. return (this.buildInfo && this.buildInfo.exportsArgument) || "exports";
  361. }
  362. /**
  363. * @deprecated moved to .buildInfo.moduleArgument
  364. * @returns {string} name of the module argument
  365. */
  366. get moduleArgument() {
  367. return (this.buildInfo && this.buildInfo.moduleArgument) || "module";
  368. }
  369. /**
  370. * @param {ModuleGraph} moduleGraph the module graph
  371. * @param {boolean} strict the importing module is strict
  372. * @returns {"namespace" | "default-only" | "default-with-named" | "dynamic"} export type
  373. * "namespace": Exports is already a namespace object. namespace = exports.
  374. * "dynamic": Check at runtime if __esModule is set. When set: namespace = { ...exports, default: exports }. When not set: namespace = { default: exports }.
  375. * "default-only": Provide a namespace object with only default export. namespace = { default: exports }
  376. * "default-with-named": Provide a namespace object with named and default export. namespace = { ...exports, default: exports }
  377. */
  378. getExportsType(moduleGraph, strict) {
  379. switch (this.buildMeta && this.buildMeta.exportsType) {
  380. case "flagged":
  381. return strict ? "default-with-named" : "namespace";
  382. case "namespace":
  383. return "namespace";
  384. case "default":
  385. switch (this.buildMeta.defaultObject) {
  386. case "redirect":
  387. return "default-with-named";
  388. case "redirect-warn":
  389. return strict ? "default-only" : "default-with-named";
  390. default:
  391. return "default-only";
  392. }
  393. case "dynamic": {
  394. if (strict) return "default-with-named";
  395. // Try to figure out value of __esModule by following reexports
  396. const handleDefault = () => {
  397. switch (this.buildMeta.defaultObject) {
  398. case "redirect":
  399. case "redirect-warn":
  400. return "default-with-named";
  401. default:
  402. return "default-only";
  403. }
  404. };
  405. const exportInfo = moduleGraph.getReadOnlyExportInfo(
  406. this,
  407. "__esModule"
  408. );
  409. if (exportInfo.provided === false) {
  410. return handleDefault();
  411. }
  412. const target = exportInfo.getTarget(moduleGraph);
  413. if (
  414. !target ||
  415. !target.export ||
  416. target.export.length !== 1 ||
  417. target.export[0] !== "__esModule"
  418. ) {
  419. return "dynamic";
  420. }
  421. switch (
  422. target.module.buildMeta &&
  423. target.module.buildMeta.exportsType
  424. ) {
  425. case "flagged":
  426. case "namespace":
  427. return "namespace";
  428. case "default":
  429. return handleDefault();
  430. default:
  431. return "dynamic";
  432. }
  433. }
  434. default:
  435. return strict ? "default-with-named" : "dynamic";
  436. }
  437. }
  438. /**
  439. * @param {Dependency} presentationalDependency dependency being tied to module.
  440. * This is a Dependency without edge in the module graph. It's only for presentation.
  441. * @returns {void}
  442. */
  443. addPresentationalDependency(presentationalDependency) {
  444. if (this.presentationalDependencies === undefined) {
  445. this.presentationalDependencies = [];
  446. }
  447. this.presentationalDependencies.push(presentationalDependency);
  448. }
  449. /**
  450. * @param {Dependency} codeGenerationDependency dependency being tied to module.
  451. * This is a Dependency where the code generation result of the referenced module is needed during code generation.
  452. * The Dependency should also be added to normal dependencies via addDependency.
  453. * @returns {void}
  454. */
  455. addCodeGenerationDependency(codeGenerationDependency) {
  456. if (this.codeGenerationDependencies === undefined) {
  457. this.codeGenerationDependencies = [];
  458. }
  459. this.codeGenerationDependencies.push(codeGenerationDependency);
  460. }
  461. /**
  462. * Removes all dependencies and blocks
  463. * @returns {void}
  464. */
  465. clearDependenciesAndBlocks() {
  466. if (this.presentationalDependencies !== undefined) {
  467. this.presentationalDependencies.length = 0;
  468. }
  469. if (this.codeGenerationDependencies !== undefined) {
  470. this.codeGenerationDependencies.length = 0;
  471. }
  472. super.clearDependenciesAndBlocks();
  473. }
  474. /**
  475. * @param {WebpackError} warning the warning
  476. * @returns {void}
  477. */
  478. addWarning(warning) {
  479. if (this._warnings === undefined) {
  480. this._warnings = [];
  481. }
  482. this._warnings.push(warning);
  483. }
  484. /**
  485. * @returns {Iterable<WebpackError> | undefined} list of warnings if any
  486. */
  487. getWarnings() {
  488. return this._warnings;
  489. }
  490. /**
  491. * @returns {number} number of warnings
  492. */
  493. getNumberOfWarnings() {
  494. return this._warnings !== undefined ? this._warnings.length : 0;
  495. }
  496. /**
  497. * @param {WebpackError} error the error
  498. * @returns {void}
  499. */
  500. addError(error) {
  501. if (this._errors === undefined) {
  502. this._errors = [];
  503. }
  504. this._errors.push(error);
  505. }
  506. /**
  507. * @returns {Iterable<WebpackError> | undefined} list of errors if any
  508. */
  509. getErrors() {
  510. return this._errors;
  511. }
  512. /**
  513. * @returns {number} number of errors
  514. */
  515. getNumberOfErrors() {
  516. return this._errors !== undefined ? this._errors.length : 0;
  517. }
  518. /**
  519. * removes all warnings and errors
  520. * @returns {void}
  521. */
  522. clearWarningsAndErrors() {
  523. if (this._warnings !== undefined) {
  524. this._warnings.length = 0;
  525. }
  526. if (this._errors !== undefined) {
  527. this._errors.length = 0;
  528. }
  529. }
  530. /**
  531. * @param {ModuleGraph} moduleGraph the module graph
  532. * @returns {boolean} true, if the module is optional
  533. */
  534. isOptional(moduleGraph) {
  535. let hasConnections = false;
  536. for (const r of moduleGraph.getIncomingConnections(this)) {
  537. if (
  538. !r.dependency ||
  539. !r.dependency.optional ||
  540. !r.isTargetActive(undefined)
  541. ) {
  542. return false;
  543. }
  544. hasConnections = true;
  545. }
  546. return hasConnections;
  547. }
  548. /**
  549. * @param {ChunkGraph} chunkGraph the chunk graph
  550. * @param {Chunk} chunk a chunk
  551. * @param {Chunk=} ignoreChunk chunk to be ignored
  552. * @returns {boolean} true, if the module is accessible from "chunk" when ignoring "ignoreChunk"
  553. */
  554. isAccessibleInChunk(chunkGraph, chunk, ignoreChunk) {
  555. // Check if module is accessible in ALL chunk groups
  556. for (const chunkGroup of chunk.groupsIterable) {
  557. if (!this.isAccessibleInChunkGroup(chunkGraph, chunkGroup)) return false;
  558. }
  559. return true;
  560. }
  561. /**
  562. * @param {ChunkGraph} chunkGraph the chunk graph
  563. * @param {ChunkGroup} chunkGroup a chunk group
  564. * @param {Chunk=} ignoreChunk chunk to be ignored
  565. * @returns {boolean} true, if the module is accessible from "chunkGroup" when ignoring "ignoreChunk"
  566. */
  567. isAccessibleInChunkGroup(chunkGraph, chunkGroup, ignoreChunk) {
  568. const queue = new Set([chunkGroup]);
  569. // Check if module is accessible from all items of the queue
  570. queueFor: for (const cg of queue) {
  571. // 1. If module is in one of the chunks of the group we can continue checking the next items
  572. // because it's accessible.
  573. for (const chunk of cg.chunks) {
  574. if (chunk !== ignoreChunk && chunkGraph.isModuleInChunk(this, chunk))
  575. continue queueFor;
  576. }
  577. // 2. If the chunk group is initial, we can break here because it's not accessible.
  578. if (chunkGroup.isInitial()) return false;
  579. // 3. Enqueue all parents because it must be accessible from ALL parents
  580. for (const parent of chunkGroup.parentsIterable) queue.add(parent);
  581. }
  582. // When we processed through the whole list and we didn't bailout, the module is accessible
  583. return true;
  584. }
  585. /**
  586. * @param {Chunk} chunk a chunk
  587. * @param {ModuleGraph} moduleGraph the module graph
  588. * @param {ChunkGraph} chunkGraph the chunk graph
  589. * @returns {boolean} true, if the module has any reason why "chunk" should be included
  590. */
  591. hasReasonForChunk(chunk, moduleGraph, chunkGraph) {
  592. // check for each reason if we need the chunk
  593. for (const [
  594. fromModule,
  595. connections
  596. ] of moduleGraph.getIncomingConnectionsByOriginModule(this)) {
  597. if (!connections.some(c => c.isTargetActive(chunk.runtime))) continue;
  598. for (const originChunk of chunkGraph.getModuleChunksIterable(
  599. fromModule
  600. )) {
  601. // return true if module this is not reachable from originChunk when ignoring chunk
  602. if (!this.isAccessibleInChunk(chunkGraph, originChunk, chunk))
  603. return true;
  604. }
  605. }
  606. return false;
  607. }
  608. /**
  609. * @param {ModuleGraph} moduleGraph the module graph
  610. * @param {RuntimeSpec} runtime the runtime
  611. * @returns {boolean} true if at least one other module depends on this module
  612. */
  613. hasReasons(moduleGraph, runtime) {
  614. for (const c of moduleGraph.getIncomingConnections(this)) {
  615. if (c.isTargetActive(runtime)) return true;
  616. }
  617. return false;
  618. }
  619. /**
  620. * @returns {string} for debugging
  621. */
  622. toString() {
  623. return `Module[${this.debugId}: ${this.identifier()}]`;
  624. }
  625. /**
  626. * @param {NeedBuildContext} context context info
  627. * @param {function(WebpackError=, boolean=): void} callback callback function, returns true, if the module needs a rebuild
  628. * @returns {void}
  629. */
  630. needBuild(context, callback) {
  631. callback(
  632. null,
  633. !this.buildMeta ||
  634. this.needRebuild === Module.prototype.needRebuild ||
  635. deprecatedNeedRebuild(this, context)
  636. );
  637. }
  638. /**
  639. * @deprecated Use needBuild instead
  640. * @param {Map<string, number|null>} fileTimestamps timestamps of files
  641. * @param {Map<string, number|null>} contextTimestamps timestamps of directories
  642. * @returns {boolean} true, if the module needs a rebuild
  643. */
  644. needRebuild(fileTimestamps, contextTimestamps) {
  645. return true;
  646. }
  647. /**
  648. * @param {Hash} hash the hash used to track dependencies
  649. * @param {UpdateHashContext} context context
  650. * @returns {void}
  651. */
  652. updateHash(
  653. hash,
  654. context = {
  655. chunkGraph: ChunkGraph.getChunkGraphForModule(
  656. this,
  657. "Module.updateHash",
  658. "DEP_WEBPACK_MODULE_UPDATE_HASH"
  659. ),
  660. runtime: undefined
  661. }
  662. ) {
  663. const { chunkGraph, runtime } = context;
  664. hash.update(chunkGraph.getModuleGraphHash(this, runtime));
  665. if (this.presentationalDependencies !== undefined) {
  666. for (const dep of this.presentationalDependencies) {
  667. dep.updateHash(hash, context);
  668. }
  669. }
  670. super.updateHash(hash, context);
  671. }
  672. /**
  673. * @returns {void}
  674. */
  675. invalidateBuild() {
  676. // should be overridden to support this feature
  677. }
  678. /* istanbul ignore next */
  679. /**
  680. * @abstract
  681. * @returns {string} a unique identifier of the module
  682. */
  683. identifier() {
  684. const AbstractMethodError = require("./AbstractMethodError");
  685. throw new AbstractMethodError();
  686. }
  687. /* istanbul ignore next */
  688. /**
  689. * @abstract
  690. * @param {RequestShortener} requestShortener the request shortener
  691. * @returns {string} a user readable identifier of the module
  692. */
  693. readableIdentifier(requestShortener) {
  694. const AbstractMethodError = require("./AbstractMethodError");
  695. throw new AbstractMethodError();
  696. }
  697. /* istanbul ignore next */
  698. /**
  699. * @abstract
  700. * @param {WebpackOptions} options webpack options
  701. * @param {Compilation} compilation the compilation
  702. * @param {ResolverWithOptions} resolver the resolver
  703. * @param {InputFileSystem} fs the file system
  704. * @param {function(WebpackError=): void} callback callback function
  705. * @returns {void}
  706. */
  707. build(options, compilation, resolver, fs, callback) {
  708. const AbstractMethodError = require("./AbstractMethodError");
  709. throw new AbstractMethodError();
  710. }
  711. /**
  712. * @abstract
  713. * @returns {Set<string>} types available (do not mutate)
  714. */
  715. getSourceTypes() {
  716. // Better override this method to return the correct types
  717. if (this.source === Module.prototype.source) {
  718. return DEFAULT_TYPES_UNKNOWN;
  719. } else {
  720. return DEFAULT_TYPES_JS;
  721. }
  722. }
  723. /**
  724. * @abstract
  725. * @deprecated Use codeGeneration() instead
  726. * @param {DependencyTemplates} dependencyTemplates the dependency templates
  727. * @param {RuntimeTemplate} runtimeTemplate the runtime template
  728. * @param {string=} type the type of source that should be generated
  729. * @returns {Source} generated source
  730. */
  731. source(dependencyTemplates, runtimeTemplate, type = "javascript") {
  732. if (this.codeGeneration === Module.prototype.codeGeneration) {
  733. const AbstractMethodError = require("./AbstractMethodError");
  734. throw new AbstractMethodError();
  735. }
  736. const chunkGraph = ChunkGraph.getChunkGraphForModule(
  737. this,
  738. "Module.source() is deprecated. Use Compilation.codeGenerationResults.getSource(module, runtime, type) instead",
  739. "DEP_WEBPACK_MODULE_SOURCE"
  740. );
  741. /** @type {CodeGenerationContext} */
  742. const codeGenContext = {
  743. dependencyTemplates,
  744. runtimeTemplate,
  745. moduleGraph: chunkGraph.moduleGraph,
  746. chunkGraph,
  747. runtime: undefined,
  748. codeGenerationResults: undefined
  749. };
  750. const sources = this.codeGeneration(codeGenContext).sources;
  751. return type ? sources.get(type) : sources.get(first(this.getSourceTypes()));
  752. }
  753. /* istanbul ignore next */
  754. /**
  755. * @abstract
  756. * @param {string=} type the source type for which the size should be estimated
  757. * @returns {number} the estimated size of the module (must be non-zero)
  758. */
  759. size(type) {
  760. const AbstractMethodError = require("./AbstractMethodError");
  761. throw new AbstractMethodError();
  762. }
  763. /**
  764. * @param {LibIdentOptions} options options
  765. * @returns {string | null} an identifier for library inclusion
  766. */
  767. libIdent(options) {
  768. return null;
  769. }
  770. /**
  771. * @returns {string | null} absolute path which should be used for condition matching (usually the resource path)
  772. */
  773. nameForCondition() {
  774. return null;
  775. }
  776. /**
  777. * @param {ConcatenationBailoutReasonContext} context context
  778. * @returns {string | undefined} reason why this module can't be concatenated, undefined when it can be concatenated
  779. */
  780. getConcatenationBailoutReason(context) {
  781. return `Module Concatenation is not implemented for ${this.constructor.name}`;
  782. }
  783. /**
  784. * @param {ModuleGraph} moduleGraph the module graph
  785. * @returns {ConnectionState} how this module should be connected to referencing modules when consumed for side-effects only
  786. */
  787. getSideEffectsConnectionState(moduleGraph) {
  788. return true;
  789. }
  790. /**
  791. * @param {CodeGenerationContext} context context for code generation
  792. * @returns {CodeGenerationResult} result
  793. */
  794. codeGeneration(context) {
  795. // Best override this method
  796. const sources = new Map();
  797. for (const type of this.getSourceTypes()) {
  798. if (type !== "unknown") {
  799. sources.set(
  800. type,
  801. this.source(
  802. context.dependencyTemplates,
  803. context.runtimeTemplate,
  804. type
  805. )
  806. );
  807. }
  808. }
  809. return {
  810. sources,
  811. runtimeRequirements: new Set([
  812. RuntimeGlobals.module,
  813. RuntimeGlobals.exports,
  814. RuntimeGlobals.require
  815. ])
  816. };
  817. }
  818. /**
  819. * @param {Chunk} chunk the chunk which condition should be checked
  820. * @param {Compilation} compilation the compilation
  821. * @returns {boolean} true, if the chunk is ok for the module
  822. */
  823. chunkCondition(chunk, compilation) {
  824. return true;
  825. }
  826. hasChunkCondition() {
  827. return this.chunkCondition !== Module.prototype.chunkCondition;
  828. }
  829. /**
  830. * Assuming this module is in the cache. Update the (cached) module with
  831. * the fresh module from the factory. Usually updates internal references
  832. * and properties.
  833. * @param {Module} module fresh module
  834. * @returns {void}
  835. */
  836. updateCacheModule(module) {
  837. this.type = module.type;
  838. this.layer = module.layer;
  839. this.context = module.context;
  840. this.factoryMeta = module.factoryMeta;
  841. this.resolveOptions = module.resolveOptions;
  842. }
  843. /**
  844. * Module should be unsafe cached. Get data that's needed for that.
  845. * This data will be passed to restoreFromUnsafeCache later.
  846. * @returns {object} cached data
  847. */
  848. getUnsafeCacheData() {
  849. return {
  850. factoryMeta: this.factoryMeta,
  851. resolveOptions: this.resolveOptions
  852. };
  853. }
  854. /**
  855. * restore unsafe cache data
  856. * @param {object} unsafeCacheData data from getUnsafeCacheData
  857. * @param {NormalModuleFactory} normalModuleFactory the normal module factory handling the unsafe caching
  858. */
  859. _restoreFromUnsafeCache(unsafeCacheData, normalModuleFactory) {
  860. this.factoryMeta = unsafeCacheData.factoryMeta;
  861. this.resolveOptions = unsafeCacheData.resolveOptions;
  862. }
  863. /**
  864. * Assuming this module is in the cache. Remove internal references to allow freeing some memory.
  865. */
  866. cleanupForCache() {
  867. this.factoryMeta = undefined;
  868. this.resolveOptions = undefined;
  869. }
  870. /**
  871. * @returns {Source | null} the original source for the module before webpack transformation
  872. */
  873. originalSource() {
  874. return null;
  875. }
  876. /**
  877. * @param {LazySet<string>} fileDependencies set where file dependencies are added to
  878. * @param {LazySet<string>} contextDependencies set where context dependencies are added to
  879. * @param {LazySet<string>} missingDependencies set where missing dependencies are added to
  880. * @param {LazySet<string>} buildDependencies set where build dependencies are added to
  881. */
  882. addCacheDependencies(
  883. fileDependencies,
  884. contextDependencies,
  885. missingDependencies,
  886. buildDependencies
  887. ) {}
  888. serialize(context) {
  889. const { write } = context;
  890. write(this.type);
  891. write(this.layer);
  892. write(this.context);
  893. write(this.resolveOptions);
  894. write(this.factoryMeta);
  895. write(this.useSourceMap);
  896. write(this.useSimpleSourceMap);
  897. write(
  898. this._warnings !== undefined && this._warnings.length === 0
  899. ? undefined
  900. : this._warnings
  901. );
  902. write(
  903. this._errors !== undefined && this._errors.length === 0
  904. ? undefined
  905. : this._errors
  906. );
  907. write(this.buildMeta);
  908. write(this.buildInfo);
  909. write(this.presentationalDependencies);
  910. write(this.codeGenerationDependencies);
  911. super.serialize(context);
  912. }
  913. deserialize(context) {
  914. const { read } = context;
  915. this.type = read();
  916. this.layer = read();
  917. this.context = read();
  918. this.resolveOptions = read();
  919. this.factoryMeta = read();
  920. this.useSourceMap = read();
  921. this.useSimpleSourceMap = read();
  922. this._warnings = read();
  923. this._errors = read();
  924. this.buildMeta = read();
  925. this.buildInfo = read();
  926. this.presentationalDependencies = read();
  927. this.codeGenerationDependencies = read();
  928. super.deserialize(context);
  929. }
  930. }
  931. makeSerializable(Module, "webpack/lib/Module");
  932. // TODO remove in webpack 6
  933. Object.defineProperty(Module.prototype, "hasEqualsChunks", {
  934. get() {
  935. throw new Error(
  936. "Module.hasEqualsChunks was renamed (use hasEqualChunks instead)"
  937. );
  938. }
  939. });
  940. // TODO remove in webpack 6
  941. Object.defineProperty(Module.prototype, "isUsed", {
  942. get() {
  943. throw new Error(
  944. "Module.isUsed was renamed (use getUsedName, isExportUsed or isModuleUsed instead)"
  945. );
  946. }
  947. });
  948. // TODO remove in webpack 6
  949. Object.defineProperty(Module.prototype, "errors", {
  950. get: util.deprecate(
  951. /**
  952. * @this {Module}
  953. * @returns {WebpackError[]} array
  954. */
  955. function () {
  956. if (this._errors === undefined) {
  957. this._errors = [];
  958. }
  959. return this._errors;
  960. },
  961. "Module.errors was removed (use getErrors instead)",
  962. "DEP_WEBPACK_MODULE_ERRORS"
  963. )
  964. });
  965. // TODO remove in webpack 6
  966. Object.defineProperty(Module.prototype, "warnings", {
  967. get: util.deprecate(
  968. /**
  969. * @this {Module}
  970. * @returns {WebpackError[]} array
  971. */
  972. function () {
  973. if (this._warnings === undefined) {
  974. this._warnings = [];
  975. }
  976. return this._warnings;
  977. },
  978. "Module.warnings was removed (use getWarnings instead)",
  979. "DEP_WEBPACK_MODULE_WARNINGS"
  980. )
  981. });
  982. // TODO remove in webpack 6
  983. Object.defineProperty(Module.prototype, "used", {
  984. get() {
  985. throw new Error(
  986. "Module.used was refactored (use ModuleGraph.getUsedExports instead)"
  987. );
  988. },
  989. set(value) {
  990. throw new Error(
  991. "Module.used was refactored (use ModuleGraph.setUsedExports instead)"
  992. );
  993. }
  994. });
  995. module.exports = Module;