mirror of https://github.com/webpack/webpack.git
chore(Chunk): add type annotations for Chunk
This commit is contained in:
parent
fc3774a8c2
commit
f1618aed04
192
lib/Chunk.js
192
lib/Chunk.js
|
@ -1,6 +1,6 @@
|
||||||
/*
|
/*
|
||||||
MIT License http://www.opensource.org/licenses/mit-license.php
|
MIT License http://www.opensource.org/licenses/mit-license.php
|
||||||
Author Tobias Koppers @sokra
|
Author Tobias Koppers @sokra
|
||||||
*/
|
*/
|
||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
|
@ -13,18 +13,52 @@ const ERR_CHUNK_ENTRY = "Chunk.entry was removed. Use hasRuntime()";
|
||||||
const ERR_CHUNK_INITIAL =
|
const ERR_CHUNK_INITIAL =
|
||||||
"Chunk.initial was removed. Use canBeInitial/isOnlyInitial()";
|
"Chunk.initial was removed. Use canBeInitial/isOnlyInitial()";
|
||||||
|
|
||||||
|
/** @typedef {import("./Module.js")} Module */
|
||||||
|
/** @typedef {import("./ChunkGroup")} ChunkGroup */
|
||||||
|
/** @typedef {import("./ModuleReason.js")} ModuleReason */
|
||||||
|
/** @typedef {import("webpack-sources").Source} Source */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {Object} Identifiable an object who contains an identifier function property
|
||||||
|
* @property {() => string} identifier the resource or unique identifier of something
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {Object} WithId an object who has an id property
|
||||||
|
* @property {string} id the id of the object
|
||||||
|
*/
|
||||||
|
|
||||||
|
/** @typedef {(a: Module, b: Module) => -1|0|1} ModuleSortPredicate */
|
||||||
|
/** @typedef {(m: Module) => boolean} ModuleFilterPredicate */
|
||||||
|
/** @typedef {(c: Chunk) => boolean} ChunkFilterPredicate */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {WithId} a object that contains an ID property
|
||||||
|
* @param {WithId} b object that contains an ID property
|
||||||
|
* @returns {-1|0|1} sort value
|
||||||
|
*/
|
||||||
const sortById = (a, b) => {
|
const sortById = (a, b) => {
|
||||||
if (a.id < b.id) return -1;
|
if (a.id < b.id) return -1;
|
||||||
if (b.id < a.id) return 1;
|
if (b.id < a.id) return 1;
|
||||||
return 0;
|
return 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {Identifiable} a first object with ident fn
|
||||||
|
* @param {Identifiable} b second object with ident fn
|
||||||
|
* @returns {-1|0|1} The order number of the sort
|
||||||
|
*/
|
||||||
const sortByIdentifier = (a, b) => {
|
const sortByIdentifier = (a, b) => {
|
||||||
if (a.identifier() > b.identifier()) return 1;
|
if (a.identifier() > b.identifier()) return 1;
|
||||||
if (a.identifier() < b.identifier()) return -1;
|
if (a.identifier() < b.identifier()) return -1;
|
||||||
return 0;
|
return 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {string} a concatenation of module identifiers sorted
|
||||||
|
* @param {SortableSet} set to pull module identifiers from
|
||||||
|
*/
|
||||||
const getModulesIdent = set => {
|
const getModulesIdent = set => {
|
||||||
set.sort();
|
set.sort();
|
||||||
let str = "";
|
let str = "";
|
||||||
|
@ -34,51 +68,105 @@ const getModulesIdent = set => {
|
||||||
return str;
|
return str;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @template {T}
|
||||||
|
* @param {Set<T>} set the set to convert to array
|
||||||
|
* @returns {Array<T>} the array returned from Array.from(set)
|
||||||
|
*/
|
||||||
const getArray = set => Array.from(set);
|
const getArray = set => Array.from(set);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {Set<Module>} set the Set to get the count/size of
|
||||||
|
* @returns {number} the size of the modules
|
||||||
|
*/
|
||||||
const getModulesSize = set => {
|
const getModulesSize = set => {
|
||||||
let count = 0;
|
let size = 0;
|
||||||
for (const module of set) {
|
for (const module of set) {
|
||||||
count += module.size();
|
size += module.size();
|
||||||
}
|
}
|
||||||
return count;
|
return size;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A Chunk is a unit of encapsulation for Modules.
|
||||||
|
* Chunks are "rendered" into bundles that get emitted when the build completes.
|
||||||
|
*/
|
||||||
class Chunk {
|
class Chunk {
|
||||||
|
/**
|
||||||
|
* @param {string=} name of chunk being created, is optional (for subclasses)
|
||||||
|
*/
|
||||||
constructor(name) {
|
constructor(name) {
|
||||||
|
/** @type {number | null} */
|
||||||
this.id = null;
|
this.id = null;
|
||||||
|
/** @type {number[] | null} */
|
||||||
this.ids = null;
|
this.ids = null;
|
||||||
|
/** @type {number} */
|
||||||
this.debugId = debugId++;
|
this.debugId = debugId++;
|
||||||
|
/** @type {string} */
|
||||||
this.name = name;
|
this.name = name;
|
||||||
|
/** @type {boolean} */
|
||||||
this.preventIntegration = false;
|
this.preventIntegration = false;
|
||||||
|
/** @type {Module=} */
|
||||||
this.entryModule = undefined;
|
this.entryModule = undefined;
|
||||||
|
//TODO make these typed generics for Module[] and ChunkGroup[] and their sort being (T, T): => 1,-1,0
|
||||||
|
//See https://github.com/webpack/webpack/pull/7046
|
||||||
|
/** @private */
|
||||||
this._modules = new SortableSet(undefined, sortByIdentifier);
|
this._modules = new SortableSet(undefined, sortByIdentifier);
|
||||||
|
/** @private */
|
||||||
this._groups = new SortableSet(undefined, sortById);
|
this._groups = new SortableSet(undefined, sortById);
|
||||||
|
/** @type {Source[]} */
|
||||||
this.files = [];
|
this.files = [];
|
||||||
|
/** @type {boolean} */
|
||||||
this.rendered = false;
|
this.rendered = false;
|
||||||
|
/** @type {string=} */
|
||||||
this.hash = undefined;
|
this.hash = undefined;
|
||||||
|
/** @type {Object} */
|
||||||
this.contentHash = Object.create(null);
|
this.contentHash = Object.create(null);
|
||||||
|
/** @type {string=} */
|
||||||
this.renderedHash = undefined;
|
this.renderedHash = undefined;
|
||||||
|
/** @type {string=} */
|
||||||
this.chunkReason = undefined;
|
this.chunkReason = undefined;
|
||||||
|
/** @type {boolean} */
|
||||||
this.extraAsync = false;
|
this.extraAsync = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Chunk.entry has been deprecated. Please use .hasRuntime() instead
|
||||||
|
* @returns {never} Throws an error trying to access this property
|
||||||
|
*/
|
||||||
get entry() {
|
get entry() {
|
||||||
throw new Error(ERR_CHUNK_ENTRY);
|
throw new Error(ERR_CHUNK_ENTRY);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated .entry has been deprecated. Please use .hasRuntime() instead
|
||||||
|
* @param {never} data The data that was attempting to be set
|
||||||
|
* @returns {never} Throws an error trying to access this property
|
||||||
|
*/
|
||||||
set entry(data) {
|
set entry(data) {
|
||||||
throw new Error(ERR_CHUNK_ENTRY);
|
throw new Error(ERR_CHUNK_ENTRY);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Chunk.initial was removed. Use canBeInitial/isOnlyInitial()
|
||||||
|
* @returns {never} Throws an error trying to access this property
|
||||||
|
*/
|
||||||
get initial() {
|
get initial() {
|
||||||
throw new Error(ERR_CHUNK_INITIAL);
|
throw new Error(ERR_CHUNK_INITIAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Chunk.initial was removed. Use canBeInitial/isOnlyInitial()
|
||||||
|
* @param {never} data The data attempting to be set
|
||||||
|
* @returns {never} Throws an error trying to access this property
|
||||||
|
*/
|
||||||
set initial(data) {
|
set initial(data) {
|
||||||
throw new Error(ERR_CHUNK_INITIAL);
|
throw new Error(ERR_CHUNK_INITIAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {boolean} whether or not the Chunk will have a runtime
|
||||||
|
*/
|
||||||
hasRuntime() {
|
hasRuntime() {
|
||||||
for (const chunkGroup of this._groups) {
|
for (const chunkGroup of this._groups) {
|
||||||
// We only need to check the first one
|
// We only need to check the first one
|
||||||
|
@ -87,6 +175,9 @@ class Chunk {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {boolean} whether or not this chunk can be an initial chunk
|
||||||
|
*/
|
||||||
canBeInitial() {
|
canBeInitial() {
|
||||||
for (const chunkGroup of this._groups) {
|
for (const chunkGroup of this._groups) {
|
||||||
if (chunkGroup.isInitial()) return true;
|
if (chunkGroup.isInitial()) return true;
|
||||||
|
@ -94,6 +185,9 @@ class Chunk {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {boolean} whether this chunk can only be an initial chunk
|
||||||
|
*/
|
||||||
isOnlyInitial() {
|
isOnlyInitial() {
|
||||||
if (this._groups.size <= 0) return false;
|
if (this._groups.size <= 0) return false;
|
||||||
for (const chunkGroup of this._groups) {
|
for (const chunkGroup of this._groups) {
|
||||||
|
@ -102,10 +196,17 @@ class Chunk {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {boolean} if this chunk contains the entry module
|
||||||
|
*/
|
||||||
hasEntryModule() {
|
hasEntryModule() {
|
||||||
return !!this.entryModule;
|
return !!this.entryModule;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {Module} module the module that will be added to this chunk.
|
||||||
|
* @returns {boolean} returns true if the chunk doesn't have the module and it was added
|
||||||
|
*/
|
||||||
addModule(module) {
|
addModule(module) {
|
||||||
if (!this._modules.has(module)) {
|
if (!this._modules.has(module)) {
|
||||||
this._modules.add(module);
|
this._modules.add(module);
|
||||||
|
@ -114,6 +215,10 @@ class Chunk {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {Module} module the module that will be removed from this chunk
|
||||||
|
* @returns {boolean} returns true if chunk exists and is successfully deleted
|
||||||
|
*/
|
||||||
removeModule(module) {
|
removeModule(module) {
|
||||||
if (this._modules.delete(module)) {
|
if (this._modules.delete(module)) {
|
||||||
module.removeChunk(this);
|
module.removeChunk(this);
|
||||||
|
@ -122,42 +227,74 @@ class Chunk {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {Module[]} modules the new modules to be set
|
||||||
|
* @returns {void} set new modules to this chunk and return nothing
|
||||||
|
*/
|
||||||
setModules(modules) {
|
setModules(modules) {
|
||||||
this._modules = new SortableSet(modules, sortByIdentifier);
|
this._modules = new SortableSet(modules, sortByIdentifier);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {number} the amount of modules in chunk
|
||||||
|
*/
|
||||||
getNumberOfModules() {
|
getNumberOfModules() {
|
||||||
return this._modules.size;
|
return this._modules.size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {SortableSet} return the modules SortableSet for this chunk
|
||||||
|
*/
|
||||||
get modulesIterable() {
|
get modulesIterable() {
|
||||||
return this._modules;
|
return this._modules;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {ChunkGroup} chunkGroup the chunkGroup the chunk is being added
|
||||||
|
* @returns {boolean} returns true if chunk is not apart of chunkGroup and is added successfully
|
||||||
|
*/
|
||||||
addGroup(chunkGroup) {
|
addGroup(chunkGroup) {
|
||||||
if (this._groups.has(chunkGroup)) return false;
|
if (this._groups.has(chunkGroup)) return false;
|
||||||
this._groups.add(chunkGroup);
|
this._groups.add(chunkGroup);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {ChunkGroup} chunkGroup the chunkGroup the chunk is being removed from
|
||||||
|
* @returns {boolean} returns true if chunk does exist in chunkGroup and is removed
|
||||||
|
*/
|
||||||
removeGroup(chunkGroup) {
|
removeGroup(chunkGroup) {
|
||||||
if (!this._groups.has(chunkGroup)) return false;
|
if (!this._groups.has(chunkGroup)) return false;
|
||||||
this._groups.delete(chunkGroup);
|
this._groups.delete(chunkGroup);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {ChunkGroup} chunkGroup the chunkGroup to check
|
||||||
|
* @returns {boolean} returns true if chunk has chunkGroup reference and exists in chunkGroup
|
||||||
|
*/
|
||||||
isInGroup(chunkGroup) {
|
isInGroup(chunkGroup) {
|
||||||
return this._groups.has(chunkGroup);
|
return this._groups.has(chunkGroup);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {number} the amount of groups said chunk is in
|
||||||
|
*/
|
||||||
getNumberOfGroups() {
|
getNumberOfGroups() {
|
||||||
return this._groups.size;
|
return this._groups.size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {SortableSet} the chunkGroups that said chunk is referenced in
|
||||||
|
*/
|
||||||
get groupsIterable() {
|
get groupsIterable() {
|
||||||
return this._groups;
|
return this._groups;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {Chunk} otherChunk the chunk to compare itself with
|
||||||
|
* @returns {-1|0|1} this is a comparitor function like sort and returns -1, 0, or 1 based on sort order
|
||||||
|
*/
|
||||||
compareTo(otherChunk) {
|
compareTo(otherChunk) {
|
||||||
this._modules.sort();
|
this._modules.sort();
|
||||||
otherChunk._modules.sort();
|
otherChunk._modules.sort();
|
||||||
|
@ -177,6 +314,10 @@ class Chunk {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {Module} module Module to check
|
||||||
|
* @returns {boolean} returns true if module does exist in this chunk
|
||||||
|
*/
|
||||||
containsModule(module) {
|
containsModule(module) {
|
||||||
return this._modules.has(module);
|
return this._modules.has(module);
|
||||||
}
|
}
|
||||||
|
@ -189,7 +330,7 @@ class Chunk {
|
||||||
return this._modules.getFromUnorderedCache(getModulesIdent);
|
return this._modules.getFromUnorderedCache(getModulesIdent);
|
||||||
}
|
}
|
||||||
|
|
||||||
remove(reason) {
|
remove() {
|
||||||
// cleanup modules
|
// cleanup modules
|
||||||
// Array.from is used here to create a clone, because removeChunk modifies this._modules
|
// Array.from is used here to create a clone, because removeChunk modifies this._modules
|
||||||
for (const module of Array.from(this._modules)) {
|
for (const module of Array.from(this._modules)) {
|
||||||
|
@ -200,12 +341,24 @@ class Chunk {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {Module} module module to move
|
||||||
|
* @param {Chunk} otherChunk other chunk to move it to
|
||||||
|
* @returns {void}
|
||||||
|
*/
|
||||||
moveModule(module, otherChunk) {
|
moveModule(module, otherChunk) {
|
||||||
GraphHelpers.disconnectChunkAndModule(this, module);
|
GraphHelpers.disconnectChunkAndModule(this, module);
|
||||||
GraphHelpers.connectChunkAndModule(otherChunk, module);
|
GraphHelpers.connectChunkAndModule(otherChunk, module);
|
||||||
module.rewriteChunkInReasons(this, [otherChunk]);
|
module.rewriteChunkInReasons(this, [otherChunk]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {Chunk} otherChunk the chunk to integrate with
|
||||||
|
* @param {ModuleReason} reason reason why the module is being integrated
|
||||||
|
* @returns {boolean} returns true or false if integration succeeds or fails
|
||||||
|
*/
|
||||||
integrate(otherChunk, reason) {
|
integrate(otherChunk, reason) {
|
||||||
if (!this.canBeIntegrated(otherChunk)) {
|
if (!this.canBeIntegrated(otherChunk)) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -236,6 +389,10 @@ class Chunk {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {Chunk} newChunk the new chunk that will be split out of, and then chunk raphi twil=
|
||||||
|
* @returns {void}
|
||||||
|
*/
|
||||||
split(newChunk) {
|
split(newChunk) {
|
||||||
for (const chunkGroup of this._groups) {
|
for (const chunkGroup of this._groups) {
|
||||||
chunkGroup.insertChunk(newChunk, this);
|
chunkGroup.insertChunk(newChunk, this);
|
||||||
|
@ -282,6 +439,12 @@ class Chunk {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {number} size the size
|
||||||
|
* @param {Object} options the options passed in
|
||||||
|
* @returns {number} the multiplier returned
|
||||||
|
*/
|
||||||
addMultiplierAndOverhead(size, options) {
|
addMultiplierAndOverhead(size, options) {
|
||||||
const overhead =
|
const overhead =
|
||||||
typeof options.chunkOverhead === "number" ? options.chunkOverhead : 10000;
|
typeof options.chunkOverhead === "number" ? options.chunkOverhead : 10000;
|
||||||
|
@ -292,10 +455,17 @@ class Chunk {
|
||||||
return size * multiplicator + overhead;
|
return size * multiplicator + overhead;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {number} the size of all modules
|
||||||
|
*/
|
||||||
modulesSize() {
|
modulesSize() {
|
||||||
return this._modules.getFromUnorderedCache(getModulesSize);
|
return this._modules.getFromUnorderedCache(getModulesSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {Object} options the size display options
|
||||||
|
* @returns {number} the chunk size
|
||||||
|
*/
|
||||||
size(options) {
|
size(options) {
|
||||||
return this.addMultiplierAndOverhead(this.modulesSize(), options);
|
return this.addMultiplierAndOverhead(this.modulesSize(), options);
|
||||||
}
|
}
|
||||||
|
@ -317,6 +487,10 @@ class Chunk {
|
||||||
return this.addMultiplierAndOverhead(integratedModulesSize, options);
|
return this.addMultiplierAndOverhead(integratedModulesSize, options);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {ModuleSortPredicate=} sortByFn a predicate function used to sort modules
|
||||||
|
* @returns {void}
|
||||||
|
*/
|
||||||
sortModules(sortByFn) {
|
sortModules(sortByFn) {
|
||||||
this._modules.sortWith(sortByFn || sortById);
|
this._modules.sortWith(sortByFn || sortById);
|
||||||
}
|
}
|
||||||
|
@ -452,6 +626,12 @@ class 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(filterFn, filterChunkFn) {
|
hasModuleInGraph(filterFn, filterChunkFn) {
|
||||||
const queue = new Set(this.groupsIterable);
|
const queue = new Set(this.groupsIterable);
|
||||||
const chunksProcessed = new Set();
|
const chunksProcessed = new Set();
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
"use strict";
|
"use strict";
|
||||||
|
//TODO: Make this a generic type
|
||||||
|
//https://github.com/Microsoft/TypeScript/issues/23385
|
||||||
|
//https://github.com/Microsoft/TypeScript/issues/23384
|
||||||
class SortableSet extends Set {
|
class SortableSet extends Set {
|
||||||
constructor(initialIterable, defaultSort) {
|
constructor(initialIterable, defaultSort) {
|
||||||
super(initialIterable);
|
super(initialIterable);
|
||||||
|
|
Loading…
Reference in New Issue