mirror of https://github.com/webpack/webpack.git
				
				
				
			
		
			
				
	
	
		
			502 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			502 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| /*
 | |
| 	MIT License http://www.opensource.org/licenses/mit-license.php
 | |
| 	Author Tobias Koppers @sokra
 | |
| */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| const SortableSet = require("./SortableSet");
 | |
| 
 | |
| /** @typedef {import("../Compilation")} Compilation */
 | |
| /** @typedef {import("../Entrypoint").EntryOptions} EntryOptions */
 | |
| 
 | |
| /** @typedef {string | SortableSet<string> | undefined} RuntimeSpec */
 | |
| /** @typedef {RuntimeSpec | boolean} RuntimeCondition */
 | |
| 
 | |
| /**
 | |
|  * @param {Compilation} compilation the compilation
 | |
|  * @param {string} name name of the entry
 | |
|  * @param {EntryOptions=} options optionally already received entry options
 | |
|  * @returns {RuntimeSpec} runtime
 | |
|  */
 | |
| exports.getEntryRuntime = (compilation, name, options) => {
 | |
| 	let dependOn;
 | |
| 	let runtime;
 | |
| 	if (options) {
 | |
| 		({ dependOn, runtime } = options);
 | |
| 	} else {
 | |
| 		const entry = compilation.entries.get(name);
 | |
| 		if (!entry) return name;
 | |
| 		({ dependOn, runtime } = entry.options);
 | |
| 	}
 | |
| 	if (dependOn) {
 | |
| 		/** @type {RuntimeSpec} */
 | |
| 		let result = undefined;
 | |
| 		const queue = new Set(dependOn);
 | |
| 		for (const name of queue) {
 | |
| 			const dep = compilation.entries.get(name);
 | |
| 			if (!dep) continue;
 | |
| 			const { dependOn, runtime } = dep.options;
 | |
| 			if (dependOn) {
 | |
| 				for (const name of dependOn) {
 | |
| 					queue.add(name);
 | |
| 				}
 | |
| 			} else {
 | |
| 				result = mergeRuntimeOwned(result, runtime || name);
 | |
| 			}
 | |
| 		}
 | |
| 		return result || name;
 | |
| 	} else {
 | |
| 		return runtime || name;
 | |
| 	}
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec} runtime runtime
 | |
|  * @param {function(string): void} fn functor
 | |
|  * @returns {void}
 | |
|  */
 | |
| exports.forEachRuntime = (runtime, fn) => {
 | |
| 	if (runtime === undefined) {
 | |
| 		fn(undefined);
 | |
| 	} else if (typeof runtime === "string") {
 | |
| 		fn(runtime);
 | |
| 	} else {
 | |
| 		for (const r of runtime) {
 | |
| 			fn(r);
 | |
| 		}
 | |
| 	}
 | |
| };
 | |
| 
 | |
| const getRuntimesKey = set => {
 | |
| 	set.sort();
 | |
| 	return Array.from(set).join("\n");
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec} runtime runtime(s)
 | |
|  * @returns {string} key of runtimes
 | |
|  */
 | |
| const getRuntimeKey = runtime => {
 | |
| 	if (runtime === undefined) return "*";
 | |
| 	if (typeof runtime === "string") return runtime;
 | |
| 	return runtime.getFromUnorderedCache(getRuntimesKey);
 | |
| };
 | |
| exports.getRuntimeKey = getRuntimeKey;
 | |
| 
 | |
| /**
 | |
|  * @param {string} key key of runtimes
 | |
|  * @returns {RuntimeSpec} runtime(s)
 | |
|  */
 | |
| const keyToRuntime = key => {
 | |
| 	if (key === "*") return undefined;
 | |
| 	const items = key.split("\n");
 | |
| 	if (items.length === 1) return items[0];
 | |
| 	return new SortableSet(items);
 | |
| };
 | |
| exports.keyToRuntime = keyToRuntime;
 | |
| 
 | |
| const getRuntimesString = set => {
 | |
| 	set.sort();
 | |
| 	return Array.from(set).join("+");
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec} runtime runtime(s)
 | |
|  * @returns {string} readable version
 | |
|  */
 | |
| const runtimeToString = runtime => {
 | |
| 	if (runtime === undefined) return "*";
 | |
| 	if (typeof runtime === "string") return runtime;
 | |
| 	return runtime.getFromUnorderedCache(getRuntimesString);
 | |
| };
 | |
| exports.runtimeToString = runtimeToString;
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeCondition} runtimeCondition runtime condition
 | |
|  * @returns {string} readable version
 | |
|  */
 | |
