blob: fe41e7111d9914baf55f33690057340ee38a7a44 [file] [log] [blame]
#!/usr/bin/env vpython3
#
# Copyright 2019 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.
"""
Helper to manage DEPS. Use this script to update node_modules instead of
running npm install manually. To upgrade a dependency, change the version
number in DEPS below and run this script.
"""
import os
import os.path as path
import json
import subprocess
import sys
from collections import OrderedDict
scripts_path = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
sys.path.append(scripts_path)
import devtools_paths
LICENSES = [
"MIT",
"Apache-2.0",
"BSD",
"BSD-2-Clause",
"BSD-3-Clause",
"CC0-1.0",
"CC-BY-3.0",
"CC-BY-4.0",
"ISC",
"MPL-2.0",
"Python-2.0",
"W3C",
]
# List all DEPS here.
DEPS = {
"@istanbuljs/schema": "0.1.3",
"@types/chai": "4.3.0",
"@types/codemirror": "0.0.108",
"@types/estree": "0.0.50",
"@types/filesystem": "0.0.32",
"@types/istanbul-lib-coverage": "2.0.4",
"@types/istanbul-lib-instrument": "1.7.4",
"@types/istanbul-lib-report": "3.0.0",
"@types/istanbul-lib-source-maps": "4.0.1",
"@types/istanbul-reports": "3.0.1",
"@types/karma-chai-sinon": "0.1.16",
"@types/node": "18.11.9",
"@types/marked": "4.0.1",
"@types/mocha": "9.0.0",
"@types/rimraf": "3.0.2",
"@types/sinon": "10.0.6",
"@types/webidl2": "23.13.6",
"@typescript-eslint/parser": "5.30.6",
"@typescript-eslint/eslint-plugin": "5.30.6",
"@web/rollup-plugin-import-meta-assets": "1.0.7",
"chai": "4.3.4",
"chokidar": "3.5.2",
"clean-css": "5.2.2",
"convert-source-map": "1.8.0",
# This should be match with esbuild in DEPS.
"esbuild": "0.14.13",
"escodegen": "2.0.0",
"eslint": "8.19.0",
"eslint-plugin-import": "2.25.4",
"eslint-plugin-lit-a11y": "2.1.0",
"eslint-plugin-mocha": "10.0.3",
"eslint-plugin-rulesdir": "0.2.1",
"istanbul-lib-instrument": "5.1.0",
"istanbul-lib-report": "3.0.0",
"karma": "6.4.0",
"karma-chai": "0.1.0",
"karma-chrome-launcher": "3.1.0",
"karma-coverage": "2.1.0",
"karma-mocha": "2.0.1",
"karma-mocha-reporter": "2.2.5",
"karma-sinon": "1.0.5",
"karma-sourcemap-loader": "0.3.8",
"karma-spec-reporter": "0.0.32",
"license-checker": "25.0.1",
"mocha": "9.1.3",
"postcss": "8.4.5",
"puppeteer": "14.0.0",
"recast": "0.20.5",
"rimraf": "3.0.2",
"rollup": "2.63.0",
"rollup-plugin-minify-html-template-literals": "1.2.0",
"rollup-plugin-terser": "7.0.2",
"sinon": "12.0.1",
"source-map-support": "0.5.21",
"stylelint": "14.2.0",
"stylelint-config-standard": "24.0.0",
"svgo": "2.8.0",
"terser": "5.10.0",
"typescript": "4.9.3",
"ws": "8.4.0",
"yargs": "17.3.1",
"glob": "7.1.7",
"webidl2": "24.2.2",
"@webref/idl": "3.12.0",
}
ADDITIONAL_NPM_ARGS = [
# This is to avoid downloading esbuild-* package.
'--omit', 'optional', '--ignore-scripts'
]
def load_json_file(location):
# By default, json load uses a standard Python dictionary, which is not ordered.
# To prevent subsequent invocations of this script to erroneously alter the order
# of keys defined in package.json files, we should use an `OrderedDict`. This
# ensures not only that we use a strict ordering, it will also make sure we maintain
# the order defined by the NPM packages themselves. That in turn is important, since
# NPM packages can define `exports`, where the order of entrypoints is crucial for
# how an NPM package is loaded. If you would change the order, it could break loading
# that package.
return json.load(location, object_pairs_hook=OrderedDict)
def exec_command(cmd):
try:
new_env = os.environ.copy()
# Prevent large files from being checked in to git.
new_env["PUPPETEER_SKIP_CHROMIUM_DOWNLOAD"] = "true"
cmd_proc_result = subprocess.check_call(cmd,
cwd=devtools_paths.root_path(),
env=new_env)
except Exception as error:
print(error)
return True
return False
def ensure_licenses():
cmd = [
devtools_paths.node_path(),
devtools_paths.license_checker_path(),
'--onlyAllow',
('%s' % (';'.join(LICENSES)))
]
return exec_command(cmd)
def strip_private_fields():
# npm adds private fields which need to be stripped.
packages = []
for root, _, filenames in os.walk(devtools_paths.node_modules_path()):
if 'package.json' in filenames:
packages.append(path.join(root, 'package.json'))
for pkg in packages:
with open(pkg, 'r+') as pkg_file:
try:
pkg_data = load_json_file(pkg_file)
updated_pkg_data = pkg_data.copy()
# Remove anything that begins with an underscore, as these are
# the private fields in a package.json
for key in pkg_data.keys():
if key.find('_') == 0:
updated_pkg_data.pop(key)
pkg_file.truncate(0)
pkg_file.seek(0)
json.dump(updated_pkg_data, pkg_file, indent=2, separators=(',', ': '))
pkg_file.write('\n')
except:
print('Unable to fix: %s, %s' % (pkg, sys.exc_info()))
return True
return False
# Required to keep the package-lock.json in sync with the package.json dependencies
def install_missing_deps():
with open(devtools_paths.package_lock_json_path(), 'r+') as pkg_lock_file:
try:
pkg_lock_data = load_json_file(pkg_lock_file)
existing_deps = pkg_lock_data['dependencies']
new_deps = []
# Find any new DEPS and add them in.
for dep, version in DEPS.items():
if not dep in existing_deps or not existing_deps[dep]['version'] == version:
new_deps.append("%s@%s" % (dep, version))
# Now install.
if len(new_deps) > 0:
cmd = [
'npm',
'install',
'--save-dev',
] + ADDITIONAL_NPM_ARGS
cmd.extend(new_deps)
return exec_command(cmd)
except Exception as exception:
print('Unable to install: %s' % exception)
return True
return False
def append_package_json_entries():
with open(devtools_paths.package_json_path(), 'r+') as pkg_file:
try:
pkg_data = load_json_file(pkg_file)
# Replace the dev deps.
pkg_data['devDependencies'] = DEPS
pkg_file.truncate(0)
pkg_file.seek(0)
json.dump(pkg_data, pkg_file, indent=2, separators=(',', ': '))
pkg_file.write('\n')
except:
print('Unable to fix: %s' % sys.exc_info()[0])
return True
return False
def remove_package_json_entries():
with open(devtools_paths.package_json_path(), 'r+') as pkg_file:
try:
pkg_data = load_json_file(pkg_file)
# Remove the dependencies and devDependencies from the root package.json
# so that they can't be used to overwrite the node_modules managed by this file.
for key in pkg_data.keys():
if key.find('dependencies') == 0 or key.find(
'devDependencies') == 0:
pkg_data.pop(key)
pkg_file.truncate(0)
pkg_file.seek(0)
json.dump(pkg_data, pkg_file, indent=2, separators=(',', ': '))
pkg_file.write('\n')
except:
print('Unable to fix: %s' % pkg)
return True
return False
def addClangFormat():
with open(path.join(devtools_paths.node_modules_path(), '.clang-format'), 'w+') as clang_format_file:
try:
clang_format_file.write('DisableFormat: true\n')
except:
print('Unable to write .clang-format file')
return True
return False
def addOwnersFile():
with open(path.join(devtools_paths.node_modules_path(), 'OWNERS'),
'w+') as owners_file:
try:
owners_file.write('file://config/owner/INFRA_OWNERS\n')
except:
print('Unable to write OWNERS file')
return True
return False
def addChromiumReadme():
with open(path.join(devtools_paths.node_modules_path(), 'README.chromium'),
'w+') as readme_file:
try:
readme_file.write('This directory hosts all packages downloaded from NPM that are used in either the build system or infrastructure scripts.\n')
readme_file.write('If you want to make any changes to this directory, please see "scripts/deps/manage_node_deps.py".\n')
except:
print('Unable to write README.chromium file')
return True
return False
def run_npm_command(npm_command_args=None):
for (name, version) in DEPS.items():
if (version.find('^') == 0):
print('Versions must be locked to a specific version; remove ^ from the start of the version.')
return True
run_custom_command = npm_command_args is not None
if append_package_json_entries():
return True
if install_missing_deps():
return True
runs_analysis_command = False
if run_custom_command:
runs_analysis_command = npm_command_args[:1] == [
'outdated'
] or npm_command_args[:1] == ['audit'
] or npm_command_args[:1] == ['ls']
# By default, run the CI version of npm, which prevents updates to the versions of modules.
# However, when we are analyzing the installed NPM dependencies, we don't need to run
# the installation process again.
if not runs_analysis_command:
if exec_command([
'npm',
'ci',
] + ADDITIONAL_NPM_ARGS):
return True
# To minimize disk usage for Chrome DevTools node_modules, always try to dedupe dependencies.
# We need to perform this every time, as the order of dependencies added could lead to a
# non-optimal dependency tree, resulting in unnecessary disk usage.
if exec_command([
'npm',
'dedupe',
] + ADDITIONAL_NPM_ARGS):
return True
if run_custom_command:
custom_command_result = exec_command(['npm'] + npm_command_args)
if remove_package_json_entries():
return True
if strip_private_fields():
return True
if addClangFormat():
return True
if addOwnersFile():
return True
if addChromiumReadme():
return True
if run_custom_command:
return custom_command_result
return ensure_licenses()
npm_args = None
if (len(sys.argv[1:]) > 0):
npm_args = sys.argv[1:]
npm_errors_found = run_npm_command(npm_args)
if npm_errors_found:
print('npm command failed')
exit(1)