blob: 01633e208a3e34177ffa73641001e57460a6c5f0 [file] [log] [blame]
#!/usr/bin/env python3
#
# Copyright 2025 The Chromium Authors
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Run Error Prone."""
import argparse
import sys
import compile_java
from util import server_utils
# Add a check here to cause the suggested fix to be applied while compiling.
# Use this when trying to enable more checks.
ERRORPRONE_CHECKS_TO_APPLY = [
# Be sure to first update "android/errorprone" within
# build/config/siso/android.star to set "remote": False.
]
# Checks to disable in tests.
TESTONLY_ERRORPRONE_WARNINGS_TO_DISABLE = [
# Can hurt readability to enforce this on test classes.
'FieldCanBeStatic',
# These are allowed in tests.
'NoStreams',
# Too much effort to enable.
'UnusedVariable',
]
# Full list of checks: https://errorprone.info/bugpatterns
ERRORPRONE_WARNINGS_TO_DISABLE = [
# High priority to enable in non-tests:
'ReturnValueIgnored',
'StaticAssignmentInConstructor',
# Still to look into:
'AnnotationPosition',
'AvoidObjectArrays',
'BanSerializableRead',
'BooleanParameter',
'CannotMockMethod',
'CatchingUnchecked',
'CheckedExceptionNotThrown',
'ConstantField',
'ConstantPatternCompile',
'DeduplicateConstants',
'DefaultLocale',
'DepAnn',
'DifferentNameButSame',
'ExpectedExceptionChecker',
'ForEachIterable',
'FunctionalInterfaceClash',
'IdentifierName',
'ImmutableMemberCollection',
'InconsistentOverloads',
'InitializeInline',
'InterruptedExceptionSwallowed',
'Interruption',
'MethodCanBeStatic',
'MissingDefault',
'MixedArrayDimensions',
'MockitoDoSetup',
'NegativeBoolean',
'NonCanonicalStaticMemberImport',
'NonFinalStaticField',
'PreferJavaTimeOverload',
'PreferredInterfaceType',
'PrimitiveArrayPassedToVarargsMethod',
'PrivateConstructorForUtilityClass',
'RedundantThrows',
'ReturnsNullCollection',
'StringFormatWithLiteral',
'SuppressWarningsWithoutExplanation',
'SystemExitOutsideMain',
'SystemOut',
'TestExceptionChecker',
'ThrowSpecificExceptions',
'ThrowsUncheckedException',
'TooManyParameters',
'TryFailRefactoring',
'TypeParameterNaming',
'UngroupedOverloads',
'UnnecessaryAnonymousClass',
'UnnecessaryBoxedAssignment',
'UnnecessaryDefaultInEnumSwitch',
'UnnecessaryFinal',
'UnsafeLocaleUsage',
'UnusedException',
'UseEnumSwitch',
'UsingJsr305CheckReturnValue',
'Var',
'Varifier',
'YodaCondition',
# Low priority.
'CatchAndPrintStackTrace',
'EqualsHashCode',
'JavaUtilDate',
'OverrideThrowableToString',
'ParameterComment',
'PatternMatchingInstanceof',
'StatementSwitchToExpressionSwitch',
'UndefinedEquals',
'StaticAssignmentOfThrowable', # Want in non-test
'StaticMockMember',
'StringCaseLocaleUsage',
'StringCharset',
'ThreadLocalUsage',
'TypeParameterUnusedInFormals',
'UnnecessaryBoxedVariable',
'UnnecessarilyFullyQualified',
'UnsafeReflectiveConstructionCast',
# Never Enable:
#
# Chromium uses assert statements.
'AssertFalse',
# Debatable whether it makes code less readable by forcing larger names for
# "Builder".
'BadImport',
# Such modifiers in nested classes do not hurt readability IMO.
'EffectivelyPrivate',
# Android APIs sometimes throw random exceptions that are safe to ignore.
'EmptyCatch',
# Just use Android Studio refactors to inline things.
'InlineMeInliner',
'InlineMeSuggester',
# We already have presubmit checks for this. We don't want it to fail
# local compiles.
'RemoveUnusedImports',
# Several instances of using a string right before the String.format(),
# which seems better than inlining.
'InlineFormatString',
# Assigning to fields marked as @Mock or @Spy. Suggested fix is to delete
# assignments, which would break tests in many cases.
'UnnecessaryAssignment',
# Android platform default is always UTF-8.
# https://developer.android.com/reference/java/nio/charset/Charset.html#defaultCharset()
'DefaultCharset',
# If google-java-format is not going to do this, it's not worth our time.
'StringConcatToTextBlock',
# We don't use Dagger.
'RefersToDaggerCodegen',
# Only has false positives (would not want to enable this).
'UnicodeEscape',
# Does not apply to Android because it assumes no desugaring.
'UnnecessaryLambda',
# These are best practices that I doubt are worth the churn / overhead.
'MixedMutabilityReturnType',
'MutablePublicArray',
'NonApiType',
# Not that useful.
'ClassNewInstance',
# Low priority corner cases with String.split.
# Linking Guava and using Splitter was rejected
# in the https://chromium-review.googlesource.com/c/chromium/src/+/871630.
'StringSplitter',
# There are lots of times when we just want to post a task.
'FutureReturnValueIgnored',
# Just false positives in our code.
'ThreadJoinLoop',
# Fine to run the auto-fix from time to time (which replaces assert
# statements with Truth assertions), but because using assert statements is
# normal in non-test code, they also show up in test helpers, which are
# arguably false-positives.
'UseCorrectAssertInTests',
# NullAway makes these redundant.
'FieldMissingNullable',
'ParameterMissingNullable',
'ReturnMissingNullable',
# Style guide difference between google3 & chromium.
'MissingBraces',
# Does not seem to take into account R8 backports. Redundant with Android
# Lint anyways.
'AndroidJdkLibsChecker',
'Java8ApiChecker',
# Style guide difference between google3 & chromium.
'UnnecessaryTestMethodPrefix',
# Too many suggestions where it's not actually necessary.
'CanIgnoreReturnValueSuggester',
# These are all for Javadoc, which we don't really care about.
'InvalidBlockTag',
'InvalidInlineTag',
'InvalidLink',
'InvalidParam',
'MalformedInlineTag',
'MissingSummary',
'NotJavadoc',
'UnescapedEntity',
'UnrecognisedJavadocTag',
]
# Full list of checks: https://errorprone.info/bugpatterns
# Only those marked as "experimental" need to be listed here in order to be
# enabled.
ERRORPRONE_WARNINGS_TO_ENABLE = [
'BinderIdentityRestoredDangerously',
'EmptyIf',
'EqualsBrokenForNull',
'FieldCanBeFinal',
'FieldCanBeLocal',
'FieldCanBeStatic',
'InvalidThrows',
'LongLiteralLowerCaseSuffix',
'MultiVariableDeclaration',
'RedundantOverride',
'StaticQualifiedUsingExpression',
'TimeUnitMismatch',
'UnnecessaryStaticImport',
'UseBinds',
'WildcardImport',
'NoStreams',
]
def main():
parser = argparse.ArgumentParser()
parser.add_argument('--use-build-server',
action='store_true',
help='Always use the build server.')
parser.add_argument('--testonly',
action='store_true',
help='Disable some Error Prone checks')
parser.add_argument('--enable-nullaway',
action='store_true',
help='Enable NullAway (requires --enable-errorprone)')
parser.add_argument('--stamp',
required=True,
help='Path of output .stamp file')
options, compile_java_argv = parser.parse_known_args()
compile_java_argv += ['--jar-path', options.stamp]
# Use the build server for errorprone runs.
if server_utils.MaybeRunCommand(
name=options.stamp,
argv=sys.argv,
stamp_file=options.stamp,
use_build_server=options.use_build_server):
compile_java.main(compile_java_argv, write_depfile_only=True)
return
# All errorprone args are passed space-separated in a single arg.
errorprone_flags = ['-Xplugin:ErrorProne']
if options.enable_nullaway:
# See: https://github.com/uber/NullAway/wiki/Configuration
# Check nullability only for classes marked with @NullMarked (this is our
# migration story).
errorprone_flags += ['-XepOpt:NullAway:OnlyNullMarked']
errorprone_flags += [
'-XepOpt:NullAway:CustomContractAnnotations='
'org.chromium.build.annotations.Contract,'
'org.chromium.support_lib_boundary.util.Contract'
]
# TODO(agrieve): Re-enable once this is fixed:
# https://github.com/uber/NullAway/issues/1104
# errorprone_flags += ['-XepOpt:NullAway:CheckContracts=true']
# TODO(agrieve): Re-enable once we sort out nullability of
# ObservableSuppliers. https://crbug.com/430320400
# Make it a warning to use assumeNonNull() with a @NonNull.
#errorprone_flags += [('-XepOpt:NullAway:CastToNonNullMethod='
# 'org.chromium.build.NullUtil.assumeNonNull')]
# Detect "assert foo != null" as a null check.
errorprone_flags += ['-XepOpt:NullAway:AssertsEnabled=true']
# Do not ignore @Nullable & @NonNull in non-@NullMarked classes.
errorprone_flags += [
'-XepOpt:NullAway:AcknowledgeRestrictiveAnnotations=true'
]
# Treat @RecentlyNullable the same as @Nullable.
errorprone_flags += ['-XepOpt:Nullaway:AcknowledgeAndroidRecent=true']
# Enable experimental checking of @Nullable generics.
# https://github.com/uber/NullAway/wiki/JSpecify-Support
errorprone_flags += ['-XepOpt:NullAway:JSpecifyMode=true']
# Treat these the same as constructors.
# These are in addition to the default list in "DEFAULT_KNOWN_INITIALIZERS":
# https://github.com/uber/NullAway/blob/d5cb4f1190a96045d85b92c6d119e4595840cc8a/nullaway/src/main/java/com/uber/nullaway/ErrorProneCLIFlagsConfig.java#L128
init_methods = [
'android.app.backup.BackupAgent.onCreate',
'android.content.ContentProvider.attachInfo',
'android.content.ContentProvider.onCreate',
'android.content.ContextWrapper.attachBaseContext',
'androidx.preference.PreferenceFragmentCompat.onCreatePreferences',
]
errorprone_flags += [
'-XepOpt:NullAway:KnownInitializers=' + ','.join(init_methods)
]
# Exclude fields with these annotations from null-checking.
mock_annotations = [
'org.mockito.Captor',
'org.mockito.Mock',
'org.mockito.Spy',
]
errorprone_flags += [
'-XepOpt:NullAway:ExcludedFieldAnnotations=' +
','.join(mock_annotations)
]
# Make everything a warning so that when treat_warnings_as_errors is false,
# they do not fail the build.
errorprone_flags += ['-XepAllErrorsAsWarnings']
# Don't check generated files (those tagged with @Generated).
errorprone_flags += ['-XepDisableWarningsInGeneratedCode']
errorprone_flags.extend('-Xep:{}:OFF'.format(x)
for x in ERRORPRONE_WARNINGS_TO_DISABLE)
errorprone_flags.extend('-Xep:{}:WARN'.format(x)
for x in ERRORPRONE_WARNINGS_TO_ENABLE)
if options.testonly:
errorprone_flags.extend('-Xep:{}:OFF'.format(x)
for x in TESTONLY_ERRORPRONE_WARNINGS_TO_DISABLE)
errorprone_flags += ['-XepCompilingTestOnlyCode']
# TODO(40661145): Enable stricter CheckReturnValue checks:
# packages = 'org.chromium,com.google,java.util.regex'
# errorprone_flags += ['-XepOpt:CheckReturnValue:Packages=' + packages]
# errorprone_flags += ['-XepOpt:CheckReturnValue:CheckAllConstructors=true']
if ERRORPRONE_CHECKS_TO_APPLY:
to_apply = list(ERRORPRONE_CHECKS_TO_APPLY)
if options.testonly:
to_apply = [
x for x in to_apply
if x not in TESTONLY_ERRORPRONE_WARNINGS_TO_DISABLE
]
errorprone_flags += [
'-XepPatchLocation:IN_PLACE', '-XepPatchChecks:,' + ','.join(to_apply)
]
# These are required to use JDK 16, and are taken directly from
# https://errorprone.info/docs/installation
javac_args = [
'-J--add-exports=jdk.compiler/com.sun.tools.javac.api=ALL-UNNAMED',
'-J--add-exports=jdk.compiler/com.sun.tools.javac.file=ALL-UNNAMED',
'-J--add-exports=jdk.compiler/com.sun.tools.javac.main=ALL-UNNAMED',
'-J--add-exports=jdk.compiler/com.sun.tools.javac.model=ALL-UNNAMED',
'-J--add-exports=jdk.compiler/com.sun.tools.javac.parser=ALL-UNNAMED',
'-J--add-exports=jdk.compiler/com.sun.tools.javac.processing='
'ALL-UNNAMED',
'-J--add-exports=jdk.compiler/com.sun.tools.javac.tree=ALL-UNNAMED',
'-J--add-exports=jdk.compiler/com.sun.tools.javac.util=ALL-UNNAMED',
'-J--add-opens=jdk.compiler/com.sun.tools.javac.code=ALL-UNNAMED',
'-J--add-opens=jdk.compiler/com.sun.tools.javac.comp=ALL-UNNAMED',
]
javac_args += ['-XDcompilePolicy=simple', ' '.join(errorprone_flags)]
javac_args += ['-XDshould-stop.ifError=FLOW']
# This flag quits errorprone after checks and before code generation, since
# we do not need errorprone outputs, this speeds up errorprone by 4 seconds
# for chrome_java.
if not ERRORPRONE_CHECKS_TO_APPLY:
javac_args += ['-XDshould-stop.ifNoError=FLOW']
compile_java.main(compile_java_argv,
extra_javac_args=javac_args,
use_errorprone=True)
if __name__ == '__main__':
main()