123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728 |
- /*
- MIT License http://www.opensource.org/licenses/mit-license.php
- Author Tobias Koppers @sokra
- */
- "use strict";
- const util = require("util");
- const Entrypoint = require("./Entrypoint");
- const ModuleGraphConnection = require("./ModuleGraphConnection");
- const { first } = require("./util/SetHelpers");
- const SortableSet = require("./util/SortableSet");
- const {
- compareModulesById,
- compareIterables,
- compareModulesByIdentifier,
- concatComparators,
- compareSelect,
- compareIds
- } = require("./util/comparators");
- const createHash = require("./util/createHash");
- const findGraphRoots = require("./util/findGraphRoots");
- const {
- RuntimeSpecMap,
- RuntimeSpecSet,
- runtimeToString,
- mergeRuntime,
- forEachRuntime
- } = require("./util/runtime");
- /** @typedef {import("./AsyncDependenciesBlock")} AsyncDependenciesBlock */
- /** @typedef {import("./Chunk")} Chunk */
- /** @typedef {import("./ChunkGroup")} ChunkGroup */
- /** @typedef {import("./Module")} Module */
- /** @typedef {import("./ModuleGraph")} ModuleGraph */
- /** @typedef {import("./RuntimeModule")} RuntimeModule */
- /** @typedef {typeof import("./util/Hash")} Hash */
- /** @typedef {import("./util/runtime").RuntimeSpec} RuntimeSpec */
- /** @type {ReadonlySet<string>} */
- const EMPTY_SET = new Set();
- const ZERO_BIG_INT = BigInt(0);
- const compareModuleIterables = compareIterables(compareModulesByIdentifier);
- /** @typedef {(c: Chunk, chunkGraph: ChunkGraph) => boolean} ChunkFilterPredicate */
- /** @typedef {(m: Module) => boolean} ModuleFilterPredicate */
- /**
- * @typedef {Object} ChunkSizeOptions
- * @property {number=} chunkOverhead constant overhead for a chunk
- * @property {number=} entryChunkMultiplicator multiplicator for initial chunks
- */
- class ModuleHashInfo {
- constructor(hash, renderedHash) {
- this.hash = hash;
- this.renderedHash = renderedHash;
- }
- }
- /** @template T @typedef {(set: SortableSet<T>) => T[]} SetToArrayFunction<T> */
- /**
- * @template T
- * @param {SortableSet<T>} set the set
- * @returns {T[]} set as array
- */
- const getArray = set => {
- return Array.from(set);
- };
- /**
- * @param {SortableSet<Chunk>} chunks the chunks
- * @returns {RuntimeSpecSet} runtimes
- */
- const getModuleRuntimes = chunks => {
- const runtimes = new RuntimeSpecSet();
- for (const chunk of chunks) {
- runtimes.add(chunk.runtime);
- }
- return runtimes;
- };
- /**
- * @param {SortableSet<Module>} set the set
- * @returns {Map<string, SortableSet<Module>>} modules by source type
- */
- const modulesBySourceType = set => {
- /** @type {Map<string, SortableSet<Module>>} */
- const map = new Map();
- for (const module of set) {
- for (const sourceType of module.getSourceTypes()) {
- let innerSet = map.get(sourceType);
- if (innerSet === undefined) {
- innerSet = new SortableSet();
- map.set(sourceType, innerSet);
- }
- innerSet.add(module);
- }
- }
- for (const [key, innerSet] of map) {
- // When all modules have the source type, we reuse the original SortableSet
- // to benefit from the shared cache (especially for sorting)
- if (innerSet.size === set.size) {
- map.set(key, set);
- }
- }
- return map;
- };
- /** @type {WeakMap<Function, any>} */
- const createOrderedArrayFunctionMap = new WeakMap();
- /**
- * @template T
- * @param {function(T, T): -1|0|1} comparator comparator function
- * @returns {SetToArrayFunction<T>} set as ordered array
- */
- const createOrderedArrayFunction = comparator => {
- /** @type {SetToArrayFunction<T>} */
- let fn = createOrderedArrayFunctionMap.get(comparator);
- if (fn !== undefined) return fn;
- fn = set => {
- set.sortWith(comparator);
- return Array.from(set);
- };
- createOrderedArrayFunctionMap.set(comparator, fn);
- return fn;
- };
- /**
- * @param {Iterable<Module>} modules the modules to get the count/size of
- * @returns {number} the size of the modules
- */
- const getModulesSize = modules => {
- let size = 0;
- for (const module of modules) {
- for (const type of module.getSourceTypes()) {
- size += module.size(type);
- }
- }
- return size;
- };
- /**
- * @param {Iterable<Module>} modules the sortable Set to get the size of
- * @returns {Record<string, number>} the sizes of the modules
- */
- const getModulesSizes = modules => {
- let sizes = Object.create(null);
- for (const module of modules) {
- for (const type of module.getSourceTypes()) {
- sizes[type] = (sizes[type] || 0) + module.size(type);
- }
- }
- return sizes;
- };
- /**
- * @param {Chunk} a chunk
- * @param {Chunk} b chunk
- * @returns {boolean} true, if a is always a parent of b
- */
- const isAvailableChunk = (a, b) => {
- const queue = new Set(b.groupsIterable);
- for (const chunkGroup of queue) {
- if (a.isInGroup(chunkGroup)) continue;
- if (chunkGroup.isInitial()) return false;
- for (const parent of chunkGroup.parentsIterable) {
- queue.add(parent);
- }
- }
- return true;
- };
- class ChunkGraphModule {
- constructor() {
- /** @type {SortableSet<Chunk>} */
- this.chunks = new SortableSet();
- /** @type {Set<Chunk> | undefined} */
- this.entryInChunks = undefined;
- /** @type {Set<Chunk> | undefined} */
- this.runtimeInChunks = undefined;
- /** @type {RuntimeSpecMap<ModuleHashInfo>} */
- this.hashes = undefined;
- /** @type {string | number} */
- this.id = null;
- /** @type {RuntimeSpecMap<Set<string>> | undefined} */
- this.runtimeRequirements = undefined;
- /** @type {RuntimeSpecMap<string>} */
- this.graphHashes = undefined;
- /** @type {RuntimeSpecMap<string>} */
- this.graphHashesWithConnections = undefined;
- }
- }
- class ChunkGraphChunk {
- constructor() {
- /** @type {SortableSet<Module>} */
- this.modules = new SortableSet();
- /** @type {Map<Module, Entrypoint>} */
- this.entryModules = new Map();
- /** @type {SortableSet<RuntimeModule>} */
- this.runtimeModules = new SortableSet();
- /** @type {Set<RuntimeModule> | undefined} */
- this.fullHashModules = undefined;
- /** @type {Set<RuntimeModule> | undefined} */
- this.dependentHashModules = undefined;
- /** @type {Set<string> | undefined} */
- this.runtimeRequirements = undefined;
- /** @type {Set<string>} */
- this.runtimeRequirementsInTree = new Set();
- }
- }
- class ChunkGraph {
- /**
- * @param {ModuleGraph} moduleGraph the module graph
- * @param {string | Hash} hashFunction the hash function to use
- */
- constructor(moduleGraph, hashFunction = "md4") {
- /** @private @type {WeakMap<Module, ChunkGraphModule>} */
- this._modules = new WeakMap();
- /** @private @type {WeakMap<Chunk, ChunkGraphChunk>} */
- this._chunks = new WeakMap();
- /** @private @type {WeakMap<AsyncDependenciesBlock, ChunkGroup>} */
- this._blockChunkGroups = new WeakMap();
- /** @private @type {Map<string, string | number>} */
- this._runtimeIds = new Map();
- /** @type {ModuleGraph} */
- this.moduleGraph = moduleGraph;
- this._hashFunction = hashFunction;
- this._getGraphRoots = this._getGraphRoots.bind(this);
- }
- /**
- * @private
- * @param {Module} module the module
- * @returns {ChunkGraphModule} internal module
- */
- _getChunkGraphModule(module) {
- let cgm = this._modules.get(module);
- if (cgm === undefined) {
- cgm = new ChunkGraphModule();
- this._modules.set(module, cgm);
- }
- return cgm;
- }
- /**
- * @private
- * @param {Chunk} chunk the chunk
- * @returns {ChunkGraphChunk} internal chunk
- */
- _getChunkGraphChunk(chunk) {
- let cgc = this._chunks.get(chunk);
- if (cgc === undefined) {
- cgc = new ChunkGraphChunk();
- this._chunks.set(chunk, cgc);
- }
- return cgc;
- }
- /**
- * @param {SortableSet<Module>} set the sortable Set to get the roots of
- * @returns {Module[]} the graph roots
- */
- _getGraphRoots(set) {
- const { moduleGraph } = this;
- return Array.from(
- findGraphRoots(set, module => {
- /** @type {Set<Module>} */
- const set = new Set();
- const addDependencies = module => {
- for (const connection of moduleGraph.getOutgoingConnections(module)) {
- if (!connection.module) continue;
- const activeState = connection.getActiveState(undefined);
- if (activeState === false) continue;
- if (activeState === ModuleGraphConnection.TRANSITIVE_ONLY) {
- addDependencies(connection.module);
- continue;
- }
- set.add(connection.module);
- }
- };
- addDependencies(module);
- return set;
- })
- ).sort(compareModulesByIdentifier);
- }
- /**
- * @param {Chunk} chunk the new chunk
- * @param {Module} module the module
- * @returns {void}
- */
- connectChunkAndModule(chunk, module) {
- const cgm = this._getChunkGraphModule(module);
- const cgc = this._getChunkGraphChunk(chunk);
- cgm.chunks.add(chunk);
- cgc.modules.add(module);
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {Module} module the module
- * @returns {void}
- */
- disconnectChunkAndModule(chunk, module) {
- const cgm = this._getChunkGraphModule(module);
- const cgc = this._getChunkGraphChunk(chunk);
- cgc.modules.delete(module);
- cgm.chunks.delete(chunk);
- }
- /**
- * @param {Chunk} chunk the chunk which will be disconnected
- * @returns {void}
- */
- disconnectChunk(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- for (const module of cgc.modules) {
- const cgm = this._getChunkGraphModule(module);
- cgm.chunks.delete(chunk);
- }
- cgc.modules.clear();
- chunk.disconnectFromGroups();
- ChunkGraph.clearChunkGraphForChunk(chunk);
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {Iterable<Module>} modules the modules
- * @returns {void}
- */
- attachModules(chunk, modules) {
- const cgc = this._getChunkGraphChunk(chunk);
- for (const module of modules) {
- cgc.modules.add(module);
- }
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {Iterable<RuntimeModule>} modules the runtime modules
- * @returns {void}
- */
- attachRuntimeModules(chunk, modules) {
- const cgc = this._getChunkGraphChunk(chunk);
- for (const module of modules) {
- cgc.runtimeModules.add(module);
- }
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {Iterable<RuntimeModule>} modules the modules that require a full hash
- * @returns {void}
- */
- attachFullHashModules(chunk, modules) {
- const cgc = this._getChunkGraphChunk(chunk);
- if (cgc.fullHashModules === undefined) cgc.fullHashModules = new Set();
- for (const module of modules) {
- cgc.fullHashModules.add(module);
- }
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {Iterable<RuntimeModule>} modules the modules that require a full hash
- * @returns {void}
- */
- attachDependentHashModules(chunk, modules) {
- const cgc = this._getChunkGraphChunk(chunk);
- if (cgc.dependentHashModules === undefined)
- cgc.dependentHashModules = new Set();
- for (const module of modules) {
- cgc.dependentHashModules.add(module);
- }
- }
- /**
- * @param {Module} oldModule the replaced module
- * @param {Module} newModule the replacing module
- * @returns {void}
- */
- replaceModule(oldModule, newModule) {
- const oldCgm = this._getChunkGraphModule(oldModule);
- const newCgm = this._getChunkGraphModule(newModule);
- for (const chunk of oldCgm.chunks) {
- const cgc = this._getChunkGraphChunk(chunk);
- cgc.modules.delete(oldModule);
- cgc.modules.add(newModule);
- newCgm.chunks.add(chunk);
- }
- oldCgm.chunks.clear();
- if (oldCgm.entryInChunks !== undefined) {
- if (newCgm.entryInChunks === undefined) {
- newCgm.entryInChunks = new Set();
- }
- for (const chunk of oldCgm.entryInChunks) {
- const cgc = this._getChunkGraphChunk(chunk);
- const old = cgc.entryModules.get(oldModule);
- /** @type {Map<Module, Entrypoint>} */
- const newEntryModules = new Map();
- for (const [m, cg] of cgc.entryModules) {
- if (m === oldModule) {
- newEntryModules.set(newModule, old);
- } else {
- newEntryModules.set(m, cg);
- }
- }
- cgc.entryModules = newEntryModules;
- newCgm.entryInChunks.add(chunk);
- }
- oldCgm.entryInChunks = undefined;
- }
- if (oldCgm.runtimeInChunks !== undefined) {
- if (newCgm.runtimeInChunks === undefined) {
- newCgm.runtimeInChunks = new Set();
- }
- for (const chunk of oldCgm.runtimeInChunks) {
- const cgc = this._getChunkGraphChunk(chunk);
- cgc.runtimeModules.delete(/** @type {RuntimeModule} */ (oldModule));
- cgc.runtimeModules.add(/** @type {RuntimeModule} */ (newModule));
- newCgm.runtimeInChunks.add(chunk);
- if (
- cgc.fullHashModules !== undefined &&
- cgc.fullHashModules.has(/** @type {RuntimeModule} */ (oldModule))
- ) {
- cgc.fullHashModules.delete(/** @type {RuntimeModule} */ (oldModule));
- cgc.fullHashModules.add(/** @type {RuntimeModule} */ (newModule));
- }
- if (
- cgc.dependentHashModules !== undefined &&
- cgc.dependentHashModules.has(/** @type {RuntimeModule} */ (oldModule))
- ) {
- cgc.dependentHashModules.delete(
- /** @type {RuntimeModule} */ (oldModule)
- );
- cgc.dependentHashModules.add(
- /** @type {RuntimeModule} */ (newModule)
- );
- }
- }
- oldCgm.runtimeInChunks = undefined;
- }
- }
- /**
- * @param {Module} module the checked module
- * @param {Chunk} chunk the checked chunk
- * @returns {boolean} true, if the chunk contains the module
- */
- isModuleInChunk(module, chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.modules.has(module);
- }
- /**
- * @param {Module} module the checked module
- * @param {ChunkGroup} chunkGroup the checked chunk group
- * @returns {boolean} true, if the chunk contains the module
- */
- isModuleInChunkGroup(module, chunkGroup) {
- for (const chunk of chunkGroup.chunks) {
- if (this.isModuleInChunk(module, chunk)) return true;
- }
- return false;
- }
- /**
- * @param {Module} module the checked module
- * @returns {boolean} true, if the module is entry of any chunk
- */
- isEntryModule(module) {
- const cgm = this._getChunkGraphModule(module);
- return cgm.entryInChunks !== undefined;
- }
- /**
- * @param {Module} module the module
- * @returns {Iterable<Chunk>} iterable of chunks (do not modify)
- */
- getModuleChunksIterable(module) {
- const cgm = this._getChunkGraphModule(module);
- return cgm.chunks;
- }
- /**
- * @param {Module} module the module
- * @param {function(Chunk, Chunk): -1|0|1} sortFn sort function
- * @returns {Iterable<Chunk>} iterable of chunks (do not modify)
- */
- getOrderedModuleChunksIterable(module, sortFn) {
- const cgm = this._getChunkGraphModule(module);
- cgm.chunks.sortWith(sortFn);
- return cgm.chunks;
- }
- /**
- * @param {Module} module the module
- * @returns {Chunk[]} array of chunks (cached, do not modify)
- */
- getModuleChunks(module) {
- const cgm = this._getChunkGraphModule(module);
- return cgm.chunks.getFromCache(getArray);
- }
- /**
- * @param {Module} module the module
- * @returns {number} the number of chunk which contain the module
- */
- getNumberOfModuleChunks(module) {
- const cgm = this._getChunkGraphModule(module);
- return cgm.chunks.size;
- }
- /**
- * @param {Module} module the module
- * @returns {RuntimeSpecSet} runtimes
- */
- getModuleRuntimes(module) {
- const cgm = this._getChunkGraphModule(module);
- return cgm.chunks.getFromUnorderedCache(getModuleRuntimes);
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {number} the number of modules which are contained in this chunk
- */
- getNumberOfChunkModules(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.modules.size;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {number} the number of full hash modules which are contained in this chunk
- */
- getNumberOfChunkFullHashModules(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.fullHashModules === undefined ? 0 : cgc.fullHashModules.size;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {Iterable<Module>} return the modules for this chunk
- */
- getChunkModulesIterable(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.modules;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {string} sourceType source type
- * @returns {Iterable<Module> | undefined} return the modules for this chunk
- */
- getChunkModulesIterableBySourceType(chunk, sourceType) {
- const cgc = this._getChunkGraphChunk(chunk);
- const modulesWithSourceType = cgc.modules
- .getFromUnorderedCache(modulesBySourceType)
- .get(sourceType);
- return modulesWithSourceType;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {function(Module, Module): -1|0|1} comparator comparator function
- * @returns {Iterable<Module>} return the modules for this chunk
- */
- getOrderedChunkModulesIterable(chunk, comparator) {
- const cgc = this._getChunkGraphChunk(chunk);
- cgc.modules.sortWith(comparator);
- return cgc.modules;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {string} sourceType source type
- * @param {function(Module, Module): -1|0|1} comparator comparator function
- * @returns {Iterable<Module> | undefined} return the modules for this chunk
- */
- getOrderedChunkModulesIterableBySourceType(chunk, sourceType, comparator) {
- const cgc = this._getChunkGraphChunk(chunk);
- const modulesWithSourceType = cgc.modules
- .getFromUnorderedCache(modulesBySourceType)
- .get(sourceType);
- if (modulesWithSourceType === undefined) return undefined;
- modulesWithSourceType.sortWith(comparator);
- return modulesWithSourceType;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {Module[]} return the modules for this chunk (cached, do not modify)
- */
- getChunkModules(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.modules.getFromUnorderedCache(getArray);
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {function(Module, Module): -1|0|1} comparator comparator function
- * @returns {Module[]} return the modules for this chunk (cached, do not modify)
- */
- getOrderedChunkModules(chunk, comparator) {
- const cgc = this._getChunkGraphChunk(chunk);
- const arrayFunction = createOrderedArrayFunction(comparator);
- return cgc.modules.getFromUnorderedCache(arrayFunction);
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {ModuleFilterPredicate} filterFn function used to filter modules
- * @param {boolean} includeAllChunks all chunks or only async chunks
- * @returns {Record<string|number, (string|number)[]>} chunk to module ids object
- */
- getChunkModuleIdMap(chunk, filterFn, includeAllChunks = false) {
- /** @type {Record<string|number, (string|number)[]>} */
- const chunkModuleIdMap = Object.create(null);
- for (const asyncChunk of includeAllChunks
- ? chunk.getAllReferencedChunks()
- : chunk.getAllAsyncChunks()) {
- /** @type {(string|number)[]} */
- let array;
- for (const module of this.getOrderedChunkModulesIterable(
- asyncChunk,
- compareModulesById(this)
- )) {
- if (filterFn(module)) {
- if (array === undefined) {
- array = [];
- chunkModuleIdMap[asyncChunk.id] = array;
- }
- const moduleId = this.getModuleId(module);
- array.push(moduleId);
- }
- }
- }
- return chunkModuleIdMap;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {ModuleFilterPredicate} filterFn function used to filter modules
- * @param {number} hashLength length of the hash
- * @param {boolean} includeAllChunks all chunks or only async chunks
- * @returns {Record<string|number, Record<string|number, string>>} chunk to module id to module hash object
- */
- getChunkModuleRenderedHashMap(
- chunk,
- filterFn,
- hashLength = 0,
- includeAllChunks = false
- ) {
- /** @type {Record<string|number, Record<string|number, string>>} */
- const chunkModuleHashMap = Object.create(null);
- for (const asyncChunk of includeAllChunks
- ? chunk.getAllReferencedChunks()
- : chunk.getAllAsyncChunks()) {
- /** @type {Record<string|number, string>} */
- let idToHashMap;
- for (const module of this.getOrderedChunkModulesIterable(
- asyncChunk,
- compareModulesById(this)
- )) {
- if (filterFn(module)) {
- if (idToHashMap === undefined) {
- idToHashMap = Object.create(null);
- chunkModuleHashMap[asyncChunk.id] = idToHashMap;
- }
- const moduleId = this.getModuleId(module);
- const hash = this.getRenderedModuleHash(module, asyncChunk.runtime);
- idToHashMap[moduleId] = hashLength ? hash.slice(0, hashLength) : hash;
- }
- }
- }
- return chunkModuleHashMap;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {ChunkFilterPredicate} filterFn function used to filter chunks
- * @returns {Record<string|number, boolean>} chunk map
- */
- getChunkConditionMap(chunk, filterFn) {
- const map = Object.create(null);
- for (const c of chunk.getAllReferencedChunks()) {
- map[c.id] = filterFn(c, this);
- }
- return map;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {ModuleFilterPredicate} filterFn predicate function used to filter modules
- * @param {ChunkFilterPredicate=} filterChunkFn predicate function used to filter chunks
- * @returns {boolean} return true if module exists in graph
- */
- hasModuleInGraph(chunk, filterFn, filterChunkFn) {
- const queue = new Set(chunk.groupsIterable);
- const chunksProcessed = new Set();
- for (const chunkGroup of queue) {
- for (const innerChunk of chunkGroup.chunks) {
- if (!chunksProcessed.has(innerChunk)) {
- chunksProcessed.add(innerChunk);
- if (!filterChunkFn || filterChunkFn(innerChunk, this)) {
- for (const module of this.getChunkModulesIterable(innerChunk)) {
- if (filterFn(module)) {
- return true;
- }
- }
- }
- }
- }
- for (const child of chunkGroup.childrenIterable) {
- queue.add(child);
- }
- }
- return false;
- }
- /**
- * @param {Chunk} chunkA first chunk
- * @param {Chunk} chunkB second chunk
- * @returns {-1|0|1} this is a comparator function like sort and returns -1, 0, or 1 based on sort order
- */
- compareChunks(chunkA, chunkB) {
- const cgcA = this._getChunkGraphChunk(chunkA);
- const cgcB = this._getChunkGraphChunk(chunkB);
- if (cgcA.modules.size > cgcB.modules.size) return -1;
- if (cgcA.modules.size < cgcB.modules.size) return 1;
- cgcA.modules.sortWith(compareModulesByIdentifier);
- cgcB.modules.sortWith(compareModulesByIdentifier);
- return compareModuleIterables(cgcA.modules, cgcB.modules);
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {number} total size of all modules in the chunk
- */
- getChunkModulesSize(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.modules.getFromUnorderedCache(getModulesSize);
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {Record<string, number>} total sizes of all modules in the chunk by source type
- */
- getChunkModulesSizes(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.modules.getFromUnorderedCache(getModulesSizes);
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {Module[]} root modules of the chunks (ordered by identifier)
- */
- getChunkRootModules(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.modules.getFromUnorderedCache(this._getGraphRoots);
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {ChunkSizeOptions} options options object
- * @returns {number} total size of the chunk
- */
- getChunkSize(chunk, options = {}) {
- const cgc = this._getChunkGraphChunk(chunk);
- const modulesSize = cgc.modules.getFromUnorderedCache(getModulesSize);
- const chunkOverhead =
- typeof options.chunkOverhead === "number" ? options.chunkOverhead : 10000;
- const entryChunkMultiplicator =
- typeof options.entryChunkMultiplicator === "number"
- ? options.entryChunkMultiplicator
- : 10;
- return (
- chunkOverhead +
- modulesSize * (chunk.canBeInitial() ? entryChunkMultiplicator : 1)
- );
- }
- /**
- * @param {Chunk} chunkA chunk
- * @param {Chunk} chunkB chunk
- * @param {ChunkSizeOptions} options options object
- * @returns {number} total size of the chunk or false if chunks can't be integrated
- */
- getIntegratedChunksSize(chunkA, chunkB, options = {}) {
- const cgcA = this._getChunkGraphChunk(chunkA);
- const cgcB = this._getChunkGraphChunk(chunkB);
- const allModules = new Set(cgcA.modules);
- for (const m of cgcB.modules) allModules.add(m);
- let modulesSize = getModulesSize(allModules);
- const chunkOverhead =
- typeof options.chunkOverhead === "number" ? options.chunkOverhead : 10000;
- const entryChunkMultiplicator =
- typeof options.entryChunkMultiplicator === "number"
- ? options.entryChunkMultiplicator
- : 10;
- return (
- chunkOverhead +
- modulesSize *
- (chunkA.canBeInitial() || chunkB.canBeInitial()
- ? entryChunkMultiplicator
- : 1)
- );
- }
- /**
- * @param {Chunk} chunkA chunk
- * @param {Chunk} chunkB chunk
- * @returns {boolean} true, if chunks could be integrated
- */
- canChunksBeIntegrated(chunkA, chunkB) {
- if (chunkA.preventIntegration || chunkB.preventIntegration) {
- return false;
- }
- const hasRuntimeA = chunkA.hasRuntime();
- const hasRuntimeB = chunkB.hasRuntime();
- if (hasRuntimeA !== hasRuntimeB) {
- if (hasRuntimeA) {
- return isAvailableChunk(chunkA, chunkB);
- } else if (hasRuntimeB) {
- return isAvailableChunk(chunkB, chunkA);
- } else {
- return false;
- }
- }
- if (
- this.getNumberOfEntryModules(chunkA) > 0 ||
- this.getNumberOfEntryModules(chunkB) > 0
- ) {
- return false;
- }
- return true;
- }
- /**
- * @param {Chunk} chunkA the target chunk
- * @param {Chunk} chunkB the chunk to integrate
- * @returns {void}
- */
- integrateChunks(chunkA, chunkB) {
- // Decide for one name (deterministic)
- if (chunkA.name && chunkB.name) {
- if (
- this.getNumberOfEntryModules(chunkA) > 0 ===
- this.getNumberOfEntryModules(chunkB) > 0
- ) {
- // When both chunks have entry modules or none have one, use
- // shortest name
- if (chunkA.name.length !== chunkB.name.length) {
- chunkA.name =
- chunkA.name.length < chunkB.name.length ? chunkA.name : chunkB.name;
- } else {
- chunkA.name = chunkA.name < chunkB.name ? chunkA.name : chunkB.name;
- }
- } else if (this.getNumberOfEntryModules(chunkB) > 0) {
- // Pick the name of the chunk with the entry module
- chunkA.name = chunkB.name;
- }
- } else if (chunkB.name) {
- chunkA.name = chunkB.name;
- }
- // Merge id name hints
- for (const hint of chunkB.idNameHints) {
- chunkA.idNameHints.add(hint);
- }
- // Merge runtime
- chunkA.runtime = mergeRuntime(chunkA.runtime, chunkB.runtime);
- // getChunkModules is used here to create a clone, because disconnectChunkAndModule modifies
- for (const module of this.getChunkModules(chunkB)) {
- this.disconnectChunkAndModule(chunkB, module);
- this.connectChunkAndModule(chunkA, module);
- }
- for (const [module, chunkGroup] of Array.from(
- this.getChunkEntryModulesWithChunkGroupIterable(chunkB)
- )) {
- this.disconnectChunkAndEntryModule(chunkB, module);
- this.connectChunkAndEntryModule(chunkA, module, chunkGroup);
- }
- for (const chunkGroup of chunkB.groupsIterable) {
- chunkGroup.replaceChunk(chunkB, chunkA);
- chunkA.addGroup(chunkGroup);
- chunkB.removeGroup(chunkGroup);
- }
- ChunkGraph.clearChunkGraphForChunk(chunkB);
- }
- /**
- * @param {Chunk} chunk the chunk to upgrade
- * @returns {void}
- */
- upgradeDependentToFullHashModules(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- if (cgc.dependentHashModules === undefined) return;
- if (cgc.fullHashModules === undefined) {
- cgc.fullHashModules = cgc.dependentHashModules;
- } else {
- for (const m of cgc.dependentHashModules) {
- cgc.fullHashModules.add(m);
- }
- cgc.dependentHashModules = undefined;
- }
- }
- /**
- * @param {Module} module the checked module
- * @param {Chunk} chunk the checked chunk
- * @returns {boolean} true, if the chunk contains the module as entry
- */
- isEntryModuleInChunk(module, chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.entryModules.has(module);
- }
- /**
- * @param {Chunk} chunk the new chunk
- * @param {Module} module the entry module
- * @param {Entrypoint=} entrypoint the chunk group which must be loaded before the module is executed
- * @returns {void}
- */
- connectChunkAndEntryModule(chunk, module, entrypoint) {
- const cgm = this._getChunkGraphModule(module);
- const cgc = this._getChunkGraphChunk(chunk);
- if (cgm.entryInChunks === undefined) {
- cgm.entryInChunks = new Set();
- }
- cgm.entryInChunks.add(chunk);
- cgc.entryModules.set(module, entrypoint);
- }
- /**
- * @param {Chunk} chunk the new chunk
- * @param {RuntimeModule} module the runtime module
- * @returns {void}
- */
- connectChunkAndRuntimeModule(chunk, module) {
- const cgm = this._getChunkGraphModule(module);
- const cgc = this._getChunkGraphChunk(chunk);
- if (cgm.runtimeInChunks === undefined) {
- cgm.runtimeInChunks = new Set();
- }
- cgm.runtimeInChunks.add(chunk);
- cgc.runtimeModules.add(module);
- }
- /**
- * @param {Chunk} chunk the new chunk
- * @param {RuntimeModule} module the module that require a full hash
- * @returns {void}
- */
- addFullHashModuleToChunk(chunk, module) {
- const cgc = this._getChunkGraphChunk(chunk);
- if (cgc.fullHashModules === undefined) cgc.fullHashModules = new Set();
- cgc.fullHashModules.add(module);
- }
- /**
- * @param {Chunk} chunk the new chunk
- * @param {RuntimeModule} module the module that require a full hash
- * @returns {void}
- */
- addDependentHashModuleToChunk(chunk, module) {
- const cgc = this._getChunkGraphChunk(chunk);
- if (cgc.dependentHashModules === undefined)
- cgc.dependentHashModules = new Set();
- cgc.dependentHashModules.add(module);
- }
- /**
- * @param {Chunk} chunk the new chunk
- * @param {Module} module the entry module
- * @returns {void}
- */
- disconnectChunkAndEntryModule(chunk, module) {
- const cgm = this._getChunkGraphModule(module);
- const cgc = this._getChunkGraphChunk(chunk);
- cgm.entryInChunks.delete(chunk);
- if (cgm.entryInChunks.size === 0) {
- cgm.entryInChunks = undefined;
- }
- cgc.entryModules.delete(module);
- }
- /**
- * @param {Chunk} chunk the new chunk
- * @param {RuntimeModule} module the runtime module
- * @returns {void}
- */
- disconnectChunkAndRuntimeModule(chunk, module) {
- const cgm = this._getChunkGraphModule(module);
- const cgc = this._getChunkGraphChunk(chunk);
- cgm.runtimeInChunks.delete(chunk);
- if (cgm.runtimeInChunks.size === 0) {
- cgm.runtimeInChunks = undefined;
- }
- cgc.runtimeModules.delete(module);
- }
- /**
- * @param {Module} module the entry module, it will no longer be entry
- * @returns {void}
- */
- disconnectEntryModule(module) {
- const cgm = this._getChunkGraphModule(module);
- for (const chunk of cgm.entryInChunks) {
- const cgc = this._getChunkGraphChunk(chunk);
- cgc.entryModules.delete(module);
- }
- cgm.entryInChunks = undefined;
- }
- /**
- * @param {Chunk} chunk the chunk, for which all entries will be removed
- * @returns {void}
- */
- disconnectEntries(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- for (const module of cgc.entryModules.keys()) {
- const cgm = this._getChunkGraphModule(module);
- cgm.entryInChunks.delete(chunk);
- if (cgm.entryInChunks.size === 0) {
- cgm.entryInChunks = undefined;
- }
- }
- cgc.entryModules.clear();
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {number} the amount of entry modules in chunk
- */
- getNumberOfEntryModules(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.entryModules.size;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {number} the amount of entry modules in chunk
- */
- getNumberOfRuntimeModules(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.runtimeModules.size;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {Iterable<Module>} iterable of modules (do not modify)
- */
- getChunkEntryModulesIterable(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.entryModules.keys();
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {Iterable<Chunk>} iterable of chunks
- */
- getChunkEntryDependentChunksIterable(chunk) {
- /** @type {Set<Chunk>} */
- const set = new Set();
- for (const chunkGroup of chunk.groupsIterable) {
- if (chunkGroup instanceof Entrypoint) {
- const entrypointChunk = chunkGroup.getEntrypointChunk();
- const cgc = this._getChunkGraphChunk(entrypointChunk);
- for (const chunkGroup of cgc.entryModules.values()) {
- for (const c of chunkGroup.chunks) {
- if (c !== chunk && c !== entrypointChunk && !c.hasRuntime()) {
- set.add(c);
- }
- }
- }
- }
- }
- return set;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {boolean} true, when it has dependent chunks
- */
- hasChunkEntryDependentChunks(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- for (const chunkGroup of cgc.entryModules.values()) {
- for (const c of chunkGroup.chunks) {
- if (c !== chunk) {
- return true;
- }
- }
- }
- return false;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {Iterable<RuntimeModule>} iterable of modules (do not modify)
- */
- getChunkRuntimeModulesIterable(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.runtimeModules;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {RuntimeModule[]} array of modules in order of execution
- */
- getChunkRuntimeModulesInOrder(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- const array = Array.from(cgc.runtimeModules);
- array.sort(
- concatComparators(
- compareSelect(
- /**
- * @param {RuntimeModule} r runtime module
- * @returns {number=} stage
- */
- r => r.stage,
- compareIds
- ),
- compareModulesByIdentifier
- )
- );
- return array;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {Iterable<RuntimeModule> | undefined} iterable of modules (do not modify)
- */
- getChunkFullHashModulesIterable(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.fullHashModules;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {ReadonlySet<RuntimeModule> | undefined} set of modules (do not modify)
- */
- getChunkFullHashModulesSet(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.fullHashModules;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {Iterable<RuntimeModule> | undefined} iterable of modules (do not modify)
- */
- getChunkDependentHashModulesIterable(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.dependentHashModules;
- }
- /** @typedef {[Module, Entrypoint | undefined]} EntryModuleWithChunkGroup */
- /**
- * @param {Chunk} chunk the chunk
- * @returns {Iterable<EntryModuleWithChunkGroup>} iterable of modules (do not modify)
- */
- getChunkEntryModulesWithChunkGroupIterable(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.entryModules;
- }
- /**
- * @param {AsyncDependenciesBlock} depBlock the async block
- * @returns {ChunkGroup} the chunk group
- */
- getBlockChunkGroup(depBlock) {
- return this._blockChunkGroups.get(depBlock);
- }
- /**
- * @param {AsyncDependenciesBlock} depBlock the async block
- * @param {ChunkGroup} chunkGroup the chunk group
- * @returns {void}
- */
- connectBlockAndChunkGroup(depBlock, chunkGroup) {
- this._blockChunkGroups.set(depBlock, chunkGroup);
- chunkGroup.addBlock(depBlock);
- }
- /**
- * @param {ChunkGroup} chunkGroup the chunk group
- * @returns {void}
- */
- disconnectChunkGroup(chunkGroup) {
- for (const block of chunkGroup.blocksIterable) {
- this._blockChunkGroups.delete(block);
- }
- // TODO refactor by moving blocks list into ChunkGraph
- chunkGroup._blocks.clear();
- }
- /**
- * @param {Module} module the module
- * @returns {string | number} the id of the module
- */
- getModuleId(module) {
- const cgm = this._getChunkGraphModule(module);
- return cgm.id;
- }
- /**
- * @param {Module} module the module
- * @param {string | number} id the id of the module
- * @returns {void}
- */
- setModuleId(module, id) {
- const cgm = this._getChunkGraphModule(module);
- cgm.id = id;
- }
- /**
- * @param {string} runtime runtime
- * @returns {string | number} the id of the runtime
- */
- getRuntimeId(runtime) {
- return this._runtimeIds.get(runtime);
- }
- /**
- * @param {string} runtime runtime
- * @param {string | number} id the id of the runtime
- * @returns {void}
- */
- setRuntimeId(runtime, id) {
- this._runtimeIds.set(runtime, id);
- }
- /**
- * @template T
- * @param {Module} module the module
- * @param {RuntimeSpecMap<T>} hashes hashes data
- * @param {RuntimeSpec} runtime the runtime
- * @returns {T} hash
- */
- _getModuleHashInfo(module, hashes, runtime) {
- if (!hashes) {
- throw new Error(
- `Module ${module.identifier()} has no hash info for runtime ${runtimeToString(
- runtime
- )} (hashes not set at all)`
- );
- } else if (runtime === undefined) {
- const hashInfoItems = new Set(hashes.values());
- if (hashInfoItems.size !== 1) {
- throw new Error(
- `No unique hash info entry for unspecified runtime for ${module.identifier()} (existing runtimes: ${Array.from(
- hashes.keys(),
- r => runtimeToString(r)
- ).join(", ")}).
- Caller might not support runtime-dependent code generation (opt-out via optimization.usedExports: "global").`
- );
- }
- return first(hashInfoItems);
- } else {
- const hashInfo = hashes.get(runtime);
- if (!hashInfo) {
- throw new Error(
- `Module ${module.identifier()} has no hash info for runtime ${runtimeToString(
- runtime
- )} (available runtimes ${Array.from(
- hashes.keys(),
- runtimeToString
- ).join(", ")})`
- );
- }
- return hashInfo;
- }
- }
- /**
- * @param {Module} module the module
- * @param {RuntimeSpec} runtime the runtime
- * @returns {boolean} true, if the module has hashes for this runtime
- */
- hasModuleHashes(module, runtime) {
- const cgm = this._getChunkGraphModule(module);
- const hashes = cgm.hashes;
- return hashes && hashes.has(runtime);
- }
- /**
- * @param {Module} module the module
- * @param {RuntimeSpec} runtime the runtime
- * @returns {string} hash
- */
- getModuleHash(module, runtime) {
- const cgm = this._getChunkGraphModule(module);
- const hashes = cgm.hashes;
- return this._getModuleHashInfo(module, hashes, runtime).hash;
- }
- /**
- * @param {Module} module the module
- * @param {RuntimeSpec} runtime the runtime
- * @returns {string} hash
- */
- getRenderedModuleHash(module, runtime) {
- const cgm = this._getChunkGraphModule(module);
- const hashes = cgm.hashes;
- return this._getModuleHashInfo(module, hashes, runtime).renderedHash;
- }
- /**
- * @param {Module} module the module
- * @param {RuntimeSpec} runtime the runtime
- * @param {string} hash the full hash
- * @param {string} renderedHash the shortened hash for rendering
- * @returns {void}
- */
- setModuleHashes(module, runtime, hash, renderedHash) {
- const cgm = this._getChunkGraphModule(module);
- if (cgm.hashes === undefined) {
- cgm.hashes = new RuntimeSpecMap();
- }
- cgm.hashes.set(runtime, new ModuleHashInfo(hash, renderedHash));
- }
- /**
- * @param {Module} module the module
- * @param {RuntimeSpec} runtime the runtime
- * @param {Set<string>} items runtime requirements to be added (ownership of this Set is given to ChunkGraph when transferOwnership not false)
- * @param {boolean} transferOwnership true: transfer ownership of the items object, false: items is immutable and shared and won't be modified
- * @returns {void}
- */
- addModuleRuntimeRequirements(
- module,
- runtime,
- items,
- transferOwnership = true
- ) {
- const cgm = this._getChunkGraphModule(module);
- const runtimeRequirementsMap = cgm.runtimeRequirements;
- if (runtimeRequirementsMap === undefined) {
- const map = new RuntimeSpecMap();
- // TODO avoid cloning item and track ownership instead
- map.set(runtime, transferOwnership ? items : new Set(items));
- cgm.runtimeRequirements = map;
- return;
- }
- runtimeRequirementsMap.update(runtime, runtimeRequirements => {
- if (runtimeRequirements === undefined) {
- return transferOwnership ? items : new Set(items);
- } else if (!transferOwnership || runtimeRequirements.size >= items.size) {
- for (const item of items) runtimeRequirements.add(item);
- return runtimeRequirements;
- } else {
- for (const item of runtimeRequirements) items.add(item);
- return items;
- }
- });
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {Set<string>} items runtime requirements to be added (ownership of this Set is given to ChunkGraph)
- * @returns {void}
- */
- addChunkRuntimeRequirements(chunk, items) {
- const cgc = this._getChunkGraphChunk(chunk);
- const runtimeRequirements = cgc.runtimeRequirements;
- if (runtimeRequirements === undefined) {
- cgc.runtimeRequirements = items;
- } else if (runtimeRequirements.size >= items.size) {
- for (const item of items) runtimeRequirements.add(item);
- } else {
- for (const item of runtimeRequirements) items.add(item);
- cgc.runtimeRequirements = items;
- }
- }
- /**
- * @param {Chunk} chunk the chunk
- * @param {Iterable<string>} items runtime requirements to be added
- * @returns {void}
- */
- addTreeRuntimeRequirements(chunk, items) {
- const cgc = this._getChunkGraphChunk(chunk);
- const runtimeRequirements = cgc.runtimeRequirementsInTree;
- for (const item of items) runtimeRequirements.add(item);
- }
- /**
- * @param {Module} module the module
- * @param {RuntimeSpec} runtime the runtime
- * @returns {ReadonlySet<string>} runtime requirements
- */
- getModuleRuntimeRequirements(module, runtime) {
- const cgm = this._getChunkGraphModule(module);
- const runtimeRequirements =
- cgm.runtimeRequirements && cgm.runtimeRequirements.get(runtime);
- return runtimeRequirements === undefined ? EMPTY_SET : runtimeRequirements;
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {ReadonlySet<string>} runtime requirements
- */
- getChunkRuntimeRequirements(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- const runtimeRequirements = cgc.runtimeRequirements;
- return runtimeRequirements === undefined ? EMPTY_SET : runtimeRequirements;
- }
- /**
- * @param {Module} module the module
- * @param {RuntimeSpec} runtime the runtime
- * @param {boolean} withConnections include connections
- * @returns {string} hash
- */
- getModuleGraphHash(module, runtime, withConnections = true) {
- const cgm = this._getChunkGraphModule(module);
- return withConnections
- ? this._getModuleGraphHashWithConnections(cgm, module, runtime)
- : this._getModuleGraphHashBigInt(cgm, module, runtime).toString(16);
- }
- /**
- * @param {Module} module the module
- * @param {RuntimeSpec} runtime the runtime
- * @param {boolean} withConnections include connections
- * @returns {bigint} hash
- */
- getModuleGraphHashBigInt(module, runtime, withConnections = true) {
- const cgm = this._getChunkGraphModule(module);
- return withConnections
- ? BigInt(
- `0x${this._getModuleGraphHashWithConnections(cgm, module, runtime)}`
- )
- : this._getModuleGraphHashBigInt(cgm, module, runtime);
- }
- /**
- * @param {ChunkGraphModule} cgm the ChunkGraphModule
- * @param {Module} module the module
- * @param {RuntimeSpec} runtime the runtime
- * @returns {bigint} hash as big int
- */
- _getModuleGraphHashBigInt(cgm, module, runtime) {
- if (cgm.graphHashes === undefined) {
- cgm.graphHashes = new RuntimeSpecMap();
- }
- const graphHash = cgm.graphHashes.provide(runtime, () => {
- const hash = createHash(this._hashFunction);
- hash.update(`${cgm.id}${this.moduleGraph.isAsync(module)}`);
- this.moduleGraph.getExportsInfo(module).updateHash(hash, runtime);
- return BigInt(`0x${/** @type {string} */ (hash.digest("hex"))}`);
- });
- return graphHash;
- }
- /**
- * @param {ChunkGraphModule} cgm the ChunkGraphModule
- * @param {Module} module the module
- * @param {RuntimeSpec} runtime the runtime
- * @returns {string} hash
- */
- _getModuleGraphHashWithConnections(cgm, module, runtime) {
- if (cgm.graphHashesWithConnections === undefined) {
- cgm.graphHashesWithConnections = new RuntimeSpecMap();
- }
- const activeStateToString = state => {
- if (state === false) return "F";
- if (state === true) return "T";
- if (state === ModuleGraphConnection.TRANSITIVE_ONLY) return "O";
- throw new Error("Not implemented active state");
- };
- const strict = module.buildMeta && module.buildMeta.strictHarmonyModule;
- return cgm.graphHashesWithConnections.provide(runtime, () => {
- const graphHash = this._getModuleGraphHashBigInt(
- cgm,
- module,
- runtime
- ).toString(16);
- const connections = this.moduleGraph.getOutgoingConnections(module);
- /** @type {Set<Module>} */
- const activeNamespaceModules = new Set();
- /** @type {Map<string, Module | Set<Module>>} */
- const connectedModules = new Map();
- const processConnection = (connection, stateInfo) => {
- const module = connection.module;
- stateInfo += module.getExportsType(this.moduleGraph, strict);
- // cspell:word Tnamespace
- if (stateInfo === "Tnamespace") activeNamespaceModules.add(module);
- else {
- const oldModule = connectedModules.get(stateInfo);
- if (oldModule === undefined) {
- connectedModules.set(stateInfo, module);
- } else if (oldModule instanceof Set) {
- oldModule.add(module);
- } else if (oldModule !== module) {
- connectedModules.set(stateInfo, new Set([oldModule, module]));
- }
- }
- };
- if (runtime === undefined || typeof runtime === "string") {
- for (const connection of connections) {
- const state = connection.getActiveState(runtime);
- if (state === false) continue;
- processConnection(connection, state === true ? "T" : "O");
- }
- } else {
- // cspell:word Tnamespace
- for (const connection of connections) {
- const states = new Set();
- let stateInfo = "";
- forEachRuntime(
- runtime,
- runtime => {
- const state = connection.getActiveState(runtime);
- states.add(state);
- stateInfo += activeStateToString(state) + runtime;
- },
- true
- );
- if (states.size === 1) {
- const state = first(states);
- if (state === false) continue;
- stateInfo = activeStateToString(state);
- }
- processConnection(connection, stateInfo);
- }
- }
- // cspell:word Tnamespace
- if (activeNamespaceModules.size === 0 && connectedModules.size === 0)
- return graphHash;
- const connectedModulesInOrder =
- connectedModules.size > 1
- ? Array.from(connectedModules).sort(([a], [b]) => (a < b ? -1 : 1))
- : connectedModules;
- const hash = createHash(this._hashFunction);
- const addModuleToHash = module => {
- hash.update(
- this._getModuleGraphHashBigInt(
- this._getChunkGraphModule(module),
- module,
- runtime
- ).toString(16)
- );
- };
- const addModulesToHash = modules => {
- let xor = ZERO_BIG_INT;
- for (const m of modules) {
- xor =
- xor ^
- this._getModuleGraphHashBigInt(
- this._getChunkGraphModule(m),
- m,
- runtime
- );
- }
- hash.update(xor.toString(16));
- };
- if (activeNamespaceModules.size === 1)
- addModuleToHash(activeNamespaceModules.values().next().value);
- else if (activeNamespaceModules.size > 1)
- addModulesToHash(activeNamespaceModules);
- for (const [stateInfo, modules] of connectedModulesInOrder) {
- hash.update(stateInfo);
- if (modules instanceof Set) {
- addModulesToHash(modules);
- } else {
- addModuleToHash(modules);
- }
- }
- hash.update(graphHash);
- return /** @type {string} */ (hash.digest("hex"));
- });
- }
- /**
- * @param {Chunk} chunk the chunk
- * @returns {ReadonlySet<string>} runtime requirements
- */
- getTreeRuntimeRequirements(chunk) {
- const cgc = this._getChunkGraphChunk(chunk);
- return cgc.runtimeRequirementsInTree;
- }
- // TODO remove in webpack 6
- /**
- * @param {Module} module the module
- * @param {string} deprecateMessage message for the deprecation message
- * @param {string} deprecationCode code for the deprecation
- * @returns {ChunkGraph} the chunk graph
- */
- static getChunkGraphForModule(module, deprecateMessage, deprecationCode) {
- const fn = deprecateGetChunkGraphForModuleMap.get(deprecateMessage);
- if (fn) return fn(module);
- const newFn = util.deprecate(
- /**
- * @param {Module} module the module
- * @returns {ChunkGraph} the chunk graph
- */
- module => {
- const chunkGraph = chunkGraphForModuleMap.get(module);
- if (!chunkGraph)
- throw new Error(
- deprecateMessage +
- ": There was no ChunkGraph assigned to the Module for backward-compat (Use the new API)"
- );
- return chunkGraph;
- },
- deprecateMessage + ": Use new ChunkGraph API",
- deprecationCode
- );
- deprecateGetChunkGraphForModuleMap.set(deprecateMessage, newFn);
- return newFn(module);
- }
- // TODO remove in webpack 6
- /**
- * @param {Module} module the module
- * @param {ChunkGraph} chunkGraph the chunk graph
- * @returns {void}
- */
- static setChunkGraphForModule(module, chunkGraph) {
- chunkGraphForModuleMap.set(module, chunkGraph);
- }
- // TODO remove in webpack 6
- /**
- * @param {Module} module the module
- * @returns {void}
- */
- static clearChunkGraphForModule(module) {
- chunkGraphForModuleMap.delete(module);
- }
- // TODO remove in webpack 6
- /**
- * @param {Chunk} chunk the chunk
- * @param {string} deprecateMessage message for the deprecation message
- * @param {string} deprecationCode code for the deprecation
- * @returns {ChunkGraph} the chunk graph
- */
- static getChunkGraphForChunk(chunk, deprecateMessage, deprecationCode) {
- const fn = deprecateGetChunkGraphForChunkMap.get(deprecateMessage);
- if (fn) return fn(chunk);
- const newFn = util.deprecate(
- /**
- * @param {Chunk} chunk the chunk
- * @returns {ChunkGraph} the chunk graph
- */
- chunk => {
- const chunkGraph = chunkGraphForChunkMap.get(chunk);
- if (!chunkGraph)
- throw new Error(
- deprecateMessage +
- "There was no ChunkGraph assigned to the Chunk for backward-compat (Use the new API)"
- );
- return chunkGraph;
- },
- deprecateMessage + ": Use new ChunkGraph API",
- deprecationCode
- );
- deprecateGetChunkGraphForChunkMap.set(deprecateMessage, newFn);
- return newFn(chunk);
- }
- // TODO remove in webpack 6
- /**
- * @param {Chunk} chunk the chunk
- * @param {ChunkGraph} chunkGraph the chunk graph
- * @returns {void}
- */
- static setChunkGraphForChunk(chunk, chunkGraph) {
- chunkGraphForChunkMap.set(chunk, chunkGraph);
- }
- // TODO remove in webpack 6
- /**
- * @param {Chunk} chunk the chunk
- * @returns {void}
- */
- static clearChunkGraphForChunk(chunk) {
- chunkGraphForChunkMap.delete(chunk);
- }
- }
- // TODO remove in webpack 6
- /** @type {WeakMap<Module, ChunkGraph>} */
- const chunkGraphForModuleMap = new WeakMap();
- // TODO remove in webpack 6
- /** @type {WeakMap<Chunk, ChunkGraph>} */
- const chunkGraphForChunkMap = new WeakMap();
- // TODO remove in webpack 6
- /** @type {Map<string, (module: Module) => ChunkGraph>} */
- const deprecateGetChunkGraphForModuleMap = new Map();
- // TODO remove in webpack 6
- /** @type {Map<string, (chunk: Chunk) => ChunkGraph>} */
- const deprecateGetChunkGraphForChunkMap = new Map();
- module.exports = ChunkGraph;
|