123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838 |
- /*
- MIT License http://www.opensource.org/licenses/mit-license.php
- Author Tobias Koppers @sokra
- */
- "use strict";
- const ChunkGraph = require("./ChunkGraph");
- const Entrypoint = require("./Entrypoint");
- const { intersect } = require("./util/SetHelpers");
- const SortableSet = require("./util/SortableSet");
- const StringXor = require("./util/StringXor");
- const {
- compareModulesByIdentifier,
- compareChunkGroupsByIndex,
- compareModulesById
- } = require("./util/comparators");
- const { createArrayToSetDeprecationSet } = require("./util/deprecation");
- const { mergeRuntime } = require("./util/runtime");
- /** @typedef {import("webpack-sources").Source} Source */
- /** @typedef {import("./ChunkGraph").ChunkFilterPredicate} ChunkFilterPredicate */
- /** @typedef {import("./ChunkGraph").ChunkSizeOptions} ChunkSizeOptions */
- /** @typedef {import("./ChunkGraph").ModuleFilterPredicate} ModuleFilterPredicate */
- /** @typedef {import("./ChunkGroup")} ChunkGroup */
- /** @typedef {import("./ChunkGroup").ChunkGroupOptions} ChunkGroupOptions */
- /** @typedef {import("./Compilation")} Compilation */
- /** @typedef {import("./Compilation").AssetInfo} AssetInfo */
- /** @typedef {import("./Compilation").PathData} PathData */
- /** @typedef {import("./Entrypoint").EntryOptions} EntryOptions */
- /** @typedef {import("./Module")} Module */
- /** @typedef {import("./ModuleGraph")} ModuleGraph */
- /** @typedef {import("./util/Hash")} Hash */
- /** @typedef {import("./util/runtime").RuntimeSpec} RuntimeSpec */
- /** @typedef {number | string} ChunkId */
- const ChunkFilesSet = createArrayToSetDeprecationSet("chunk.files");
- /**
- * @typedef {Object} WithId an object who has an id property *
- * @property {string | number} id the id of the object
- */
- /**
- * @deprecated
- * @typedef {Object} ChunkMaps
- * @property {Record<string|number, string>} hash
- * @property {Record<string|number, Record<string, string>>} contentHash
- * @property {Record<string|number, string>} name
- */
- /**
- * @deprecated
- * @typedef {Object} ChunkModuleMaps
- * @property {Record<string|number, (string|number)[]>} id
- * @property {Record<string|number, string>} hash
- */
- let debugId = 1000;
- /**
- * A Chunk is a unit of encapsulation for Modules.
- * Chunks are "rendered" into bundles that get emitted when the build completes.
- */
- class Chunk {
- /**
- * @param {string=} name of chunk being created, is optional (for subclasses)
- * @param {boolean} backCompat enable backward-compatibility
- */
- constructor(name, backCompat = true) {
- /** @type {ChunkId | null} */
- this.id = null;
- /** @type {ChunkId[] | null} */
- this.ids = null;
- /** @type {number} */
- this.debugId = debugId++;
- /** @type {string | undefined} */
- this.name = name;
- /** @type {SortableSet<string>} */
- this.idNameHints = new SortableSet();
- /** @type {boolean} */
- this.preventIntegration = false;
- /** @type {(string | function(PathData, AssetInfo=): string) | undefined} */
- this.filenameTemplate = undefined;
- /** @type {(string | function(PathData, AssetInfo=): string) | undefined} */
- this.cssFilenameTemplate = undefined;
- /** @private @type {SortableSet<ChunkGroup>} */
- this._groups = new SortableSet(undefined, compareChunkGroupsByIndex);
- /** @type {RuntimeSpec} */
- this.runtime = undefined;
- /** @type {Set<string>} */
- this.files = backCompat ? new ChunkFilesSet() : new Set();
- /** @type {Set<string>} */
- this.auxiliaryFiles = new Set();
- /** @type {boolean} */
- this.rendered = false;
- /** @type {string=} */
- this.hash = undefined;
- /** @type {Record<string, string>} */
- this.contentHash = Object.create(null);
- /** @type {string=} */
- this.renderedHash = undefined;
- /** @type {string=} */
- this.chunkReason = undefined;
- /** @type {boolean} */
- this.extraAsync = false;
- }
- // TODO remove in webpack 6
- // BACKWARD-COMPAT START
- get entryModule() {
- const entryModules = Array.from(
- ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.entryModule",
- "DEP_WEBPACK_CHUNK_ENTRY_MODULE"
- ).getChunkEntryModulesIterable(this)
- );
- if (entryModules.length === 0) {
- return undefined;
- } else if (entryModules.length === 1) {
- return entryModules[0];
- } else {
- throw new Error(
- "Module.entryModule: Multiple entry modules are not supported by the deprecated API (Use the new ChunkGroup API)"
- );
- }
- }
- /**
- * @returns {boolean} true, if the chunk contains an entry module
- */
- hasEntryModule() {
- return (
- ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.hasEntryModule",
- "DEP_WEBPACK_CHUNK_HAS_ENTRY_MODULE"
- ).getNumberOfEntryModules(this) > 0
- );
- }
- /**
- * @param {Module} module the module
- * @returns {boolean} true, if the chunk could be added
- */
- addModule(module) {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.addModule",
- "DEP_WEBPACK_CHUNK_ADD_MODULE"
- );
- if (chunkGraph.isModuleInChunk(module, this)) return false;
- chunkGraph.connectChunkAndModule(this, module);
- return true;
- }
- /**
- * @param {Module} module the module
- * @returns {void}
- */
- removeModule(module) {
- ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.removeModule",
- "DEP_WEBPACK_CHUNK_REMOVE_MODULE"
- ).disconnectChunkAndModule(this, module);
- }
- /**
- * @returns {number} the number of module which are contained in this chunk
- */
- getNumberOfModules() {
- return ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.getNumberOfModules",
- "DEP_WEBPACK_CHUNK_GET_NUMBER_OF_MODULES"
- ).getNumberOfChunkModules(this);
- }
- get modulesIterable() {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.modulesIterable",
- "DEP_WEBPACK_CHUNK_MODULES_ITERABLE"
- );
- return chunkGraph.getOrderedChunkModulesIterable(
- this,
- compareModulesByIdentifier
- );
- }
- /**
- * @param {Chunk} otherChunk the chunk to compare with
- * @returns {-1|0|1} the comparison result
- */
- compareTo(otherChunk) {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.compareTo",
- "DEP_WEBPACK_CHUNK_COMPARE_TO"
- );
- return chunkGraph.compareChunks(this, otherChunk);
- }
- /**
- * @param {Module} module the module
- * @returns {boolean} true, if the chunk contains the module
- */
- containsModule(module) {
- return ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.containsModule",
- "DEP_WEBPACK_CHUNK_CONTAINS_MODULE"
- ).isModuleInChunk(module, this);
- }
- /**
- * @returns {Module[]} the modules for this chunk
- */
- getModules() {
- return ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.getModules",
- "DEP_WEBPACK_CHUNK_GET_MODULES"
- ).getChunkModules(this);
- }
- /**
- * @returns {void}
- */
- remove() {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.remove",
- "DEP_WEBPACK_CHUNK_REMOVE"
- );
- chunkGraph.disconnectChunk(this);
- this.disconnectFromGroups();
- }
- /**
- * @param {Module} module the module
- * @param {Chunk} otherChunk the target chunk
- * @returns {void}
- */
- moveModule(module, otherChunk) {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.moveModule",
- "DEP_WEBPACK_CHUNK_MOVE_MODULE"
- );
- chunkGraph.disconnectChunkAndModule(this, module);
- chunkGraph.connectChunkAndModule(otherChunk, module);
- }
- /**
- * @param {Chunk} otherChunk the other chunk
- * @returns {boolean} true, if the specified chunk has been integrated
- */
- integrate(otherChunk) {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.integrate",
- "DEP_WEBPACK_CHUNK_INTEGRATE"
- );
- if (chunkGraph.canChunksBeIntegrated(this, otherChunk)) {
- chunkGraph.integrateChunks(this, otherChunk);
- return true;
- } else {
- return false;
- }
- }
- /**
- * @param {Chunk} otherChunk the other chunk
- * @returns {boolean} true, if chunks could be integrated
- */
- canBeIntegrated(otherChunk) {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.canBeIntegrated",
- "DEP_WEBPACK_CHUNK_CAN_BE_INTEGRATED"
- );
- return chunkGraph.canChunksBeIntegrated(this, otherChunk);
- }
- /**
- * @returns {boolean} true, if this chunk contains no module
- */
- isEmpty() {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.isEmpty",
- "DEP_WEBPACK_CHUNK_IS_EMPTY"
- );
- return chunkGraph.getNumberOfChunkModules(this) === 0;
- }
- /**
- * @returns {number} total size of all modules in this chunk
- */
- modulesSize() {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.modulesSize",
- "DEP_WEBPACK_CHUNK_MODULES_SIZE"
- );
- return chunkGraph.getChunkModulesSize(this);
- }
- /**
- * @param {ChunkSizeOptions} options options object
- * @returns {number} total size of this chunk
- */
- size(options = {}) {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.size",
- "DEP_WEBPACK_CHUNK_SIZE"
- );
- return chunkGraph.getChunkSize(this, options);
- }
- /**
- * @param {Chunk} otherChunk the other chunk
- * @param {ChunkSizeOptions} options options object
- * @returns {number} total size of the chunk or false if the chunk can't be integrated
- */
- integratedSize(otherChunk, options) {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.integratedSize",
- "DEP_WEBPACK_CHUNK_INTEGRATED_SIZE"
- );
- return chunkGraph.getIntegratedChunksSize(this, otherChunk, options);
- }
- /**
- * @param {ModuleFilterPredicate} filterFn function used to filter modules
- * @returns {ChunkModuleMaps} module map information
- */
- getChunkModuleMaps(filterFn) {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.getChunkModuleMaps",
- "DEP_WEBPACK_CHUNK_GET_CHUNK_MODULE_MAPS"
- );
- /** @type {Record<string|number, (string|number)[]>} */
- const chunkModuleIdMap = Object.create(null);
- /** @type {Record<string|number, string>} */
- const chunkModuleHashMap = Object.create(null);
- for (const asyncChunk of this.getAllAsyncChunks()) {
- /** @type {ChunkId[] | undefined} */
- let array;
- for (const module of chunkGraph.getOrderedChunkModulesIterable(
- asyncChunk,
- compareModulesById(chunkGraph)
- )) {
- if (filterFn(module)) {
- if (array === undefined) {
- array = [];
- chunkModuleIdMap[/** @type {ChunkId} */ (asyncChunk.id)] = array;
- }
- const moduleId = chunkGraph.getModuleId(module);
- array.push(moduleId);
- chunkModuleHashMap[moduleId] = chunkGraph.getRenderedModuleHash(
- module,
- undefined
- );
- }
- }
- }
- return {
- id: chunkModuleIdMap,
- hash: chunkModuleHashMap
- };
- }
- /**
- * @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(filterFn, filterChunkFn) {
- const chunkGraph = ChunkGraph.getChunkGraphForChunk(
- this,
- "Chunk.hasModuleInGraph",
- "DEP_WEBPACK_CHUNK_HAS_MODULE_IN_GRAPH"
- );
- return chunkGraph.hasModuleInGraph(this, filterFn, filterChunkFn);
- }
- /**
- * @deprecated
- * @param {boolean} realHash whether the full hash or the rendered hash is to be used
- * @returns {ChunkMaps} the chunk map information
- */
- getChunkMaps(realHash) {
- /** @type {Record<string|number, string>} */
- const chunkHashMap = Object.create(null);
- /** @type {Record<string|number, Record<string, string>>} */
- const chunkContentHashMap = Object.create(null);
- /** @type {Record<string|number, string>} */
- const chunkNameMap = Object.create(null);
- for (const chunk of this.getAllAsyncChunks()) {
- const id = /** @type {ChunkId} */ (chunk.id);
- chunkHashMap[id] =
- /** @type {string} */
- (realHash ? chunk.hash : chunk.renderedHash);
- for (const key of Object.keys(chunk.contentHash)) {
- if (!chunkContentHashMap[key]) {
- chunkContentHashMap[key] = Object.create(null);
- }
- chunkContentHashMap[key][id] = chunk.contentHash[key];
- }
- if (chunk.name) {
- chunkNameMap[id] = chunk.name;
- }
- }
- return {
- hash: chunkHashMap,
- contentHash: chunkContentHashMap,
- name: chunkNameMap
- };
- }
- // BACKWARD-COMPAT END
- /**
- * @returns {boolean} whether or not the Chunk will have a runtime
- */
- hasRuntime() {
- for (const chunkGroup of this._groups) {
- if (
- chunkGroup instanceof Entrypoint &&
- chunkGroup.getRuntimeChunk() === this
- ) {
- return true;
- }
- }
- return false;
- }
- /**
- * @returns {boolean} whether or not this chunk can be an initial chunk
- */
- canBeInitial() {
- for (const chunkGroup of this._groups) {
- if (chunkGroup.isInitial()) return true;
- }
- return false;
- }
- /**
- * @returns {boolean} whether this chunk can only be an initial chunk
- */
- isOnlyInitial() {
- if (this._groups.size <= 0) return false;
- for (const chunkGroup of this._groups) {
- if (!chunkGroup.isInitial()) return false;
- }
- return true;
- }
- /**
- * @returns {EntryOptions | undefined} the entry options for this chunk
- */
- getEntryOptions() {
- for (const chunkGroup of this._groups) {
- if (chunkGroup instanceof Entrypoint) {
- return chunkGroup.options;
- }
- }
- return undefined;
- }
- /**
- * @param {ChunkGroup} chunkGroup the chunkGroup the chunk is being added
- * @returns {void}
- */
- addGroup(chunkGroup) {
- this._groups.add(chunkGroup);
- }
- /**
- * @param {ChunkGroup} chunkGroup the chunkGroup the chunk is being removed from
- * @returns {void}
- */
- removeGroup(chunkGroup) {
- this._groups.delete(chunkGroup);
- }
- /**
- * @param {ChunkGroup} chunkGroup the chunkGroup to check
- * @returns {boolean} returns true if chunk has chunkGroup reference and exists in chunkGroup
- */
- isInGroup(chunkGroup) {
- return this._groups.has(chunkGroup);
- }
- /**
- * @returns {number} the amount of groups that the said chunk is in
- */
- getNumberOfGroups() {
- return this._groups.size;
- }
- /**
- * @returns {SortableSet<ChunkGroup>} the chunkGroups that the said chunk is referenced in
- */
- get groupsIterable() {
- this._groups.sort();
- return this._groups;
- }
- /**
- * @returns {void}
- */
- disconnectFromGroups() {
- for (const chunkGroup of this._groups) {
- chunkGroup.removeChunk(this);
- }
- }
- /**
- * @param {Chunk} newChunk the new chunk that will be split out of
- * @returns {void}
- */
- split(newChunk) {
- for (const chunkGroup of this._groups) {
- chunkGroup.insertChunk(newChunk, this);
- newChunk.addGroup(chunkGroup);
- }
- for (const idHint of this.idNameHints) {
- newChunk.idNameHints.add(idHint);
- }
- newChunk.runtime = mergeRuntime(newChunk.runtime, this.runtime);
- }
- /**
- * @param {Hash} hash hash (will be modified)
- * @param {ChunkGraph} chunkGraph the chunk graph
- * @returns {void}
- */
- updateHash(hash, chunkGraph) {
- hash.update(
- `${this.id} ${this.ids ? this.ids.join() : ""} ${this.name || ""} `
- );
- const xor = new StringXor();
- for (const m of chunkGraph.getChunkModulesIterable(this)) {
- xor.add(chunkGraph.getModuleHash(m, this.runtime));
- }
- xor.updateHash(hash);
- const entryModules =
- chunkGraph.getChunkEntryModulesWithChunkGroupIterable(this);
- for (const [m, chunkGroup] of entryModules) {
- hash.update(
- `entry${chunkGraph.getModuleId(m)}${
- /** @type {ChunkGroup} */ (chunkGroup).id
- }`
- );
- }
- }
- /**
- * @returns {Set<Chunk>} a set of all the async chunks
- */
- getAllAsyncChunks() {
- const queue = new Set();
- const chunks = new Set();
- const initialChunks = intersect(
- Array.from(this.groupsIterable, g => new Set(g.chunks))
- );
- const initialQueue = new Set(this.groupsIterable);
- for (const chunkGroup of initialQueue) {
- for (const child of chunkGroup.childrenIterable) {
- if (child instanceof Entrypoint) {
- initialQueue.add(child);
- } else {
- queue.add(child);
- }
- }
- }
- for (const chunkGroup of queue) {
- for (const chunk of chunkGroup.chunks) {
- if (!initialChunks.has(chunk)) {
- chunks.add(chunk);
- }
- }
- for (const child of chunkGroup.childrenIterable) {
- queue.add(child);
- }
- }
- return chunks;
- }
- /**
- * @returns {Set<Chunk>} a set of all the initial chunks (including itself)
- */
- getAllInitialChunks() {
- const chunks = new Set();
- const queue = new Set(this.groupsIterable);
- for (const group of queue) {
- if (group.isInitial()) {
- for (const c of group.chunks) chunks.add(c);
- for (const g of group.childrenIterable) queue.add(g);
- }
- }
- return chunks;
- }
- /**
- * @returns {Set<Chunk>} a set of all the referenced chunks (including itself)
- */
- getAllReferencedChunks() {
- const queue = new Set(this.groupsIterable);
- const chunks = new Set();
- for (const chunkGroup of queue) {
- for (const chunk of chunkGroup.chunks) {
- chunks.add(chunk);
- }
- for (const child of chunkGroup.childrenIterable) {
- queue.add(child);
- }
- }
- return chunks;
- }
- /**
- * @returns {Set<Entrypoint>} a set of all the referenced entrypoints
- */
- getAllReferencedAsyncEntrypoints() {
- const queue = new Set(this.groupsIterable);
- const entrypoints = new Set();
- for (const chunkGroup of queue) {
- for (const entrypoint of chunkGroup.asyncEntrypointsIterable) {
- entrypoints.add(entrypoint);
- }
- for (const child of chunkGroup.childrenIterable) {
- queue.add(child);
- }
- }
- return entrypoints;
- }
- /**
- * @returns {boolean} true, if the chunk references async chunks
- */
- hasAsyncChunks() {
- const queue = new Set();
- const initialChunks = intersect(
- Array.from(this.groupsIterable, g => new Set(g.chunks))
- );
- for (const chunkGroup of this.groupsIterable) {
- for (const child of chunkGroup.childrenIterable) {
- queue.add(child);
- }
- }
- for (const chunkGroup of queue) {
- for (const chunk of chunkGroup.chunks) {
- if (!initialChunks.has(chunk)) {
- return true;
- }
- }
- for (const child of chunkGroup.childrenIterable) {
- queue.add(child);
- }
- }
- return false;
- }
- /**
- * @param {ChunkGraph} chunkGraph the chunk graph
- * @param {ChunkFilterPredicate=} filterFn function used to filter chunks
- * @returns {Record<string, (string | number)[]>} a record object of names to lists of child ids(?)
- */
- getChildIdsByOrders(chunkGraph, filterFn) {
- /** @type {Map<string, {order: number, group: ChunkGroup}[]>} */
- const lists = new Map();
- for (const group of this.groupsIterable) {
- if (group.chunks[group.chunks.length - 1] === this) {
- for (const childGroup of group.childrenIterable) {
- for (const key of Object.keys(childGroup.options)) {
- if (key.endsWith("Order")) {
- const name = key.slice(0, key.length - "Order".length);
- let list = lists.get(name);
- if (list === undefined) {
- list = [];
- lists.set(name, list);
- }
- list.push({
- order:
- /** @type {number} */
- (
- childGroup.options[
- /** @type {keyof ChunkGroupOptions} */ (key)
- ]
- ),
- group: childGroup
- });
- }
- }
- }
- }
- }
- /** @type {Record<string, (string | number)[]>} */
- const result = Object.create(null);
- for (const [name, list] of lists) {
- list.sort((a, b) => {
- const cmp = b.order - a.order;
- if (cmp !== 0) return cmp;
- return a.group.compareTo(chunkGraph, b.group);
- });
- /** @type {Set<string | number>} */
- const chunkIdSet = new Set();
- for (const item of list) {
- for (const chunk of item.group.chunks) {
- if (filterFn && !filterFn(chunk, chunkGraph)) continue;
- chunkIdSet.add(/** @type {ChunkId} */ (chunk.id));
- }
- }
- if (chunkIdSet.size > 0) {
- result[name] = Array.from(chunkIdSet);
- }
- }
- return result;
- }
- /**
- * @param {ChunkGraph} chunkGraph the chunk graph
- * @param {string} type option name
- * @returns {{ onChunks: Chunk[], chunks: Set<Chunk> }[] | undefined} referenced chunks for a specific type
- */
- getChildrenOfTypeInOrder(chunkGraph, type) {
- const list = [];
- for (const group of this.groupsIterable) {
- for (const childGroup of group.childrenIterable) {
- const order =
- childGroup.options[/** @type {keyof ChunkGroupOptions} */ (type)];
- if (order === undefined) continue;
- list.push({
- order,
- group,
- childGroup
- });
- }
- }
- if (list.length === 0) return undefined;
- list.sort((a, b) => {
- const cmp =
- /** @type {number} */ (b.order) - /** @type {number} */ (a.order);
- if (cmp !== 0) return cmp;
- return a.group.compareTo(chunkGraph, b.group);
- });
- const result = [];
- let lastEntry;
- for (const { group, childGroup } of list) {
- if (lastEntry && lastEntry.onChunks === group.chunks) {
- for (const chunk of childGroup.chunks) {
- lastEntry.chunks.add(chunk);
- }
- } else {
- result.push(
- (lastEntry = {
- onChunks: group.chunks,
- chunks: new Set(childGroup.chunks)
- })
- );
- }
- }
- return result;
- }
- /**
- * @param {ChunkGraph} chunkGraph the chunk graph
- * @param {boolean=} includeDirectChildren include direct children (by default only children of async children are included)
- * @param {ChunkFilterPredicate=} filterFn function used to filter chunks
- * @returns {Record<string|number, Record<string, (string | number)[]>>} a record object of names to lists of child ids(?) by chunk id
- */
- getChildIdsByOrdersMap(chunkGraph, includeDirectChildren, filterFn) {
- /** @type {Record<string|number, Record<string, (string | number)[]>>} */
- const chunkMaps = Object.create(null);
- /**
- * @param {Chunk} chunk a chunk
- * @returns {void}
- */
- const addChildIdsByOrdersToMap = chunk => {
- const data = chunk.getChildIdsByOrders(chunkGraph, filterFn);
- for (const key of Object.keys(data)) {
- let chunkMap = chunkMaps[key];
- if (chunkMap === undefined) {
- chunkMaps[key] = chunkMap = Object.create(null);
- }
- chunkMap[/** @type {ChunkId} */ (chunk.id)] = data[key];
- }
- };
- if (includeDirectChildren) {
- /** @type {Set<Chunk>} */
- const chunks = new Set();
- for (const chunkGroup of this.groupsIterable) {
- for (const chunk of chunkGroup.chunks) {
- chunks.add(chunk);
- }
- }
- for (const chunk of chunks) {
- addChildIdsByOrdersToMap(chunk);
- }
- }
- for (const chunk of this.getAllAsyncChunks()) {
- addChildIdsByOrdersToMap(chunk);
- }
- return chunkMaps;
- }
- }
- module.exports = Chunk;
|