| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | /* | 
					
						
							|  |  |  | 	MIT License http://www.opensource.org/licenses/mit-license.php
 | 
					
						
							|  |  |  | 	Author Tobias Koppers @sokra | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2018-07-30 23:08:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | "use strict"; | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-11 12:27:09 +08:00
										 |  |  | const asyncLib = require("neo-async"); | 
					
						
							| 
									
										
										
										
											2018-07-30 23:08:51 +08:00
										 |  |  | const { SyncHook, MultiHook } = require("tapable"); | 
					
						
							| 
									
										
										
										
											2018-12-09 19:54:17 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-13 18:10:42 +08:00
										 |  |  | const ConcurrentCompilationError = require("./ConcurrentCompilationError"); | 
					
						
							| 
									
										
										
										
											2018-07-30 23:08:51 +08:00
										 |  |  | const MultiStats = require("./MultiStats"); | 
					
						
							|  |  |  | const MultiWatching = require("./MultiWatching"); | 
					
						
							| 
									
										
										
										
											2022-03-03 01:06:07 +08:00
										 |  |  | const WebpackError = require("./WebpackError"); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | const ArrayQueue = require("./util/ArrayQueue"); | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-05 18:15:03 +08:00
										 |  |  | /** @template T @typedef {import("tapable").AsyncSeriesHook<T>} AsyncSeriesHook<T> */ | 
					
						
							|  |  |  | /** @template T @template R @typedef {import("tapable").SyncBailHook<T, R>} SyncBailHook<T, R> */ | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | /** @typedef {import("../declarations/WebpackOptions").WatchOptions} WatchOptions */ | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | /** @typedef {import("./Compiler")} Compiler */ | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | /** @typedef {import("./Stats")} Stats */ | 
					
						
							| 
									
										
										
										
											2018-12-09 19:54:17 +08:00
										 |  |  | /** @typedef {import("./Watching")} Watching */ | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | /** @typedef {import("./logging/Logger").Logger} Logger */ | 
					
						
							| 
									
										
										
										
											2019-06-11 19:09:42 +08:00
										 |  |  | /** @typedef {import("./util/fs").InputFileSystem} InputFileSystem */ | 
					
						
							|  |  |  | /** @typedef {import("./util/fs").IntermediateFileSystem} IntermediateFileSystem */ | 
					
						
							|  |  |  | /** @typedef {import("./util/fs").OutputFileSystem} OutputFileSystem */ | 
					
						
							| 
									
										
										
										
											2020-07-08 16:21:31 +08:00
										 |  |  | /** @typedef {import("./util/fs").WatchFileSystem} WatchFileSystem */ | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @template T | 
					
						
							|  |  |  |  * @callback Callback | 
					
						
							| 
									
										
										
										
											2024-02-21 22:05:29 +08:00
										 |  |  |  * @param {Error | null} err | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  |  * @param {T=} result | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @callback RunWithDependenciesHandler | 
					
						
							|  |  |  |  * @param {Compiler} compiler | 
					
						
							|  |  |  |  * @param {Callback<MultiStats>} callback | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2024-06-11 21:09:50 +08:00
										 |  |  |  * @typedef {object} MultiCompilerOptions | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  |  * @property {number=} parallelism how many Compilers are allows to run at the same time in parallel | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-23 20:03:37 +08:00
										 |  |  | const CLASS_NAME = "MultiCompiler"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-26 14:27:44 +08:00
										 |  |  | module.exports = class MultiCompiler { | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 	/** | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 	 * @param {Compiler[] | Record<string, Compiler>} compilers child compilers | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 	 * @param {MultiCompilerOptions} options options | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 	constructor(compilers, options) { | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 		if (!Array.isArray(compilers)) { | 
					
						
							| 
									
										
										
										
											2024-02-21 22:05:29 +08:00
										 |  |  | 			/** @type {Compiler[]} */ | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 			compilers = Object.keys(compilers).map(name => { | 
					
						
							| 
									
										
										
										
											2024-02-21 22:05:29 +08:00
										 |  |  | 				/** @type {Record<string, Compiler>} */ | 
					
						
							|  |  |  | 				(compilers)[name].name = name; | 
					
						
							|  |  |  | 				return /** @type {Record<string, Compiler>} */ (compilers)[name]; | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 			}); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-30 20:25:40 +08:00
										 |  |  | 		this.hooks = Object.freeze({ | 
					
						
							| 
									
										
										
										
											2018-12-09 19:54:17 +08:00
										 |  |  | 			/** @type {SyncHook<[MultiStats]>} */ | 
					
						
							| 
									
										
										
										
											2017-11-28 17:18:46 +08:00
										 |  |  | 			done: new SyncHook(["stats"]), | 
					
						
							| 
									
										
										
										
											2020-07-15 21:49:34 +08:00
										 |  |  | 			/** @type {MultiHook<SyncHook<[string | null, number]>>} */ | 
					
						
							| 
									
										
										
										
											2017-12-29 18:23:14 +08:00
										 |  |  | 			invalid: new MultiHook(compilers.map(c => c.hooks.invalid)), | 
					
						
							| 
									
										
										
										
											2018-12-09 19:54:17 +08:00
										 |  |  | 			/** @type {MultiHook<AsyncSeriesHook<[Compiler]>>} */ | 
					
						
							| 
									
										
										
										
											2017-12-29 18:23:14 +08:00
										 |  |  | 			run: new MultiHook(compilers.map(c => c.hooks.run)), | 
					
						
							| 
									
										
										
										
											2018-12-09 19:54:17 +08:00
										 |  |  | 			/** @type {SyncHook<[]>} */ | 
					
						
							| 
									
										
										
										
											2017-12-29 18:23:14 +08:00
										 |  |  | 			watchClose: new SyncHook([]), | 
					
						
							| 
									
										
										
										
											2018-12-09 19:54:17 +08:00
										 |  |  | 			/** @type {MultiHook<AsyncSeriesHook<[Compiler]>>} */ | 
					
						
							| 
									
										
										
										
											2019-08-01 14:20:34 +08:00
										 |  |  | 			watchRun: new MultiHook(compilers.map(c => c.hooks.watchRun)), | 
					
						
							| 
									
										
										
										
											2025-04-03 00:02:22 +08:00
										 |  |  | 			/** @type {MultiHook<SyncBailHook<[string, string, EXPECTED_ANY[] | undefined], true | void>>} */ | 
					
						
							| 
									
										
										
										
											2019-08-01 14:20:34 +08:00
										 |  |  | 			infrastructureLog: new MultiHook( | 
					
						
							|  |  |  | 				compilers.map(c => c.hooks.infrastructureLog) | 
					
						
							|  |  |  | 			) | 
					
						
							| 
									
										
										
										
											2018-07-30 20:25:40 +08:00
										 |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		this.compilers = compilers; | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		/** @type {MultiCompilerOptions} */ | 
					
						
							|  |  |  | 		this._options = { | 
					
						
							|  |  |  | 			parallelism: options.parallelism || Infinity | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 		/** @type {WeakMap<Compiler, string[]>} */ | 
					
						
							|  |  |  | 		this.dependencies = new WeakMap(); | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 		this.running = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 		/** @type {(Stats | null)[]} */ | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 		const compilerStats = this.compilers.map(() => null); | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		let doneCompilers = 0; | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 		for (let index = 0; index < this.compilers.length; index++) { | 
					
						
							|  |  |  | 			const compiler = this.compilers[index]; | 
					
						
							|  |  |  | 			const compilerIndex = index; | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 			let compilerDone = false; | 
					
						
							| 
									
										
										
										
											2024-07-31 13:38:50 +08:00
										 |  |  | 			// eslint-disable-next-line no-loop-func
 | 
					
						
							| 
									
										
										
										
											2025-04-23 20:03:37 +08:00
										 |  |  | 			compiler.hooks.done.tap(CLASS_NAME, stats => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				if (!compilerDone) { | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 					compilerDone = true; | 
					
						
							|  |  |  | 					doneCompilers++; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-01-22 20:52:43 +08:00
										 |  |  | 				compilerStats[compilerIndex] = stats; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				if (doneCompilers === this.compilers.length) { | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 					this.hooks.done.call( | 
					
						
							|  |  |  | 						new MultiStats(/** @type {Stats[]} */ (compilerStats)) | 
					
						
							|  |  |  | 					); | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			}); | 
					
						
							| 
									
										
										
										
											2024-07-31 13:38:50 +08:00
										 |  |  | 			// eslint-disable-next-line no-loop-func
 | 
					
						
							| 
									
										
										
										
											2025-04-23 20:03:37 +08:00
										 |  |  | 			compiler.hooks.invalid.tap(CLASS_NAME, () => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				if (compilerDone) { | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 					compilerDone = false; | 
					
						
							|  |  |  | 					doneCompilers--; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			}); | 
					
						
							| 
									
										
										
										
											2018-01-22 20:52:43 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-03-03 01:06:07 +08:00
										 |  |  | 		this._validateCompilersOptions(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_validateCompilersOptions() { | 
					
						
							|  |  |  | 		if (this.compilers.length < 2) return; | 
					
						
							| 
									
										
										
										
											2024-04-13 00:24:21 +08:00
										 |  |  | 		/** | 
					
						
							|  |  |  | 		 * @param {Compiler} compiler compiler | 
					
						
							|  |  |  | 		 * @param {WebpackError} warning warning | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2022-03-03 01:06:07 +08:00
										 |  |  | 		const addWarning = (compiler, warning) => { | 
					
						
							| 
									
										
										
										
											2025-04-23 20:03:37 +08:00
										 |  |  | 			compiler.hooks.thisCompilation.tap(CLASS_NAME, compilation => { | 
					
						
							| 
									
										
										
										
											2022-03-03 01:06:07 +08:00
										 |  |  | 				compilation.warnings.push(warning); | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2024-05-08 00:01:41 +08:00
										 |  |  | 		const cacheNames = new Set(); | 
					
						
							| 
									
										
										
										
											2022-03-03 01:06:07 +08:00
										 |  |  | 		for (const compiler of this.compilers) { | 
					
						
							|  |  |  | 			if (compiler.options.cache && "name" in compiler.options.cache) { | 
					
						
							| 
									
										
										
										
											2024-04-13 00:24:21 +08:00
										 |  |  | 				const name = compiler.options.cache.name; | 
					
						
							| 
									
										
										
										
											2024-05-08 00:01:41 +08:00
										 |  |  | 				if (cacheNames.has(name)) { | 
					
						
							|  |  |  | 					addWarning( | 
					
						
							|  |  |  | 						compiler, | 
					
						
							|  |  |  | 						new WebpackError( | 
					
						
							|  |  |  | 							`${ | 
					
						
							|  |  |  | 								compiler.name | 
					
						
							|  |  |  | 									? `Compiler with name "${compiler.name}" doesn't use unique cache name. ` | 
					
						
							|  |  |  | 									: "" | 
					
						
							|  |  |  | 							}Please set unique "cache.name" option. Name "${name}" already used.`
 | 
					
						
							|  |  |  | 						) | 
					
						
							|  |  |  | 					); | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					cacheNames.add(name); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2022-03-03 01:06:07 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-04-24 05:55:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 17:27:46 +08:00
										 |  |  | 	get options() { | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		return Object.assign( | 
					
						
							|  |  |  | 			this.compilers.map(c => c.options), | 
					
						
							|  |  |  | 			this._options | 
					
						
							|  |  |  | 		); | 
					
						
							| 
									
										
										
										
											2020-02-17 17:27:46 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	get outputPath() { | 
					
						
							|  |  |  | 		let commonPath = this.compilers[0].outputPath; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		for (const compiler of this.compilers) { | 
					
						
							|  |  |  | 			while ( | 
					
						
							|  |  |  | 				compiler.outputPath.indexOf(commonPath) !== 0 && | 
					
						
							|  |  |  | 				/[/\\]/.test(commonPath) | 
					
						
							|  |  |  | 			) { | 
					
						
							| 
									
										
										
										
											2017-07-24 17:54:06 +08:00
										 |  |  | 				commonPath = commonPath.replace(/[/\\][^/\\]*$/, ""); | 
					
						
							| 
									
										
										
										
											2014-06-19 05:02:33 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (!commonPath && this.compilers[0].outputPath[0] === "/") return "/"; | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		return commonPath; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	get inputFileSystem() { | 
					
						
							|  |  |  | 		throw new Error("Cannot read inputFileSystem of a MultiCompiler"); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-11 19:09:42 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {InputFileSystem} value the new input file system | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	set inputFileSystem(value) { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		for (const compiler of this.compilers) { | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 			compiler.inputFileSystem = value; | 
					
						
							| 
									
										
										
										
											2018-01-22 20:52:43 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-31 09:37:24 +08:00
										 |  |  | 	get outputFileSystem() { | 
					
						
							|  |  |  | 		throw new Error("Cannot read outputFileSystem of a MultiCompiler"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-11 19:09:42 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {OutputFileSystem} value the new output file system | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	set outputFileSystem(value) { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		for (const compiler of this.compilers) { | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 			compiler.outputFileSystem = value; | 
					
						
							| 
									
										
										
										
											2018-01-22 20:52:43 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-31 09:37:24 +08:00
										 |  |  | 	get watchFileSystem() { | 
					
						
							|  |  |  | 		throw new Error("Cannot read watchFileSystem of a MultiCompiler"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 16:21:31 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {WatchFileSystem} value the new watch file system | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	set watchFileSystem(value) { | 
					
						
							|  |  |  | 		for (const compiler of this.compilers) { | 
					
						
							|  |  |  | 			compiler.watchFileSystem = value; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-11 19:09:42 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {IntermediateFileSystem} value the new intermediate file system | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	set intermediateFileSystem(value) { | 
					
						
							|  |  |  | 		for (const compiler of this.compilers) { | 
					
						
							|  |  |  | 			compiler.intermediateFileSystem = value; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-31 09:37:24 +08:00
										 |  |  | 	get intermediateFileSystem() { | 
					
						
							|  |  |  | 		throw new Error("Cannot read outputFileSystem of a MultiCompiler"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 	/** | 
					
						
							| 
									
										
										
										
											2025-03-12 09:56:14 +08:00
										 |  |  | 	 * @param {string | (() => string)} name name of the logger, or function called once to get the logger name | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 	 * @returns {Logger} a logger with that name | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2019-07-31 22:57:26 +08:00
										 |  |  | 	getInfrastructureLogger(name) { | 
					
						
							|  |  |  | 		return this.compilers[0].getInfrastructureLogger(name); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {Compiler} compiler the child compiler | 
					
						
							|  |  |  | 	 * @param {string[]} dependencies its dependencies | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	setDependencies(compiler, dependencies) { | 
					
						
							|  |  |  | 		this.dependencies.set(compiler, dependencies); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {Callback<MultiStats>} callback signals when the validation is complete | 
					
						
							|  |  |  | 	 * @returns {boolean} true if the dependencies are valid | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 	validateDependencies(callback) { | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 		/** @type {Set<{source: Compiler, target: Compiler}>} */ | 
					
						
							| 
									
										
										
										
											2017-09-17 22:59:35 +08:00
										 |  |  | 		const edges = new Set(); | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 		/** @type {string[]} */ | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 		const missing = []; | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 		/** | 
					
						
							|  |  |  | 		 * @param {Compiler} compiler compiler | 
					
						
							|  |  |  | 		 * @returns {boolean} target was found | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		const targetFound = compiler => { | 
					
						
							|  |  |  | 			for (const edge of edges) { | 
					
						
							|  |  |  | 				if (edge.target === compiler) { | 
					
						
							| 
									
										
										
										
											2017-09-17 22:59:35 +08:00
										 |  |  | 					return true; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2024-02-21 22:05:29 +08:00
										 |  |  | 		/** | 
					
						
							|  |  |  | 		 * @param {{source: Compiler, target: Compiler}} e1 edge 1 | 
					
						
							|  |  |  | 		 * @param {{source: Compiler, target: Compiler}} e2 edge 2 | 
					
						
							|  |  |  | 		 * @returns {number} result | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2024-07-31 11:31:11 +08:00
										 |  |  | 		const sortEdges = (e1, e2) => | 
					
						
							|  |  |  | 			/** @type {string} */ | 
					
						
							|  |  |  | 			(e1.source.name).localeCompare(/** @type {string} */ (e2.source.name)) || | 
					
						
							|  |  |  | 			/** @type {string} */ | 
					
						
							|  |  |  | 			(e1.target.name).localeCompare(/** @type {string} */ (e2.target.name)); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		for (const source of this.compilers) { | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 			const dependencies = this.dependencies.get(source); | 
					
						
							|  |  |  | 			if (dependencies) { | 
					
						
							|  |  |  | 				for (const dep of dependencies) { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					const target = this.compilers.find(c => c.name === dep); | 
					
						
							|  |  |  | 					if (!target) { | 
					
						
							| 
									
										
										
										
											2017-09-09 00:46:47 +08:00
										 |  |  | 						missing.push(dep); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 					} else { | 
					
						
							| 
									
										
										
										
											2017-09-17 22:59:35 +08:00
										 |  |  | 						edges.add({ | 
					
						
							| 
									
										
										
										
											2017-09-09 00:46:47 +08:00
										 |  |  | 							source, | 
					
						
							|  |  |  | 							target | 
					
						
							|  |  |  | 						}); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-01-13 23:48:25 +08:00
										 |  |  | 		/** @type {string[]} */ | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		const errors = missing.map(m => `Compiler dependency \`${m}\` not found.`); | 
					
						
							|  |  |  | 		const stack = this.compilers.filter(c => !targetFound(c)); | 
					
						
							|  |  |  | 		while (stack.length > 0) { | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 			const current = stack.pop(); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			for (const edge of edges) { | 
					
						
							|  |  |  | 				if (edge.source === current) { | 
					
						
							| 
									
										
										
										
											2017-09-17 22:59:35 +08:00
										 |  |  | 					edges.delete(edge); | 
					
						
							|  |  |  | 					const target = edge.target; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					if (!targetFound(target)) { | 
					
						
							| 
									
										
										
										
											2017-09-17 22:59:35 +08:00
										 |  |  | 						stack.push(target); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (edges.size > 0) { | 
					
						
							| 
									
										
										
										
											2021-01-13 23:48:25 +08:00
										 |  |  | 			/** @type {string[]} */ | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			const lines = Array.from(edges) | 
					
						
							|  |  |  | 				.sort(sortEdges) | 
					
						
							|  |  |  | 				.map(edge => `${edge.source.name} -> ${edge.target.name}`); | 
					
						
							| 
									
										
										
										
											2017-09-17 21:38:19 +08:00
										 |  |  | 			lines.unshift("Circular dependency found in compiler dependencies."); | 
					
						
							|  |  |  | 			errors.unshift(lines.join("\n")); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (errors.length > 0) { | 
					
						
							| 
									
										
										
										
											2017-09-11 21:55:28 +08:00
										 |  |  | 			const message = errors.join("\n"); | 
					
						
							|  |  |  | 			callback(new Error(message)); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 	// TODO webpack 6 remove
 | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 	/** | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 	 * @deprecated This method should have been private | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 	 * @param {Compiler[]} compilers the child compilers | 
					
						
							|  |  |  | 	 * @param {RunWithDependenciesHandler} fn a handler to run for each compiler | 
					
						
							|  |  |  | 	 * @param {Callback<MultiStats>} callback the compiler's handler | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	runWithDependencies(compilers, fn, callback) { | 
					
						
							| 
									
										
										
										
											2018-01-05 14:41:09 +08:00
										 |  |  | 		const fulfilledNames = new Set(); | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		let remainingCompilers = compilers; | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 		/** | 
					
						
							|  |  |  | 		 * @param {string} d dependency | 
					
						
							|  |  |  | 		 * @returns {boolean} when dependency was fulfilled | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		const isDependencyFulfilled = d => fulfilledNames.has(d); | 
					
						
							| 
									
										
										
										
											2024-02-21 22:05:29 +08:00
										 |  |  | 		/** | 
					
						
							|  |  |  | 		 * @returns {Compiler[]} compilers | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		const getReadyCompilers = () => { | 
					
						
							| 
									
										
										
										
											2024-07-31 04:09:42 +08:00
										 |  |  | 			const readyCompilers = []; | 
					
						
							|  |  |  | 			const list = remainingCompilers; | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 			remainingCompilers = []; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			for (const c of list) { | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 				const dependencies = this.dependencies.get(c); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				const ready = | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 					!dependencies || dependencies.every(isDependencyFulfilled); | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 				if (ready) { | 
					
						
							|  |  |  | 					readyCompilers.push(c); | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					remainingCompilers.push(c); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			return readyCompilers; | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 		/** | 
					
						
							|  |  |  | 		 * @param {Callback<MultiStats>} callback callback | 
					
						
							|  |  |  | 		 * @returns {void} | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		const runCompilers = callback => { | 
					
						
							| 
									
										
										
										
											2024-02-21 22:05:29 +08:00
										 |  |  | 			if (remainingCompilers.length === 0) return callback(null); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			asyncLib.map( | 
					
						
							|  |  |  | 				getReadyCompilers(), | 
					
						
							|  |  |  | 				(compiler, callback) => { | 
					
						
							|  |  |  | 					fn(compiler, err => { | 
					
						
							|  |  |  | 						if (err) return callback(err); | 
					
						
							|  |  |  | 						fulfilledNames.add(compiler.name); | 
					
						
							|  |  |  | 						runCompilers(callback); | 
					
						
							|  |  |  | 					}); | 
					
						
							|  |  |  | 				}, | 
					
						
							| 
									
										
										
										
											2024-02-21 22:05:29 +08:00
										 |  |  | 				(err, results) => { | 
					
						
							| 
									
										
										
										
											2024-02-21 22:55:02 +08:00
										 |  |  | 					callback(err, /** @type {TODO} */ (results)); | 
					
						
							| 
									
										
										
										
											2024-02-21 22:05:29 +08:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			); | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		}; | 
					
						
							|  |  |  | 		runCompilers(callback); | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template SetupResult | 
					
						
							| 
									
										
										
										
											2025-03-12 09:56:14 +08:00
										 |  |  | 	 * @param {(compiler: Compiler, index: number, doneCallback: Callback<Stats>, isBlocked: () => boolean, setChanged: () => void, setInvalid: () => void) => SetupResult} setup setup a single compiler | 
					
						
							|  |  |  | 	 * @param {(compiler: Compiler, setupResult: SetupResult, callback: Callback<Stats>) => void} run run/continue a single compiler | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 	 * @param {Callback<MultiStats>} callback callback when all compilers are done, result includes Stats of all changed compilers | 
					
						
							|  |  |  | 	 * @returns {SetupResult[]} result of setup | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	_runGraph(setup, run, callback) { | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 		/** @typedef {{ compiler: Compiler, setupResult: undefined | SetupResult, result: undefined | Stats, state: "pending" | "blocked" | "queued" | "starting" | "running" | "running-outdated" | "done", children: Node[], parents: Node[] }} Node */ | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// State transitions for nodes:
 | 
					
						
							|  |  |  | 		// -> blocked (initial)
 | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 		// blocked -> starting [running++] (when all parents done)
 | 
					
						
							|  |  |  | 		// queued -> starting [running++] (when processing the queue)
 | 
					
						
							|  |  |  | 		// starting -> running (when run has been called)
 | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 		// running -> done [running--] (when compilation is done)
 | 
					
						
							|  |  |  | 		// done -> pending (when invalidated from file change)
 | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 		// pending -> blocked [add to queue] (when invalidated from aggregated changes)
 | 
					
						
							|  |  |  | 		// done -> blocked [add to queue] (when invalidated, from parent invalidation)
 | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 		// running -> running-outdated (when invalidated, either from change or parent invalidation)
 | 
					
						
							|  |  |  | 		// running-outdated -> blocked [running--] (when compilation is done)
 | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/** @type {Node[]} */ | 
					
						
							|  |  |  | 		const nodes = this.compilers.map(compiler => ({ | 
					
						
							|  |  |  | 			compiler, | 
					
						
							| 
									
										
										
										
											2021-07-13 16:50:29 +08:00
										 |  |  | 			setupResult: undefined, | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			result: undefined, | 
					
						
							|  |  |  | 			state: "blocked", | 
					
						
							|  |  |  | 			children: [], | 
					
						
							|  |  |  | 			parents: [] | 
					
						
							|  |  |  | 		})); | 
					
						
							|  |  |  | 		/** @type {Map<string, Node>} */ | 
					
						
							|  |  |  | 		const compilerToNode = new Map(); | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 		for (const node of nodes) { | 
					
						
							|  |  |  | 			compilerToNode.set(/** @type {string} */ (node.compiler.name), node); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		for (const node of nodes) { | 
					
						
							|  |  |  | 			const dependencies = this.dependencies.get(node.compiler); | 
					
						
							|  |  |  | 			if (!dependencies) continue; | 
					
						
							|  |  |  | 			for (const dep of dependencies) { | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 				const parent = /** @type {Node} */ (compilerToNode.get(dep)); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 				node.parents.push(parent); | 
					
						
							|  |  |  | 				parent.children.push(node); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 		/** @type {ArrayQueue<Node>} */ | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		const queue = new ArrayQueue(); | 
					
						
							|  |  |  | 		for (const node of nodes) { | 
					
						
							|  |  |  | 			if (node.parents.length === 0) { | 
					
						
							|  |  |  | 				node.state = "queued"; | 
					
						
							|  |  |  | 				queue.enqueue(node); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		let errored = false; | 
					
						
							|  |  |  | 		let running = 0; | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 		const parallelism = /** @type {number} */ (this._options.parallelism); | 
					
						
							| 
									
										
										
										
											2021-02-25 03:12:32 +08:00
										 |  |  | 		/** | 
					
						
							|  |  |  | 		 * @param {Node} node node | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 		 * @param {(Error | null)=} err error | 
					
						
							| 
									
										
										
										
											2021-02-25 03:12:32 +08:00
										 |  |  | 		 * @param {Stats=} stats result | 
					
						
							|  |  |  | 		 * @returns {void} | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		const nodeDone = (node, err, stats) => { | 
					
						
							|  |  |  | 			if (errored) return; | 
					
						
							|  |  |  | 			if (err) { | 
					
						
							|  |  |  | 				errored = true; | 
					
						
							|  |  |  | 				return asyncLib.each( | 
					
						
							|  |  |  | 					nodes, | 
					
						
							|  |  |  | 					(node, callback) => { | 
					
						
							|  |  |  | 						if (node.compiler.watching) { | 
					
						
							|  |  |  | 							node.compiler.watching.close(callback); | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							callback(); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					() => callback(err) | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			node.result = stats; | 
					
						
							| 
									
										
										
										
											2021-02-25 03:12:32 +08:00
										 |  |  | 			running--; | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			if (node.state === "running") { | 
					
						
							|  |  |  | 				node.state = "done"; | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 				for (const child of node.children) { | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 					if (child.state === "blocked") queue.enqueue(child); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 			} else if (node.state === "running-outdated") { | 
					
						
							|  |  |  | 				node.state = "blocked"; | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 				queue.enqueue(node); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 			processQueue(); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2021-02-25 03:12:32 +08:00
										 |  |  | 		/** | 
					
						
							|  |  |  | 		 * @param {Node} node node | 
					
						
							|  |  |  | 		 * @returns {void} | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 		const nodeInvalidFromParent = node => { | 
					
						
							|  |  |  | 			if (node.state === "done") { | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 				node.state = "blocked"; | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 			} else if (node.state === "running") { | 
					
						
							|  |  |  | 				node.state = "running-outdated"; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			for (const child of node.children) { | 
					
						
							|  |  |  | 				nodeInvalidFromParent(child); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 		/** | 
					
						
							|  |  |  | 		 * @param {Node} node node | 
					
						
							|  |  |  | 		 * @returns {void} | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		const nodeInvalid = node => { | 
					
						
							|  |  |  | 			if (node.state === "done") { | 
					
						
							|  |  |  | 				node.state = "pending"; | 
					
						
							|  |  |  | 			} else if (node.state === "running") { | 
					
						
							|  |  |  | 				node.state = "running-outdated"; | 
					
						
							| 
									
										
										
										
											2021-02-25 03:12:32 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			for (const child of node.children) { | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 				nodeInvalidFromParent(child); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2021-02-25 03:12:32 +08:00
										 |  |  | 		/** | 
					
						
							|  |  |  | 		 * @param {Node} node node | 
					
						
							|  |  |  | 		 * @returns {void} | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		const nodeChange = node => { | 
					
						
							|  |  |  | 			nodeInvalid(node); | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 			if (node.state === "pending") { | 
					
						
							|  |  |  | 				node.state = "blocked"; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 			if (node.state === "blocked") { | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 				queue.enqueue(node); | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 				processQueue(); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 		/** @type {SetupResult[]} */ | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		const setupResults = []; | 
					
						
							| 
									
										
										
										
											2024-08-02 02:36:27 +08:00
										 |  |  | 		for (const [i, node] of nodes.entries()) { | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			setupResults.push( | 
					
						
							| 
									
										
										
										
											2021-07-13 16:50:29 +08:00
										 |  |  | 				(node.setupResult = setup( | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 					node.compiler, | 
					
						
							|  |  |  | 					i, | 
					
						
							|  |  |  | 					nodeDone.bind(null, node), | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 					() => node.state !== "starting" && node.state !== "running", | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 					() => nodeChange(node), | 
					
						
							|  |  |  | 					() => nodeInvalid(node) | 
					
						
							| 
									
										
										
										
											2021-07-13 16:50:29 +08:00
										 |  |  | 				)) | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			); | 
					
						
							| 
									
										
										
										
											2024-08-02 02:36:27 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 		let processing = true; | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		const processQueue = () => { | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 			if (processing) return; | 
					
						
							|  |  |  | 			processing = true; | 
					
						
							|  |  |  | 			process.nextTick(processQueueWorker); | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 		const processQueueWorker = () => { | 
					
						
							| 
									
										
										
										
											2024-07-31 13:38:50 +08:00
										 |  |  | 			// eslint-disable-next-line no-unmodified-loop-condition
 | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			while (running < parallelism && queue.length > 0 && !errored) { | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 				const node = /** @type {Node} */ (queue.dequeue()); | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 				if ( | 
					
						
							|  |  |  | 					node.state === "queued" || | 
					
						
							|  |  |  | 					(node.state === "blocked" && | 
					
						
							|  |  |  | 						node.parents.every(p => p.state === "done")) | 
					
						
							|  |  |  | 				) { | 
					
						
							|  |  |  | 					running++; | 
					
						
							|  |  |  | 					node.state = "starting"; | 
					
						
							| 
									
										
										
										
											2024-02-17 01:39:12 +08:00
										 |  |  | 					run( | 
					
						
							|  |  |  | 						node.compiler, | 
					
						
							|  |  |  | 						/** @type {SetupResult} */ (node.setupResult), | 
					
						
							|  |  |  | 						nodeDone.bind(null, node) | 
					
						
							|  |  |  | 					); | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 					node.state = "running"; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 			processing = false; | 
					
						
							| 
									
										
										
										
											2021-03-17 18:04:41 +08:00
										 |  |  | 			if ( | 
					
						
							|  |  |  | 				!errored && | 
					
						
							|  |  |  | 				running === 0 && | 
					
						
							|  |  |  | 				nodes.every(node => node.state === "done") | 
					
						
							|  |  |  | 			) { | 
					
						
							| 
									
										
										
										
											2021-02-15 04:57:52 +08:00
										 |  |  | 				const stats = []; | 
					
						
							|  |  |  | 				for (const node of nodes) { | 
					
						
							|  |  |  | 					const result = node.result; | 
					
						
							|  |  |  | 					if (result) { | 
					
						
							|  |  |  | 						node.result = undefined; | 
					
						
							|  |  |  | 						stats.push(result); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				if (stats.length > 0) { | 
					
						
							|  |  |  | 					callback(null, new MultiStats(stats)); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2021-05-17 19:49:43 +08:00
										 |  |  | 		processQueueWorker(); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		return setupResults; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 	/** | 
					
						
							| 
									
										
										
										
											2019-11-08 18:13:16 +08:00
										 |  |  | 	 * @param {WatchOptions|WatchOptions[]} watchOptions the watcher's options | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 	 * @param {Callback<MultiStats>} handler signals when the call finishes | 
					
						
							|  |  |  | 	 * @returns {MultiWatching} a compiler watcher | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	watch(watchOptions, handler) { | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 		if (this.running) { | 
					
						
							|  |  |  | 			return handler(new ConcurrentCompilationError()); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-02-25 03:12:32 +08:00
										 |  |  | 		this.running = true; | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-13 18:16:39 +08:00
										 |  |  | 		if (this.validateDependencies(handler)) { | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			const watchings = this._runGraph( | 
					
						
							|  |  |  | 				(compiler, idx, callback, isBlocked, setChanged, setInvalid) => { | 
					
						
							|  |  |  | 					const watching = compiler.watch( | 
					
						
							|  |  |  | 						Array.isArray(watchOptions) ? watchOptions[idx] : watchOptions, | 
					
						
							|  |  |  | 						callback | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 					if (watching) { | 
					
						
							|  |  |  | 						watching._onInvalid = setInvalid; | 
					
						
							|  |  |  | 						watching._onChange = setChanged; | 
					
						
							|  |  |  | 						watching._isBlocked = isBlocked; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					return watching; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				}, | 
					
						
							| 
									
										
										
										
											2021-07-13 16:50:29 +08:00
										 |  |  | 				(compiler, watching, callback) => { | 
					
						
							|  |  |  | 					if (compiler.watching !== watching) return; | 
					
						
							|  |  |  | 					if (!watching.running) watching.invalidate(); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 				handler | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			); | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			return new MultiWatching(watchings, this); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 		return new MultiWatching([], this); | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {Callback<MultiStats>} callback signals when the call finishes | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	run(callback) { | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 		if (this.running) { | 
					
						
							|  |  |  | 			return callback(new ConcurrentCompilationError()); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-02-25 03:12:32 +08:00
										 |  |  | 		this.running = true; | 
					
						
							| 
									
										
										
										
											2018-03-07 18:45:38 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (this.validateDependencies(callback)) { | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 			this._runGraph( | 
					
						
							|  |  |  | 				() => {}, | 
					
						
							| 
									
										
										
										
											2021-07-13 16:50:29 +08:00
										 |  |  | 				(compiler, setupResult, callback) => compiler.run(callback), | 
					
						
							| 
									
										
										
										
											2021-02-14 02:53:50 +08:00
										 |  |  | 				(err, stats) => { | 
					
						
							|  |  |  | 					this.running = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					if (callback !== undefined) { | 
					
						
							|  |  |  | 						return callback(err, stats); | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	purgeInputFileSystem() { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		for (const compiler of this.compilers) { | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 			if (compiler.inputFileSystem && compiler.inputFileSystem.purge) { | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 				compiler.inputFileSystem.purge(); | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-01-22 20:52:43 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {Callback<void>} callback signals when the compiler closes | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 	close(callback) { | 
					
						
							|  |  |  | 		asyncLib.each( | 
					
						
							|  |  |  | 			this.compilers, | 
					
						
							|  |  |  | 			(compiler, callback) => { | 
					
						
							|  |  |  | 				compiler.close(callback); | 
					
						
							|  |  |  | 			}, | 
					
						
							| 
									
										
										
										
											2024-02-21 22:05:29 +08:00
										 |  |  | 			error => { | 
					
						
							|  |  |  | 				callback(error); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-09-27 13:22:19 +08:00
										 |  |  | 		); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-06-12 04:52:02 +08:00
										 |  |  | }; |