webpack/lib/NormalModuleFactory.js

278 lines
8.2 KiB
JavaScript
Raw Normal View History

2013-01-31 01:49:25 +08:00
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
var async = require("async");
var objectAssign = require('object-assign');
2013-01-31 01:49:25 +08:00
var Tapable = require("tapable");
var NormalModule = require("./NormalModule");
var RawModule = require("./RawModule");
var Parser = require("./Parser");
var RuleSet = require("./RuleSet");
function loaderToIdent(data) {
if(!data.options)
return data.loader;
if(typeof data.options === "string")
return data.loader + "?" + data.options;
if(typeof data.options !== "object")
throw new Error("loader options must be string or object");
if(data.options.ident)
return data.loader + "??" + data.options.ident;
return data.loader + "?" + JSON.stringify(data.options);
}
2013-01-31 01:49:25 +08:00
function identToLoaderRequest(resultString) {
var idx = resultString.indexOf("?");
var options;
if(idx >= 0) {
options = resultString.substr(idx + 1);
resultString = resultString.substr(0, idx);
return {
loader: resultString,
options: options
};
} else {
return {
loader: resultString
}
}
}
function NormalModuleFactory(context, resolvers, options) {
2013-01-31 01:49:25 +08:00
Tapable.call(this);
this.resolvers = resolvers;
this.ruleSet = new RuleSet(options.rules || options.loaders);
2013-02-04 20:59:43 +08:00
this.context = context || "";
this.parserCache = {};
this.plugin("factory", function() {
2015-11-11 06:31:03 +08:00
var _this = this;
return function(result, callback) {
2016-01-27 00:56:44 +08:00
var resolver = _this.applyPluginsWaterfall0("resolver", null);
// Ignored
if(!resolver) return callback();
resolver(result, function onDoneResolving(err, data) {
if(err) return callback(err);
// Ignored
if(!data) return callback();
// direct module
if(typeof data.source === "function")
return callback(null, data);
2015-11-11 06:31:03 +08:00
_this.applyPluginsAsyncWaterfall("after-resolve", data, function(err, result) {
if(err) return callback(err);
// Ignored
if(!result) return callback();
2015-11-11 06:31:03 +08:00
var createdModule = _this.applyPluginsBailResult("create-module", result);
if(!createdModule) {
2016-02-10 02:32:50 +08:00
if(!result.request) {
return callback(new Error("Empty dependency (no request)"));
}
createdModule = new NormalModule(
result.request,
result.userRequest,
result.rawRequest,
result.loaders,
result.resource,
result.parser
);
}
2015-05-17 00:27:59 +08:00
2016-01-27 00:56:44 +08:00
createdModule = _this.applyPluginsWaterfall0("module", createdModule);
2015-05-17 00:27:59 +08:00
return callback(null, createdModule);
2015-11-11 06:31:03 +08:00
});
});
};
});
this.plugin("resolver", function() {
2015-11-11 06:31:03 +08:00
var _this = this;
return function(data, callback) {
2016-01-26 01:28:56 +08:00
var contextInfo = data.contextInfo;
var context = data.context;
var request = data.request;
var noAutoLoaders = /^-?!/.test(request);
var noPrePostAutoLoaders = /^!!/.test(request);
var noPostAutoLoaders = /^-!/.test(request);
var elements = request.replace(/^-?!+/, "").replace(/!!+/g, "!").split("!");
var resource = elements.pop();
elements = elements.map(identToLoaderRequest);
async.parallel([
function(callback) {
2016-01-26 01:28:56 +08:00
_this.resolveRequestArray(contextInfo, context, elements, _this.resolvers.loader, callback);
2015-11-11 06:31:03 +08:00
},
function(callback) {
if(resource === "" || resource[0] === "?")
return callback(null, resource);
2016-01-26 01:28:56 +08:00
_this.resolvers.normal.resolve(contextInfo, context, resource, function(err, result) {
if(err) return callback(err);
callback(null, result);
});
2015-11-11 06:31:03 +08:00
}
], function(err, results) {
if(err) return callback(err);
var loaders = results[0];
resource = results[1];
// translate option idents
try {
loaders.forEach(function(item) {
if(typeof item.options === "string" && /^\?/.test(item.options)) {
item.options = _this.ruleSet.findOptionsByIdent(item.options.substr(1));
}
})
} catch(e) {
return callback(e);
}
if(resource === false)
return callback(null,
new RawModule("/* (ignored) */",
"ignored " + context + " " + request,
request + " (ignored)")); // ignored
var userRequest = loaders.map(loaderToIdent).concat([resource]).join("!");
var resourcePath = resource;
var resourceQuery = "";
var queryIndex = resourcePath.indexOf("?");
if(queryIndex >= 0) {
resourceQuery = resourcePath.substr(queryIndex);
resourcePath = resourcePath.substr(0, queryIndex);
}
var result = _this.ruleSet.exec({
resource: resourcePath,
resourceQuery: resourceQuery,
issuer: contextInfo.issuer
});
var settings = {};
2016-09-21 01:39:07 +08:00
var useLoadersPost = [];
var useLoaders = [];
var useLoadersPre = [];
result.forEach(function(r) {
if(r.type === "use") {
2016-09-21 01:39:07 +08:00
if(r.enforce === "post" && !noPostAutoLoaders && !noPrePostAutoLoaders)
useLoadersPost.push(r.value);
else if(r.enforce === "pre" && !noPrePostAutoLoaders)
useLoadersPre.push(r.value);
else if(!r.enforce && !noAutoLoaders && !noPrePostAutoLoaders)
useLoaders.push(r.value);
} else {
settings[r.type] = r.value;
}
});
async.parallel([
2016-09-21 01:39:07 +08:00
_this.resolveRequestArray.bind(_this, contextInfo, _this.context, useLoadersPost, _this.resolvers.loader),
_this.resolveRequestArray.bind(_this, contextInfo, _this.context, useLoaders, _this.resolvers.loader),
2016-09-21 01:39:07 +08:00
_this.resolveRequestArray.bind(_this, contextInfo, _this.context, useLoadersPre, _this.resolvers.loader)
], function(err, results) {
if(err) return callback(err);
loaders = results[0].concat(loaders).concat(results[1]).concat(results[2]);
onDoneResolving();
});
2015-07-13 06:20:09 +08:00
function onDoneResolving() {
callback(null, {
context: context,
request: loaders.map(loaderToIdent).concat([resource]).join("!"),
dependencies: data.dependencies,
userRequest: userRequest,
rawRequest: request,
loaders: loaders,
resource: resource,
parser: _this.getParser(settings.parser)
});
}
2015-11-11 06:31:03 +08:00
});
};
});
2013-01-31 01:49:25 +08:00
}
module.exports = NormalModuleFactory;
NormalModuleFactory.prototype = Object.create(Tapable.prototype);
NormalModuleFactory.prototype.constructor = NormalModuleFactory;
NormalModuleFactory.prototype.create = function(data, callback) {
2015-11-11 06:31:03 +08:00
var _this = this;
var context = data.context || this.context;
var dependencies = data.dependencies;
var request = dependencies[0].request;
var contextInfo = data.contextInfo || {};
2015-11-11 06:31:03 +08:00
_this.applyPluginsAsyncWaterfall("before-resolve", {
contextInfo: contextInfo,
2013-01-31 01:49:25 +08:00
context: context,
request: request,
dependencies: dependencies
2013-01-31 01:49:25 +08:00
}, function(err, result) {
if(err) return callback(err);
// Ignored
if(!result) return callback();
2016-01-27 00:56:44 +08:00
var factory = _this.applyPluginsWaterfall0("factory", null);
2013-01-31 01:49:25 +08:00
// Ignored
if(!factory) return callback();
factory(result, callback);
2015-11-11 06:31:03 +08:00
});
2013-01-31 01:49:25 +08:00
};
2016-01-26 01:28:56 +08:00
NormalModuleFactory.prototype.resolveRequestArray = function resolveRequestArray(contextInfo, context, array, resolver, callback) {
2013-01-31 01:49:25 +08:00
if(array.length === 0) return callback(null, []);
async.map(array, function(item, callback) {
resolver.resolve(contextInfo, context, item.loader, function(err, result) {
if(err && /^[^/]*$/.test(item.loader) && !/-loader$/.test(item.loader)) {
return resolver.resolve(contextInfo, context, item.loader + "-loader", function(err2) {
if(!err2) {
err.message = err.message + "\n" +
"BREAKING CHANGE: It's no longer allowed to omit the '-loader' suffix when using loaders.\n" +
" You need to specify '" + item.loader + "-loader' instead of '" + item.loader + "'.";
}
callback(err);
});
}
if(err) return callback(err);
2016-12-14 19:03:24 +08:00
var optionsOnly = item.options ? {
options: item.options
} : undefined;
return callback(null, objectAssign({}, item, identToLoaderRequest(result), optionsOnly));
});
2013-01-31 01:49:25 +08:00
}, callback);
};
NormalModuleFactory.prototype.getParser = function getParser(parserOptions) {
var ident = "null"
if(parserOptions) {
if(parserOptions.ident)
ident = parserOptions.ident;
else
ident = JSON.stringify(parserOptions);
}
var parser = this.parserCache[ident];
if(parser)
return parser;
return this.parserCache[ident] = this.createParser(parserOptions);
};
NormalModuleFactory.prototype.createParser = function createParser(parserOptions) {
var parser = new Parser();
this.applyPlugins("parser", parser, parserOptions || {});
return parser;
};