| exports.runtimeConditionToString = runtimeCondition => {
 | |
| 	if (runtimeCondition === true) return "true";
 | |
| 	if (runtimeCondition === false) return "false";
 | |
| 	return runtimeToString(runtimeCondition);
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec} a first
 | |
|  * @param {RuntimeSpec} b second
 | |
|  * @returns {boolean} true, when they are equal
 | |
|  */
 | |
| exports.runtimeEqual = (a, b) => {
 | |
| 	if (a === b) {
 | |
| 		return true;
 | |
| 	} else if (
 | |
| 		a === undefined ||
 | |
| 		b === undefined ||
 | |
| 		typeof a === "string" ||
 | |
| 		typeof b === "string"
 | |
| 	) {
 | |
| 		return false;
 | |
| 	} else if (a.size !== b.size) {
 | |
| 		return false;
 | |
| 	} else {
 | |
| 		a.sort();
 | |
| 		b.sort();
 | |
| 		const aIt = a[Symbol.iterator]();
 | |
| 		const bIt = b[Symbol.iterator]();
 | |
| 		for (;;) {
 | |
| 			const aV = aIt.next();
 | |
| 			if (aV.done) return true;
 | |
| 			const bV = bIt.next();
 | |
| 			if (aV.value !== bV.value) return false;
 | |
| 		}
 | |
| 	}
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec} a first
 | |
|  * @param {RuntimeSpec} b second
 | |
|  * @returns {-1|0|1} compare
 | |
|  */
 | |
| exports.compareRuntime = (a, b) => {
 | |
| 	if (a === b) {
 | |
| 		return 0;
 | |
| 	} else if (a === undefined) {
 | |
| 		return -1;
 | |
| 	} else if (b === undefined) {
 | |
| 		return 1;
 | |
| 	} else {
 | |
| 		const aKey = getRuntimeKey(a);
 | |
| 		const bKey = getRuntimeKey(b);
 | |
| 		if (aKey < bKey) return -1;
 | |
| 		if (aKey > bKey) return 1;
 | |
| 		return 0;
 | |
| 	}
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec} a first
 | |
|  * @param {RuntimeSpec} b second
 | |
|  * @returns {RuntimeSpec} merged
 | |
|  */
 | |
| const mergeRuntime = (a, b) => {
 | |
| 	if (a === undefined) {
 | |
| 		return b;
 | |
| 	} else if (b === undefined) {
 | |
| 		return a;
 | |
| 	} else if (a === b) {
 | |
| 		return a;
 | |
| 	} else if (typeof a === "string") {
 | |
| 		if (typeof b === "string") {
 | |
| 			const set = new SortableSet();
 | |
| 			set.add(a);
 | |
| 			set.add(b);
 | |
| 			return set;
 | |
| 		} else if (b.has(a)) {
 | |
| 			return b;
 | |
| 		} else {
 | |
| 			const set = new SortableSet(b);
 | |
| 			set.add(a);
 | |
| 			return set;
 | |
| 		}
 | |
| 	} else {
 | |
| 		if (typeof b === "string") {
 | |
| 			if (a.has(b)) return a;
 | |
| 			const set = new SortableSet(a);
 | |
| 			set.add(b);
 | |
| 			return set;
 | |
| 		} else {
 | |
| 			const set = new SortableSet(a);
 | |
| 			for (const item of b) set.add(item);
 | |
| 			if (set.size === a.size) return a;
 | |
| 			return set;
 | |
| 		}
 | |
| 	}
 | |
| };
 | |
| exports.mergeRuntime = mergeRuntime;
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeCondition} a first
 | |
|  * @param {RuntimeCondition} b second
 | |
|  * @param {RuntimeSpec} runtime full runtime
 | |
|  * @returns {RuntimeCondition} result
 | |
|  */
 | |
| exports.mergeRuntimeCondition = (a, b, runtime) => {
 | |
| 	if (a === false) return b;
 | |
| 	if (b === false) return a;
 | |
| 	if (a === true || b === true) return true;
 | |
| 	const merged = mergeRuntime(a, b);
 | |
| 	if (merged === undefined) return undefined;
 | |
| 	if (typeof merged === "string") {
 | |
| 		if (typeof runtime === "string" && merged === runtime) return true;
 | |
| 		return merged;
 | |
| 	}
 | |
| 	if (typeof runtime === "string" || runtime === undefined) return merged;
 | |
| 	if (merged.size === runtime.size) return true;
 | |
| 	return merged;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec | true} a first
 | |
|  * @param {RuntimeSpec | true} b second
 | |
|  * @param {RuntimeSpec} runtime full runtime
 | |
|  * @returns {RuntimeSpec | true} result
 | |
