| 
									
										
										
										
											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"); | 
					
						
							| 
									
										
										
										
											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 */ | 
					
						
							| 
									
										
										
										
											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 */ | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 18:41:15 +08:00
										 |  |  | /** @typedef {number} CompilerStatus */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | const STATUS_PENDING = 0; | 
					
						
							|  |  |  | const STATUS_DONE = 1; | 
					
						
							|  |  |  | const STATUS_NEW = 2; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @template T | 
					
						
							|  |  |  |  * @callback Callback | 
					
						
							|  |  |  |  * @param {Error=} err | 
					
						
							|  |  |  |  * @param {T=} result | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @callback RunWithDependenciesHandler | 
					
						
							|  |  |  |  * @param {Compiler} compiler | 
					
						
							|  |  |  |  * @param {Callback<MultiStats>} callback | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	constructor(compilers) { | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 		if (!Array.isArray(compilers)) { | 
					
						
							|  |  |  | 			compilers = Object.keys(compilers).map(name => { | 
					
						
							|  |  |  | 				compilers[name].name = name; | 
					
						
							|  |  |  | 				return compilers[name]; | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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"]), | 
					
						
							| 
									
										
										
										
											2018-12-09 19:54:17 +08:00
										 |  |  | 			/** @type {MultiHook<SyncHook<[string, string]>>} */ | 
					
						
							| 
									
										
										
										
											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)), | 
					
						
							| 
									
										
										
										
											2019-08-05 18:15:03 +08:00
										 |  |  | 			/** @type {MultiHook<SyncBailHook<[string, string, any[]], true>>} */ | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/** @type {Stats[]} */ | 
					
						
							|  |  |  | 		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; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:15:37 +08:00
										 |  |  | 			// eslint-disable-next-line no-loop-func
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			compiler.hooks.done.tap("MultiCompiler", stats => { | 
					
						
							|  |  |  | 				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) { | 
					
						
							| 
									
										
										
										
											2017-11-28 17:18:46 +08:00
										 |  |  | 					this.hooks.done.call(new MultiStats(compilerStats)); | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			}); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:15:37 +08:00
										 |  |  | 			// eslint-disable-next-line no-loop-func
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			compiler.hooks.invalid.tap("MultiCompiler", () => { | 
					
						
							|  |  |  | 				if (compilerDone) { | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 					compilerDone = false; | 
					
						
							|  |  |  | 					doneCompilers--; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			}); | 
					
						
							| 
									
										
										
										
											2018-01-22 20:52:43 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-04-24 05:55:50 +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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	get outputFileSystem() { | 
					
						
							|  |  |  | 		throw new Error("Cannot read outputFileSystem of a MultiCompiler"); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-11 19:09:42 +08:00
										 |  |  | 	get intermediateFileSystem() { | 
					
						
							|  |  |  | 		throw new Error("Cannot read outputFileSystem of a MultiCompiler"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @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
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 = []; | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2017-09-17 21:38:19 +08:00
										 |  |  | 		const sortEdges = (e1, e2) => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			return ( | 
					
						
							|  |  |  | 				e1.source.name.localeCompare(e2.source.name) || | 
					
						
							|  |  |  | 				e1.target.name.localeCompare(e2.target.name) | 
					
						
							|  |  |  | 			); | 
					
						
							| 
									
										
										
										
											2017-09-17 21:38:19 +08:00
										 |  |  | 		}; | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  | 			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; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		const isDependencyFulfilled = d => fulfilledNames.has(d); | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		const getReadyCompilers = () => { | 
					
						
							|  |  |  | 			let readyCompilers = []; | 
					
						
							|  |  |  | 			let list = remainingCompilers; | 
					
						
							|  |  |  | 			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; | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		const runCompilers = callback => { | 
					
						
							|  |  |  | 			if (remainingCompilers.length === 0) return callback(); | 
					
						
							|  |  |  | 			asyncLib.map( | 
					
						
							|  |  |  | 				getReadyCompilers(), | 
					
						
							|  |  |  | 				(compiler, callback) => { | 
					
						
							|  |  |  | 					fn(compiler, err => { | 
					
						
							|  |  |  | 						if (err) return callback(err); | 
					
						
							|  |  |  | 						fulfilledNames.add(compiler.name); | 
					
						
							|  |  |  | 						runCompilers(callback); | 
					
						
							|  |  |  | 					}); | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				callback | 
					
						
							|  |  |  | 			); | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		}; | 
					
						
							|  |  |  | 		runCompilers(callback); | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 18:34:59 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {WatchOptions[]} watchOptions the watcher's options | 
					
						
							|  |  |  | 	 * @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()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-09 19:54:17 +08:00
										 |  |  | 		/** @type {Watching[]} */ | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 		const watchings = []; | 
					
						
							| 
									
										
										
										
											2018-12-09 19:54:17 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/** @type {Stats[]} */ | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 		const allStats = this.compilers.map(() => null); | 
					
						
							| 
									
										
										
										
											2018-12-10 18:41:15 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/** @type {CompilerStatus[]} */ | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 		const compilerStatus = this.compilers.map(() => STATUS_PENDING); | 
					
						
							| 
									
										
										
										
											2018-03-07 18:45:38 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-13 18:16:39 +08:00
										 |  |  | 		if (this.validateDependencies(handler)) { | 
					
						
							| 
									
										
										
										
											2018-03-07 18:45:38 +08:00
										 |  |  | 			this.running = true; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			this.runWithDependencies( | 
					
						
							|  |  |  | 				this.compilers, | 
					
						
							|  |  |  | 				(compiler, callback) => { | 
					
						
							|  |  |  | 					const compilerIdx = this.compilers.indexOf(compiler); | 
					
						
							|  |  |  | 					let firstRun = true; | 
					
						
							|  |  |  | 					let watching = compiler.watch( | 
					
						
							|  |  |  | 						Array.isArray(watchOptions) | 
					
						
							|  |  |  | 							? watchOptions[compilerIdx] | 
					
						
							|  |  |  | 							: watchOptions, | 
					
						
							|  |  |  | 						(err, stats) => { | 
					
						
							| 
									
										
										
										
											2018-03-13 18:16:39 +08:00
										 |  |  | 							if (err) handler(err); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 							if (stats) { | 
					
						
							|  |  |  | 								allStats[compilerIdx] = stats; | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 								compilerStatus[compilerIdx] = STATUS_NEW; | 
					
						
							|  |  |  | 								if (compilerStatus.every(status => status !== STATUS_PENDING)) { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 									const freshStats = allStats.filter((s, idx) => { | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 										return compilerStatus[idx] === STATUS_NEW; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 									}); | 
					
						
							| 
									
										
										
										
											2018-12-09 21:26:35 +08:00
										 |  |  | 									compilerStatus.fill(STATUS_DONE); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 									const multiStats = new MultiStats(freshStats); | 
					
						
							| 
									
										
										
										
											2018-03-13 18:16:39 +08:00
										 |  |  | 									handler(null, multiStats); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 								} | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							if (firstRun && !err) { | 
					
						
							|  |  |  | 								firstRun = false; | 
					
						
							|  |  |  | 								callback(); | 
					
						
							|  |  |  | 							} | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 						} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					); | 
					
						
							|  |  |  | 					watchings.push(watching); | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				() => { | 
					
						
							|  |  |  | 					// ignore
 | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		return new MultiWatching(watchings, this); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											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()); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-03-07 18:45:38 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		const finalCallback = (err, stats) => { | 
					
						
							|  |  |  | 			this.running = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 			if (callback !== undefined) { | 
					
						
							|  |  |  | 				return callback(err, stats); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-03-07 18:45:38 +08:00
										 |  |  | 		}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		const allStats = this.compilers.map(() => null); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (this.validateDependencies(callback)) { | 
					
						
							| 
									
										
										
										
											2018-03-07 18:45:38 +08:00
										 |  |  | 			this.running = true; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			this.runWithDependencies( | 
					
						
							|  |  |  | 				this.compilers, | 
					
						
							|  |  |  | 				(compiler, callback) => { | 
					
						
							|  |  |  | 					const compilerIdx = this.compilers.indexOf(compiler); | 
					
						
							|  |  |  | 					compiler.run((err, stats) => { | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 						if (err) { | 
					
						
							|  |  |  | 							return callback(err); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						allStats[compilerIdx] = stats; | 
					
						
							|  |  |  | 						callback(); | 
					
						
							|  |  |  | 					}); | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				err => { | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 					if (err) { | 
					
						
							|  |  |  | 						return finalCallback(err); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-03-07 18:45:38 +08:00
										 |  |  | 					finalCallback(null, new MultiStats(allStats)); | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			callback | 
					
						
							|  |  |  | 		); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-06-12 04:52:02 +08:00
										 |  |  | }; |