Rico Wind | 62d0320 | 2018-11-30 13:43:49 +0100 | [diff] [blame] | 1 | # Copyright (c) 2018, the R8 project authors. Please see the AUTHORS file |
| 2 | # for details. All rights reserved. Use of this source code is governed by a |
| 3 | # BSD-style license that can be found in the LICENSE file. |
| 4 | |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 5 | from os import path |
Morten Krogh-Jespersen | f997786 | 2021-09-14 12:36:28 +0200 | [diff] [blame] | 6 | import datetime |
Christoffer Quist Adamsen | 4411f07 | 2020-03-05 13:28:31 +0100 | [diff] [blame] | 7 | from subprocess import check_output, Popen, PIPE, STDOUT |
Morten Krogh-Jespersen | 145c0ea | 2023-06-27 13:43:49 +0200 | [diff] [blame] | 8 | import inspect |
Rico Wind | 141383f | 2023-07-03 13:25:50 +0200 | [diff] [blame] | 9 | import os |
| 10 | import sys |
| 11 | # Add both current path to allow us to package import utils and the tools |
| 12 | # dir to allow transitive (for utils) dependendies to be loaded. |
Morten Krogh-Jespersen | 145c0ea | 2023-06-27 13:43:49 +0200 | [diff] [blame] | 13 | sys.path.append(path.dirname(inspect.getfile(lambda: None))) |
Rico Wind | 141383f | 2023-07-03 13:25:50 +0200 | [diff] [blame] | 14 | sys.path.append(os.path.join( |
| 15 | path.dirname(inspect.getfile(lambda: None)), 'tools')) |
Morten Krogh-Jespersen | 145c0ea | 2023-06-27 13:43:49 +0200 | [diff] [blame] | 16 | from tools.utils import EnsureDepFromGoogleCloudStorage |
Søren Gjesse | b5ae53d | 2025-02-19 09:31:52 +0100 | [diff] [blame] | 17 | from tools.jdk import GetJavaExecutable |
| 18 | |
| 19 | |
| 20 | KOTLIN_FMT_JAR = path.join( |
| 21 | 'third_party', |
| 22 | 'google', |
| 23 | 'google-kotlin-format', |
| 24 | '0.54', |
| 25 | 'ktfmt-0.54-jar-with-dependencies.jar') |
| 26 | |
| 27 | KOTLIN_FMT_SHA1 = path.join( |
| 28 | 'third_party', 'google', 'google-kotlin-format', '0.54.tar.gz.sha1') |
| 29 | KOTLIN_FMT_TGZ = path.join( |
| 30 | 'third_party', 'google', 'google-kotlin-format', '0.54.tar.gz.sha1') |
Søren Gjesse | ae696a1 | 2025-02-19 11:48:28 +0100 | [diff] [blame] | 31 | KOTLIN_FMT_IGNORE = { |
| 32 | 'src/test/java/com/android/tools/r8/kotlin/metadata/inline_class_fun_descriptor_classes_app/main.kt' |
| 33 | } |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 34 | KOTLIN_FMT_BATCH_SIZE = 100 |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 35 | |
| 36 | FMT_CMD = path.join( |
| 37 | 'third_party', |
Christoffer Quist Adamsen | bfe52fd | 2022-02-15 14:32:52 +0100 | [diff] [blame] | 38 | 'google', |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 39 | 'google-java-format', |
Søren Gjesse | b557f92 | 2024-11-22 09:31:01 +0100 | [diff] [blame] | 40 | '1.24.0', |
| 41 | 'google-java-format-1.24.0', |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 42 | 'scripts', |
| 43 | 'google-java-format-diff.py') |
| 44 | |
Søren Gjesse | ffc0619 | 2022-06-03 11:11:27 +0200 | [diff] [blame] | 45 | FMT_CMD_JDK17 = path.join('tools','google-java-format-diff.py') |
Morten Krogh-Jespersen | 145c0ea | 2023-06-27 13:43:49 +0200 | [diff] [blame] | 46 | FMT_SHA1 = path.join( |
Søren Gjesse | b557f92 | 2024-11-22 09:31:01 +0100 | [diff] [blame] | 47 | 'third_party', 'google', 'google-java-format', '1.24.0.tar.gz.sha1') |
Morten Krogh-Jespersen | 145c0ea | 2023-06-27 13:43:49 +0200 | [diff] [blame] | 48 | FMT_TGZ = path.join( |
Søren Gjesse | b557f92 | 2024-11-22 09:31:01 +0100 | [diff] [blame] | 49 | 'third_party', 'google', 'google-java-format', '1.24.0.tar.gz') |
Søren Gjesse | ffc0619 | 2022-06-03 11:11:27 +0200 | [diff] [blame] | 50 | |
Rico Wind | 62d0320 | 2018-11-30 13:43:49 +0100 | [diff] [blame] | 51 | def CheckDoNotMerge(input_api, output_api): |
| 52 | for l in input_api.change.FullDescriptionText().splitlines(): |
| 53 | if l.lower().startswith('do not merge'): |
| 54 | msg = 'Your cl contains: \'Do not merge\' - this will break WIP bots' |
| 55 | return [output_api.PresubmitPromptWarning(msg, [])] |
| 56 | return [] |
| 57 | |
Morten Krogh-Jespersen | ebc876f | 2020-04-01 10:58:02 +0200 | [diff] [blame] | 58 | def CheckFormatting(input_api, output_api, branch): |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 59 | seen_kotlin_error = False |
| 60 | seen_java_error = False |
| 61 | pending_kotlin_files = [] |
Søren Gjesse | b5ae53d | 2025-02-19 09:31:52 +0100 | [diff] [blame] | 62 | EnsureDepFromGoogleCloudStorage( |
| 63 | KOTLIN_FMT_JAR, KOTLIN_FMT_TGZ, KOTLIN_FMT_SHA1, 'google-kotlin-format') |
| 64 | EnsureDepFromGoogleCloudStorage( |
| 65 | FMT_CMD, FMT_TGZ, FMT_SHA1, 'google-java-format') |
Rico Wind | 62d0320 | 2018-11-30 13:43:49 +0100 | [diff] [blame] | 66 | results = [] |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 67 | for f in input_api.AffectedFiles(): |
| 68 | path = f.LocalPath() |
Søren Gjesse | b5ae53d | 2025-02-19 09:31:52 +0100 | [diff] [blame] | 69 | if not path.endswith('.java') and not path.endswith('.kt'): |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 70 | continue |
Søren Gjesse | e86a4a3 | 2025-02-19 12:41:34 +0100 | [diff] [blame] | 71 | if path.endswith('.kt'): |
| 72 | if path in KOTLIN_FMT_IGNORE: |
| 73 | continue |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 74 | pending_kotlin_files.append(path) |
| 75 | if len(pending_kotlin_files) == KOTLIN_FMT_BATCH_SIZE: |
Rico Wind | f387fb2 | 2025-02-20 09:01:53 +0100 | [diff] [blame] | 76 | seen_kotlin_error = (CheckKotlinFormatting(pending_kotlin_files, output_api, results) |
| 77 | or seen_kotlin_error) |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 78 | pending_kotlin_files = [] |
Søren Gjesse | e86a4a3 | 2025-02-19 12:41:34 +0100 | [diff] [blame] | 79 | else: |
Rico Wind | f387fb2 | 2025-02-20 09:01:53 +0100 | [diff] [blame] | 80 | seen_java_error = (CheckJavaFormatting(path, branch, output_api, results) |
| 81 | or seen_java_error) |
Søren Gjesse | e86a4a3 | 2025-02-19 12:41:34 +0100 | [diff] [blame] | 82 | # Check remaining Kotlin files if any. |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 83 | if len(pending_kotlin_files) > 0: |
Rico Wind | f387fb2 | 2025-02-20 09:01:53 +0100 | [diff] [blame] | 84 | seen_kotlin_error = (CheckKotlinFormatting(pending_kotlin_files, output_api, results) |
| 85 | or seen_kotlin_error) |
Søren Gjesse | e86a4a3 | 2025-02-19 12:41:34 +0100 | [diff] [blame] | 86 | # Provide the reformatting commands if needed. |
| 87 | if seen_kotlin_error: |
| 88 | results.append(output_api.PresubmitError(KotlinFormatPresubmitMessage())) |
| 89 | if seen_java_error: |
| 90 | results.append(output_api.PresubmitError(JavaFormatPresubMessage())) |
| 91 | |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 92 | # Comment this out to easily presumbit changes |
| 93 | # results.append(output_api.PresubmitError("TESTING")) |
| 94 | return results |
| 95 | |
| 96 | |
| 97 | def CheckKotlinFormatting(paths, output_api, results): |
| 98 | cmd = [GetJavaExecutable(), '-jar', KOTLIN_FMT_JAR, '--google-style', '-n'] |
| 99 | cmd.extend(paths) |
| 100 | result = check_output(cmd) |
| 101 | if len(result) > 0: |
Søren Gjesse | e86a4a3 | 2025-02-19 12:41:34 +0100 | [diff] [blame] | 102 | with_format_error = result.splitlines() |
| 103 | for path in with_format_error: |
| 104 | results.append( |
| 105 | output_api.PresubmitError( |
| 106 | "File {path} needs formatting".format(path=path.decode('utf-8')))) |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 107 | return len(result) > 0 |
| 108 | |
| 109 | |
| 110 | def KotlinFormatPresubmitMessage(): |
| 111 | return """Please fix the Kotlin formatting by running: |
Ian Zerny | 67f4945 | 2023-05-25 13:12:45 +0200 | [diff] [blame] | 112 | |
Søren Gjesse | b5ae53d | 2025-02-19 09:31:52 +0100 | [diff] [blame] | 113 | git diff $(git cl upstream) --name-only "*.kt" | xargs {java} -jar {fmt_jar} --google-style |
| 114 | |
| 115 | or fix formatting, commit and upload: |
| 116 | |
| 117 | git diff $(git cl upstream) --name-only "*.kt" | xargs {java} -jar {fmt_jar} --google-style && git commit -a --amend --no-edit && git cl upload |
| 118 | |
| 119 | or bypass the checks with: |
| 120 | |
| 121 | git cl upload --bypass-hooks |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 122 | """.format(java=GetJavaExecutable(), fmt_jar=KOTLIN_FMT_JAR) |
Søren Gjesse | b5ae53d | 2025-02-19 09:31:52 +0100 | [diff] [blame] | 123 | |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 124 | |
| 125 | def CheckJavaFormatting(path, branch, output_api, results): |
| 126 | diff = check_output( |
| 127 | ['git', 'diff', '--no-prefix', '-U0', branch, '--', path]) |
| 128 | |
| 129 | proc = Popen(FMT_CMD, stdin=PIPE, stdout=PIPE, stderr=STDOUT) |
| 130 | (stdout, stderr) = proc.communicate(input=diff) |
| 131 | if len(stdout) > 0: |
| 132 | results.append(output_api.PresubmitError(stdout.decode('utf-8'))) |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 133 | return len(stdout) > 0 |
| 134 | |
| 135 | |
| 136 | def JavaFormatPresubMessage(): |
| 137 | return """Please fix the Java formatting by running: |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 138 | |
| 139 | git diff -U0 $(git cl upstream) | %s -p1 -i |
| 140 | |
Søren Gjesse | 641b9ab | 2020-10-08 13:28:37 +0200 | [diff] [blame] | 141 | or fix formatting, commit and upload: |
| 142 | |
| 143 | git diff -U0 $(git cl upstream) | %s -p1 -i && git commit -a --amend --no-edit && git cl upload |
| 144 | |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 145 | or bypass the checks with: |
| 146 | |
Søren Gjesse | 641b9ab | 2020-10-08 13:28:37 +0200 | [diff] [blame] | 147 | git cl upload --bypass-hooks |
Søren Gjesse | ffc0619 | 2022-06-03 11:11:27 +0200 | [diff] [blame] | 148 | |
| 149 | If formatting fails with 'No enum constant javax.lang.model.element.Modifier.SEALED' try |
| 150 | |
| 151 | git diff -U0 $(git cl upstream) | %s %s %s -p1 -i && git commit -a --amend --no-edit && git cl upload |
| 152 | """ % ( |
| 153 | FMT_CMD, |
| 154 | FMT_CMD, |
| 155 | FMT_CMD_JDK17, |
| 156 | '--google-java-format-jar', |
Søren Gjesse | 853e863 | 2025-02-19 12:37:50 +0100 | [diff] [blame] | 157 | 'third_party/google/google-java-format/1.24.0/google-java-format-1.24.0-all-deps.jar') |
| 158 | |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 159 | |
Morten Krogh-Jespersen | ebc876f | 2020-04-01 10:58:02 +0200 | [diff] [blame] | 160 | def CheckDeterministicDebuggingChanged(input_api, output_api, branch): |
Morten Krogh-Jespersen | 30d1f1b | 2020-03-26 11:39:19 +0100 | [diff] [blame] | 161 | for f in input_api.AffectedFiles(): |
| 162 | path = f.LocalPath() |
| 163 | if not path.endswith('InternalOptions.java'): |
| 164 | continue |
Morten Krogh-Jespersen | 30d1f1b | 2020-03-26 11:39:19 +0100 | [diff] [blame] | 165 | diff = check_output( |
Ian Zerny | 67f4945 | 2023-05-25 13:12:45 +0200 | [diff] [blame] | 166 | ['git', 'diff', '--no-prefix', '-U0', branch, '--', path]).decode('utf-8') |
Morten Krogh-Jespersen | 30d1f1b | 2020-03-26 11:39:19 +0100 | [diff] [blame] | 167 | if 'DETERMINISTIC_DEBUGGING' in diff: |
| 168 | return [output_api.PresubmitError(diff)] |
| 169 | return [] |
| 170 | |
Morten Krogh-Jespersen | 3285e46 | 2020-04-01 13:10:46 +0200 | [diff] [blame] | 171 | def CheckForAddedDisassemble(input_api, output_api): |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 172 | results = [] |
Morten Krogh-Jespersen | 3285e46 | 2020-04-01 13:10:46 +0200 | [diff] [blame] | 173 | for (file, line_nr, line) in input_api.RightHandSideLines(): |
Ian Zerny | a84bc1b | 2020-08-11 09:47:00 +0200 | [diff] [blame] | 174 | if file.LocalPath().endswith('.java') and '.disassemble()' in line: |
Morten Krogh-Jespersen | 3285e46 | 2020-04-01 13:10:46 +0200 | [diff] [blame] | 175 | results.append( |
| 176 | output_api.PresubmitError( |
Ian Zerny | a84bc1b | 2020-08-11 09:47:00 +0200 | [diff] [blame] | 177 | 'Test call to disassemble\n%s:%s %s' % (file.LocalPath(), line_nr, line))) |
Morten Krogh-Jespersen | ebc876f | 2020-04-01 10:58:02 +0200 | [diff] [blame] | 178 | return results |
| 179 | |
Morten Krogh-Jespersen | 3285e46 | 2020-04-01 13:10:46 +0200 | [diff] [blame] | 180 | def CheckForCopyRight(input_api, output_api, branch): |
| 181 | results = [] |
| 182 | for f in input_api.AffectedSourceFiles(None): |
| 183 | # Check if it is a new file. |
| 184 | if f.OldContents(): |
| 185 | continue |
| 186 | contents = f.NewContents() |
| 187 | if (not contents) or (len(contents) == 0): |
| 188 | continue |
Christoffer Quist Adamsen | c1e18d6 | 2020-04-15 08:45:01 +0200 | [diff] [blame] | 189 | if not CopyRightInContents(f, contents): |
Morten Krogh-Jespersen | 3285e46 | 2020-04-01 13:10:46 +0200 | [diff] [blame] | 190 | results.append( |
Morten Krogh-Jespersen | f997786 | 2021-09-14 12:36:28 +0200 | [diff] [blame] | 191 | output_api.PresubmitError('Could not find correctly formatted ' |
| 192 | 'copyright in file: %s' % f)) |
Morten Krogh-Jespersen | 3285e46 | 2020-04-01 13:10:46 +0200 | [diff] [blame] | 193 | return results |
| 194 | |
Christoffer Quist Adamsen | c1e18d6 | 2020-04-15 08:45:01 +0200 | [diff] [blame] | 195 | def CopyRightInContents(f, contents): |
Søren Gjesse | 09c3dd0 | 2021-06-17 08:21:51 +0200 | [diff] [blame] | 196 | expected = '//' |
| 197 | if f.LocalPath().endswith('.py') or f.LocalPath().endswith('.sh'): |
| 198 | expected = '#' |
Morten Krogh-Jespersen | f997786 | 2021-09-14 12:36:28 +0200 | [diff] [blame] | 199 | expected = expected + ' Copyright (c) ' + str(datetime.datetime.now().year) |
Morten Krogh-Jespersen | 3285e46 | 2020-04-01 13:10:46 +0200 | [diff] [blame] | 200 | for content_line in contents: |
Christoffer Quist Adamsen | c1e18d6 | 2020-04-15 08:45:01 +0200 | [diff] [blame] | 201 | if expected in content_line: |
Morten Krogh-Jespersen | 3285e46 | 2020-04-01 13:10:46 +0200 | [diff] [blame] | 202 | return True |
| 203 | return False |
| 204 | |
Morten Krogh-Jespersen | ebc876f | 2020-04-01 10:58:02 +0200 | [diff] [blame] | 205 | def CheckChange(input_api, output_api): |
| 206 | branch = ( |
| 207 | check_output(['git', 'cl', 'upstream']) |
Ian Zerny | 4eff3b6 | 2023-05-15 17:09:03 +0200 | [diff] [blame] | 208 | .decode('utf-8') |
Morten Krogh-Jespersen | ebc876f | 2020-04-01 10:58:02 +0200 | [diff] [blame] | 209 | .strip() |
| 210 | .replace('refs/heads/', '')) |
| 211 | results = [] |
Rico Wind | 62d0320 | 2018-11-30 13:43:49 +0100 | [diff] [blame] | 212 | results.extend(CheckDoNotMerge(input_api, output_api)) |
Morten Krogh-Jespersen | ebc876f | 2020-04-01 10:58:02 +0200 | [diff] [blame] | 213 | results.extend(CheckFormatting(input_api, output_api, branch)) |
| 214 | results.extend( |
| 215 | CheckDeterministicDebuggingChanged(input_api, output_api, branch)) |
Morten Krogh-Jespersen | 3285e46 | 2020-04-01 13:10:46 +0200 | [diff] [blame] | 216 | results.extend(CheckForAddedDisassemble(input_api, output_api)) |
| 217 | results.extend(CheckForCopyRight(input_api, output_api, branch)) |
Rico Wind | 62d0320 | 2018-11-30 13:43:49 +0100 | [diff] [blame] | 218 | return results |
Ian Zerny | daac9c5 | 2020-03-03 10:57:17 +0100 | [diff] [blame] | 219 | |
| 220 | def CheckChangeOnCommit(input_api, output_api): |
| 221 | return CheckChange(input_api, output_api) |
| 222 | |
| 223 | def CheckChangeOnUpload(input_api, output_api): |
| 224 | return CheckChange(input_api, output_api) |