/* MIT License http://www.opensource.org/licenses/mit-license.php Author Tobias Koppers @sokra */ "use strict"; const asyncLib = require("async"); const path = require("path"); const Tapable = require("tapable"); const ContextModule = require("./ContextModule"); const ContextElementDependency = require("./dependencies/ContextElementDependency"); module.exports = class ContextModuleFactory extends Tapable { constructor(resolvers) { super(); this.resolvers = resolvers; } create(data, callback) { const context = data.context; const dependencies = data.dependencies; const dependency = dependencies[0]; this.applyPluginsAsyncWaterfall("before-resolve", { context: context, request: dependency.request, recursive: dependency.recursive, regExp: dependency.regExp, async: dependency.async, dependencies: dependencies }, (err, result) => { if(err) return callback(err); // Ignored if(!result) return callback(); const context = result.context; const request = result.request; const recursive = result.recursive; const regExp = result.regExp; const asyncContext = result.async; const dependencies = result.dependencies; 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; } const resolvers = this.resolvers; asyncLib.parallel([ function(callback) { resolvers.context.resolve({}, context, resource, function(err, result) { if(err) return callback(err); callback(null, result); }); }, function(callback) { asyncLib.map(loaders, function(loader, callback) { resolvers.loader.resolve({}, context, loader, function(err, result) { if(err) return callback(err); callback(null, result); }); }, callback); } ], (err, result) => { if(err) return callback(err); this.applyPluginsAsyncWaterfall("after-resolve", { loaders: loadersPrefix + result[1].join("!") + (result[1].length > 0 ? "!" : ""), resource: result[0], recursive: recursive, regExp: regExp, async: asyncContext, dependencies: dependencies, resolveDependencies: this.resolveDependencies.bind(this) }, function(err, result) { if(err) return callback(err); // Ignored if(!result) return callback(); return callback(null, new ContextModule(result.resolveDependencies, result.resource, result.recursive, result.regExp, result.loaders, result.async, dependency.chunkName)); }); }); }); } resolveDependencies(fs, resource, recursive, regExp, callback) { const cmf = this; if(!regExp || !resource) return callback(null, []); (function addDirectory(directory, callback) { fs.readdir(directory, (err, files) => { if(err) return callback(err); files = cmf.applyPluginsWaterfall("context-module-files", files); if(!files || files.length === 0) return callback(null, []); asyncLib.map(files.filter(function(p) { return p.indexOf(".") !== 0; }), (seqment, callback) => { const subResource = path.join(directory, seqment); 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); } } if(stat.isDirectory()) { if(!recursive) return callback(); addDirectory.call(this, subResource, callback); } else if(stat.isFile()) { const obj = { context: resource, request: "." + subResource.substr(resource.length).replace(/\\/g, "/") }; this.applyPluginsAsyncWaterfall("alternatives", [obj], (err, alternatives) => { if(err) return callback(err); alternatives = alternatives.filter(function(obj) { return regExp.test(obj.request); }).map(function(obj) { const dep = new ContextElementDependency(obj.request); dep.optional = true; return dep; }); callback(null, alternatives); }); } else callback(); }); }, (err, result) => { if(err) return callback(err); if(!result) return callback(null, []); callback(null, result.filter(function(i) { return !!i; }).reduce(function(a, i) { return a.concat(i); }, [])); }); }); }.call(this, resource, callback)); } };