You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
271 lines
7.7 KiB
271 lines
7.7 KiB
var entries = require('object.entries');
|
|
var path = require('path');
|
|
var fse = require('fs-extra');
|
|
var _ = require('lodash');
|
|
|
|
const emitCountMap = new Map();
|
|
const compilerHookMap = new WeakMap();
|
|
|
|
const standardizeFilePaths = (file) => {
|
|
file.name = file.name.replace(/\\/g, '/');
|
|
file.path = file.path.replace(/\\/g, '/');
|
|
return file;
|
|
};
|
|
|
|
function ManifestPlugin(opts) {
|
|
this.opts = _.assign({
|
|
publicPath: null,
|
|
basePath: '',
|
|
fileName: 'manifest.json',
|
|
transformExtensions: /^(gz|map)$/i,
|
|
writeToFileEmit: false,
|
|
seed: null,
|
|
filter: null,
|
|
map: null,
|
|
generate: null,
|
|
sort: null,
|
|
serialize: function(manifest) {
|
|
return JSON.stringify(manifest, null, 2);
|
|
},
|
|
}, opts || {});
|
|
}
|
|
|
|
ManifestPlugin.getCompilerHooks = (compiler) => {
|
|
var hooks = compilerHookMap.get(compiler);
|
|
if (hooks === undefined) {
|
|
const SyncWaterfallHook = require('tapable').SyncWaterfallHook;
|
|
hooks = {
|
|
afterEmit: new SyncWaterfallHook(['manifest'])
|
|
};
|
|
compilerHookMap.set(compiler, hooks);
|
|
}
|
|
return hooks;
|
|
}
|
|
|
|
ManifestPlugin.prototype.getFileType = function(str) {
|
|
str = str.replace(/\?.*/, '');
|
|
var split = str.split('.');
|
|
var ext = split.pop();
|
|
if (this.opts.transformExtensions.test(ext)) {
|
|
ext = split.pop() + '.' + ext;
|
|
}
|
|
return ext;
|
|
};
|
|
|
|
ManifestPlugin.prototype.apply = function(compiler) {
|
|
var moduleAssets = {};
|
|
|
|
var outputFolder = compiler.options.output.path;
|
|
var outputFile = path.resolve(outputFolder, this.opts.fileName);
|
|
var outputName = path.relative(outputFolder, outputFile);
|
|
|
|
var moduleAsset = function (module, file) {
|
|
if (module.userRequest) {
|
|
moduleAssets[file] = path.join(
|
|
path.dirname(file),
|
|
path.basename(module.userRequest)
|
|
);
|
|
}
|
|
};
|
|
|
|
|
|
var emit = function(compilation, compileCallback) {
|
|
const emitCount = emitCountMap.get(outputFile) - 1
|
|
emitCountMap.set(outputFile, emitCount);
|
|
|
|
var seed = this.opts.seed || {};
|
|
|
|
var publicPath = this.opts.publicPath != null ? this.opts.publicPath : compilation.options.output.publicPath;
|
|
var stats = compilation.getStats().toJson({
|
|
// Disable data generation of everything we don't use
|
|
all: false,
|
|
// Add asset Information
|
|
assets: true,
|
|
// Show cached assets (setting this to `false` only shows emitted files)
|
|
cachedAssets: true,
|
|
});
|
|
|
|
var files = compilation.chunks.reduce(function(files, chunk) {
|
|
return chunk.files.reduce(function (files, path) {
|
|
var name = chunk.name ? chunk.name : null;
|
|
|
|
if (name) {
|
|
name = name + '.' + this.getFileType(path);
|
|
} else {
|
|
// For nameless chunks, just map the files directly.
|
|
name = path;
|
|
}
|
|
|
|
// Webpack 4: .isOnlyInitial()
|
|
// Webpack 3: .isInitial()
|
|
// Webpack 1/2: .initial
|
|
return files.concat({
|
|
path: path,
|
|
chunk: chunk,
|
|
name: name,
|
|
isInitial: chunk.isOnlyInitial ? chunk.isOnlyInitial() : (chunk.isInitial ? chunk.isInitial() : chunk.initial),
|
|
isChunk: true,
|
|
isAsset: false,
|
|
isModuleAsset: false
|
|
});
|
|
}.bind(this), files);
|
|
}.bind(this), []);
|
|
|
|
// module assets don't show up in assetsByChunkName.
|
|
// we're getting them this way;
|
|
files = stats.assets.reduce(function (files, asset) {
|
|
var name = moduleAssets[asset.name];
|
|
if (name) {
|
|
return files.concat({
|
|
path: asset.name,
|
|
name: name,
|
|
isInitial: false,
|
|
isChunk: false,
|
|
isAsset: true,
|
|
isModuleAsset: true
|
|
});
|
|
}
|
|
|
|
var isEntryAsset = asset.chunks.length > 0;
|
|
if (isEntryAsset) {
|
|
return files;
|
|
}
|
|
|
|
return files.concat({
|
|
path: asset.name,
|
|
name: asset.name,
|
|
isInitial: false,
|
|
isChunk: false,
|
|
isAsset: true,
|
|
isModuleAsset: false
|
|
});
|
|
}, files);
|
|
|
|
files = files.filter(function (file) {
|
|
// Don't add hot updates to manifest
|
|
var isUpdateChunk = file.path.indexOf('hot-update') >= 0;
|
|
// Don't add manifest from another instance
|
|
var isManifest = emitCountMap.get(path.join(outputFolder, file.name)) !== undefined;
|
|
|
|
return !isUpdateChunk && !isManifest;
|
|
});
|
|
|
|
// Append optional basepath onto all references.
|
|
// This allows output path to be reflected in the manifest.
|
|
if (this.opts.basePath) {
|
|
files = files.map(function(file) {
|
|
file.name = this.opts.basePath + file.name;
|
|
return file;
|
|
}.bind(this));
|
|
}
|
|
|
|
if (publicPath) {
|
|
// Similar to basePath but only affects the value (similar to how
|
|
// output.publicPath turns require('foo/bar') into '/public/foo/bar', see
|
|
// https://github.com/webpack/docs/wiki/configuration#outputpublicpath
|
|
files = files.map(function(file) {
|
|
file.path = publicPath + file.path;
|
|
return file;
|
|
}.bind(this));
|
|
}
|
|
|
|
files = files.map(standardizeFilePaths);
|
|
|
|
if (this.opts.filter) {
|
|
files = files.filter(this.opts.filter);
|
|
}
|
|
|
|
if (this.opts.map) {
|
|
files = files.map(this.opts.map).map(standardizeFilePaths);
|
|
}
|
|
|
|
if (this.opts.sort) {
|
|
files = files.sort(this.opts.sort);
|
|
}
|
|
|
|
var manifest;
|
|
if (this.opts.generate) {
|
|
const entrypointsArray = Array.from(
|
|
compilation.entrypoints instanceof Map ?
|
|
// Webpack 4+
|
|
compilation.entrypoints.entries() :
|
|
// Webpack 3
|
|
entries(compilation.entrypoints)
|
|
);
|
|
const entrypoints = entrypointsArray.reduce(
|
|
(e, [name, entrypoint]) => Object.assign(e, { [name]: entrypoint.getFiles() }),
|
|
{}
|
|
);
|
|
manifest = this.opts.generate(seed, files, entrypoints);
|
|
} else {
|
|
manifest = files.reduce(function (manifest, file) {
|
|
manifest[file.name] = file.path;
|
|
return manifest;
|
|
}, seed);
|
|
}
|
|
|
|
const isLastEmit = emitCount === 0
|
|
if (isLastEmit) {
|
|
var output = this.opts.serialize(manifest);
|
|
|
|
compilation.assets[outputName] = {
|
|
source: function() {
|
|
return output;
|
|
},
|
|
size: function() {
|
|
return output.length;
|
|
}
|
|
};
|
|
|
|
if (this.opts.writeToFileEmit) {
|
|
fse.outputFileSync(outputFile, output);
|
|
}
|
|
}
|
|
|
|
if (compiler.hooks) {
|
|
ManifestPlugin.getCompilerHooks(compiler).afterEmit.call(manifest);
|
|
} else {
|
|
compilation.applyPluginsAsync('webpack-manifest-plugin-after-emit', manifest, compileCallback);
|
|
}
|
|
}.bind(this);
|
|
|
|
function beforeRun (compiler, callback) {
|
|
let emitCount = emitCountMap.get(outputFile) || 0;
|
|
emitCountMap.set(outputFile, emitCount + 1);
|
|
|
|
if (callback) {
|
|
callback();
|
|
}
|
|
}
|
|
|
|
if (compiler.hooks) {
|
|
const pluginOptions = {
|
|
name: 'ManifestPlugin',
|
|
stage: Infinity
|
|
};
|
|
|
|
// Preserve exposure of custom hook in Webpack 4 for back compatability.
|
|
// Going forward, plugins should call `ManifestPlugin.getCompilerHooks(compiler)` directy.
|
|
if (!Object.isFrozen(compiler.hooks)) {
|
|
compiler.hooks.webpackManifestPluginAfterEmit = ManifestPlugin.getCompilerHooks(compiler).afterEmit;
|
|
}
|
|
|
|
compiler.hooks.compilation.tap(pluginOptions, function (compilation) {
|
|
compilation.hooks.moduleAsset.tap(pluginOptions, moduleAsset);
|
|
});
|
|
compiler.hooks.emit.tap(pluginOptions, emit);
|
|
|
|
compiler.hooks.run.tap(pluginOptions, beforeRun);
|
|
compiler.hooks.watchRun.tap(pluginOptions, beforeRun);
|
|
} else {
|
|
compiler.plugin('compilation', function (compilation) {
|
|
compilation.plugin('module-asset', moduleAsset);
|
|
});
|
|
compiler.plugin('emit', emit);
|
|
|
|
compiler.plugin('before-run', beforeRun);
|
|
compiler.plugin('watch-run', beforeRun);
|
|
}
|
|
};
|
|
|
|
module.exports = ManifestPlugin;
|