| #!/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) |