|  */
 | |
| exports.mergeRuntimeConditionNonFalse = (a, b, runtime) => {
 | |
| 	if (a === true || b === true) return true;
 | |
| 	const merged = mergeRuntime(a, b);
 | |
| 	if (merged === undefined) return undefined;
 | |
| 	if (typeof merged === "string") {
 | |
| 		if (typeof runtime === "string" && merged === runtime) return true;
 | |
| 		return merged;
 | |
| 	}
 | |
| 	if (typeof runtime === "string" || runtime === undefined) return merged;
 | |
| 	if (merged.size === runtime.size) return true;
 | |
| 	return merged;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec} a first (may be modified)
 | |
|  * @param {RuntimeSpec} b second
 | |
|  * @returns {RuntimeSpec} merged
 | |
|  */
 | |
| const mergeRuntimeOwned = (a, b) => {
 | |
| 	if (b === undefined) {
 | |
| 		return a;
 | |
| 	} else if (a === b) {
 | |
| 		return a;
 | |
| 	} else if (a === undefined) {
 | |
| 		if (typeof b === "string") {
 | |
| 			const set = new SortableSet();
 | |
| 			set.add(b);
 | |
| 			return set;
 | |
| 		} else {
 | |
| 			return new SortableSet(b);
 | |
| 		}
 | |
| 	} else if (typeof a === "string") {
 | |
| 		if (typeof b === "string") {
 | |
| 			const set = new SortableSet();
 | |
| 			set.add(a);
 | |
| 			set.add(b);
 | |
| 			return set;
 | |
| 		} else {
 | |
| 			const set = new SortableSet(b);
 | |
| 			set.add(a);
 | |
| 			return set;
 | |
| 		}
 | |
| 	} else {
 | |
| 		if (typeof b === "string") {
 | |
| 			a.add(b);
 | |
| 			return a;
 | |
| 		} else {
 | |
| 			for (const item of b) a.add(item);
 | |
| 			return a;
 | |
| 		}
 | |
| 	}
 | |
| };
 | |
| exports.mergeRuntimeOwned = mergeRuntimeOwned;
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec} a first
 | |
|  * @param {RuntimeSpec} b second
 | |
|  * @returns {RuntimeSpec} merged
 | |
|  */
 | |
| exports.intersectRuntime = (a, b) => {
 | |
| 	if (a === undefined) {
 | |
| 		return b;
 | |
| 	} else if (b === undefined) {
 | |
| 		return a;
 | |
| 	} else if (a === b) {
 | |
| 		return a;
 | |
| 	} else if (typeof a === "string") {
 | |
| 		if (typeof b === "string") {
 | |
| 			return undefined;
 | |
| 		} else if (b.has(a)) {
 | |
| 			return a;
 | |
| 		} else {
 | |
| 			return undefined;
 | |
| 		}
 | |
| 	} else {
 | |
| 		if (typeof b === "string") {
 | |
| 			if (a.has(b)) return b;
 | |
| 			return undefined;
 | |
| 		} else {
 | |
| 			const set = new SortableSet();
 | |
| 			for (const item of b) {
 | |
| 				if (a.has(item)) set.add(item);
 | |
| 			}
 | |
| 			if (set.size === 0) return undefined;
 | |
| 			if (set.size === 1) for (const item of set) return item;
 | |
| 			return set;
 | |
| 		}
 | |
| 	}
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec} a first
 | |
|  * @param {RuntimeSpec} b second
 | |
|  * @returns {RuntimeSpec} result
 | |
|  */
 | |
| const subtractRuntime = (a, b) => {
 | |
| 	if (a === undefined) {
 | |
| 		return undefined;
 | |
| 	} else if (b === undefined) {
 | |
| 		return a;
 | |
| 	} else if (a === b) {
 | |
| 		return undefined;
 | |
| 	} else if (typeof a === "string") {
 | |
| 		if (typeof b === "string") {
 | |
| 			return undefined;
 | |
| 		} else if (b.has(a)) {
 | |
| 			return undefined;
 | |
| 		} else {
 | |
| 			return a;
 | |
| 		}
 | |
| 	} else {
 | |
| 		if (typeof b === "string") {
 | |
| 			if (!a.has(b)) return a;
 | |
| 			if (a.size === 2) {
 | |
| 				for (const item of a) {
 | |
| 					if (item !== b) return item;
 | |
| 				}
 | |
| 			}
 | |
| 			const set = new SortableSet(a);
 | |
| 			set.delete(b);
 | |
| 		} else {
 | |
| 			const set = new SortableSet();
 | |
| 			for (const item of a) {
 | |
| 				if (!b.has(item)) set.add(item);
 | |
| 			}
 | |
| 			if (set.size === 0) return undefined;
 | |
| 			if (set.size === 1) for (const item of set) return item;
 | |
| 			return set;
 | |
| 		}
 | |
| 	}
 | |
| };
 | |
