| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | /* | 
					
						
							|  |  |  | 	MIT License http://www.opensource.org/licenses/mit-license.php
 | 
					
						
							|  |  |  | 	Author Tobias Koppers @sokra | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | "use strict"; | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-28 17:18:46 +08:00
										 |  |  | const Tapable = require("tapable").Tapable; | 
					
						
							|  |  |  | const SyncHook = require("tapable").SyncHook; | 
					
						
							| 
									
										
										
										
											2017-12-29 18:23:14 +08:00
										 |  |  | const MultiHook = require("tapable").MultiHook; | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | const asyncLib = require("async"); | 
					
						
							|  |  |  | const MultiWatching = require("./MultiWatching"); | 
					
						
							|  |  |  | const MultiStats = require("./MultiStats"); | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | module.exports = class MultiCompiler extends Tapable { | 
					
						
							|  |  |  | 	constructor(compilers) { | 
					
						
							|  |  |  | 		super(); | 
					
						
							| 
									
										
										
										
											2017-11-28 17:18:46 +08:00
										 |  |  | 		this.hooks = { | 
					
						
							|  |  |  | 			done: new SyncHook(["stats"]), | 
					
						
							| 
									
										
										
										
											2017-12-29 18:23:14 +08:00
										 |  |  | 			invalid: new MultiHook(compilers.map(c => c.hooks.invalid)), | 
					
						
							|  |  |  | 			run: new MultiHook(compilers.map(c => c.hooks.run)), | 
					
						
							|  |  |  | 			watchClose: new SyncHook([]), | 
					
						
							|  |  |  | 			watchRun: new MultiHook(compilers.map(c => c.hooks.watchRun)) | 
					
						
							| 
									
										
										
										
											2017-11-28 17:18:46 +08:00
										 |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		if(!Array.isArray(compilers)) { | 
					
						
							|  |  |  | 			compilers = Object.keys(compilers).map((name) => { | 
					
						
							|  |  |  | 				compilers[name].name = name; | 
					
						
							|  |  |  | 				return compilers[name]; | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		this.compilers = compilers; | 
					
						
							|  |  |  | 		let doneCompilers = 0; | 
					
						
							|  |  |  | 		let compilerStats = []; | 
					
						
							|  |  |  | 		this.compilers.forEach((compiler, idx) => { | 
					
						
							|  |  |  | 			let compilerDone = false; | 
					
						
							| 
									
										
										
										
											2017-12-14 04:35:39 +08:00
										 |  |  | 			compiler.hooks.done.tap("MultiCompiler", stats => { | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 				if(!compilerDone) { | 
					
						
							|  |  |  | 					compilerDone = true; | 
					
						
							|  |  |  | 					doneCompilers++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				compilerStats[idx] = stats; | 
					
						
							|  |  |  | 				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
										 |  |  | 				} | 
					
						
							|  |  |  | 			}); | 
					
						
							| 
									
										
										
										
											2017-12-14 04:35:39 +08:00
										 |  |  | 			compiler.hooks.invalid.tap("MultiCompiler", () => { | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 				if(compilerDone) { | 
					
						
							|  |  |  | 					compilerDone = false; | 
					
						
							|  |  |  | 					doneCompilers--; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			}); | 
					
						
							| 
									
										
										
										
											2017-11-09 03:49:41 +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; | 
					
						
							|  |  |  | 		for(const compiler of this.compilers) { | 
					
						
							| 
									
										
										
										
											2017-07-24 17:54:06 +08:00
										 |  |  | 			while(compiler.outputPath.indexOf(commonPath) !== 0 && /[/\\]/.test(commonPath)) { | 
					
						
							|  |  |  | 				commonPath = commonPath.replace(/[/\\][^/\\]*$/, ""); | 
					
						
							| 
									
										
										
										
											2014-06-19 05:02:33 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-06-12 04:26:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 		if(!commonPath && this.compilers[0].outputPath[0] === "/") return "/"; | 
					
						
							|  |  |  | 		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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	set inputFileSystem(value) { | 
					
						
							|  |  |  | 		this.compilers.forEach(compiler => { | 
					
						
							|  |  |  | 			compiler.inputFileSystem = value; | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	set outputFileSystem(value) { | 
					
						
							|  |  |  | 		this.compilers.forEach(compiler => { | 
					
						
							|  |  |  | 			compiler.outputFileSystem = value; | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 	validateDependencies(callback) { | 
					
						
							| 
									
										
										
										
											2017-09-17 22:59:35 +08:00
										 |  |  | 		const edges = new Set(); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 		const missing = []; | 
					
						
							| 
									
										
										
										
											2017-09-17 22:59:35 +08:00
										 |  |  | 		const targetFound = (compiler) => { | 
					
						
							|  |  |  | 			for(const edge of edges) { | 
					
						
							|  |  |  | 				if(edge.target === compiler) { | 
					
						
							|  |  |  | 					return true; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2017-09-17 21:38:19 +08:00
										 |  |  | 		const sortEdges = (e1, e2) => { | 
					
						
							|  |  |  | 			return e1.source.name.localeCompare(e2.source.name) || | 
					
						
							|  |  |  | 				e1.target.name.localeCompare(e2.target.name); | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 		for(const source of this.compilers) { | 
					
						
							|  |  |  | 			if(source.dependencies) { | 
					
						
							| 
									
										
										
										
											2017-09-09 00:46:47 +08:00
										 |  |  | 				for(const dep of source.dependencies) { | 
					
						
							|  |  |  | 					const target = this.compilers.find((c) => c.name === dep); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 					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
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		const errors = missing.map((m) => `Compiler dependency \`${m}\` not found.`); | 
					
						
							| 
									
										
										
										
											2017-09-17 22:59:35 +08:00
										 |  |  | 		const stack = this.compilers.filter((c) => !targetFound(c)); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 		while(stack.length > 0) { | 
					
						
							|  |  |  | 			const current = stack.pop(); | 
					
						
							| 
									
										
										
										
											2017-09-17 22:59:35 +08:00
										 |  |  | 			for(const edge of edges) { | 
					
						
							|  |  |  | 				if(edge.source === current) { | 
					
						
							|  |  |  | 					edges.delete(edge); | 
					
						
							|  |  |  | 					const target = edge.target; | 
					
						
							|  |  |  | 					if(!targetFound(target)) { | 
					
						
							|  |  |  | 						stack.push(target); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-09-17 22:59:35 +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
										 |  |  | 		} | 
					
						
							|  |  |  | 		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; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	runWithDependencies(compilers, fn, callback) { | 
					
						
							|  |  |  | 		let fulfilledNames = {}; | 
					
						
							|  |  |  | 		let remainingCompilers = compilers; | 
					
						
							|  |  |  | 		const isDependencyFulfilled = (d) => fulfilledNames[d]; | 
					
						
							|  |  |  | 		const getReadyCompilers = () => { | 
					
						
							|  |  |  | 			let readyCompilers = []; | 
					
						
							|  |  |  | 			let list = remainingCompilers; | 
					
						
							|  |  |  | 			remainingCompilers = []; | 
					
						
							|  |  |  | 			for(const c of list) { | 
					
						
							|  |  |  | 				const ready = !c.dependencies || c.dependencies.every(isDependencyFulfilled); | 
					
						
							|  |  |  | 				if(ready) | 
					
						
							|  |  |  | 					readyCompilers.push(c); | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					remainingCompilers.push(c); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return readyCompilers; | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 		const runCompilers = (callback) => { | 
					
						
							|  |  |  | 			if(remainingCompilers.length === 0) return callback(); | 
					
						
							|  |  |  | 			asyncLib.map(getReadyCompilers(), (compiler, callback) => { | 
					
						
							|  |  |  | 				fn(compiler, (err) => { | 
					
						
							|  |  |  | 					if(err) return callback(err); | 
					
						
							|  |  |  | 					fulfilledNames[compiler.name] = true; | 
					
						
							|  |  |  | 					runCompilers(callback); | 
					
						
							|  |  |  | 				}); | 
					
						
							|  |  |  | 			}, callback); | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 		runCompilers(callback); | 
					
						
							| 
									
										
										
										
											2016-09-14 18:04:42 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	watch(watchOptions, handler) { | 
					
						
							|  |  |  | 		let watchings = []; | 
					
						
							|  |  |  | 		let allStats = this.compilers.map(() => null); | 
					
						
							|  |  |  | 		let compilerStatus = this.compilers.map(() => false); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 		if(this.validateDependencies(handler)) { | 
					
						
							|  |  |  | 			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) => { | 
					
						
							|  |  |  | 					if(err) | 
					
						
							|  |  |  | 						handler(err); | 
					
						
							|  |  |  | 					if(stats) { | 
					
						
							|  |  |  | 						allStats[compilerIdx] = stats; | 
					
						
							|  |  |  | 						compilerStatus[compilerIdx] = "new"; | 
					
						
							|  |  |  | 						if(compilerStatus.every(Boolean)) { | 
					
						
							|  |  |  | 							const freshStats = allStats.filter((s, idx) => { | 
					
						
							|  |  |  | 								return compilerStatus[idx] === "new"; | 
					
						
							|  |  |  | 							}); | 
					
						
							|  |  |  | 							compilerStatus.fill(true); | 
					
						
							|  |  |  | 							const multiStats = new MultiStats(freshStats); | 
					
						
							|  |  |  | 							handler(null, multiStats); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 					if(firstRun && !err) { | 
					
						
							|  |  |  | 						firstRun = false; | 
					
						
							|  |  |  | 						callback(); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				}); | 
					
						
							|  |  |  | 				watchings.push(watching); | 
					
						
							|  |  |  | 			}, () => { | 
					
						
							|  |  |  | 				// ignore
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 			}); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 	run(callback) { | 
					
						
							|  |  |  | 		const allStats = this.compilers.map(() => null); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 		if(this.validateDependencies(callback)) { | 
					
						
							|  |  |  | 			this.runWithDependencies(this.compilers, ((compiler, callback) => { | 
					
						
							|  |  |  | 				const compilerIdx = this.compilers.indexOf(compiler); | 
					
						
							|  |  |  | 				compiler.run((err, stats) => { | 
					
						
							|  |  |  | 					if(err) return callback(err); | 
					
						
							|  |  |  | 					allStats[compilerIdx] = stats; | 
					
						
							|  |  |  | 					callback(); | 
					
						
							|  |  |  | 				}); | 
					
						
							|  |  |  | 			}), (err) => { | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +08:00
										 |  |  | 				if(err) return callback(err); | 
					
						
							| 
									
										
										
										
											2017-09-08 01:04:16 +08:00
										 |  |  | 				callback(null, new MultiStats(allStats)); | 
					
						
							| 
									
										
										
										
											2017-06-12 03:45:55 +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() { | 
					
						
							|  |  |  | 		this.compilers.forEach((compiler) => { | 
					
						
							|  |  |  | 			if(compiler.inputFileSystem && compiler.inputFileSystem.purge) | 
					
						
							|  |  |  | 				compiler.inputFileSystem.purge(); | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-06-12 04:52:02 +08:00
										 |  |  | }; |