The Great Blink mv for source files, part 2.

Move and rename files.

NOAUTOREVERT=true
NOPRESUBMIT=true
NOTREECHECKS=true
Bug: 768828
TBR=darin@chromium.org
NOTRY=true

Change-Id: I66d3b155808bc5bdbf237b80208e1e552bcf7f28
Reviewed-on: https://chromium-review.googlesource.com/1001153
Reviewed-by: Blink Reformat <blink-reformat@chromium.org>
Commit-Queue: Blink Reformat <blink-reformat@chromium.org>
Cr-Original-Commit-Position: refs/heads/master@{#549061}
Cr-Mirrored-From: https://chromium.googlesource.com/chromium/src
Cr-Mirrored-Commit: 0aee4434a4dba42a42abaea9bfbc0cd196a63bc1
diff --git a/scripts/npm_test.js b/scripts/npm_test.js
new file mode 100644
index 0000000..17e05a1
--- /dev/null
+++ b/scripts/npm_test.js
@@ -0,0 +1,281 @@
+// Copyright 2016 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+var childProcess = require('child_process');
+var fs = require('fs');
+var path = require('path');
+var shell = require('child_process').execSync;
+
+var utils = require('./utils');
+
+var Flags = {
+  DEBUG_DEVTOOLS: '--debug-devtools',
+  DEBUG_DEVTOOLS_SHORTHAND: '-d',
+  FETCH_CONTENT_SHELL: '--fetch-content-shell',
+  CHROMIUM_PATH: '--chromium-path',  // useful for bisecting
+  TARGET: '--target',                // build sub-directory (e.g. Release, Default)
+};
+
+var IS_DEBUG_ENABLED =
+    utils.includes(process.argv, Flags.DEBUG_DEVTOOLS) || utils.includes(process.argv, Flags.DEBUG_DEVTOOLS_SHORTHAND);
+var CUSTOM_CHROMIUM_PATH = utils.parseArgs(process.argv)[Flags.CHROMIUM_PATH];
+var IS_FETCH_CONTENT_SHELL = utils.includes(process.argv, Flags.FETCH_CONTENT_SHELL);
+var TARGET = utils.parseArgs(process.argv)[Flags.TARGET] || 'Release';
+
+var CONTENT_SHELL_ZIP = 'content-shell.zip';
+var MAX_CONTENT_SHELLS = 10;
+var PLATFORM = getPlatform();
+var PYTHON = process.platform === 'win32' ? 'python.bat' : 'python';
+
+var CHROMIUM_SRC_PATH = CUSTOM_CHROMIUM_PATH || path.resolve(__dirname, '..', '..', '..', '..', '..');
+var RELEASE_PATH = path.resolve(CHROMIUM_SRC_PATH, 'out', TARGET);
+var BLINK_TEST_PATH = path.resolve(CHROMIUM_SRC_PATH, 'blink', 'tools', 'run_layout_tests.py');
+var DEVTOOLS_PATH = path.resolve(CHROMIUM_SRC_PATH, 'third_party', 'WebKit', 'Source', 'devtools');
+var CACHE_PATH = path.resolve(DEVTOOLS_PATH, '.test_cache');
+var SOURCE_PATH = path.resolve(DEVTOOLS_PATH, 'front_end');
+
+function main() {
+  if (!utils.isDir(CACHE_PATH))
+    fs.mkdirSync(CACHE_PATH);
+  deleteOldContentShells();
+
+  var hasUserCompiledContentShell = utils.isFile(getContentShellBinaryPath(RELEASE_PATH));
+  if (!IS_FETCH_CONTENT_SHELL && hasUserCompiledContentShell) {
+    var outDir = path.resolve(RELEASE_PATH, '..');
+    if (!IS_DEBUG_ENABLED)
+      compileFrontend();
+
+    runTests(outDir, IS_DEBUG_ENABLED);
+    return;
+  }
+
+  findPreviousUploadedPosition(findMostRecentChromiumCommit()).then(onUploadedCommitPosition).catch(onError);
+
+  function onError(error) {
+    console.log('Unable to run tests because of error:', error);
+    console.log(`Try removing the .test_cache folder [${CACHE_PATH}] and retrying`);
+  }
+}
+main();
+
+function compileFrontend() {
+  console.log('Compiling devtools frontend');
+  try {
+    shell(`ninja -C ${RELEASE_PATH} devtools_frontend_resources`, {cwd: CHROMIUM_SRC_PATH});
+  } catch (err) {
+    console.log(err.stdout.toString());
+    console.log('ERROR: Cannot compile frontend\n' + err);
+    process.exit(1);
+  }
+}
+
+function onUploadedCommitPosition(commitPosition) {
+  var contentShellDirPath = path.resolve(CACHE_PATH, commitPosition, 'out', TARGET);
+  var contentShellResourcesPath = path.resolve(contentShellDirPath, 'resources');
+  var contentShellPath = path.resolve(CACHE_PATH, commitPosition, 'out');
+
+  var hasCachedContentShell = utils.isFile(getContentShellBinaryPath(contentShellDirPath));
+  if (hasCachedContentShell) {
+    console.log(`Using cached content shell at: ${contentShellPath}`);
+    copyFrontend(contentShellResourcesPath);
+    return runTests(contentShellPath, true);
+  }
+  var url = `http://commondatastorage.googleapis.com/chromium-browser-snapshots/${PLATFORM}/${commitPosition
+  }/${CONTENT_SHELL_ZIP}`;
+  return prepareContentShellDirectory(commitPosition)
+      .then(() => downloadContentShell(url, commitPosition))
+      .then(extractContentShell)
+      .then(() => copyFrontend(contentShellResourcesPath))
+      .then(() => runTests(contentShellPath, true));
+}
+
+function copyFrontend(contentShellResourcesPath) {
+  var devtoolsResourcesPath = path.resolve(contentShellResourcesPath, 'inspector');
+  var copiedFrontendPath = path.resolve(devtoolsResourcesPath, 'front_end');
+  var debugFrontendPath = path.resolve(devtoolsResourcesPath, 'debug');
+  var inspectorBackendCommandsPath = path.resolve(devtoolsResourcesPath, 'InspectorBackendCommands.js');
+  var supportedCSSPropertiesPath = path.resolve(devtoolsResourcesPath, 'SupportedCSSProperties.js');
+  utils.removeRecursive(copiedFrontendPath);
+  utils.removeRecursive(debugFrontendPath);
+  utils.copyRecursive(SOURCE_PATH, devtoolsResourcesPath);
+  fs.renameSync(copiedFrontendPath, debugFrontendPath);
+  utils.copy(inspectorBackendCommandsPath, debugFrontendPath);
+  utils.copy(supportedCSSPropertiesPath, debugFrontendPath);
+}
+
+function getPlatform() {
+  if (process.platform === 'linux') {
+    return 'Linux_x64';
+  }
+  if (process.platform === 'win32') {
+    return 'Win_x64';
+  }
+  if (process.platform === 'darwin')
+    return 'Mac';
+
+  throw new Error(`Unrecognized platform detected: ${process.platform}`);
+}
+
+function findMostRecentChromiumCommit() {
+  var commitMessage = shell(`git log --max-count=1 --grep="Cr-Commit-Position"`).toString().trim();
+  var commitPosition = commitMessage.match(/Cr-Commit-Position: refs\/heads\/master@\{#([0-9]+)\}/)[1];
+  return commitPosition;
+}
+
+function deleteOldContentShells() {
+  var files = fs.readdirSync(CACHE_PATH);
+  if (files.length < MAX_CONTENT_SHELLS)
+    return;
+  files.sort((a, b) => parseInt(b, 10) - parseInt(a, 10));
+  var remainingNumberOfContentShells = MAX_CONTENT_SHELLS / 2;
+  var oldContentShellDirs = files.slice(remainingNumberOfContentShells);
+  for (var i = 0; i < oldContentShellDirs.length; i++)
+    utils.removeRecursive(path.resolve(CACHE_PATH, oldContentShellDirs[i]));
+  console.log(`Removed old content shells: ${oldContentShellDirs}`);
+}
+
+function findPreviousUploadedPosition(commitPosition) {
+  var previousPosition = commitPosition - 100;
+  var positionsListURL =
+      `http://commondatastorage.googleapis.com/chromium-browser-snapshots/?delimiter=/&prefix=${PLATFORM
+  }/&marker=${PLATFORM}/${previousPosition}/`;
+  return utils.fetch(positionsListURL).then(onPositionsList).catch(onError);
+
+  function onPositionsList(buffer) {
+    var positions = buffer.toString('binary')
+                        .match(/([^<>]+)(?=<\/Prefix><\/CommonPrefixes>)/g)
+                        .map(prefixedPosition => prefixedPosition.split('/')[1])
+                        .map(positionString => parseInt(positionString, 10));
+    var positionSet = new Set(positions);
+    var previousUploadedPosition = commitPosition;
+    while (commitPosition - previousUploadedPosition < 100) {
+      if (positionSet.has(previousUploadedPosition))
+        return previousUploadedPosition.toString();
+      previousUploadedPosition--;
+    }
+    onError();
+  }
+
+  function onError(error) {
+    if (error)
+      console.log(`Received error: ${error} trying to fetch positions list from url: ${positionsListURL}`);
+    throw new Error(`Unable to find a previous upload position for commit position: ${commitPosition}`);
+  }
+}
+
+function prepareContentShellDirectory(folder) {
+  var contentShellPath = path.join(CACHE_PATH, folder);
+  if (utils.isDir(contentShellPath))
+    utils.removeRecursive(contentShellPath);
+  fs.mkdirSync(contentShellPath);
+  return Promise.resolve(folder);
+}
+
+function downloadContentShell(url, folder) {
+  console.log('Downloading content shell from:', url);
+  console.log('NOTE: Download is ~35-65 MB depending on OS');
+  return utils.fetch(url).then(writeZip).catch(onError);
+
+  function writeZip(buffer) {
+    console.log('Completed download of content shell');
+    var contentShellZipPath = path.join(CACHE_PATH, folder, CONTENT_SHELL_ZIP);
+    fs.writeFileSync(contentShellZipPath, buffer);
+    return contentShellZipPath;
+  }
+
+  function onError(error) {
+    console.log(`Received error: ${error} trying to download content shell from url: ${url}`);
+    throw new Error('Unable to download content shell');
+  }
+}
+
+function extractContentShell(contentShellZipPath) {
+  console.log(`Extracting content shell zip: ${contentShellZipPath}`);
+  var unzipScriptPath = path.resolve(__dirname, 'unzip.py');
+  var src = contentShellZipPath;
+  var dest = path.resolve(path.dirname(src), 'out');
+  shell(`${PYTHON} ${unzipScriptPath} ${src} ${dest}`);
+  fs.unlinkSync(src);
+  var originalDirPath = path.resolve(dest, 'content-shell');
+  var newDirPath = path.resolve(dest, TARGET);
+  fs.renameSync(originalDirPath, newDirPath);
+  fs.chmodSync(getContentShellBinaryPath(newDirPath), '755');
+  if (process.platform === 'darwin') {
+    var helperPath = path.resolve(
+        newDirPath, 'Content Shell.app', 'Contents', 'Frameworks', 'Content Shell Helper.app', 'Contents', 'MacOS',
+        'Content Shell Helper');
+    fs.chmodSync(helperPath, '755');
+  }
+  return dest;
+}
+
+function getContentShellBinaryPath(dirPath) {
+  if (process.platform === 'linux')
+    return path.resolve(dirPath, 'content_shell');
+
+  if (process.platform === 'win32')
+    return path.resolve(dirPath, 'content_shell.exe');
+
+  if (process.platform === 'darwin')
+    return path.resolve(dirPath, 'Content Shell.app', 'Contents', 'MacOS', 'Content Shell');
+}
+
+function runTests(buildDirectoryPath, useDebugDevtools) {
+  var testArgs = getInspectorTests().concat([
+    '--no-pixel-tests',
+    '--build-directory',
+    buildDirectoryPath,
+    '--target',
+    TARGET,
+  ]);
+  if (useDebugDevtools)
+    testArgs.push('--additional-driver-flag=--debug-devtools');
+  else
+    console.log('TIP: You can debug a test using: npm run debug-test inspector/test-name.html');
+
+  if (IS_DEBUG_ENABLED) {
+    testArgs.push('--additional-driver-flag=--remote-debugging-port=9222');
+    testArgs.push('--time-out-ms=6000000');
+    console.log('\n=============================================');
+    var unitTest = testArgs.find(arg => arg.includes('http/tests/devtools/unit/'));
+    if (unitTest) {
+      var unitTestPath = `http://localhost:8080/${unitTest.slice('http/tests/'.length)}`;
+      var link =
+          `http://localhost:8080/inspector-sources/debug/integration_test_runner.html?experiments=true&test=${
+                                                                                                              unitTestPath
+                                                                                                            }`;
+      console.log('1) Go to: ', link);
+      console.log('2) Go to: http://localhost:9222/, click on "inspected-page.html", and copy the ws query parameter');
+      console.log('3) Open DevTools on DevTools and you can refresh to re-run the test')
+    } else {
+      console.log('Go to: http://localhost:9222/');
+      console.log('Click on link and in console execute: test()');
+    }
+    console.log('=============================================\n');
+  }
+  var args = [BLINK_TEST_PATH].concat(testArgs).concat(getTestFlags());
+  console.log(`Running layout tests with args: ${args}`);
+  childProcess.spawn(PYTHON, args, {stdio: 'inherit'});
+}
+
+function getTestFlags() {
+  var flagValues = Object.keys(Flags).map(key => Flags[key]);
+  return process.argv.slice(2).filter(arg => {
+    var flagName = utils.includes(arg, '=') ? arg.slice(0, arg.indexOf('=')) : arg;
+    return !utils.includes(flagValues, flagName) && !utils.includes(arg, 'inspector') &&
+        !utils.includes(arg, 'http/tests/devtools');
+  });
+}
+
+function getInspectorTests() {
+  var specificTests =
+      process.argv.filter(arg => utils.includes(arg, 'inspector') || utils.includes(arg, 'http/tests/devtools'));
+  if (specificTests.length)
+    return specificTests;
+  return [
+    'inspector*',
+    'http/tests/inspector*',
+    'http/tests/devtools',
+  ];
+}
\ No newline at end of file