| exports.subtractRuntime = subtractRuntime;
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeCondition} a first
 | |
|  * @param {RuntimeCondition} b second
 | |
|  * @param {RuntimeSpec} runtime runtime
 | |
|  * @returns {RuntimeCondition} result
 | |
|  */
 | |
| exports.subtractRuntimeCondition = (a, b, runtime) => {
 | |
| 	if (b === true) return false;
 | |
| 	if (b === false) return a;
 | |
| 	if (a === false) return false;
 | |
| 	const result = subtractRuntime(a === true ? runtime : a, b);
 | |
| 	return result === undefined ? false : result;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {RuntimeSpec} runtime runtime
 | |
|  * @param {function(RuntimeSpec): boolean} filter filter function
 | |
|  * @returns {boolean | RuntimeSpec} true/false if filter is constant for all runtimes, otherwise runtimes that are active
 | |
|  */
 | |
| exports.filterRuntime = (runtime, filter) => {
 | |
| 	if (runtime === undefined) return filter(undefined);
 | |
| 	if (typeof runtime === "string") return filter(runtime);
 | |
| 	let some = false;
 | |
| 	let every = true;
 | |
| 	let result = undefined;
 | |
| 	for (const r of runtime) {
 | |
| 		const v = filter(r);
 | |
| 		if (v) {
 | |
| 			some = true;
 | |
| 			result = mergeRuntimeOwned(result, r);
 | |
| 		} else {
 | |
| 			every = false;
 | |
| 		}
 | |
| 	}
 | |
| 	if (!some) return false;
 | |
| 	if (every) return true;
 | |
| 	return result;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @template T
 | |
|  */
 | |
| class RuntimeSpecMap {
 | |
| 	/**
 | |
| 	 * @param {RuntimeSpecMap<T>=} clone copy form this
 | |
| 	 */
 | |
| 	constructor(clone) {
 | |
| 		/** @type {Map<string, T>} */
 | |
| 		this._map = new Map(clone ? clone._map : undefined);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {RuntimeSpec} runtime the runtimes
 | |
| 	 * @returns {T} value
 | |
| 	 */
 | |
| 	get(runtime) {
 | |
| 		const key = getRuntimeKey(runtime);
 | |
| 		return this._map.get(key);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param {RuntimeSpec} runtime the runtimes
 | |
| 	 * @returns {boolean} true, when the runtime is stored
 | |
| 	 */
 | |
| 	has(runtime) {
 | |
| 		const key = getRuntimeKey(runtime);
 | |
| 		return this._map.has(key);
 | |
| 	}
 | |
| 
 | |
| 	set(runtime, value) {
 | |
| 		this._map.set(getRuntimeKey(runtime), value);
 | |
| 	}
 | |
| 
 | |
| 	delete(runtime) {
 | |
| 		this._map.delete(getRuntimeKey(runtime));
 | |
| 	}
 | |
| 
 | |
| 	update(runtime, fn) {
 | |
| 		const key = getRuntimeKey(runtime);
 | |
| 		const oldValue = this._map.get(key);
 | |
| 		const newValue = fn(oldValue);
 | |
| 		if (newValue !== oldValue) this._map.set(key, newValue);
 | |
| 	}
 | |
| 
 | |
| 	keys() {
 | |
| 		return Array.from(this._map.keys(), keyToRuntime);
 | |
| 	}
 | |
| 
 | |
| 	values() {
 | |
| 		return this._map.values();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| exports.RuntimeSpecMap = RuntimeSpecMap;
 | |
| 
 | |
| class RuntimeSpecSet {
 | |
| 	constructor(iterable) {
 | |
| 		/** @type {Map<string, RuntimeSpec>} */
 | |
| 		this._map = new Map();
 | |
| 		if (iterable) {
 | |
| 			for (const item of iterable) {
 | |
| 				this.add(item);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	add(runtime) {
 | |
| 		this._map.set(getRuntimeKey(runtime), runtime);
 | |
| 	}
 | |
| 
 | |
| 	has(runtime) {
 | |
| 		return this._map.has(getRuntimeKey(runtime));
 | |
| 	}
 | |
| 
 | |
| 	[Symbol.iterator]() {
 | |
| 		return this._map.values();
 | |
| 	}
 | |
| 
 | |
| 	get size() {
 | |
| 		return this._map.size;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| exports.RuntimeSpecSet = RuntimeSpecSet;
 |