@farris/build-angular
Version:
Angular Webpack Build Facade
196 lines • 30 kB
JavaScript
;
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
Object.defineProperty(exports, "__esModule", { value: true });
/**
* @license
* Copyright Google Inc. All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
const crypto_1 = require("crypto");
const webpack_sources_1 = require("webpack-sources");
const parse5 = require('parse5');
function readFile(filename, compilation) {
return new Promise((resolve, reject) => {
compilation.inputFileSystem.readFile(filename, (err, data) => {
if (err) {
reject(err);
return;
}
let content;
if (data.length >= 3 && data[0] === 0xEF && data[1] === 0xBB && data[2] === 0xBF) {
// Strip UTF-8 BOM
content = data.toString('utf8', 3);
}
else if (data.length >= 2 && data[0] === 0xFF && data[1] === 0xFE) {
// Strip UTF-16 LE BOM
content = data.toString('utf16le', 2);
}
else {
content = data.toString();
}
resolve(content);
});
});
}
class IndexHtmlWebpackPlugin {
constructor(options) {
this._options = Object.assign({ input: 'index.html', output: 'index.html', entrypoints: ['polyfills', 'main'], sri: false }, options);
}
apply(compiler) {
compiler.hooks.emit.tapPromise('index-html-webpack-plugin', (compilation) => __awaiter(this, void 0, void 0, function* () {
// Get input html file
const inputContent = yield readFile(this._options.input, compilation);
compilation
.fileDependencies.add(this._options.input);
// Get all files for selected entrypoints
let unfilteredSortedFiles = [];
for (const entryName of this._options.entrypoints) {
const entrypoint = compilation.entrypoints.get(entryName);
if (entrypoint && entrypoint.getFiles) {
unfilteredSortedFiles = unfilteredSortedFiles.concat(entrypoint.getFiles() || []);
}
}
// Filter files
const existingFiles = new Set();
const stylesheets = [];
const scripts = [];
for (const file of unfilteredSortedFiles) {
if (existingFiles.has(file)) {
continue;
}
existingFiles.add(file);
if (file.endsWith('.js')) {
scripts.push(file);
}
else if (file.endsWith('.css')) {
stylesheets.push(file);
}
}
// Find the head and body elements
const treeAdapter = parse5.treeAdapters.default;
const document = parse5.parse(inputContent, { treeAdapter, locationInfo: true });
let headElement;
let bodyElement;
for (const docChild of document.childNodes) {
if (docChild.tagName === 'html') {
for (const htmlChild of docChild.childNodes) {
if (htmlChild.tagName === 'head') {
headElement = htmlChild;
}
if (htmlChild.tagName === 'body') {
bodyElement = htmlChild;
}
}
}
}
if (!headElement || !bodyElement) {
throw new Error('Missing head and/or body elements');
}
// Determine script insertion point
let scriptInsertionPoint;
if (bodyElement.__location && bodyElement.__location.endTag) {
scriptInsertionPoint = bodyElement.__location.endTag.startOffset;
}
else {
// Less accurate fallback
// parse5 4.x does not provide locations if malformed html is present
scriptInsertionPoint = inputContent.indexOf('</body>');
}
let styleInsertionPoint;
if (headElement.__location && headElement.__location.endTag) {
styleInsertionPoint = headElement.__location.endTag.startOffset;
}
else {
// Less accurate fallback
// parse5 4.x does not provide locations if malformed html is present
styleInsertionPoint = inputContent.indexOf('</head>');
}
// Inject into the html
const indexSource = new webpack_sources_1.ReplaceSource(new webpack_sources_1.RawSource(inputContent), this._options.input);
const scriptElements = treeAdapter.createDocumentFragment();
for (const script of scripts) {
const attrs = [
{ name: 'type', value: 'text/javascript' },
{ name: 'src', value: (this._options.deployUrl || '') + script },
];
if (this._options.sri) {
const content = compilation.assets[script].source();
attrs.push(...this._generateSriAttributes(content));
}
const element = treeAdapter.createElement('script', undefined, attrs);
treeAdapter.appendChild(scriptElements, element);
}
indexSource.insert(scriptInsertionPoint, parse5.serialize(scriptElements, { treeAdapter }));
// Adjust base href if specified
if (typeof this._options.baseHref == 'string') {
let baseElement;
for (const headChild of headElement.childNodes) {
if (headChild.tagName === 'base') {
baseElement = headChild;
}
}
const baseFragment = treeAdapter.createDocumentFragment();
if (!baseElement) {
baseElement = treeAdapter.createElement('base', undefined, [
{ name: 'href', value: this._options.baseHref },
]);
treeAdapter.appendChild(baseFragment, baseElement);
indexSource.insert(headElement.__location.startTag.endOffset + 1, parse5.serialize(baseFragment, { treeAdapter }));
}
else {
let hrefAttribute;
for (const attribute of baseElement.attrs) {
if (attribute.name === 'href') {
hrefAttribute = attribute;
}
}
if (hrefAttribute) {
hrefAttribute.value = this._options.baseHref;
}
else {
baseElement.attrs.push({ name: 'href', value: this._options.baseHref });
}
treeAdapter.appendChild(baseFragment, baseElement);
indexSource.replace(baseElement.__location.startOffset, baseElement.__location.endOffset, parse5.serialize(baseFragment, { treeAdapter }));
}
}
const styleElements = treeAdapter.createDocumentFragment();
for (const stylesheet of stylesheets) {
const attrs = [
{ name: 'rel', value: 'stylesheet' },
{ name: 'href', value: (this._options.deployUrl || '') + stylesheet },
];
if (this._options.sri) {
const content = compilation.assets[stylesheet].source();
attrs.push(...this._generateSriAttributes(content));
}
const element = treeAdapter.createElement('link', undefined, attrs);
treeAdapter.appendChild(styleElements, element);
}
indexSource.insert(styleInsertionPoint, parse5.serialize(styleElements, { treeAdapter }));
// Add to compilation assets
compilation.assets[this._options.output] = indexSource;
}));
}
_generateSriAttributes(content) {
const algo = 'sha384';
const hash = crypto_1.createHash(algo)
.update(content, 'utf8')
.digest('base64');
return [
{ name: 'integrity', value: `${algo}-${hash}` },
{ name: 'crossorigin', value: 'anonymous' },
];
}
}
exports.IndexHtmlWebpackPlugin = IndexHtmlWebpackPlugin;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index-html-webpack-plugin.js","sourceRoot":"./","sources":["packages/farris_devkit/build_angular/src/angular-cli-files/plugins/index-html-webpack-plugin.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google Inc. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nimport { createHash } from 'crypto';\nimport { Compiler, compilation } from 'webpack';\nimport { RawSource, ReplaceSource } from 'webpack-sources';\n\nconst parse5 = require('parse5');\n\nexport interface IndexHtmlWebpackPluginOptions {\n  input: string;\n  output: string;\n  baseHref?: string;\n  entrypoints: string[];\n  deployUrl?: string;\n  sri: boolean;\n}\n\nfunction readFile(filename: string, compilation: compilation.Compilation): Promise<string> {\n  return new Promise<string>((resolve, reject) => {\n    compilation.inputFileSystem.readFile(filename, (err: Error, data: Buffer) => {\n      if (err) {\n        reject(err);\n\n        return;\n      }\n\n      let content;\n      if (data.length >= 3 && data[0] === 0xEF && data[1] === 0xBB && data[2] === 0xBF) {\n        // Strip UTF-8 BOM\n        content = data.toString('utf8', 3);\n      } else if (data.length >= 2 && data[0] === 0xFF && data[1] === 0xFE) {\n        // Strip UTF-16 LE BOM\n        content = data.toString('utf16le', 2);\n      } else {\n        content = data.toString();\n      }\n\n      resolve(content);\n    });\n  });\n}\n\nexport class IndexHtmlWebpackPlugin {\n  private _options: IndexHtmlWebpackPluginOptions;\n\n  constructor(options?: Partial<IndexHtmlWebpackPluginOptions>) {\n    this._options = {\n      input: 'index.html',\n      output: 'index.html',\n      entrypoints: ['polyfills', 'main'],\n      sri: false,\n      ...options,\n    };\n  }\n\n  apply(compiler: Compiler) {\n    compiler.hooks.emit.tapPromise('index-html-webpack-plugin', async compilation => {\n      // Get input html file\n      const inputContent = await readFile(this._options.input, compilation);\n      (compilation as compilation.Compilation & { fileDependencies: Set<string> })\n        .fileDependencies.add(this._options.input);\n\n\n      // Get all files for selected entrypoints\n      let unfilteredSortedFiles: string[] = [];\n      for (const entryName of this._options.entrypoints) {\n        const entrypoint = compilation.entrypoints.get(entryName);\n        if (entrypoint && entrypoint.getFiles) {\n          unfilteredSortedFiles = unfilteredSortedFiles.concat(entrypoint.getFiles() || []);\n        }\n      }\n\n      // Filter files\n      const existingFiles = new Set<string>();\n      const stylesheets: string[] = [];\n      const scripts: string[] = [];\n      for (const file of unfilteredSortedFiles) {\n        if (existingFiles.has(file)) {\n          continue;\n        }\n        existingFiles.add(file);\n\n        if (file.endsWith('.js')) {\n          scripts.push(file);\n        } else if (file.endsWith('.css')) {\n          stylesheets.push(file);\n        }\n\n      }\n\n      // Find the head and body elements\n      const treeAdapter = parse5.treeAdapters.default;\n      const document = parse5.parse(inputContent, { treeAdapter, locationInfo: true });\n      let headElement;\n      let bodyElement;\n      for (const docChild of document.childNodes) {\n        if (docChild.tagName === 'html') {\n          for (const htmlChild of docChild.childNodes) {\n            if (htmlChild.tagName === 'head') {\n              headElement = htmlChild;\n            }\n            if (htmlChild.tagName === 'body') {\n              bodyElement = htmlChild;\n            }\n          }\n        }\n      }\n\n      if (!headElement || !bodyElement) {\n        throw new Error('Missing head and/or body elements');\n      }\n\n      // Determine script insertion point\n      let scriptInsertionPoint;\n      if (bodyElement.__location && bodyElement.__location.endTag) {\n        scriptInsertionPoint = bodyElement.__location.endTag.startOffset;\n      } else {\n        // Less accurate fallback\n        // parse5 4.x does not provide locations if malformed html is present\n        scriptInsertionPoint = inputContent.indexOf('</body>');\n      }\n\n      let styleInsertionPoint;\n      if (headElement.__location && headElement.__location.endTag) {\n        styleInsertionPoint = headElement.__location.endTag.startOffset;\n      } else {\n        // Less accurate fallback\n        // parse5 4.x does not provide locations if malformed html is present\n        styleInsertionPoint = inputContent.indexOf('</head>');\n      }\n\n      // Inject into the html\n      const indexSource = new ReplaceSource(new RawSource(inputContent), this._options.input);\n\n      const scriptElements = treeAdapter.createDocumentFragment();\n      for (const script of scripts) {\n        const attrs = [\n          { name: 'type', value: 'text/javascript' },\n          { name: 'src', value: (this._options.deployUrl || '') + script },\n        ];\n\n        if (this._options.sri) {\n          const content = compilation.assets[script].source();\n          attrs.push(...this._generateSriAttributes(content));\n        }\n\n        const element = treeAdapter.createElement('script', undefined, attrs);\n        treeAdapter.appendChild(scriptElements, element);\n      }\n\n      indexSource.insert(\n        scriptInsertionPoint,\n        parse5.serialize(scriptElements, { treeAdapter }),\n      );\n\n      // Adjust base href if specified\n      if (typeof this._options.baseHref == 'string') {\n        let baseElement;\n        for (const headChild of headElement.childNodes) {\n          if (headChild.tagName === 'base') {\n            baseElement = headChild;\n          }\n        }\n\n        const baseFragment = treeAdapter.createDocumentFragment();\n\n        if (!baseElement) {\n          baseElement = treeAdapter.createElement(\n            'base',\n            undefined,\n            [\n              { name: 'href', value: this._options.baseHref },\n            ],\n          );\n\n          treeAdapter.appendChild(baseFragment, baseElement);\n          indexSource.insert(\n            headElement.__location.startTag.endOffset + 1,\n            parse5.serialize(baseFragment, { treeAdapter }),\n          );\n        } else {\n          let hrefAttribute;\n          for (const attribute of baseElement.attrs) {\n            if (attribute.name === 'href') {\n              hrefAttribute = attribute;\n            }\n          }\n          if (hrefAttribute) {\n            hrefAttribute.value = this._options.baseHref;\n          } else {\n            baseElement.attrs.push({ name: 'href', value: this._options.baseHref });\n          }\n\n          treeAdapter.appendChild(baseFragment, baseElement);\n          indexSource.replace(\n            baseElement.__location.startOffset,\n            baseElement.__location.endOffset,\n            parse5.serialize(baseFragment, { treeAdapter }),\n          );\n        }\n      }\n\n      const styleElements = treeAdapter.createDocumentFragment();\n      for (const stylesheet of stylesheets) {\n        const attrs = [\n          { name: 'rel', value: 'stylesheet' },\n          { name: 'href', value: (this._options.deployUrl || '') + stylesheet },\n        ];\n\n        if (this._options.sri) {\n          const content = compilation.assets[stylesheet].source();\n          attrs.push(...this._generateSriAttributes(content));\n        }\n\n        const element = treeAdapter.createElement('link', undefined, attrs);\n        treeAdapter.appendChild(styleElements, element);\n      }\n\n      indexSource.insert(\n        styleInsertionPoint,\n        parse5.serialize(styleElements, { treeAdapter }),\n      );\n\n      // Add to compilation assets\n      compilation.assets[this._options.output] = indexSource;\n    });\n  }\n\n  private _generateSriAttributes(content: string) {\n    const algo = 'sha384';\n    const hash = createHash(algo)\n      .update(content, 'utf8')\n      .digest('base64');\n\n    return [\n      { name: 'integrity', value: `${algo}-${hash}` },\n      { name: 'crossorigin', value: 'anonymous' },\n    ];\n  }\n}\n"]}