123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344 |
- /*
- MIT License http://www.opensource.org/licenses/mit-license.php
- Author Tobias Koppers @sokra
- */
- var fs = require('fs');
- var ConcatSource = require("webpack-sources").ConcatSource;
- var async = require("async");
- var ExtractedModule = require("./ExtractedModule");
- var Chunk = require("webpack/lib/Chunk");
- var OrderUndefinedError = require("./OrderUndefinedError");
- var loaderUtils = require("loader-utils");
- var schemaTester = require('./schema/validator');
- var loaderSchema = require('./schema/loader-schema');
- var pluginSchema = require('./schema/plugin-schema.json');
- var NS = fs.realpathSync(__dirname);
- var nextId = 0;
- function ExtractTextPluginCompilation() {
- this.modulesByIdentifier = {};
- }
- ExtractTextPlugin.prototype.mergeNonInitialChunks = function(chunk, intoChunk, checkedChunks) {
- if(!intoChunk) {
- checkedChunks = [];
- chunk.chunks.forEach(function(c) {
- if(c.isInitial()) return;
- this.mergeNonInitialChunks(c, chunk, checkedChunks);
- }, this);
- } else if(checkedChunks.indexOf(chunk) < 0) {
- checkedChunks.push(chunk);
- chunk.modules.slice().forEach(function(module) {
- intoChunk.addModule(module);
- module.addChunk(intoChunk);
- });
- chunk.chunks.forEach(function(c) {
- if(c.isInitial()) return;
- this.mergeNonInitialChunks(c, intoChunk, checkedChunks);
- }, this);
- }
- };
- ExtractTextPluginCompilation.prototype.addModule = function(identifier, originalModule, source, additionalInformation, sourceMap, prevModules) {
- var m;
- if(!this.modulesByIdentifier[identifier]) {
- m = this.modulesByIdentifier[identifier] = new ExtractedModule(identifier, originalModule, source, sourceMap, additionalInformation, prevModules);
- } else {
- m = this.modulesByIdentifier[identifier];
- m.addPrevModules(prevModules);
- if(originalModule.index2 < m.getOriginalModule().index2) {
- m.setOriginalModule(originalModule);
- }
- }
- return m;
- };
- ExtractTextPluginCompilation.prototype.addResultToChunk = function(identifier, result, originalModule, extractedChunk) {
- if(!Array.isArray(result)) {
- result = [[identifier, result]];
- }
- var counterMap = {};
- var prevModules = [];
- result.forEach(function(item) {
- var c = counterMap[item[0]];
- var module = this.addModule.call(this, item[0] + (c || ""), originalModule, item[1], item[2], item[3], prevModules.slice());
- extractedChunk.addModule(module);
- module.addChunk(extractedChunk);
- counterMap[item[0]] = (c || 0) + 1;
- prevModules.push(module);
- }, this);
- };
- ExtractTextPlugin.prototype.renderExtractedChunk = function(chunk) {
- var source = new ConcatSource();
- chunk.modules.forEach(function(module) {
- var moduleSource = module.source();
- source.add(this.applyAdditionalInformation(moduleSource, module.additionalInformation));
- }, this);
- return source;
- };
- function isInvalidOrder(a, b) {
- var bBeforeA = a.getPrevModules().indexOf(b) >= 0;
- var aBeforeB = b.getPrevModules().indexOf(a) >= 0;
- return aBeforeB && bBeforeA;
- }
- function getOrder(a, b) {
- var aOrder = a.getOrder();
- var bOrder = b.getOrder();
- if(aOrder < bOrder) return -1;
- if(aOrder > bOrder) return 1;
- var aIndex = a.getOriginalModule().index2;
- var bIndex = b.getOriginalModule().index2;
- if(aIndex < bIndex) return -1;
- if(aIndex > bIndex) return 1;
- var bBeforeA = a.getPrevModules().indexOf(b) >= 0;
- var aBeforeB = b.getPrevModules().indexOf(a) >= 0;
- if(aBeforeB && !bBeforeA) return -1;
- if(!aBeforeB && bBeforeA) return 1;
- var ai = a.identifier();
- var bi = b.identifier();
- if(ai < bi) return -1;
- if(ai > bi) return 1;
- return 0;
- }
- function ExtractTextPlugin(options) {
- if(arguments.length > 1) {
- throw new Error("Breaking change: ExtractTextPlugin now only takes a single argument. Either an options " +
- "object *or* the name of the result file.\n" +
- "Example: if your old code looked like this:\n" +
- " new ExtractTextPlugin('css/[name].css', { disable: false, allChunks: true })\n\n" +
- "You would change it to:\n" +
- " new ExtractTextPlugin({ filename: 'css/[name].css', disable: false, allChunks: true })\n\n" +
- "The available options are:\n" +
- " filename: string\n" +
- " allChunks: boolean\n" +
- " disable: boolean\n");
- }
- if(isString(options)) {
- options = { filename: options };
- } else {
- schemaTester(pluginSchema, options);
- }
- this.filename = options.filename;
- this.id = options.id != null ? options.id : ++nextId;
- this.options = {};
- mergeOptions(this.options, options);
- delete this.options.filename;
- delete this.options.id;
- }
- module.exports = ExtractTextPlugin;
- function getLoaderObject(loader) {
- if (isString(loader)) {
- return {loader: loader};
- }
- return loader;
- }
- function mergeOptions(a, b) {
- if(!b) return a;
- Object.keys(b).forEach(function(key) {
- a[key] = b[key];
- });
- return a;
- }
- function isString(a) {
- return typeof a === "string";
- }
- function isFunction(a) {
- return isType('Function', a);
- }
- function isType(type, obj) {
- return Object.prototype.toString.call(obj) === '[object ' + type + ']';
- }
- ExtractTextPlugin.loader = function(options) {
- return { loader: require.resolve("./loader"), options: options };
- };
- ExtractTextPlugin.prototype.applyAdditionalInformation = function(source, info) {
- if(info) {
- return new ConcatSource(
- "@media " + info[0] + " {",
- source,
- "}"
- );
- }
- return source;
- };
- ExtractTextPlugin.prototype.loader = function(options) {
- return ExtractTextPlugin.loader(mergeOptions({id: this.id}, options));
- };
- ExtractTextPlugin.prototype.extract = function(options) {
- if(arguments.length > 1) {
- throw new Error("Breaking change: extract now only takes a single argument. Either an options " +
- "object *or* the loader(s).\n" +
- "Example: if your old code looked like this:\n" +
- " ExtractTextPlugin.extract('style-loader', 'css-loader')\n\n" +
- "You would change it to:\n" +
- " ExtractTextPlugin.extract({ fallback: 'style-loader', use: 'css-loader' })\n\n" +
- "The available options are:\n" +
- " use: string | object | loader[]\n" +
- " fallback: string | object | loader[]\n" +
- " publicPath: string\n");
- }
- if(options.fallbackLoader) {
- console.warn('fallbackLoader option has been deprecated - replace with "fallback"');
- }
- if(options.loader) {
- console.warn('loader option has been deprecated - replace with "use"');
- }
- if(Array.isArray(options) || isString(options) || typeof options.options === "object" || typeof options.query === 'object') {
- options = { loader: options };
- } else {
- schemaTester(loaderSchema, options);
- }
- var loader = options.use || options.loader;
- var before = options.fallback || options.fallbackLoader || [];
- if(isString(loader)) {
- loader = loader.split("!");
- }
- if(isString(before)) {
- before = before.split("!");
- } else if(!Array.isArray(before)) {
- before = [before];
- }
- options = mergeOptions({omit: before.length, remove: true}, options);
- delete options.loader;
- delete options.use;
- delete options.fallback;
- delete options.fallbackLoader;
- return [this.loader(options)]
- .concat(before, loader)
- .map(getLoaderObject);
- }
- ExtractTextPlugin.extract = ExtractTextPlugin.prototype.extract.bind(ExtractTextPlugin);
- ExtractTextPlugin.prototype.apply = function(compiler) {
- var options = this.options;
- compiler.plugin("this-compilation", function(compilation) {
- var extractCompilation = new ExtractTextPluginCompilation();
- compilation.plugin("normal-module-loader", function(loaderContext, module) {
- loaderContext[NS] = function(content, opt) {
- if(options.disable)
- return false;
- if(!Array.isArray(content) && content != null)
- throw new Error("Exported value was not extracted as an array: " + JSON.stringify(content));
- module[NS] = {
- content: content,
- options: opt || {}
- };
- return options.allChunks || module[NS + "/extract"]; // eslint-disable-line no-path-concat
- };
- });
- var filename = this.filename;
- var id = this.id;
- var extractedChunks, entryChunks, initialChunks;
- compilation.plugin("optimize-tree", function(chunks, modules, callback) {
- extractedChunks = chunks.map(function() {
- return new Chunk();
- });
- chunks.forEach(function(chunk, i) {
- var extractedChunk = extractedChunks[i];
- extractedChunk.index = i;
- extractedChunk.originalChunk = chunk;
- extractedChunk.name = chunk.name;
- extractedChunk.entrypoints = chunk.entrypoints;
- chunk.chunks.forEach(function(c) {
- extractedChunk.addChunk(extractedChunks[chunks.indexOf(c)]);
- });
- chunk.parents.forEach(function(c) {
- extractedChunk.addParent(extractedChunks[chunks.indexOf(c)]);
- });
- });
- async.forEach(chunks, function(chunk, callback) {
- var extractedChunk = extractedChunks[chunks.indexOf(chunk)];
- var shouldExtract = !!(options.allChunks || chunk.isInitial());
- async.forEach(chunk.modules.slice(), function(module, callback) {
- var meta = module[NS];
- if(meta && (!meta.options.id || meta.options.id === id)) {
- var wasExtracted = Array.isArray(meta.content);
- if(shouldExtract !== wasExtracted) {
- module[NS + "/extract"] = shouldExtract; // eslint-disable-line no-path-concat
- compilation.rebuildModule(module, function(err) {
- if(err) {
- compilation.errors.push(err);
- return callback();
- }
- meta = module[NS];
- if(!Array.isArray(meta.content)) {
- err = new Error(module.identifier() + " doesn't export content");
- compilation.errors.push(err);
- return callback();
- }
- if(meta.content)
- extractCompilation.addResultToChunk(module.identifier(), meta.content, module, extractedChunk);
- callback();
- });
- } else {
- if(meta.content)
- extractCompilation.addResultToChunk(module.identifier(), meta.content, module, extractedChunk);
- callback();
- }
- } else callback();
- }, function(err) {
- if(err) return callback(err);
- callback();
- });
- }, function(err) {
- if(err) return callback(err);
- extractedChunks.forEach(function(extractedChunk) {
- if(extractedChunk.isInitial())
- this.mergeNonInitialChunks(extractedChunk);
- }, this);
- extractedChunks.forEach(function(extractedChunk) {
- if(!extractedChunk.isInitial()) {
- extractedChunk.modules.slice().forEach(function(module) {
- extractedChunk.removeModule(module);
- });
- }
- });
- compilation.applyPlugins("optimize-extracted-chunks", extractedChunks);
- callback();
- }.bind(this));
- }.bind(this));
- compilation.plugin("additional-assets", function(callback) {
- extractedChunks.forEach(function(extractedChunk) {
- if(extractedChunk.modules.length) {
- extractedChunk.modules.sort(function(a, b) {
- if(!options.ignoreOrder && isInvalidOrder(a, b)) {
- compilation.errors.push(new OrderUndefinedError(a.getOriginalModule()));
- compilation.errors.push(new OrderUndefinedError(b.getOriginalModule()));
- }
- return getOrder(a, b);
- });
- var chunk = extractedChunk.originalChunk;
- var source = this.renderExtractedChunk(extractedChunk);
- var getPath = (format) => compilation.getPath(format, {
- chunk: chunk
- }).replace(/\[(?:(\w+):)?contenthash(?::([a-z]+\d*))?(?::(\d+))?\]/ig, function() {
- return loaderUtils.getHashDigest(source.source(), arguments[1], arguments[2], parseInt(arguments[3], 10));
- });
- var file = (isFunction(filename)) ? filename(getPath) : getPath(filename);
-
- compilation.assets[file] = source;
- chunk.files.push(file);
- }
- }, this);
- callback();
- }.bind(this));
- }.bind(this));
- };
|