blob: 0e1b78d890e0df65ef25d25fef050833e02a682f [file] [log] [blame]
# 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.
"""
import os
import os.path as path
import json
import shutil
import subprocess
import sys
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",
"ISC",
]
# List all DEPS here.
DEPS = {
"@types/chai": "4.2.0",
"@types/mocha": "5.2.7",
"chai": "4.2.0",
"escodegen": "1.12.0",
"eslint": "6.0.1",
"esprima": "git+https://git@github.com/ChromeDevTools/esprima.git#4d0f0e18bd8d3731e5f931bf573af3394cbf7cbe",
"handlebars": "4.3.1",
"karma": "4.2.0",
"karma-chai": "0.1.0",
"karma-chrome-launcher": "3.1.0",
"karma-coverage-istanbul-instrumenter": "1.0.1",
"karma-coverage-istanbul-reporter": "2.1.0",
"karma-mocha": "1.3.0",
"karma-typescript": "4.1.1",
"license-checker": "25.0.1",
"mocha": "6.2.0",
"puppeteer": "2.0.0",
"rollup": "1.23.1",
"typescript": "3.5.3",
"yargs": "15.0.2"
}
def exec_command(cmd):
try:
cmd_proc_result = subprocess.check_call(cmd, cwd=devtools_paths.root_path())
except CalledProcessError as error:
print(error.output)
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.
pattern = path.join(devtools_paths.node_modules_path(), 'package.json')
packages = []
for root, dirnames, filenames in os.walk(devtools_paths.node_modules_path()):
for filename in filter(lambda f: f == 'package.json', filenames):
packages.append(path.join(root, filename))
for pkg in packages:
with open(pkg, 'r+') as pkg_file:
prop_removal_count = 0
try:
pkg_data = json.load(pkg_file)
# 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(u'_') == 0:
pkg_data.pop(key)
prop_removal_count = prop_removal_count + 1
pkg_file.truncate(0)
pkg_file.seek(0)
json.dump(pkg_data, pkg_file, indent=2, sort_keys=True)
print("(%s): %s" % (prop_removal_count, pkg))
except:
print('Unable to fix: %s' % pkg)
return True
return False
def install_missing_deps():
with open(devtools_paths.package_lock_json_path(), 'r+') as pkg_lock_file:
try:
pkg_lock_data = json.load(pkg_lock_file)
existing_deps = pkg_lock_data[u'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']
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 = json.load(pkg_file)
# Replace the dev deps.
pkg_data[u'devDependencies'] = DEPS
pkg_file.truncate(0)
pkg_file.seek(0)
json.dump(pkg_data, pkg_file, indent=2, sort_keys=True)
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 = json.load(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(u'dependencies') == 0 or key.find(u'devDependencies') == 0:
pkg_data.pop(key)
pkg_file.truncate(0)
pkg_file.seek(0)
json.dump(pkg_data, pkg_file, indent=2, sort_keys=True)
except:
print('Unable to fix: %s' % pkg)
return True
return False
def install_deps():
for (name, version) in DEPS.items():
if (version.find(u'^') == 0):
print('Versions must be locked to a specific version; remove ^ from the start of the version.')
return True
if append_package_json_entries():
return True
if install_missing_deps():
return True
# Run the CI version of npm, which prevents updates to the versions of modules.
if exec_command(['npm', 'ci']):
return True
if strip_private_fields():
return True
if remove_package_json_entries():
return True
return ensure_licenses()
npm_errors_found = install_deps()
if npm_errors_found:
print('npm installation failed')
else:
print('npm installation successful')