12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178 |
- // @ts-check
- // Import types
- /** @typedef {import("./typings").HtmlTagObject} HtmlTagObject */
- /** @typedef {import("./typings").Options} HtmlWebpackOptions */
- /** @typedef {import("./typings").ProcessedOptions} ProcessedHtmlWebpackOptions */
- /** @typedef {import("./typings").TemplateParameter} TemplateParameter */
- /** @typedef {import("webpack/lib/Compiler.js")} WebpackCompiler */
- /** @typedef {import("webpack/lib/Compilation.js")} WebpackCompilation */
- 'use strict';
- const promisify = require('util').promisify;
- const vm = require('vm');
- const fs = require('fs');
- const _ = require('lodash');
- const path = require('path');
- const { CachedChildCompilation } = require('./lib/cached-child-compiler');
- const { createHtmlTagObject, htmlTagObjectToString, HtmlTagArray } = require('./lib/html-tags');
- const prettyError = require('./lib/errors.js');
- const chunkSorter = require('./lib/chunksorter.js');
- const getHtmlWebpackPluginHooks = require('./lib/hooks.js').getHtmlWebpackPluginHooks;
- const { assert } = require('console');
- const fsReadFileAsync = promisify(fs.readFile);
- class HtmlWebpackPlugin {
- /**
- * @param {HtmlWebpackOptions} [options]
- */
- constructor (options) {
- /** @type {HtmlWebpackOptions} */
- this.userOptions = options || {};
- this.version = HtmlWebpackPlugin.version;
- }
- apply (compiler) {
- // Wait for configuration preset plugions to apply all configure webpack defaults
- compiler.hooks.initialize.tap('HtmlWebpackPlugin', () => {
- const userOptions = this.userOptions;
- // Default options
- /** @type {ProcessedHtmlWebpackOptions} */
- const defaultOptions = {
- template: 'auto',
- templateContent: false,
- templateParameters: templateParametersGenerator,
- filename: 'index.html',
- publicPath: userOptions.publicPath === undefined ? 'auto' : userOptions.publicPath,
- hash: false,
- inject: userOptions.scriptLoading === 'blocking' ? 'body' : 'head',
- scriptLoading: 'defer',
- compile: true,
- favicon: false,
- minify: 'auto',
- cache: true,
- showErrors: true,
- chunks: 'all',
- excludeChunks: [],
- chunksSortMode: 'auto',
- meta: {},
- base: false,
- title: 'Webpack App',
- xhtml: false
- };
- /** @type {ProcessedHtmlWebpackOptions} */
- const options = Object.assign(defaultOptions, userOptions);
- this.options = options;
- // Assert correct option spelling
- assert(options.scriptLoading === 'defer' || options.scriptLoading === 'blocking' || options.scriptLoading === 'module', 'scriptLoading needs to be set to "defer", "blocking" or "module"');
- assert(options.inject === true || options.inject === false || options.inject === 'head' || options.inject === 'body', 'inject needs to be set to true, false, "head" or "body');
- // Default metaOptions if no template is provided
- if (!userOptions.template && options.templateContent === false && options.meta) {
- const defaultMeta = {
- // TODO remove in the next major release
- // From https://developer.mozilla.org/en-US/docs/Mozilla/Mobile/Viewport_meta_tag
- viewport: 'width=device-width, initial-scale=1'
- };
- options.meta = Object.assign({}, options.meta, defaultMeta, userOptions.meta);
- }
- // entryName to fileName conversion function
- const userOptionFilename = userOptions.filename || defaultOptions.filename;
- const filenameFunction = typeof userOptionFilename === 'function'
- ? userOptionFilename
- // Replace '[name]' with entry name
- : (entryName) => userOptionFilename.replace(/\[name\]/g, entryName);
- /** output filenames for the given entry names */
- const entryNames = Object.keys(compiler.options.entry);
- const outputFileNames = new Set((entryNames.length ? entryNames : ['main']).map(filenameFunction));
- /** Option for every entry point */
- const entryOptions = Array.from(outputFileNames).map((filename) => ({
- ...options,
- filename
- }));
- // Hook all options into the webpack compiler
- entryOptions.forEach((instanceOptions) => {
- hookIntoCompiler(compiler, instanceOptions, this);
- });
- });
- }
- /**
- * Once webpack is done with compiling the template into a NodeJS code this function
- * evaluates it to generate the html result
- *
- * The evaluateCompilationResult is only a class function to allow spying during testing.
- * Please change that in a further refactoring
- *
- * @param {string} source
- * @param {string} templateFilename
- * @returns {Promise<string | (() => string | Promise<string>)>}
- */
- evaluateCompilationResult (source, publicPath, templateFilename) {
- if (!source) {
- return Promise.reject(new Error('The child compilation didn\'t provide a result'));
- }
- // The LibraryTemplatePlugin stores the template result in a local variable.
- // By adding it to the end the value gets extracted during evaluation
- if (source.indexOf('HTML_WEBPACK_PLUGIN_RESULT') >= 0) {
- source += ';\nHTML_WEBPACK_PLUGIN_RESULT';
- }
- const templateWithoutLoaders = templateFilename.replace(/^.+!/, '').replace(/\?.+$/, '');
- const vmContext = vm.createContext({
- ...global,
- HTML_WEBPACK_PLUGIN: true,
- require: require,
- htmlWebpackPluginPublicPath: publicPath,
- __filename: templateWithoutLoaders,
- __dirname: path.dirname(templateWithoutLoaders),
- AbortController: global.AbortController,
- AbortSignal: global.AbortSignal,
- Blob: global.Blob,
- Buffer: global.Buffer,
- ByteLengthQueuingStrategy: global.ByteLengthQueuingStrategy,
- BroadcastChannel: global.BroadcastChannel,
- CompressionStream: global.CompressionStream,
- CountQueuingStrategy: global.CountQueuingStrategy,
- Crypto: global.Crypto,
- CryptoKey: global.CryptoKey,
- CustomEvent: global.CustomEvent,
- DecompressionStream: global.DecompressionStream,
- Event: global.Event,
- EventTarget: global.EventTarget,
- File: global.File,
- FormData: global.FormData,
- Headers: global.Headers,
- MessageChannel: global.MessageChannel,
- MessageEvent: global.MessageEvent,
- MessagePort: global.MessagePort,
- PerformanceEntry: global.PerformanceEntry,
- PerformanceMark: global.PerformanceMark,
- PerformanceMeasure: global.PerformanceMeasure,
- PerformanceObserver: global.PerformanceObserver,
- PerformanceObserverEntryList: global.PerformanceObserverEntryList,
- PerformanceResourceTiming: global.PerformanceResourceTiming,
- ReadableByteStreamController: global.ReadableByteStreamController,
- ReadableStream: global.ReadableStream,
- ReadableStreamBYOBReader: global.ReadableStreamBYOBReader,
- ReadableStreamBYOBRequest: global.ReadableStreamBYOBRequest,
- ReadableStreamDefaultController: global.ReadableStreamDefaultController,
- ReadableStreamDefaultReader: global.ReadableStreamDefaultReader,
- Response: global.Response,
- Request: global.Request,
- SubtleCrypto: global.SubtleCrypto,
- DOMException: global.DOMException,
- TextDecoder: global.TextDecoder,
- TextDecoderStream: global.TextDecoderStream,
- TextEncoder: global.TextEncoder,
- TextEncoderStream: global.TextEncoderStream,
- TransformStream: global.TransformStream,
- TransformStreamDefaultController: global.TransformStreamDefaultController,
- URL: global.URL,
- URLSearchParams: global.URLSearchParams,
- WebAssembly: global.WebAssembly,
- WritableStream: global.WritableStream,
- WritableStreamDefaultController: global.WritableStreamDefaultController,
- WritableStreamDefaultWriter: global.WritableStreamDefaultWriter
- });
- const vmScript = new vm.Script(source, { filename: templateWithoutLoaders });
- // Evaluate code and cast to string
- let newSource;
- try {
- newSource = vmScript.runInContext(vmContext);
- } catch (e) {
- return Promise.reject(e);
- }
- if (typeof newSource === 'object' && newSource.__esModule && newSource.default) {
- newSource = newSource.default;
- }
- return typeof newSource === 'string' || typeof newSource === 'function'
- ? Promise.resolve(newSource)
- : Promise.reject(new Error('The loader "' + templateWithoutLoaders + '" didn\'t return html.'));
- }
- }
- /**
- * connect the html-webpack-plugin to the webpack compiler lifecycle hooks
- *
- * @param {import('webpack').Compiler} compiler
- * @param {ProcessedHtmlWebpackOptions} options
- * @param {HtmlWebpackPlugin} plugin
- */
- function hookIntoCompiler (compiler, options, plugin) {
- const webpack = compiler.webpack;
- // Instance variables to keep caching information
- // for multiple builds
- let assetJson;
- /**
- * store the previous generated asset to emit them even if the content did not change
- * to support watch mode for third party plugins like the clean-webpack-plugin or the compression plugin
- * @type {Array<{html: string, name: string}>}
- */
- let previousEmittedAssets = [];
- options.template = getFullTemplatePath(options.template, compiler.context);
- // Inject child compiler plugin
- const childCompilerPlugin = new CachedChildCompilation(compiler);
- if (!options.templateContent) {
- childCompilerPlugin.addEntry(options.template);
- }
- // convert absolute filename into relative so that webpack can
- // generate it at correct location
- const filename = options.filename;
- if (path.resolve(filename) === path.normalize(filename)) {
- const outputPath = /** @type {string} - Once initialized the path is always a string */(compiler.options.output.path);
- options.filename = path.relative(outputPath, filename);
- }
- // Check if webpack is running in production mode
- // @see https://github.com/webpack/webpack/blob/3366421f1784c449f415cda5930a8e445086f688/lib/WebpackOptionsDefaulter.js#L12-L14
- const isProductionLikeMode = compiler.options.mode === 'production' || !compiler.options.mode;
- const minify = options.minify;
- if (minify === true || (minify === 'auto' && isProductionLikeMode)) {
- /** @type { import('html-minifier-terser').Options } */
- options.minify = {
- // https://www.npmjs.com/package/html-minifier-terser#options-quick-reference
- collapseWhitespace: true,
- keepClosingSlash: true,
- removeComments: true,
- removeRedundantAttributes: true,
- removeScriptTypeAttributes: true,
- removeStyleLinkTypeAttributes: true,
- useShortDoctype: true
- };
- }
- compiler.hooks.thisCompilation.tap('HtmlWebpackPlugin',
- /**
- * Hook into the webpack compilation
- * @param {WebpackCompilation} compilation
- */
- (compilation) => {
- compilation.hooks.processAssets.tapAsync(
- {
- name: 'HtmlWebpackPlugin',
- stage:
- /**
- * Generate the html after minification and dev tooling is done
- */
- webpack.Compilation.PROCESS_ASSETS_STAGE_OPTIMIZE_INLINE
- },
- /**
- * Hook into the process assets hook
- * @param {WebpackCompilation} compilationAssets
- * @param {(err?: Error) => void} callback
- */
- (compilationAssets, callback) => {
- // Get all entry point names for this html file
- const entryNames = Array.from(compilation.entrypoints.keys());
- const filteredEntryNames = filterChunks(entryNames, options.chunks, options.excludeChunks);
- const sortedEntryNames = sortEntryChunks(filteredEntryNames, options.chunksSortMode, compilation);
- const templateResult = options.templateContent
- ? { mainCompilationHash: compilation.hash }
- : childCompilerPlugin.getCompilationEntryResult(options.template);
- if ('error' in templateResult) {
- compilation.errors.push(prettyError(templateResult.error, compiler.context).toString());
- }
- // If the child compilation was not executed during a previous main compile run
- // it is a cached result
- const isCompilationCached = templateResult.mainCompilationHash !== compilation.hash;
- /** The public path used inside the html file */
- const htmlPublicPath = getPublicPath(compilation, options.filename, options.publicPath);
- /** Generated file paths from the entry point names */
- const assets = htmlWebpackPluginAssets(compilation, sortedEntryNames, htmlPublicPath);
- // If the template and the assets did not change we don't have to emit the html
- const newAssetJson = JSON.stringify(getAssetFiles(assets));
- if (isCompilationCached && options.cache && assetJson === newAssetJson) {
- previousEmittedAssets.forEach(({ name, html }) => {
- compilation.emitAsset(name, new webpack.sources.RawSource(html, false));
- });
- return callback();
- } else {
- previousEmittedAssets = [];
- assetJson = newAssetJson;
- }
- // The html-webpack plugin uses a object representation for the html-tags which will be injected
- // to allow altering them more easily
- // Just before they are converted a third-party-plugin author might change the order and content
- const assetsPromise = getFaviconPublicPath(options.favicon, compilation, assets.publicPath)
- .then((faviconPath) => {
- assets.favicon = faviconPath;
- return getHtmlWebpackPluginHooks(compilation).beforeAssetTagGeneration.promise({
- assets: assets,
- outputName: options.filename,
- plugin: plugin
- });
- });
- // Turn the js and css paths into grouped HtmlTagObjects
- const assetTagGroupsPromise = assetsPromise
- // And allow third-party-plugin authors to reorder and change the assetTags before they are grouped
- .then(({ assets }) => getHtmlWebpackPluginHooks(compilation).alterAssetTags.promise({
- assetTags: {
- scripts: generatedScriptTags(assets.js),
- styles: generateStyleTags(assets.css),
- meta: [
- ...generateBaseTag(options.base),
- ...generatedMetaTags(options.meta),
- ...generateFaviconTags(assets.favicon)
- ]
- },
- outputName: options.filename,
- publicPath: htmlPublicPath,
- plugin: plugin
- }))
- .then(({ assetTags }) => {
- // Inject scripts to body unless it set explicitly to head
- const scriptTarget = options.inject === 'head' ||
- (options.inject !== 'body' && options.scriptLoading !== 'blocking') ? 'head' : 'body';
- // Group assets to `head` and `body` tag arrays
- const assetGroups = generateAssetGroups(assetTags, scriptTarget);
- // Allow third-party-plugin authors to reorder and change the assetTags once they are grouped
- return getHtmlWebpackPluginHooks(compilation).alterAssetTagGroups.promise({
- headTags: assetGroups.headTags,
- bodyTags: assetGroups.bodyTags,
- outputName: options.filename,
- publicPath: htmlPublicPath,
- plugin: plugin
- });
- });
- // Turn the compiled template into a nodejs function or into a nodejs string
- const templateEvaluationPromise = Promise.resolve()
- .then(() => {
- if ('error' in templateResult) {
- return options.showErrors ? prettyError(templateResult.error, compiler.context).toHtml() : 'ERROR';
- }
- // Allow to use a custom function / string instead
- if (options.templateContent !== false) {
- return options.templateContent;
- }
- // Once everything is compiled evaluate the html factory
- // and replace it with its content
- return ('compiledEntry' in templateResult)
- ? plugin.evaluateCompilationResult(templateResult.compiledEntry.content, htmlPublicPath, options.template)
- : Promise.reject(new Error('Child compilation contained no compiledEntry'));
- });
- const templateExectutionPromise = Promise.all([assetsPromise, assetTagGroupsPromise, templateEvaluationPromise])
- // Execute the template
- .then(([assetsHookResult, assetTags, compilationResult]) => typeof compilationResult !== 'function'
- ? compilationResult
- : executeTemplate(compilationResult, assetsHookResult.assets, { headTags: assetTags.headTags, bodyTags: assetTags.bodyTags }, compilation));
- const injectedHtmlPromise = Promise.all([assetTagGroupsPromise, templateExectutionPromise])
- // Allow plugins to change the html before assets are injected
- .then(([assetTags, html]) => {
- const pluginArgs = { html, headTags: assetTags.headTags, bodyTags: assetTags.bodyTags, plugin: plugin, outputName: options.filename };
- return getHtmlWebpackPluginHooks(compilation).afterTemplateExecution.promise(pluginArgs);
- })
- .then(({ html, headTags, bodyTags }) => {
- return postProcessHtml(html, assets, { headTags, bodyTags });
- });
- const emitHtmlPromise = injectedHtmlPromise
- // Allow plugins to change the html after assets are injected
- .then((html) => {
- const pluginArgs = { html, plugin: plugin, outputName: options.filename };
- return getHtmlWebpackPluginHooks(compilation).beforeEmit.promise(pluginArgs)
- .then(result => result.html);
- })
- .catch(err => {
- // In case anything went wrong the promise is resolved
- // with the error message and an error is logged
- compilation.errors.push(prettyError(err, compiler.context).toString());
- return options.showErrors ? prettyError(err, compiler.context).toHtml() : 'ERROR';
- })
- .then(html => {
- const filename = options.filename.replace(/\[templatehash([^\]]*)\]/g, require('util').deprecate(
- (match, options) => `[contenthash${options}]`,
- '[templatehash] is now [contenthash]')
- );
- const replacedFilename = replacePlaceholdersInFilename(filename, html, compilation);
- // Add the evaluated html code to the webpack assets
- compilation.emitAsset(replacedFilename.path, new webpack.sources.RawSource(html, false), replacedFilename.info);
- previousEmittedAssets.push({ name: replacedFilename.path, html });
- return replacedFilename.path;
- })
- .then((finalOutputName) => getHtmlWebpackPluginHooks(compilation).afterEmit.promise({
- outputName: finalOutputName,
- plugin: plugin
- }).catch(err => {
- console.error(err);
- return null;
- }).then(() => null));
- // Once all files are added to the webpack compilation
- // let the webpack compiler continue
- emitHtmlPromise.then(() => {
- callback();
- });
- });
- });
- /**
- * Generate the template parameters for the template function
- * @param {WebpackCompilation} compilation
- * @param {{
- publicPath: string,
- js: Array<string>,
- css: Array<string>,
- manifest?: string,
- favicon?: string
- }} assets
- * @param {{
- headTags: HtmlTagObject[],
- bodyTags: HtmlTagObject[]
- }} assetTags
- * @returns {Promise<{[key: any]: any}>}
- */
- function getTemplateParameters (compilation, assets, assetTags) {
- const templateParameters = options.templateParameters;
- if (templateParameters === false) {
- return Promise.resolve({});
- }
- if (typeof templateParameters !== 'function' && typeof templateParameters !== 'object') {
- throw new Error('templateParameters has to be either a function or an object');
- }
- const templateParameterFunction = typeof templateParameters === 'function'
- // A custom function can overwrite the entire template parameter preparation
- ? templateParameters
- // If the template parameters is an object merge it with the default values
- : (compilation, assets, assetTags, options) => Object.assign({},
- templateParametersGenerator(compilation, assets, assetTags, options),
- templateParameters
- );
- const preparedAssetTags = {
- headTags: prepareAssetTagGroupForRendering(assetTags.headTags),
- bodyTags: prepareAssetTagGroupForRendering(assetTags.bodyTags)
- };
- return Promise
- .resolve()
- .then(() => templateParameterFunction(compilation, assets, preparedAssetTags, options));
- }
- /**
- * This function renders the actual html by executing the template function
- *
- * @param {(templateParameters) => string | Promise<string>} templateFunction
- * @param {{
- publicPath: string,
- js: Array<string>,
- css: Array<string>,
- manifest?: string,
- favicon?: string
- }} assets
- * @param {{
- headTags: HtmlTagObject[],
- bodyTags: HtmlTagObject[]
- }} assetTags
- * @param {WebpackCompilation} compilation
- *
- * @returns Promise<string>
- */
- function executeTemplate (templateFunction, assets, assetTags, compilation) {
- // Template processing
- const templateParamsPromise = getTemplateParameters(compilation, assets, assetTags);
- return templateParamsPromise.then((templateParams) => {
- try {
- // If html is a promise return the promise
- // If html is a string turn it into a promise
- return templateFunction(templateParams);
- } catch (e) {
- compilation.errors.push(new Error('Template execution failed: ' + e));
- return Promise.reject(e);
- }
- });
- }
- /**
- * Html Post processing
- *
- * @param {any} html
- * The input html
- * @param {any} assets
- * @param {{
- headTags: HtmlTagObject[],
- bodyTags: HtmlTagObject[]
- }} assetTags
- * The asset tags to inject
- *
- * @returns {Promise<string>}
- */
- function postProcessHtml (html, assets, assetTags) {
- if (typeof html !== 'string') {
- return Promise.reject(new Error('Expected html to be a string but got ' + JSON.stringify(html)));
- }
- const htmlAfterInjection = options.inject
- ? injectAssetsIntoHtml(html, assets, assetTags)
- : html;
- const htmlAfterMinification = minifyHtml(htmlAfterInjection);
- return Promise.resolve(htmlAfterMinification);
- }
- /*
- * Pushes the content of the given filename to the compilation assets
- * @param {string} filename
- * @param {WebpackCompilation} compilation
- *
- * @returns {string} file basename
- */
- function addFileToAssets (filename, compilation) {
- filename = path.resolve(compilation.compiler.context, filename);
- return fsReadFileAsync(filename)
- .then(source => new webpack.sources.RawSource(source, false))
- .catch(() => Promise.reject(new Error('HtmlWebpackPlugin: could not load file ' + filename)))
- .then(rawSource => {
- const basename = path.basename(filename);
- compilation.fileDependencies.add(filename);
- compilation.emitAsset(basename, rawSource);
- return basename;
- });
- }
- /**
- * Replace [contenthash] in filename
- *
- * @see https://survivejs.com/webpack/optimizing/adding-hashes-to-filenames/
- *
- * @param {string} filename
- * @param {string|Buffer} fileContent
- * @param {WebpackCompilation} compilation
- * @returns {{ path: string, info: {} }}
- */
- function replacePlaceholdersInFilename (filename, fileContent, compilation) {
- if (/\[\\*([\w:]+)\\*\]/i.test(filename) === false) {
- return { path: filename, info: {} };
- }
- const hash = compiler.webpack.util.createHash(compilation.outputOptions.hashFunction);
- hash.update(fileContent);
- if (compilation.outputOptions.hashSalt) {
- hash.update(compilation.outputOptions.hashSalt);
- }
- const contentHash = hash.digest(compilation.outputOptions.hashDigest).slice(0, compilation.outputOptions.hashDigestLength);
- return compilation.getPathWithInfo(
- filename,
- {
- contentHash,
- chunk: {
- hash: contentHash,
- contentHash
- }
- }
- );
- }
- /**
- * Helper to sort chunks
- * @param {string[]} entryNames
- * @param {string|((entryNameA: string, entryNameB: string) => number)} sortMode
- * @param {WebpackCompilation} compilation
- */
- function sortEntryChunks (entryNames, sortMode, compilation) {
- // Custom function
- if (typeof sortMode === 'function') {
- return entryNames.sort(sortMode);
- }
- // Check if the given sort mode is a valid chunkSorter sort mode
- if (typeof chunkSorter[sortMode] !== 'undefined') {
- return chunkSorter[sortMode](entryNames, compilation, options);
- }
- throw new Error('"' + sortMode + '" is not a valid chunk sort mode');
- }
- /**
- * Return all chunks from the compilation result which match the exclude and include filters
- * @param {any} chunks
- * @param {string[]|'all'} includedChunks
- * @param {string[]} excludedChunks
- */
- function filterChunks (chunks, includedChunks, excludedChunks) {
- return chunks.filter(chunkName => {
- // Skip if the chunks should be filtered and the given chunk was not added explicity
- if (Array.isArray(includedChunks) && includedChunks.indexOf(chunkName) === -1) {
- return false;
- }
- // Skip if the chunks should be filtered and the given chunk was excluded explicity
- if (Array.isArray(excludedChunks) && excludedChunks.indexOf(chunkName) !== -1) {
- return false;
- }
- // Add otherwise
- return true;
- });
- }
- /**
- * Generate the relative or absolute base url to reference images, css, and javascript files
- * from within the html file - the publicPath
- *
- * @param {WebpackCompilation} compilation
- * @param {string} childCompilationOutputName
- * @param {string | 'auto'} customPublicPath
- * @returns {string}
- */
- function getPublicPath (compilation, childCompilationOutputName, customPublicPath) {
- const compilationHash = compilation.hash;
- /**
- * @type {string} the configured public path to the asset root
- * if a path publicPath is set in the current webpack config use it otherwise
- * fallback to a relative path
- */
- const webpackPublicPath = compilation.getAssetPath(compilation.outputOptions.publicPath, { hash: compilationHash });
- // Webpack 5 introduced "auto" as default value
- const isPublicPathDefined = webpackPublicPath !== 'auto';
- let publicPath =
- // If the html-webpack-plugin options contain a custom public path uset it
- customPublicPath !== 'auto'
- ? customPublicPath
- : (isPublicPathDefined
- // If a hard coded public path exists use it
- ? webpackPublicPath
- // If no public path was set get a relative url path
- : path.relative(path.resolve(compilation.options.output.path, path.dirname(childCompilationOutputName)), compilation.options.output.path)
- .split(path.sep).join('/')
- );
- if (publicPath.length && publicPath.substr(-1, 1) !== '/') {
- publicPath += '/';
- }
- return publicPath;
- }
- /**
- * The htmlWebpackPluginAssets extracts the asset information of a webpack compilation
- * for all given entry names
- * @param {WebpackCompilation} compilation
- * @param {string[]} entryNames
- * @param {string | 'auto'} publicPath
- * @returns {{
- publicPath: string,
- js: Array<string>,
- css: Array<string>,
- manifest?: string,
- favicon?: string
- }}
- */
- function htmlWebpackPluginAssets (compilation, entryNames, publicPath) {
- const compilationHash = compilation.hash;
- /**
- * @type {{
- publicPath: string,
- js: Array<string>,
- css: Array<string>,
- manifest?: string,
- favicon?: string
- }}
- */
- const assets = {
- // The public path
- publicPath,
- // Will contain all js and mjs files
- js: [],
- // Will contain all css files
- css: [],
- // Will contain the html5 appcache manifest files if it exists
- manifest: Object.keys(compilation.assets).find(assetFile => path.extname(assetFile) === '.appcache'),
- // Favicon
- favicon: undefined
- };
- // Append a hash for cache busting
- if (options.hash && assets.manifest) {
- assets.manifest = appendHash(assets.manifest, compilationHash);
- }
- // Extract paths to .js, .mjs and .css files from the current compilation
- const entryPointPublicPathMap = {};
- const extensionRegexp = /\.(css|js|mjs)(\?|$)/;
- for (let i = 0; i < entryNames.length; i++) {
- const entryName = entryNames[i];
- /** entryPointUnfilteredFiles - also includes hot module update files */
- const entryPointUnfilteredFiles = compilation.entrypoints.get(entryName).getFiles();
- const entryPointFiles = entryPointUnfilteredFiles.filter((chunkFile) => {
- const asset = compilation.getAsset(chunkFile);
- if (!asset) {
- return true;
- }
- // Prevent hot-module files from being included:
- const assetMetaInformation = asset.info || {};
- return !(assetMetaInformation.hotModuleReplacement || assetMetaInformation.development);
- });
- // Prepend the publicPath and append the hash depending on the
- // webpack.output.publicPath and hashOptions
- // E.g. bundle.js -> /bundle.js?hash
- const entryPointPublicPaths = entryPointFiles
- .map(chunkFile => {
- const entryPointPublicPath = publicPath + urlencodePath(chunkFile);
- return options.hash
- ? appendHash(entryPointPublicPath, compilationHash)
- : entryPointPublicPath;
- });
- entryPointPublicPaths.forEach((entryPointPublicPath) => {
- const extMatch = extensionRegexp.exec(entryPointPublicPath);
- // Skip if the public path is not a .css, .mjs or .js file
- if (!extMatch) {
- return;
- }
- // Skip if this file is already known
- // (e.g. because of common chunk optimizations)
- if (entryPointPublicPathMap[entryPointPublicPath]) {
- return;
- }
- entryPointPublicPathMap[entryPointPublicPath] = true;
- // ext will contain .js or .css, because .mjs recognizes as .js
- const ext = extMatch[1] === 'mjs' ? 'js' : extMatch[1];
- assets[ext].push(entryPointPublicPath);
- });
- }
- return assets;
- }
- /**
- * Converts a favicon file from disk to a webpack resource
- * and returns the url to the resource
- *
- * @param {string|false} faviconFilePath
- * @param {WebpackCompilation} compilation
- * @param {string} publicPath
- * @returns {Promise<string|undefined>}
- */
- function getFaviconPublicPath (faviconFilePath, compilation, publicPath) {
- if (!faviconFilePath) {
- return Promise.resolve(undefined);
- }
- return addFileToAssets(faviconFilePath, compilation)
- .then((faviconName) => {
- const faviconPath = publicPath + faviconName;
- if (options.hash) {
- return appendHash(faviconPath, compilation.hash);
- }
- return faviconPath;
- });
- }
- /**
- * Generate all tags script for the given file paths
- * @param {Array<string>} jsAssets
- * @returns {Array<HtmlTagObject>}
- */
- function generatedScriptTags (jsAssets) {
- return jsAssets.map(scriptAsset => ({
- tagName: 'script',
- voidTag: false,
- meta: { plugin: 'html-webpack-plugin' },
- attributes: {
- defer: options.scriptLoading === 'defer',
- type: options.scriptLoading === 'module' ? 'module' : undefined,
- src: scriptAsset
- }
- }));
- }
- /**
- * Generate all style tags for the given file paths
- * @param {Array<string>} cssAssets
- * @returns {Array<HtmlTagObject>}
- */
- function generateStyleTags (cssAssets) {
- return cssAssets.map(styleAsset => ({
- tagName: 'link',
- voidTag: true,
- meta: { plugin: 'html-webpack-plugin' },
- attributes: {
- href: styleAsset,
- rel: 'stylesheet'
- }
- }));
- }
- /**
- * Generate an optional base tag
- * @param { false
- | string
- | {[attributeName: string]: string} // attributes e.g. { href:"http://example.com/page.html" target:"_blank" }
- } baseOption
- * @returns {Array<HtmlTagObject>}
- */
- function generateBaseTag (baseOption) {
- if (baseOption === false) {
- return [];
- } else {
- return [{
- tagName: 'base',
- voidTag: true,
- meta: { plugin: 'html-webpack-plugin' },
- attributes: (typeof baseOption === 'string') ? {
- href: baseOption
- } : baseOption
- }];
- }
- }
- /**
- * Generate all meta tags for the given meta configuration
- * @param {false | {
- [name: string]:
- false // disabled
- | string // name content pair e.g. {viewport: 'width=device-width, initial-scale=1, shrink-to-fit=no'}`
- | {[attributeName: string]: string|boolean} // custom properties e.g. { name:"viewport" content:"width=500, initial-scale=1" }
- }} metaOptions
- * @returns {Array<HtmlTagObject>}
- */
- function generatedMetaTags (metaOptions) {
- if (metaOptions === false) {
- return [];
- }
- // Make tags self-closing in case of xhtml
- // Turn { "viewport" : "width=500, initial-scale=1" } into
- // [{ name:"viewport" content:"width=500, initial-scale=1" }]
- const metaTagAttributeObjects = Object.keys(metaOptions)
- .map((metaName) => {
- const metaTagContent = metaOptions[metaName];
- return (typeof metaTagContent === 'string') ? {
- name: metaName,
- content: metaTagContent
- } : metaTagContent;
- })
- .filter((attribute) => attribute !== false);
- // Turn [{ name:"viewport" content:"width=500, initial-scale=1" }] into
- // the html-webpack-plugin tag structure
- return metaTagAttributeObjects.map((metaTagAttributes) => {
- if (metaTagAttributes === false) {
- throw new Error('Invalid meta tag');
- }
- return {
- tagName: 'meta',
- voidTag: true,
- meta: { plugin: 'html-webpack-plugin' },
- attributes: metaTagAttributes
- };
- });
- }
- /**
- * Generate a favicon tag for the given file path
- * @param {string| undefined} faviconPath
- * @returns {Array<HtmlTagObject>}
- */
- function generateFaviconTags (faviconPath) {
- if (!faviconPath) {
- return [];
- }
- return [{
- tagName: 'link',
- voidTag: true,
- meta: { plugin: 'html-webpack-plugin' },
- attributes: {
- rel: 'icon',
- href: faviconPath
- }
- }];
- }
- /**
- * Group assets to head and bottom tags
- *
- * @param {{
- scripts: Array<HtmlTagObject>;
- styles: Array<HtmlTagObject>;
- meta: Array<HtmlTagObject>;
- }} assetTags
- * @param {"body" | "head"} scriptTarget
- * @returns {{
- headTags: Array<HtmlTagObject>;
- bodyTags: Array<HtmlTagObject>;
- }}
- */
- function generateAssetGroups (assetTags, scriptTarget) {
- /** @type {{ headTags: Array<HtmlTagObject>; bodyTags: Array<HtmlTagObject>; }} */
- const result = {
- headTags: [
- ...assetTags.meta,
- ...assetTags.styles
- ],
- bodyTags: []
- };
- // Add script tags to head or body depending on
- // the htmlPluginOptions
- if (scriptTarget === 'body') {
- result.bodyTags.push(...assetTags.scripts);
- } else {
- // If script loading is blocking add the scripts to the end of the head
- // If script loading is non-blocking add the scripts infront of the css files
- const insertPosition = options.scriptLoading === 'blocking' ? result.headTags.length : assetTags.meta.length;
- result.headTags.splice(insertPosition, 0, ...assetTags.scripts);
- }
- return result;
- }
- /**
- * Add toString methods for easier rendering
- * inside the template
- *
- * @param {Array<HtmlTagObject>} assetTagGroup
- * @returns {Array<HtmlTagObject>}
- */
- function prepareAssetTagGroupForRendering (assetTagGroup) {
- const xhtml = options.xhtml;
- return HtmlTagArray.from(assetTagGroup.map((assetTag) => {
- const copiedAssetTag = Object.assign({}, assetTag);
- copiedAssetTag.toString = function () {
- return htmlTagObjectToString(this, xhtml);
- };
- return copiedAssetTag;
- }));
- }
- /**
- * Injects the assets into the given html string
- *
- * @param {string} html
- * The input html
- * @param {any} assets
- * @param {{
- headTags: HtmlTagObject[],
- bodyTags: HtmlTagObject[]
- }} assetTags
- * The asset tags to inject
- *
- * @returns {string}
- */
- function injectAssetsIntoHtml (html, assets, assetTags) {
- const htmlRegExp = /(<html[^>]*>)/i;
- const headRegExp = /(<\/head\s*>)/i;
- const bodyRegExp = /(<\/body\s*>)/i;
- const metaViewportRegExp = /<meta[^>]+name=["']viewport["'][^>]*>/i;
- const body = assetTags.bodyTags.map((assetTagObject) => htmlTagObjectToString(assetTagObject, options.xhtml));
- const head = assetTags.headTags.filter((item) => {
- if (item.tagName === 'meta' && item.attributes && item.attributes.name === 'viewport' && metaViewportRegExp.test(html)) {
- return false;
- }
- return true;
- }).map((assetTagObject) => htmlTagObjectToString(assetTagObject, options.xhtml));
- if (body.length) {
- if (bodyRegExp.test(html)) {
- // Append assets to body element
- html = html.replace(bodyRegExp, match => body.join('') + match);
- } else {
- // Append scripts to the end of the file if no <body> element exists:
- html += body.join('');
- }
- }
- if (head.length) {
- // Create a head tag if none exists
- if (!headRegExp.test(html)) {
- if (!htmlRegExp.test(html)) {
- html = '<head></head>' + html;
- } else {
- html = html.replace(htmlRegExp, match => match + '<head></head>');
- }
- }
- // Append assets to head element
- html = html.replace(headRegExp, match => head.join('') + match);
- }
- // Inject manifest into the opening html tag
- if (assets.manifest) {
- html = html.replace(/(<html[^>]*)(>)/i, (match, start, end) => {
- // Append the manifest only if no manifest was specified
- if (/\smanifest\s*=/.test(match)) {
- return match;
- }
- return start + ' manifest="' + assets.manifest + '"' + end;
- });
- }
- return html;
- }
- /**
- * Appends a cache busting hash to the query string of the url
- * E.g. http://localhost:8080/ -> http://localhost:8080/?50c9096ba6183fd728eeb065a26ec175
- * @param {string} url
- * @param {string} hash
- */
- function appendHash (url, hash) {
- if (!url) {
- return url;
- }
- return url + (url.indexOf('?') === -1 ? '?' : '&') + hash;
- }
- /**
- * Encode each path component using `encodeURIComponent` as files can contain characters
- * which needs special encoding in URLs like `+ `.
- *
- * Valid filesystem characters which need to be encoded for urls:
- *
- * # pound, % percent, & ampersand, { left curly bracket, } right curly bracket,
- * \ back slash, < left angle bracket, > right angle bracket, * asterisk, ? question mark,
- * blank spaces, $ dollar sign, ! exclamation point, ' single quotes, " double quotes,
- * : colon, @ at sign, + plus sign, ` backtick, | pipe, = equal sign
- *
- * However the query string must not be encoded:
- *
- * fo:demonstration-path/very fancy+name.js?path=/home?value=abc&value=def#zzz
- * ^ ^ ^ ^ ^ ^ ^ ^^ ^ ^ ^ ^ ^
- * | | | | | | | || | | | | |
- * encoded | | encoded | | || | | | | |
- * ignored ignored ignored ignored ignored
- *
- * @param {string} filePath
- */
- function urlencodePath (filePath) {
- // People use the filepath in quite unexpected ways.
- // Try to extract the first querystring of the url:
- //
- // some+path/demo.html?value=abc?def
- //
- const queryStringStart = filePath.indexOf('?');
- const urlPath = queryStringStart === -1 ? filePath : filePath.substr(0, queryStringStart);
- const queryString = filePath.substr(urlPath.length);
- // Encode all parts except '/' which are not part of the querystring:
- const encodedUrlPath = urlPath.split('/').map(encodeURIComponent).join('/');
- return encodedUrlPath + queryString;
- }
- /**
- * Helper to return the absolute template path with a fallback loader
- * @param {string} template
- * The path to the template e.g. './index.html'
- * @param {string} context
- * The webpack base resolution path for relative paths e.g. process.cwd()
- */
- function getFullTemplatePath (template, context) {
- if (template === 'auto') {
- template = path.resolve(context, 'src/index.ejs');
- if (!fs.existsSync(template)) {
- template = path.join(__dirname, 'default_index.ejs');
- }
- }
- // If the template doesn't use a loader use the lodash template loader
- if (template.indexOf('!') === -1) {
- template = require.resolve('./lib/loader.js') + '!' + path.resolve(context, template);
- }
- // Resolve template path
- return template.replace(
- /([!])([^/\\][^!?]+|[^/\\!?])($|\?[^!?\n]+$)/,
- (match, prefix, filepath, postfix) => prefix + path.resolve(filepath) + postfix);
- }
- /**
- * Minify the given string using html-minifier-terser
- *
- * As this is a breaking change to html-webpack-plugin 3.x
- * provide an extended error message to explain how to get back
- * to the old behaviour
- *
- * @param {string} html
- */
- function minifyHtml (html) {
- if (typeof options.minify !== 'object') {
- return html;
- }
- try {
- return require('html-minifier-terser').minify(html, options.minify);
- } catch (e) {
- const isParseError = String(e.message).indexOf('Parse Error') === 0;
- if (isParseError) {
- e.message = 'html-webpack-plugin could not minify the generated output.\n' +
- 'In production mode the html minifcation is enabled by default.\n' +
- 'If you are not generating a valid html output please disable it manually.\n' +
- 'You can do so by adding the following setting to your HtmlWebpackPlugin config:\n|\n|' +
- ' minify: false\n|\n' +
- 'See https://github.com/jantimon/html-webpack-plugin#options for details.\n\n' +
- 'For parser dedicated bugs please create an issue here:\n' +
- 'https://danielruf.github.io/html-minifier-terser/' +
- '\n' + e.message;
- }
- throw e;
- }
- }
- /**
- * Helper to return a sorted unique array of all asset files out of the
- * asset object
- */
- function getAssetFiles (assets) {
- const files = _.uniq(Object.keys(assets).filter(assetType => assetType !== 'chunks' && assets[assetType]).reduce((files, assetType) => files.concat(assets[assetType]), []));
- files.sort();
- return files;
- }
- }
- /**
- * The default for options.templateParameter
- * Generate the template parameters
- *
- * Generate the template parameters for the template function
- * @param {WebpackCompilation} compilation
- * @param {{
- publicPath: string,
- js: Array<string>,
- css: Array<string>,
- manifest?: string,
- favicon?: string
- }} assets
- * @param {{
- headTags: HtmlTagObject[],
- bodyTags: HtmlTagObject[]
- }} assetTags
- * @param {ProcessedHtmlWebpackOptions} options
- * @returns {TemplateParameter}
- */
- function templateParametersGenerator (compilation, assets, assetTags, options) {
- return {
- compilation: compilation,
- webpackConfig: compilation.options,
- htmlWebpackPlugin: {
- tags: assetTags,
- files: assets,
- options: options
- }
- };
- }
- // Statics:
- /**
- * The major version number of this plugin
- */
- HtmlWebpackPlugin.version = 5;
- /**
- * A static helper to get the hooks for this plugin
- *
- * Usage: HtmlWebpackPlugin.getHooks(compilation).HOOK_NAME.tapAsync('YourPluginName', () => { ... });
- */
- HtmlWebpackPlugin.getHooks = getHtmlWebpackPluginHooks;
- HtmlWebpackPlugin.createHtmlTagObject = createHtmlTagObject;
- module.exports = HtmlWebpackPlugin;
|