| 
									
										
										
										
											2013-01-31 01:49:25 +08:00
										 |  |  | /* | 
					
						
							|  |  |  | 	MIT License http://www.opensource.org/licenses/mit-license.php
 | 
					
						
							|  |  |  | 	Author Tobias Koppers @sokra | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | "use strict"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const asyncLib = require("async"); | 
					
						
							|  |  |  | const path = require("path"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | const Tapable = require("tapable").Tapable; | 
					
						
							|  |  |  | const AsyncSeriesWaterfallHook = require("tapable").AsyncSeriesWaterfallHook; | 
					
						
							|  |  |  | const SyncWaterfallHook = require("tapable").SyncWaterfallHook; | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | const ContextModule = require("./ContextModule"); | 
					
						
							|  |  |  | const ContextElementDependency = require("./dependencies/ContextElementDependency"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-17 21:26:23 +08:00
										 |  |  | const EMPTY_RESOLVE_OPTIONS = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | module.exports = class ContextModuleFactory extends Tapable { | 
					
						
							| 
									
										
										
										
											2017-11-17 21:26:23 +08:00
										 |  |  | 	constructor(resolverFactory) { | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 		super(); | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 		this.hooks = { | 
					
						
							|  |  |  | 			beforeResolve: new AsyncSeriesWaterfallHook(["data"]), | 
					
						
							|  |  |  | 			afterResolve: new AsyncSeriesWaterfallHook(["data"]), | 
					
						
							|  |  |  | 			contextModuleFiles: new SyncWaterfallHook(["files"]), | 
					
						
							|  |  |  | 			alternatives: new AsyncSeriesWaterfallHook(["modules"]) | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 		this._pluginCompat.tap("ContextModuleFactory", options => { | 
					
						
							|  |  |  | 			switch(options.name) { | 
					
						
							|  |  |  | 				case "before-resolve": | 
					
						
							|  |  |  | 				case "after-resolve": | 
					
						
							|  |  |  | 				case "alternatives": | 
					
						
							|  |  |  | 					options.async = true; | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2017-11-17 21:26:23 +08:00
										 |  |  | 		this.resolverFactory = resolverFactory; | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	create(data, callback) { | 
					
						
							|  |  |  | 		const context = data.context; | 
					
						
							|  |  |  | 		const dependencies = data.dependencies; | 
					
						
							| 
									
										
										
										
											2017-11-17 21:26:23 +08:00
										 |  |  | 		const resolveOptions = data.resolveOptions; | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 		const dependency = dependencies[0]; | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 		this.hooks.beforeResolve.callAsync(Object.assign({ | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 			context: context, | 
					
						
							| 
									
										
										
										
											2017-11-17 21:26:23 +08:00
										 |  |  | 			dependencies: dependencies, | 
					
						
							|  |  |  | 			resolveOptions | 
					
						
							| 
									
										
										
										
											2017-10-16 18:52:04 +08:00
										 |  |  | 		}, dependency.options), (err, beforeResolveResult) => { | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 			if(err) return callback(err); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// Ignored
 | 
					
						
							| 
									
										
										
										
											2017-10-16 18:52:04 +08:00
										 |  |  | 			if(!beforeResolveResult) return callback(); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-16 18:52:04 +08:00
										 |  |  | 			const context = beforeResolveResult.context; | 
					
						
							|  |  |  | 			const request = beforeResolveResult.request; | 
					
						
							| 
									
										
										
										
											2017-11-17 21:26:23 +08:00
										 |  |  | 			const resolveOptions = beforeResolveResult.resolveOptions; | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			let loaders, resource, loadersPrefix = ""; | 
					
						
							|  |  |  | 			const idx = request.lastIndexOf("!"); | 
					
						
							|  |  |  | 			if(idx >= 0) { | 
					
						
							|  |  |  | 				loaders = request.substr(0, idx + 1); | 
					
						
							|  |  |  | 				let i; | 
					
						
							|  |  |  | 				for(i = 0; i < loaders.length && loaders[i] === "!"; i++) { | 
					
						
							|  |  |  | 					loadersPrefix += "!"; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				loaders = loaders.substr(i).replace(/!+$/, "").replace(/!!+/g, "!"); | 
					
						
							|  |  |  | 				if(loaders === "") loaders = []; | 
					
						
							|  |  |  | 				else loaders = loaders.split("!"); | 
					
						
							|  |  |  | 				resource = request.substr(idx + 1); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				loaders = []; | 
					
						
							|  |  |  | 				resource = request; | 
					
						
							| 
									
										
										
										
											2013-02-04 19:34:20 +08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-17 21:26:23 +08:00
										 |  |  | 			const contextResolver = this.resolverFactory.get("context", resolveOptions || EMPTY_RESOLVE_OPTIONS); | 
					
						
							|  |  |  | 			const loaderResolver = this.resolverFactory.get("loader", EMPTY_RESOLVE_OPTIONS); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			asyncLib.parallel([ | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 				callback => { | 
					
						
							| 
									
										
										
										
											2017-12-23 02:35:04 +08:00
										 |  |  | 					contextResolver.resolve({}, context, resource, {}, (err, result) => { | 
					
						
							| 
									
										
										
										
											2016-01-02 07:21:54 +08:00
										 |  |  | 						if(err) return callback(err); | 
					
						
							|  |  |  | 						callback(null, result); | 
					
						
							| 
									
										
										
										
											2017-01-11 17:51:58 +08:00
										 |  |  | 					}); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 				}, | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 				callback => { | 
					
						
							|  |  |  | 					asyncLib.map(loaders, (loader, callback) => { | 
					
						
							| 
									
										
										
										
											2017-12-23 02:35:04 +08:00
										 |  |  | 						loaderResolver.resolve({}, context, loader, {}, (err, result) => { | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 							if(err) return callback(err); | 
					
						
							|  |  |  | 							callback(null, result); | 
					
						
							|  |  |  | 						}); | 
					
						
							|  |  |  | 					}, callback); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			], (err, result) => { | 
					
						
							| 
									
										
										
										
											2013-01-31 01:49:25 +08:00
										 |  |  | 				if(err) return callback(err); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 				this.hooks.afterResolve.callAsync(Object.assign({ | 
					
						
							| 
									
										
										
										
											2017-10-16 18:52:04 +08:00
										 |  |  | 					addon: loadersPrefix + result[1].join("!") + (result[1].length > 0 ? "!" : ""), | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 					resource: result[0], | 
					
						
							| 
									
										
										
										
											2017-11-17 21:26:23 +08:00
										 |  |  | 					resolveDependencies: this.resolveDependencies.bind(this) | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 				}, beforeResolveResult), (err, result) => { | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 					if(err) return callback(err); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// Ignored
 | 
					
						
							|  |  |  | 					if(!result) return callback(); | 
					
						
							| 
									
										
										
										
											2013-11-06 01:19:04 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-16 18:52:04 +08:00
										 |  |  | 					return callback(null, new ContextModule(result.resolveDependencies, result)); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 				}); | 
					
						
							| 
									
										
										
										
											2016-01-02 07:21:54 +08:00
										 |  |  | 			}); | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-02-04 19:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 07:51:01 +08:00
										 |  |  | 	resolveDependencies(fs, options, callback) { | 
					
						
							| 
									
										
										
										
											2017-09-25 19:27:51 +08:00
										 |  |  | 		const cmf = this; | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 		let resource = options.resource; | 
					
						
							| 
									
										
										
										
											2017-11-16 14:20:50 +08:00
										 |  |  | 		let resourceQuery = options.resourceQuery; | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 		let recursive = options.recursive; | 
					
						
							|  |  |  | 		let regExp = options.regExp; | 
					
						
							|  |  |  | 		let include = options.include; | 
					
						
							|  |  |  | 		let exclude = options.exclude; | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 		if(!regExp || !resource) | 
					
						
							|  |  |  | 			return callback(null, []); | 
					
						
							| 
									
										
										
										
											2017-11-16 14:20:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 		const addDirectory = (directory, callback) => { | 
					
						
							| 
									
										
										
										
											2017-08-11 13:52:25 +08:00
										 |  |  | 			fs.readdir(directory, (err, files) => { | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 				if(err) return callback(err); | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 				files = cmf.hooks.contextModuleFiles.call(files); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 				if(!files || files.length === 0) return callback(null, []); | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 				asyncLib.map(files.filter(p => p.indexOf(".") !== 0), (seqment, callback) => { | 
					
						
							| 
									
										
										
										
											2013-01-31 01:49:25 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 					const subResource = path.join(directory, seqment); | 
					
						
							| 
									
										
										
										
											2013-01-31 01:49:25 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 					if(!exclude || !subResource.match(exclude)) { | 
					
						
							|  |  |  | 						fs.stat(subResource, (err, stat) => { | 
					
						
							|  |  |  | 							if(err) { | 
					
						
							| 
									
										
										
										
											2017-10-14 07:51:01 +08:00
										 |  |  | 								if(err.code === "ENOENT") { | 
					
						
							|  |  |  | 									// ENOENT is ok here because the file may have been deleted between
 | 
					
						
							|  |  |  | 									// the readdir and stat calls.
 | 
					
						
							|  |  |  | 									return callback(); | 
					
						
							|  |  |  | 								} else { | 
					
						
							|  |  |  | 									return callback(err); | 
					
						
							|  |  |  | 								} | 
					
						
							| 
									
										
										
										
											2017-10-11 16:52:48 +08:00
										 |  |  | 							} | 
					
						
							| 
									
										
										
										
											2013-01-31 01:49:25 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 							if(stat.isDirectory()) { | 
					
						
							| 
									
										
										
										
											2013-01-31 01:49:25 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 								if(!recursive) return callback(); | 
					
						
							|  |  |  | 								addDirectory.call(this, subResource, callback); | 
					
						
							| 
									
										
										
										
											2013-02-04 19:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 							} else if(stat.isFile() && (!include || subResource.match(include))) { | 
					
						
							| 
									
										
										
										
											2013-02-04 19:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 								const obj = { | 
					
						
							|  |  |  | 									context: resource, | 
					
						
							|  |  |  | 									request: "." + subResource.substr(resource.length).replace(/\\/g, "/") | 
					
						
							|  |  |  | 								}; | 
					
						
							| 
									
										
										
										
											2016-01-02 07:21:54 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 								this.hooks.alternatives.callAsync([obj], (err, alternatives) => { | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 									if(err) return callback(err); | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 									alternatives = alternatives.filter(obj => regExp.test(obj.request)).map(obj => { | 
					
						
							| 
									
										
										
										
											2017-11-16 14:20:50 +08:00
										 |  |  | 										const dep = new ContextElementDependency(obj.request + resourceQuery, obj.request); | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 										dep.optional = true; | 
					
						
							|  |  |  | 										return dep; | 
					
						
							|  |  |  | 									}); | 
					
						
							|  |  |  | 									callback(null, alternatives); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 								}); | 
					
						
							| 
									
										
										
										
											2013-02-04 19:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 							} else callback(); | 
					
						
							| 
									
										
										
										
											2013-01-31 01:49:25 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-14 05:31:15 +08:00
										 |  |  | 						}); | 
					
						
							|  |  |  | 					} else callback(); | 
					
						
							| 
									
										
										
										
											2017-08-11 13:52:25 +08:00
										 |  |  | 				}, (err, result) => { | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 					if(err) return callback(err); | 
					
						
							| 
									
										
										
										
											2013-02-04 19:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 					if(!result) return callback(null, []); | 
					
						
							| 
									
										
										
										
											2013-02-04 19:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 					callback(null, result.filter(Boolean).reduce((a, i) => a.concat(i), [])); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 				}); | 
					
						
							| 
									
										
										
										
											2017-08-11 13:52:25 +08:00
										 |  |  | 			}); | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 		}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		addDirectory(resource, callback); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-01-31 01:49:25 +08:00
										 |  |  | }; |