| 
									
										
										
										
											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"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-11 12:27:09 +08:00
										 |  |  | const asyncLib = require("neo-async"); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | const path = require("path"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-26 14:27:44 +08:00
										 |  |  | const { AsyncSeriesWaterfallHook, SyncWaterfallHook } = require("tapable"); | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | const ContextModule = require("./ContextModule"); | 
					
						
							|  |  |  | const ContextElementDependency = require("./dependencies/ContextElementDependency"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-05 13:20:24 +08:00
										 |  |  | /** @typedef {import("./Module")} Module */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-17 21:26:23 +08:00
										 |  |  | const EMPTY_RESOLVE_OPTIONS = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-26 14:27:44 +08:00
										 |  |  | module.exports = class ContextModuleFactory { | 
					
						
							| 
									
										
										
										
											2017-11-17 21:26:23 +08:00
										 |  |  | 	constructor(resolverFactory) { | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 		this.hooks = { | 
					
						
							| 
									
										
										
										
											2018-07-05 13:20:24 +08:00
										 |  |  | 			/** @type {AsyncSeriesWaterfallHook<TODO>} */ | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 			beforeResolve: new AsyncSeriesWaterfallHook(["data"]), | 
					
						
							| 
									
										
										
										
											2018-07-05 13:20:24 +08:00
										 |  |  | 			/** @type {AsyncSeriesWaterfallHook<TODO>} */ | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 			afterResolve: new AsyncSeriesWaterfallHook(["data"]), | 
					
						
							| 
									
										
										
										
											2018-07-05 13:20:24 +08:00
										 |  |  | 			/** @type {SyncWaterfallHook<string[]>} */ | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 			contextModuleFiles: new SyncWaterfallHook(["files"]), | 
					
						
							| 
									
										
										
										
											2018-07-05 13:20:24 +08:00
										 |  |  | 			/** @type {SyncWaterfallHook<TODO[]>} */ | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 			alternatives: new AsyncSeriesWaterfallHook(["modules"]) | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											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]; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		this.hooks.beforeResolve.callAsync( | 
					
						
							|  |  |  | 			Object.assign( | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					context: context, | 
					
						
							|  |  |  | 					dependencies: dependencies, | 
					
						
							|  |  |  | 					resolveOptions | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 				}, | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				dependency.options | 
					
						
							|  |  |  | 			), | 
					
						
							|  |  |  | 			(err, beforeResolveResult) => { | 
					
						
							|  |  |  | 				if (err) return callback(err); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// Ignored
 | 
					
						
							|  |  |  | 				if (!beforeResolveResult) return callback(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				const context = beforeResolveResult.context; | 
					
						
							|  |  |  | 				const request = beforeResolveResult.request; | 
					
						
							|  |  |  | 				const resolveOptions = beforeResolveResult.resolveOptions; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				let loaders, | 
					
						
							|  |  |  | 					resource, | 
					
						
							|  |  |  | 					loadersPrefix = ""; | 
					
						
							|  |  |  | 				const idx = request.lastIndexOf("!"); | 
					
						
							|  |  |  | 				if (idx >= 0) { | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 					let loadersRequest = request.substr(0, idx + 1); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					let i; | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 					for ( | 
					
						
							|  |  |  | 						i = 0; | 
					
						
							|  |  |  | 						i < loadersRequest.length && loadersRequest[i] === "!"; | 
					
						
							|  |  |  | 						i++ | 
					
						
							|  |  |  | 					) { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						loadersPrefix += "!"; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 					loadersRequest = loadersRequest | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 						.substr(i) | 
					
						
							|  |  |  | 						.replace(/!+$/, "") | 
					
						
							|  |  |  | 						.replace(/!!+/g, "!"); | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 					if (loadersRequest === "") { | 
					
						
							|  |  |  | 						loaders = []; | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						loaders = loadersRequest.split("!"); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					resource = request.substr(idx + 1); | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					loaders = []; | 
					
						
							|  |  |  | 					resource = request; | 
					
						
							| 
									
										
										
										
											2017-05-11 03:36:20 +08:00
										 |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				const contextResolver = this.resolverFactory.get( | 
					
						
							|  |  |  | 					"context", | 
					
						
							|  |  |  | 					resolveOptions || EMPTY_RESOLVE_OPTIONS | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 				const loaderResolver = this.resolverFactory.get( | 
					
						
							|  |  |  | 					"loader", | 
					
						
							|  |  |  | 					EMPTY_RESOLVE_OPTIONS | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				asyncLib.parallel( | 
					
						
							|  |  |  | 					[ | 
					
						
							|  |  |  | 						callback => { | 
					
						
							|  |  |  | 							contextResolver.resolve( | 
					
						
							|  |  |  | 								{}, | 
					
						
							|  |  |  | 								context, | 
					
						
							|  |  |  | 								resource, | 
					
						
							|  |  |  | 								{}, | 
					
						
							|  |  |  | 								(err, result) => { | 
					
						
							|  |  |  | 									if (err) return callback(err); | 
					
						
							|  |  |  | 									callback(null, result); | 
					
						
							|  |  |  | 								} | 
					
						
							|  |  |  | 							); | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						callback => { | 
					
						
							|  |  |  | 							asyncLib.map( | 
					
						
							|  |  |  | 								loaders, | 
					
						
							|  |  |  | 								(loader, callback) => { | 
					
						
							|  |  |  | 									loaderResolver.resolve( | 
					
						
							|  |  |  | 										{}, | 
					
						
							|  |  |  | 										context, | 
					
						
							|  |  |  | 										loader, | 
					
						
							|  |  |  | 										{}, | 
					
						
							|  |  |  | 										(err, result) => { | 
					
						
							|  |  |  | 											if (err) return callback(err); | 
					
						
							|  |  |  | 											callback(null, result); | 
					
						
							|  |  |  | 										} | 
					
						
							|  |  |  | 									); | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 								callback | 
					
						
							|  |  |  | 							); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					], | 
					
						
							|  |  |  | 					(err, result) => { | 
					
						
							|  |  |  | 						if (err) return callback(err); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						this.hooks.afterResolve.callAsync( | 
					
						
							|  |  |  | 							Object.assign( | 
					
						
							|  |  |  | 								{ | 
					
						
							|  |  |  | 									addon: | 
					
						
							|  |  |  | 										loadersPrefix + | 
					
						
							|  |  |  | 										result[1].join("!") + | 
					
						
							|  |  |  | 										(result[1].length > 0 ? "!" : ""), | 
					
						
							|  |  |  | 									resource: result[0], | 
					
						
							|  |  |  | 									resolveDependencies: this.resolveDependencies.bind(this) | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 								beforeResolveResult | 
					
						
							|  |  |  | 							), | 
					
						
							|  |  |  | 							(err, result) => { | 
					
						
							|  |  |  | 								if (err) return callback(err); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 								// Ignored
 | 
					
						
							|  |  |  | 								if (!result) return callback(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 								return callback( | 
					
						
							|  |  |  | 									null, | 
					
						
							|  |  |  | 									new ContextModule(result.resolveDependencies, result) | 
					
						
							|  |  |  | 								); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		); | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00: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) => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				if (err) return callback(err); | 
					
						
							| 
									
										
										
										
											2017-11-28 16:54:24 +08:00
										 |  |  | 				files = cmf.hooks.contextModuleFiles.call(files); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 				if (!files || files.length === 0) return callback(null, []); | 
					
						
							|  |  |  | 				asyncLib.map( | 
					
						
							|  |  |  | 					files.filter(p => p.indexOf(".") !== 0), | 
					
						
							| 
									
										
										
										
											2018-02-26 10:43:37 +08:00
										 |  |  | 					(segment, callback) => { | 
					
						
							|  |  |  | 						const subResource = path.join(directory, segment); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 						if (!exclude || !subResource.match(exclude)) { | 
					
						
							|  |  |  | 							fs.stat(subResource, (err, stat) => { | 
					
						
							|  |  |  | 								if (err) { | 
					
						
							|  |  |  | 									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-14 07:51:01 +08:00
										 |  |  | 								} | 
					
						
							| 
									
										
										
										
											2013-02-04 19:34:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 								if (stat.isDirectory()) { | 
					
						
							|  |  |  | 									if (!recursive) return callback(); | 
					
						
							|  |  |  | 									addDirectory.call(this, subResource, callback); | 
					
						
							|  |  |  | 								} else if ( | 
					
						
							|  |  |  | 									stat.isFile() && | 
					
						
							|  |  |  | 									(!include || subResource.match(include)) | 
					
						
							|  |  |  | 								) { | 
					
						
							|  |  |  | 									const obj = { | 
					
						
							|  |  |  | 										context: resource, | 
					
						
							|  |  |  | 										request: | 
					
						
							|  |  |  | 											"." + | 
					
						
							|  |  |  | 											subResource.substr(resource.length).replace(/\\/g, "/") | 
					
						
							|  |  |  | 									}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 									this.hooks.alternatives.callAsync( | 
					
						
							|  |  |  | 										[obj], | 
					
						
							|  |  |  | 										(err, alternatives) => { | 
					
						
							|  |  |  | 											if (err) return callback(err); | 
					
						
							|  |  |  | 											alternatives = alternatives | 
					
						
							|  |  |  | 												.filter(obj => regExp.test(obj.request)) | 
					
						
							|  |  |  | 												.map(obj => { | 
					
						
							|  |  |  | 													const dep = new ContextElementDependency( | 
					
						
							|  |  |  | 														obj.request + resourceQuery, | 
					
						
							|  |  |  | 														obj.request | 
					
						
							|  |  |  | 													); | 
					
						
							|  |  |  | 													dep.optional = true; | 
					
						
							|  |  |  | 													return dep; | 
					
						
							|  |  |  | 												}); | 
					
						
							|  |  |  | 											callback(null, alternatives); | 
					
						
							|  |  |  | 										} | 
					
						
							|  |  |  | 									); | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 								} else { | 
					
						
							|  |  |  | 									callback(); | 
					
						
							|  |  |  | 								} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 							}); | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 						} else { | 
					
						
							|  |  |  | 							callback(); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 					}, | 
					
						
							|  |  |  | 					(err, result) => { | 
					
						
							|  |  |  | 						if (err) return callback(err); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						if (!result) return callback(null, []); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						callback( | 
					
						
							|  |  |  | 							null, | 
					
						
							|  |  |  | 							result.filter(Boolean).reduce((a, i) => a.concat(i), []) | 
					
						
							|  |  |  | 						); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | }; |