blob: c7ffc7e43f57ccceb37b506ef907aded75e01f7f [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 os
import shutil
import subprocess
import sys
def node_path(options):
try:
old_sys_path = sys.path[:]
sys.path.append(
os.path.join(options.chromium_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('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 get_hook_action(options, hook_name):
"""Parses Chromium DEPS file and returns the action for the given hook.
"""
sys.path.append(
os.path.join(options.chromium_dir, 'third_party', 'depot_tools'))
import gclient_eval
filepath = os.path.join(options.chromium_dir, 'DEPS')
with open(filepath) as deps_file:
deps = gclient_eval.Parse(deps_file.read(), filepath)
for hook in deps['hooks']:
if hook['name'] == hook_name:
return hook['action']
raise RuntimeError(f'{hook_name} hook not found in DEPS')
def sync_node(options):
"""Executes the nodejs sync hook from Chromium DEPS file."""
action = get_hook_action(options, 'node_linux64')
gclient_context = os.path.join(options.chromium_dir, '..')
subprocess.check_call(action, cwd=gclient_context)
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_dom_pinned_properties(options):
print('generating DOM pinned properties dataset from .idl definitions')
subprocess.check_call([
node_path(options),
os.path.join(options.devtools_dir, 'scripts', 'webidl-properties',
'index.js'), 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'], 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),
os.path.join(options.devtools_dir, 'scripts', 'test',
'run_lint_check_js.mjs')
] + generated_source_files,
cwd=options.devtools_dir)
if __name__ == '__main__':
OPTIONS = parse_options(sys.argv[1:])
if OPTIONS.ref == ReferenceMode.Tot:
update(OPTIONS)
if OPTIONS.update_node:
sync_node(OPTIONS)
copy_files(OPTIONS)
generate_signatures(OPTIONS)
generate_dom_pinned_properties(OPTIONS)
generate_protocol_resources(OPTIONS)
run_git_cl_format(OPTIONS)
run_eslint(OPTIONS)