# Copyright 2015 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. """Top-level presubmit script for catapult. See https://www.chromium.org/developers/how-tos/depottools/presubmit-scripts for more details about the presubmit API built into depot_tools. """ import re import sys _EXCLUDED_PATHS = ( r'(.*[\\/])?\.git[\\/].*', r'.+\.png$', r'.+\.svg$', r'.+\.skp$', r'.+\.gypi$', r'.+\.gyp$', r'.+\.gn$', r'.*\.gitignore$', r'.*codereview.settings$', r'.*AUTHOR$', r'^CONTRIBUTORS\.md$', r'.*LICENSE$', r'.*OWNERS$', r'.*README\.md$', r'^dashboard[\\/]dashboard[\\/]templates[\\/].*', r'^experimental[\\/]heatmap[\\/].*', r'^perf_insights[\\/]test_data[\\/].*', r'^perf_insights[\\/]third_party[\\/].*', r'^third_party[\\/].*', r'^tracing[\\/]\.allow-devtools-save$', r'^tracing[\\/]bower\.json$', r'^tracing[\\/]\.bowerrc$', r'^tracing[\\/]tracing_examples[\\/]string_convert\.js$', r'^tracing[\\/]test_data[\\/].*', r'^tracing[\\/]third_party[\\/].*', r'^telemetry[\\/]support[\\/]html_output[\\/]results-template.html', ) _CATAPULT_BUG_ID_RE = re.compile(r'#[1-9]\d*') _RIETVELD_BUG_ID_RE = re.compile(r'[1-9]\d*') _RIETVELD_REPOSITORY_NAMES = frozenset({'chromium', 'v8'}) def CheckChangeLogBug(input_api, output_api): # Show a presubmit message if there is no BUG= line. if input_api.change.BUG is None: return [output_api.PresubmitNotifyResult( 'If this change has associated Catapult and/or Rietveld bug(s), add a ' '"BUG=<bug>(, <bug>)*" line to the patch description where <bug> can ' 'be one of the following: catapult:#NNNN, ' + ', '.join('%s:NNNNNN' % n for n in _RIETVELD_REPOSITORY_NAMES) + '.')] # Throw a presubmit error if the BUG= line is provided but empty. if input_api.change.BUG.strip() == '': return [output_api.PresubmitError( 'Empty BUG= line. Either remove it, or, preferably, change it to ' '"BUG=<bug>(, <bug>)*" where <bug> can be one of the following: ' + 'catapult:#NNNN, ' + ', '.join('%s:NNNNNN' % n for n in _RIETVELD_REPOSITORY_NAMES) + '.')] # Check that each bug in the BUG= line has the correct format. error_messages = [] catapult_bug_provided = False append_repository_order_error = False for index, bug in enumerate(input_api.change.BUG.split(',')): if index > 0: bug = bug.lstrip() # Allow spaces after commas. # Check if the bug can be split into a repository name and a bug ID (e.g. # 'catapult:#1234' -> 'catapult' and '#1234'). bug_parts = bug.split(':') if len(bug_parts) != 2: error_messages.append('Invalid bug "%s". Bugs should be provided in the ' '"<repository-name>:<bug-id>" format.' % bug) continue repository_name, bug_id = bug_parts if repository_name == 'catapult': if not _CATAPULT_BUG_ID_RE.match(bug_id): error_messages.append('Invalid bug "%s". Bugs in the Catapult ' 'repository should be provided in the ' '"catapult:#NNNN" format.' % bug) catapult_bug_provided = True elif repository_name in _RIETVELD_REPOSITORY_NAMES: if not _RIETVELD_BUG_ID_RE.match(bug_id): error_messages.append('Invalid bug "%s". Bugs in the Rietveld %s ' 'repository should be provided in the ' '"%s:NNNNNN" format.' % (bug, repository_name, repository_name)) if catapult_bug_provided: append_repository_order_error = True else: error_messages.append('Invalid bug "%s". Unknown repository "%s".' % ( bug, repository_name)) if append_repository_order_error: error_messages.append('Please list Rietveld bugs (' + ', '.join('%s:NNNNNN' % n for n in _RIETVELD_REPOSITORY_NAMES) + ') before Catapult bugs (catapult:#NNNN) so ' 'that Rietveld would display them as hyperlinks.') return map(output_api.PresubmitError, error_messages) def CheckChange(input_api, output_api): results = [] try: sys.path += [input_api.PresubmitLocalPath()] from catapult_build import js_checks from catapult_build import html_checks from catapult_build import repo_checks results += input_api.canned_checks.PanProjectChecks( input_api, output_api, excluded_paths=_EXCLUDED_PATHS) results += CheckChangeLogBug(input_api, output_api) results += js_checks.RunChecks( input_api, output_api, excluded_paths=_EXCLUDED_PATHS) results += html_checks.RunChecks( input_api, output_api, excluded_paths=_EXCLUDED_PATHS) results += repo_checks.RunChecks(input_api, output_api) finally: sys.path.remove(input_api.PresubmitLocalPath()) return results def CheckChangeOnUpload(input_api, output_api): return CheckChange(input_api, output_api) def CheckChangeOnCommit(input_api, output_api): return CheckChange(input_api, output_api)