webpack/lib/BannerPlugin.js

106 lines
3.3 KiB
JavaScript
Raw Normal View History

2013-05-13 05:16:22 +08:00
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
"use strict";
const ConcatSource = require("webpack-sources").ConcatSource;
const ModuleFilenameHelpers = require("./ModuleFilenameHelpers");
2013-05-13 05:16:22 +08:00
function wrapComment(str) {
if(!str.includes("\n")) return `/*! ${str} */`;
return `/*!\n * ${str.split("\n").join("\n * ")}\n */`;
2013-05-13 05:16:22 +08:00
}
/*The following tokens are replaced in the name parameter:
*
*[ext] the extension of the resource
*[name] the basename of the resource
*[path] the path of the resource relative to the context query parameter or option.
*[folder] the folder of the resource is in.
*[emoji] a random emoji representation of options.content
*[emoji:<length>] same as above, but with a customizable number of emojis
*[hash] the hash of options.content (Buffer) (by default it's the hex digest of the md5 hash)
*[<hashType>:hash:<digestType>:<length>] optionally one can configure
*other hashTypes, i. e. sha1, md5, sha256, sha512
*other digestTypes, i. e. hex, base26, base32, base36, base49, base52, base58, base62, base64
*and length the length in chars
*[N] the N-th match obtained from matching the current file name against options.regExp
*/
2017-04-11 10:18:05 +08:00
const REGEXP_HASH = /\[hash:?(\d+)?\]/gi,
REGEXP_CHUNKHASH = /\[chunkhash(?::(\d+))?\]/gi,
REGEXP_NAME = /\[name\]/gi,
REGEXP_ID = /\[id\]/gi,
REGEXP_FILE = /\[file\]/gi,
REGEXP_QUERY = /\[query\]/gi;
// withHashLength, getReplacer from lib/TemplatedPathPlugin.js
const withHashLength = (replacer, handlerFn) => {
return function(_, hashLength) {
const length = hashLength && parseInt(hashLength, 10);
if(length && handlerFn) {
return handlerFn(length);
}
const hash = replacer.apply(this, arguments);
return length ? hash.slice(0, length) : hash;
};
};
const getReplacer = (value, allowEmpty) => {
return function(match) {
// last argument in replacer is the entire input string
const input = arguments[arguments.length - 1];
if(value === null || value === undefined) {
if(!allowEmpty) throw new Error(`Path variable ${match} not implemented in this context: ${input}`);
return "";
} else {
return `${value}`;
}
};
};
const interpolate = (banner, file, hash, chunkHash) => {
return banner
.replace(REGEXP_HASH, withHashLength(getReplacer(hash)))
}
class BannerPlugin {
constructor(options) {
if(arguments.length > 1)
throw new Error("BannerPlugin only takes one argument (pass an options object)");
if(typeof options === "string")
options = {
banner: options
};
this.options = options || {};
this.banner = this.options.raw ? options.banner : wrapComment(options.banner);
}
apply(compiler) {
2017-04-11 10:18:05 +08:00
const options = this.options,
banner = this.banner;
compiler.plugin("compilation", (compilation) => {
compilation.plugin("optimize-chunk-assets", (chunks, callback) => {
chunks.forEach((chunk) => {
if(options.entryOnly && !chunk.isInitial()) return;
chunk.files
.filter(ModuleFilenameHelpers.matchObject.bind(undefined, options))
.forEach((file) => {
2017-04-11 10:18:05 +08:00
const chunkHash = chunk.renderedHash || chunk.hash
const comment = interpolate(banner, file, compilation.hash, chunkHash)
return compilation.assets[file] = new ConcatSource(comment, "\n", compilation.assets[file])
});
2013-05-13 05:16:22 +08:00
});
callback();
2013-05-13 05:16:22 +08:00
});
});
}
}
module.exports = BannerPlugin;