|  | # Copyright (c) 2013 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 Blink. | 
|  |  | 
|  | See http://dev.chromium.org/developers/how-tos/depottools/presubmit-scripts | 
|  | for more details about the presubmit API built into gcl. | 
|  | """ | 
|  |  | 
|  | import imp | 
|  | import inspect | 
|  | import os | 
|  |  | 
|  | try: | 
|  | # pylint: disable=C0103 | 
|  | audit_non_blink_usage = imp.load_source( | 
|  | 'audit_non_blink_usage', | 
|  | os.path.join(os.path.dirname(inspect.stack()[0][1]), 'tools/audit_non_blink_usage.py')) | 
|  | except IOError: | 
|  | # One of the presubmit upload tests tries to exec this script, which doesn't interact so well | 
|  | # with the import hack... just ignore the exception here and hope for the best. | 
|  | pass | 
|  |  | 
|  |  | 
|  | _EXCLUDED_PATHS = ( | 
|  | # This directory is created and updated via a script. | 
|  | r'^third_party[\\\/]blink[\\\/]tools[\\\/]blinkpy[\\\/]third_party[\\\/]wpt[\\\/]wpt[\\\/].*', | 
|  | ) | 
|  |  | 
|  |  | 
|  | def _CheckForWrongMojomIncludes(input_api, output_api): | 
|  | # In blink the code should either use -blink.h or -shared.h mojom | 
|  | # headers, except in public where only -shared.h headers should be | 
|  | # used to avoid exporting Blink types outside Blink. | 
|  | def source_file_filter(path): | 
|  | return input_api.FilterSourceFile(path, | 
|  | black_list=[r'third_party/blink/common/', r'third_party/blink/public/common']) | 
|  |  | 
|  | # The list of files that we specifically want to allow including | 
|  | # -blink variant files (e.g. because it has #if INSIDE_BLINK). | 
|  | allow_blink_files = [r'third_party/blink/public/platform/modules/serviceworker/web_service_worker_request.h'] | 
|  |  | 
|  | pattern = input_api.re.compile(r'#include\s+.+\.mojom(.*)\.h[>"]') | 
|  | public_folder = input_api.os_path.normpath('third_party/blink/public/') | 
|  | non_blink_mojom_errors = [] | 
|  | public_blink_mojom_errors = [] | 
|  | for f in input_api.AffectedFiles(file_filter=source_file_filter): | 
|  | for line_num, line in f.ChangedContents(): | 
|  | error_list = None | 
|  | # This is not super precise as we don't check endif, but allow | 
|  | # including Blink variant mojom files if the file has | 
|  | # '#if INSIDE_BLINK'. | 
|  | match = pattern.match(line) | 
|  | if match: | 
|  | if match.group(1) != '-shared': | 
|  | if f.LocalPath().startswith(public_folder) and \ | 
|  | not f.LocalPath() in allow_blink_files: | 
|  | error_list = public_blink_mojom_errors | 
|  | elif match.group(1) != '-blink': | 
|  | # Neither -shared.h, nor -blink.h. | 
|  | error_list = non_blink_mojom_errors | 
|  |  | 
|  | if error_list is not None: | 
|  | error_list.append('    %s:%d %s' % ( | 
|  | f.LocalPath(), line_num, line)) | 
|  |  | 
|  | results = [] | 
|  | if non_blink_mojom_errors: | 
|  | results.append(output_api.PresubmitError( | 
|  | 'Files that include non-Blink variant mojoms found. ' | 
|  | 'You must include .mojom-blink.h or .mojom-shared.h instead:', | 
|  | non_blink_mojom_errors)) | 
|  |  | 
|  | if public_blink_mojom_errors: | 
|  | results.append(output_api.PresubmitError( | 
|  | 'Public blink headers using Blink variant mojoms found. ' | 
|  | 'You must include .mojom-shared.h instead:', | 
|  | public_blink_mojom_errors)) | 
|  |  | 
|  | return results | 
|  |  | 
|  |  | 
|  | def _CommonChecks(input_api, output_api): | 
|  | """Checks common to both upload and commit.""" | 
|  | # We should figure out what license checks we actually want to use. | 
|  | license_header = r'.*' | 
|  |  | 
|  | results = [] | 
|  | results.extend(input_api.canned_checks.PanProjectChecks( | 
|  | input_api, output_api, excluded_paths=_EXCLUDED_PATHS, | 
|  | maxlen=800, license_header=license_header)) | 
|  | results.extend(_CheckForWrongMojomIncludes(input_api, output_api)) | 
|  | return results | 
|  |  | 
|  |  | 
|  | def _CheckStyle(input_api, output_api): | 
|  | style_checker_path = input_api.os_path.join(input_api.PresubmitLocalPath(), | 
|  | 'tools', 'check_blink_style.py') | 
|  | args = [input_api.python_executable, style_checker_path, '--diff-files'] | 
|  | files = [] | 
|  | for f in input_api.AffectedFiles(): | 
|  | file_path = f.LocalPath() | 
|  | # Filter out changes in LayoutTests. | 
|  | if 'web_tests' + input_api.os_path.sep in file_path and 'TestExpectations' not in file_path: | 
|  | continue | 
|  | if '/PRESUBMIT' in file_path: | 
|  | continue | 
|  | files.append(input_api.os_path.join('..', '..', file_path)) | 
|  | # Do not call check_blink_style.py with empty affected file list if all | 
|  | # input_api.AffectedFiles got filtered. | 
|  | if not files: | 
|  | return [] | 
|  | args += files | 
|  | results = [] | 
|  |  | 
|  | try: | 
|  | child = input_api.subprocess.Popen(args, | 
|  | stderr=input_api.subprocess.PIPE) | 
|  | _, stderrdata = child.communicate() | 
|  | if child.returncode != 0: | 
|  | results.append(output_api.PresubmitError( | 
|  | 'check_blink_style.py failed', [stderrdata])) | 
|  | except Exception as e: | 
|  | results.append(output_api.PresubmitNotifyResult( | 
|  | 'Could not run check_blink_style.py', [str(e)])) | 
|  |  | 
|  | return results | 
|  |  | 
|  |  | 
|  | def _CheckForPrintfDebugging(input_api, output_api): | 
|  | """Generally speaking, we'd prefer not to land patches that printf | 
|  | debug output. | 
|  | """ | 
|  | printf_re = input_api.re.compile(r'^\s*(printf\(|fprintf\(stderr,)') | 
|  | errors = input_api.canned_checks._FindNewViolationsOfRule( | 
|  | lambda _, x: not printf_re.search(x), | 
|  | input_api, None) | 
|  | errors = ['  * %s' % violation for violation in errors] | 
|  | if errors: | 
|  | return [output_api.PresubmitPromptOrNotify( | 
|  | 'printf debugging is best debugging! That said, it might ' | 
|  | 'be a good idea to drop the following occurences from ' | 
|  | 'your patch before uploading:\n%s' % '\n'.join(errors))] | 
|  | return [] | 
|  |  | 
|  |  | 
|  | def _CheckForForbiddenChromiumCode(input_api, output_api): | 
|  | """Checks that Blink uses Chromium classes and namespaces only in permitted code.""" | 
|  | # TODO(dcheng): This is pretty similar to _FindNewViolationsOfRule. Unfortunately, that can;'t | 
|  | # be used directly because it doesn't give the callable enough information (namely the path to | 
|  | # the file), so we duplicate the logic here... | 
|  | results = [] | 
|  | for f in input_api.AffectedFiles(): | 
|  | path = f.LocalPath() | 
|  | errors = audit_non_blink_usage.check(path, [(i + 1, l) for i, l in enumerate(f.NewContents())]) | 
|  | if errors: | 
|  | errors = audit_non_blink_usage.check(path, f.ChangedContents()) | 
|  | if errors: | 
|  | for line_number, disallowed_identifier in errors: | 
|  | results.append(output_api.PresubmitError( | 
|  | '%s:%d uses disallowed identifier %s' % (path, line_number, disallowed_identifier))) | 
|  | return results | 
|  |  | 
|  |  | 
|  | def CheckChangeOnUpload(input_api, output_api): | 
|  | results = [] | 
|  | results.extend(_CommonChecks(input_api, output_api)) | 
|  | results.extend(_CheckStyle(input_api, output_api)) | 
|  | results.extend(_CheckForPrintfDebugging(input_api, output_api)) | 
|  | results.extend(_CheckForForbiddenChromiumCode(input_api, output_api)) | 
|  | return results | 
|  |  | 
|  |  | 
|  | def CheckChangeOnCommit(input_api, output_api): | 
|  | results = [] | 
|  | results.extend(_CommonChecks(input_api, output_api)) | 
|  | results.extend(input_api.canned_checks.CheckTreeIsOpen( | 
|  | input_api, output_api, | 
|  | json_url='http://chromium-status.appspot.com/current?format=json')) | 
|  | results.extend(input_api.canned_checks.CheckChangeHasDescription( | 
|  | input_api, output_api)) | 
|  | return results | 
|  |  | 
|  |  | 
|  | def _ArePaintOrCompositingDirectoriesModified(change):  # pylint: disable=C0103 | 
|  | """Checks whether CL has changes to paint or compositing directories.""" | 
|  | paint_or_compositing_paths = [ | 
|  | os.path.join('third_party', 'blink', 'renderer', 'platform', 'graphics'), | 
|  | os.path.join('third_party', 'blink', 'renderer', 'core', 'layout', | 
|  | 'compositing'), | 
|  | os.path.join('third_party', 'blink', 'renderer', 'core', 'svg'), | 
|  | os.path.join('third_party', 'blink', 'renderer', 'core', 'paint'), | 
|  | os.path.join('third_party', 'blink', 'web_tests', 'FlagExpectations', | 
|  | 'enable-slimming-paint-v2'), | 
|  | os.path.join('third_party', 'blink', 'web_tests', 'flag-specific', | 
|  | 'enable-slimming-paint-v2'), | 
|  | os.path.join('third_party', 'blink', 'web_tests', 'FlagExpectations', | 
|  | 'enable-slimming-paint-v175'), | 
|  | os.path.join('third_party', 'blink', 'web_tests', 'flag-specific', | 
|  | 'enable-slimming-paint-v175'), | 
|  | ] | 
|  | for affected_file in change.AffectedFiles(): | 
|  | file_path = affected_file.LocalPath() | 
|  | if any(x in file_path for x in paint_or_compositing_paths): | 
|  | return True | 
|  | return False | 
|  |  | 
|  |  | 
|  | def _AreLayoutNGDirectoriesModified(change):  # pylint: disable=C0103 | 
|  | """Checks whether CL has changes to a layout ng directory.""" | 
|  | layout_ng_paths = [ | 
|  | os.path.join('third_party', 'blink', 'renderer', 'core', 'layout', | 
|  | 'ng'), | 
|  | os.path.join('third_party', 'blink', 'renderer', 'core', 'paint', | 
|  | 'ng'), | 
|  | os.path.join('third_party', 'blink', 'renderer', 'platform', 'fonts', | 
|  | 'shaping'), | 
|  | os.path.join('third_party', 'blink', 'web_tests', 'FlagExpectations', | 
|  | 'enable-blink-features=LayoutNG'), | 
|  | os.path.join('third_party', 'blink', 'web_tests', 'flag-specific', | 
|  | 'enable-blink-features=LayoutNG'), | 
|  | ] | 
|  | for affected_file in change.AffectedFiles(): | 
|  | file_path = affected_file.LocalPath() | 
|  | if any(x in file_path for x in layout_ng_paths): | 
|  | return True | 
|  | return False | 
|  |  | 
|  |  | 
|  | def PostUploadHook(cl, change, output_api):  # pylint: disable=C0103 | 
|  | """git cl upload will call this hook after the issue is created/modified. | 
|  |  | 
|  | This hook adds extra try bots to the CL description in order to run slimming | 
|  | paint v2 tests or LayoutNG tests in addition to the CQ try bots if the | 
|  | change contains changes in a relevant direcotry (see: | 
|  | _ArePaintOrCompositingDirectoriesModified and | 
|  | _AreLayoutNGDirectoriesModified). For more information about | 
|  | slimming-paint-v2 tests see https://crbug.com/601275 and for information | 
|  | about the LayoutNG tests see https://crbug.com/706183. | 
|  | """ | 
|  | results = [] | 
|  | if _ArePaintOrCompositingDirectoriesModified(change): | 
|  | results.extend(output_api.EnsureCQIncludeTrybotsAreAdded( | 
|  | cl, | 
|  | ['luci.chromium.try:' | 
|  | 'linux_layout_tests_slimming_paint_v2', | 
|  | # TODO(kojii): Update linux_trusty_blink_rel to luci when migrated. | 
|  | 'master.tryserver.blink:linux_trusty_blink_rel'], | 
|  | 'Automatically added linux_layout_tests_slimming_paint_v2 and ' | 
|  | 'linux_trusty_blink_rel to run on CQ due to changes in paint or ' | 
|  | 'compositing directories.')) | 
|  | if _AreLayoutNGDirectoriesModified(change): | 
|  | results.extend(output_api.EnsureCQIncludeTrybotsAreAdded( | 
|  | cl, | 
|  | ['luci.chromium.try:' | 
|  | 'linux_layout_tests_layout_ng'], | 
|  | 'Automatically added linux_layout_tests_layout_ng to run on CQ due ' | 
|  | 'to changes in LayoutNG directories.')) | 
|  | return results |