blob: 94d35f1df90dc3143ec91d0d16360f21af002ebd [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.
"""
Update manually maintained dependencies from Chromium.
"""
import argparse
import enum
import json
import os
import shutil
import subprocess
import sys
def node_path(options):
try:
old_sys_path = sys.path[:]
sys.path.append(
os.path.join(options.devtools_dir, 'third_party', 'node'))
import node
finally:
sys.path = old_sys_path
return node.GetBinaryPath()
# Files whose location within devtools-frontend matches the upstream location.
FILES = [
'v8/include/js_protocol.pdl',
'third_party/blink/renderer/core/css/css_properties.json5',
'third_party/blink/renderer/core/html/aria_properties.json5',
'third_party/blink/public/devtools_protocol/browser_protocol.pdl',
'third_party/blink/renderer/core/frame/deprecation/deprecation.json5',
]
# Files whose location within devtools-frontend differs from the upstream location.
FILE_MAPPINGS = {
# chromium_path => devtools_frontend_path
'components/variations/proto/devtools/client_variations.js':
'front_end/third_party/chromium/client-variations/ClientVariations.js',
'third_party/axe-core/axe.d.ts': 'front_end/third_party/axe-core/axe.d.ts',
'third_party/axe-core/axe.js': 'front_end/third_party/axe-core/axe.js',
'third_party/axe-core/axe.min.js':
'front_end/third_party/axe-core/axe.min.js',
'third_party/axe-core/LICENSE': 'front_end/third_party/axe-core/LICENSE',
}
for f in FILES:
FILE_MAPPINGS[f] = f
class ReferenceMode(enum.Enum):
Tot = 'tot'
WorkingTree = 'working-tree'
def __str__(self):
return self.value
def parse_options(cli_args):
parser = argparse.ArgumentParser(
description='Roll dependencies from Chromium.')
parser.add_argument(
'--ref',
type=ReferenceMode,
choices=list(ReferenceMode),
default=ReferenceMode.Tot,
help='Defaults to tot. '
'If tot, fetch origin/main of Chromium repository and use it. '
'If working-tree, use working tree as is.')
parser.add_argument('--update-node',
action="store_true",
default=False,
help='If set it syncs nodejs.')
parser.add_argument(
'--output',
default=None,
help=
'If set it outputs information about the roll in the specified file.')
parser.add_argument('chromium_dir', help='path to chromium/src directory')
parser.add_argument('devtools_dir',
help='path to devtools/devtools-frontend directory')
return parser.parse_args(cli_args)
def update(options):
subprocess.check_call(['git', 'fetch', 'origin'], cwd=options.chromium_dir)
subprocess.check_call(['git', 'checkout', 'origin/main'],
cwd=options.chromium_dir)
subprocess.check_call(['gclient', 'sync'], cwd=options.chromium_dir)
def sync_node(options):
"""Node is managed as a standard GCS deps so we run gclient sync but without hooks"""
subprocess.check_call(['gclient', 'sync', '--nohooks'],
cwd=options.devtools_dir)
def copy_files(options):
for from_path, to_path in FILE_MAPPINGS.items():
from_path = os.path.normpath(from_path)
to_path = os.path.normpath(to_path)
print('%s => %s' % (from_path, to_path))
shutil.copy(os.path.join(options.chromium_dir, from_path),
os.path.join(options.devtools_dir, to_path))
def generate_signatures(options):
print('generating JavaScript native functions signatures from .idl '
'and typescript definitions')
subprocess.check_call([
node_path(options),
os.path.join(options.devtools_dir, 'scripts', 'javascript_natives',
'index.js'), options.chromium_dir, options.devtools_dir
])
def generate_protocol_resources(options):
print('generating protocol resources')
subprocess.check_call([
os.path.join(options.devtools_dir, 'scripts', 'deps',
'generate_protocol_resources.py'), '--node-path',
node_path(options)
],
cwd=options.devtools_dir)
def run_git_cl_format(options):
print('running `git cl format` to format generated TS files')
subprocess.check_call(['git', 'cl', 'format', '--js', '--full'],
cwd=options.devtools_dir)
def run_eslint(options):
print('running eslint with --fix for generated files')
result = subprocess.check_output(
['git', 'diff', '--diff-filter=d', '--name-only'],
cwd=options.devtools_dir).strip()
generated_source_files = []
for line in result.split(b'\n'):
if line.endswith(b'.js') or line.endswith(b'.ts'):
generated_source_files.append(line)
subprocess.check_call([
node_path(options), "--experimental-strip-types",
"--no-warnings=ExperimentalWarning",
os.path.join(options.devtools_dir, 'scripts', 'test',
'run_lint_check.mjs')
] + generated_source_files,
cwd=options.devtools_dir)
def files_changed(options):
return subprocess.check_output(['git', 'diff', '--name-only'],
cwd=options.devtools_dir,
text=True).strip()
def update_deps_revision(options):
print('updating DEPS revision')
old_revision = subprocess.check_output(
['gclient', 'getdep', '--var=chromium_browser_protocol_revision'],
cwd=options.devtools_dir,
text=True).strip()
new_revision = subprocess.check_output(
['git', 'log', '-1', '--pretty=format:%H'],
cwd=options.chromium_dir,
text=True).strip()
subprocess.check_call(
[
'gclient', 'setdep',
f'--var=chromium_browser_protocol_revision={new_revision}'
],
cwd=options.devtools_dir,
)
if options.output:
with open(options.output, 'w', encoding='utf-8') as f:
json.dump(
{
'old_revision': old_revision,
'new_revision': new_revision
}, f)
if __name__ == '__main__':
OPTIONS = parse_options(sys.argv[1:])
if OPTIONS.ref == ReferenceMode.Tot:
update(OPTIONS)
elif OPTIONS.update_node:
sync_node(OPTIONS)
copy_files(OPTIONS)
generate_signatures(OPTIONS)
generate_protocol_resources(OPTIONS)
if files_changed(OPTIONS):
run_git_cl_format(OPTIONS)
run_eslint(OPTIONS)
update_deps_revision(OPTIONS)