Initial implementation of the links-updater.ts changes file formatting - it doesn't preserve line breaks inside tags. This change update links-updater.ts to avoid unnessecary formatting change. Change-Id: Ifccf588c63eefe69782e059fa63f3d1b70f570b9
		
			
				
	
	
		
			134 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			134 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
/**
 | 
						|
 * @license
 | 
						|
 * Copyright (C) 2020 The Android Open Source Project
 | 
						|
 *
 | 
						|
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
 * you may not use this file except in compliance with the License.
 | 
						|
 * You may obtain a copy of the License at
 | 
						|
 *
 | 
						|
 * http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 *
 | 
						|
 * Unless required by applicable law or agreed to in writing, software
 | 
						|
 * distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
 * See the License for the specific language governing permissions and
 | 
						|
 * limitations under the License.
 | 
						|
 */
 | 
						|
 | 
						|
import * as fs from "fs";
 | 
						|
import RewritingStream from "parse5-html-rewriting-stream";
 | 
						|
import * as dom5 from "dom5";
 | 
						|
import {HtmlFileUtils, RedirectsResolver} from "./utils";
 | 
						|
import {Node} from 'dom5';
 | 
						|
import {readMultilineParamFile} from "../utils/command-line";
 | 
						|
import { fail } from "../utils/common";
 | 
						|
import {JSONRedirects} from "./redirects";
 | 
						|
 | 
						|
/** Update links in HTML file
 | 
						|
 * input_output_param_files - is a list of paths; each path is placed on a separate line
 | 
						|
 *   The first line is the path to a first input file (relative to process working directory)
 | 
						|
 *   The second line is the path to the output file  (relative to process working directory)
 | 
						|
 *   The next 2 lines describe the second file and so on.
 | 
						|
 * redirectFile.json describes how to update links (see {@link JSONRedirects} for exact format)
 | 
						|
 * Additionaly, update some test links (related to web-component-tester)
 | 
						|
 */
 | 
						|
 | 
						|
async function main() {
 | 
						|
  if (process.argv.length < 4) {
 | 
						|
    console.info("Usage:\n\tnode links_updater.js input_output_param_files redirectFile.json\n");
 | 
						|
    process.exit(1);
 | 
						|
  }
 | 
						|
 | 
						|
  const jsonRedirects: JSONRedirects = JSON.parse(fs.readFileSync(process.argv[3], {encoding: "utf-8"}));
 | 
						|
  const redirectsResolver = new RedirectsResolver(jsonRedirects.redirects);
 | 
						|
 | 
						|
  const input = readMultilineParamFile(process.argv[2]);
 | 
						|
  const updater = new HtmlFileUpdater(redirectsResolver);
 | 
						|
  for(let i = 0; i < input.length; i += 2) {
 | 
						|
    const srcFile = input[i];
 | 
						|
    const targetFile = input[i + 1];
 | 
						|
    await updater.updateFile(srcFile, targetFile);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
/** Update all links in HTML file based on redirects.
 | 
						|
 * Additionally, update references to web-component-tester */
 | 
						|
class HtmlFileUpdater {
 | 
						|
  private static readonly Predicates = {
 | 
						|
    isScriptWithSrcTag: (node: Node) => node.tagName === "script" && dom5.hasAttribute(node, "src"),
 | 
						|
 | 
						|
    isWebComponentTesterImport: (node: Node) => HtmlFileUpdater.Predicates.isScriptWithSrcTag(node) &&
 | 
						|
        dom5.getAttribute(node, "src")!.endsWith("/bower_components/web-component-tester/browser.js"),
 | 
						|
 | 
						|
    isHtmlImport: (node: Node) => node.tagName === "link" && dom5.getAttribute(node, "rel") === "import" &&
 | 
						|
        dom5.hasAttribute(node, "href")
 | 
						|
  };
 | 
						|
 | 
						|
  public constructor(private readonly redirectsResolver: RedirectsResolver) {
 | 
						|
  }
 | 
						|
 | 
						|
  public async updateFile(srcFile: string, targetFile: string) {
 | 
						|
    const html = fs.readFileSync(srcFile, "utf-8");
 | 
						|
    const readStream = fs.createReadStream(srcFile, {encoding: "utf-8"});
 | 
						|
    const rewriterOutput = srcFile === targetFile ? targetFile + ".tmp" : targetFile;
 | 
						|
    const writeStream = fs.createWriteStream(rewriterOutput, {encoding: "utf-8"});
 | 
						|
    const rewriter = new RewritingStream();
 | 
						|
    (rewriter as any).tokenizer.preprocessor.bufferWaterline = Infinity;
 | 
						|
    rewriter.on("startTag", (tag: any) => {
 | 
						|
      if (HtmlFileUpdater.Predicates.isWebComponentTesterImport(tag)) {
 | 
						|
        dom5.setAttribute(tag, "src", "/components/wct-browser-legacy/browser.js");
 | 
						|
      } else if (HtmlFileUpdater.Predicates.isHtmlImport(tag)) {
 | 
						|
        this.updateRefAttribute(tag, srcFile, "href");
 | 
						|
      } else if (HtmlFileUpdater.Predicates.isScriptWithSrcTag(tag)) {
 | 
						|
        this.updateRefAttribute(tag, srcFile, "src");
 | 
						|
      } else {
 | 
						|
        const location = tag.sourceCodeLocation;
 | 
						|
        const raw = html.substring(location.startOffset, location.endOffset);
 | 
						|
        rewriter.emitRaw(raw);
 | 
						|
        return;
 | 
						|
      }
 | 
						|
      rewriter.emitStartTag(tag);
 | 
						|
    });
 | 
						|
    return new Promise<void>((resolve, reject) => {
 | 
						|
      writeStream.on("close", () => {
 | 
						|
        writeStream.close();
 | 
						|
        if (rewriterOutput !== targetFile) {
 | 
						|
          fs.renameSync(rewriterOutput, targetFile);
 | 
						|
        }
 | 
						|
        resolve();
 | 
						|
      });
 | 
						|
      readStream.pipe(rewriter).pipe(writeStream);
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  private getResolvedPath(parentHtml: string, href: string) {
 | 
						|
    const originalPath = '/' + HtmlFileUtils.getPathRelativeToRoot(parentHtml, href);
 | 
						|
 | 
						|
    const resolvedInfo = this.redirectsResolver.resolve(originalPath, true);
 | 
						|
    if (!resolvedInfo.insideNodeModules && resolvedInfo.target === originalPath) {
 | 
						|
      return href;
 | 
						|
    }
 | 
						|
    if (resolvedInfo.insideNodeModules) {
 | 
						|
      return '/node_modules/' + resolvedInfo.target;
 | 
						|
    }
 | 
						|
    if (href.startsWith('/')) {
 | 
						|
      return resolvedInfo.target;
 | 
						|
    }
 | 
						|
    return HtmlFileUtils.getPathRelativeToRoot(parentHtml, resolvedInfo.target);
 | 
						|
  }
 | 
						|
 | 
						|
  private updateRefAttribute(node: Node, parentHtml: string, attributeName: string) {
 | 
						|
    const ref = dom5.getAttribute(node, attributeName);
 | 
						|
    if (!ref) {
 | 
						|
      fail(`Internal error - ${node} in ${parentHtml} doesn't have attribute ${attributeName}`);
 | 
						|
    }
 | 
						|
    const newRef = this.getResolvedPath(parentHtml, ref);
 | 
						|
    if (newRef === ref) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    dom5.setAttribute(node, attributeName, newRef);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
main();
 |