org.gradle.internal.buildevents.BuildExceptionReporterTest.groovy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
The newest version!
/*
* Copyright 2016 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.gradle.internal.buildevents
import org.gradle.BuildResult
import org.gradle.StartParameter
import org.gradle.api.GradleException
import org.gradle.api.internal.artifacts.ivyservice.TypedResolveException
import org.gradle.api.invocation.Gradle
import org.gradle.api.logging.LogLevel
import org.gradle.api.logging.configuration.LoggingConfiguration
import org.gradle.api.logging.configuration.ShowStacktrace
import org.gradle.api.problems.internal.Problem
import org.gradle.api.problems.internal.ProblemAwareFailure
import org.gradle.api.tasks.TaskExecutionException
import org.gradle.execution.MultipleBuildFailures
import org.gradle.initialization.BuildClientMetaData
import org.gradle.internal.enterprise.core.GradleEnterprisePluginManager
import org.gradle.internal.exceptions.CompilationFailedIndicator
import org.gradle.internal.exceptions.ContextAwareException
import org.gradle.internal.exceptions.DefaultMultiCauseException
import org.gradle.internal.exceptions.FailureResolutionAware
import org.gradle.internal.exceptions.LocationAwareException
import org.gradle.internal.logging.DefaultLoggingConfiguration
import org.gradle.internal.logging.text.StyledTextOutputFactory
import org.gradle.internal.logging.text.TestStyledTextOutput
import spock.lang.Specification
import java.lang.reflect.Field
class BuildExceptionReporterTest extends Specification {
final TestStyledTextOutput output = new TestStyledTextOutput()
final StyledTextOutputFactory factory = Mock()
final BuildClientMetaData clientMetaData = Mock()
final GradleEnterprisePluginManager gradleEnterprisePluginManager = Mock()
final LoggingConfiguration configuration = new DefaultLoggingConfiguration()
final BuildExceptionReporter reporter = new BuildExceptionReporter(factory, configuration, clientMetaData, gradleEnterprisePluginManager)
static final String MESSAGE = ""
static final String FAILURE = ''
static final String LOCATION = ""
static final String STACKTRACE = "{info}> {normal}Run with {userinput}--stacktrace{normal} option to get the stack trace."
static final String INFO_OR_DEBUG = "{info}> {normal}Run with {userinput}--info{normal} or {userinput}--debug{normal} option to get more log output."
static final String TRY_SCAN = "{info}> {normal}Run with {userinput}--scan{normal} to get full insights."
static final String GET_HELP = "{info}> {normal}Get more help at {userinput}https://help.gradle.org{normal}."
def setup() {
factory.create(BuildExceptionReporter.class, LogLevel.ERROR) >> output
clientMetaData.describeCommand(!null, !null) >> { args -> args[0].append("[gradle ${args[1].join(' ')}]") }
}
def doesNothingWhenBuildIsSuccessful() {
expect:
reporter.buildFinished(result(null))
output.value == ''
}
def reportsBuildFailure() {
GradleException exception = new GradleException(MESSAGE);
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
$MESSAGE
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def "does not suggest to use --scan if option was on command line"() {
GradleException exception = new GradleException(MESSAGE);
def result = result(exception)
result.gradle >> Mock(Gradle) {
getStartParameter() >> Mock(StartParameter) {
isBuildScan() >> true
isNoBuildScan() >> false
}
}
gradleEnterprisePluginManager.isPresent() >> true
expect:
reporter.buildFinished(result)
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
$MESSAGE
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$GET_HELP
"""
}
def "does not suggest to use --scan if --no-scan is on command line"() {
GradleException exception = new GradleException(MESSAGE);
def result = result(exception)
result.gradle >> Mock(Gradle) {
getStartParameter() >> Mock(StartParameter) {
isBuildScan() >> false
isNoBuildScan() >> true
}
}
gradleEnterprisePluginManager.isPresent() >> true
expect:
reporter.buildFinished(result)
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
$MESSAGE
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$GET_HELP
"""
}
def reportsBuildFailureWhenFailureHasNoMessage() {
GradleException exception = new GradleException();
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
org.gradle.api.GradleException (no error message)
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def reportsLocationAwareException() {
Throwable exception = new LocationAwareException(new RuntimeException(MESSAGE, new RuntimeException("")), LOCATION, 42)
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* Where:
$LOCATION line: 42
* What went wrong:
$MESSAGE
{info}> {normal}
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def reportsLocationAwareExceptionWithNoMessage() {
Throwable exception = new LocationAwareException(new RuntimeException(new IOException()), LOCATION, 42)
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* Where:
$LOCATION line: 42
* What went wrong:
java.io.IOException
{info}> {normal}java.io.IOException (no error message)
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def reportsLocationAwareExceptionWithMultipleCauses() {
Throwable exception = new LocationAwareException(new DefaultMultiCauseException(MESSAGE, new RuntimeException(""), new RuntimeException("")), LOCATION, 42)
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* Where:
$LOCATION line: 42
* What went wrong:
$MESSAGE
{info}> {normal}
{info}> {normal}
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def reportsLocationAwareExceptionWithMultipleNestedCauses() {
def cause1 = new DefaultMultiCauseException("", new RuntimeException(""), new RuntimeException(""))
def cause2 = new DefaultMultiCauseException("", new RuntimeException(""))
Throwable exception = new LocationAwareException(new DefaultMultiCauseException(MESSAGE, cause1, cause2), LOCATION, 42)
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* Where:
$LOCATION line: 42
* What went wrong:
$MESSAGE
{info}> {normal}
{info}> {normal}
{info}> {normal}
{info}> {normal}
{info}> {normal}
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def reportsLocationAwareExceptionWhenCauseHasNoMessage() {
Throwable exception = new LocationAwareException(new RuntimeException(MESSAGE, new RuntimeException()), LOCATION, 42)
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* Where:
$LOCATION line: 42
* What went wrong:
$MESSAGE
{info}> {normal}java.lang.RuntimeException (no error message)
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def showsStacktraceOfCauseOfLocationAwareException() {
configuration.showStacktrace = ShowStacktrace.ALWAYS
Throwable exception = new LocationAwareException(new GradleException(MESSAGE, new GradleException(FAILURE)), LOCATION, 42)
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* Where:
$LOCATION line: 42
* What went wrong:
$MESSAGE
{info}> {normal}$FAILURE
* Try:
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
* Exception is:
org.gradle.api.GradleException: $MESSAGE
{stacktrace}
Caused by: org.gradle.api.GradleException: $FAILURE
{stacktrace}
"""
}
def "report multiple failures and skip help link for NonGradleCauseException"() {
def failure1 = new LocationAwareException(new TaskExecutionException(null, new TestNonGradleCauseException()), LOCATION, 42)
def failure2 = new LocationAwareException(new TaskExecutionException(null, new TestCompilationFailureException()), LOCATION, 42)
def failure3 = new RuntimeException("")
Throwable exception = new MultipleBuildFailures([failure1, failure2, failure3])
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: Build completed with 3 failures.{normal}
{failure}1: {normal}{failure}Task failed with an exception.{normal}
-----------
* Where:
$LOCATION line: 42
* What went wrong:
Execution failed for null.
{info}> {normal}org.gradle.internal.buildevents.TestNonGradleCauseException (no error message)
* Try:
$TRY_SCAN
==============================================================================
{failure}2: {normal}{failure}Task failed with an exception.{normal}
-----------
* Where:
$LOCATION line: 42
* What went wrong:
Execution failed for null.
{info}> {normal}org.gradle.internal.buildevents.TestCompilationFailureException (no error message)
* Try:
$TRY_SCAN
==============================================================================
{failure}3: {normal}{failure}Task failed with an exception.{normal}
-----------
* What went wrong:
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
==============================================================================
""";
}
def reportsBuildFailureWhenShowStacktraceEnabled() {
configuration.showStacktrace = ShowStacktrace.ALWAYS
GradleException exception = new GradleException(MESSAGE)
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
$MESSAGE
* Try:
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
* Exception is:
org.gradle.api.GradleException: $MESSAGE
{stacktrace}
"""
}
def reportsBuildFailureWhenShowFullStacktraceEnabled() {
configuration.showStacktrace = ShowStacktrace.ALWAYS_FULL
GradleException exception = new GradleException(MESSAGE)
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
$MESSAGE
* Try:
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
* Exception is:
org.gradle.api.GradleException: $MESSAGE
{stacktrace}
"""
}
def includesResolutionsFromExceptionWhenItImplementsFailureResolutionAware() {
def exception = new TestException() {
@Override
void appendResolutions(FailureResolutionAware.Context context) {
context.appendResolution { output -> output.append("resolution 1.") }
context.appendResolution { output -> output.append("resolution 2.") }
}
}
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
$MESSAGE
* Try:
{info}> {normal}resolution 1.
{info}> {normal}resolution 2.
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def doesNotSuggestGeneratingAScanWhenTheBuildIsMissing() {
def exception = new TestException() {
@Override
void appendResolutions(FailureResolutionAware.Context context) {
context.doNotSuggestResolutionsThatRequireBuildDefinition()
context.appendResolution { output -> output.append("resolution 1.") }
}
}
expect:
reporter.buildFinished(result(exception))
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
$MESSAGE
* Try:
{info}> {normal}resolution 1.
$STACKTRACE
$INFO_OR_DEBUG
$GET_HELP
"""
}
// region Duplicate Exception Branch Filtering
def "multi-cause exceptions have branches with identical root causes summarized properly"() {
def ultimateCause = new RuntimeException("ultimate cause")
def branch1 = new DefaultMultiCauseException("first failure", ultimateCause)
def branch2 = new DefaultMultiCauseException("second failure", ultimateCause)
Throwable exception = new ContextAwareException(new TypedResolveException("task dependencies", "org:example:1.0", [branch1, branch2]))
when:
reporter.buildFinished(result(exception))
print(output.value)
then:
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
Could not resolve all task dependencies for org:example:1.0.
{info}> {normal}first failure
{info}> {normal}ultimate cause
{info}> {normal}There is 1 more failure with an identical cause.
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def "multi-cause exceptions have branches with identical root causes and additional intermediate failures summarized properly"() {
def ultimateCause = new RuntimeException("ultimate cause")
def branch1 = new DefaultMultiCauseException("first failure", ultimateCause)
def branch2 = new DefaultMultiCauseException("second failure", ultimateCause)
def intermediateFailure = new DefaultMultiCauseException("intermediate failure", ultimateCause)
def branch3 = new DefaultMultiCauseException("third failure", intermediateFailure)
Throwable exception = new ContextAwareException(new TypedResolveException("task dependencies", "org:example:1.0", [branch1, branch2, branch3]))
when:
reporter.buildFinished(result(exception))
print(output.value)
then:
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
Could not resolve all task dependencies for org:example:1.0.
{info}> {normal}first failure
{info}> {normal}ultimate cause
{info}> {normal}There are 2 more failures with identical causes.
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def "multi-cause exceptions have branches with identical root causes summarized properly when ultimate cause is self-caused"() {
def ultimateCause = new RuntimeException("ultimate cause")
Field field = Throwable.class.getDeclaredField("cause")
field.setAccessible(true)
field.set(ultimateCause, ultimateCause)
def branch1 = new DefaultMultiCauseException("first failure", ultimateCause)
def branch2 = new DefaultMultiCauseException("second failure", ultimateCause)
Throwable exception = new ContextAwareException(new TypedResolveException("task dependencies", "org:example:1.0", [branch1, branch2]))
when:
reporter.buildFinished(result(exception))
print(output.value)
then:
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
Could not resolve all task dependencies for org:example:1.0.
{info}> {normal}first failure
{info}> {normal}ultimate cause
{info}> {normal}There is 1 more failure with an identical cause.
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
def "multi-cause exceptions with multiple branches with a set of identical root causes are summarized properly"() {
def ultimateCause1 = new RuntimeException("ultimate cause 1")
def branch1 = new DefaultMultiCauseException("first failure", ultimateCause1)
def branch2 = new DefaultMultiCauseException("second failure", ultimateCause1)
def intermediateFailure1 = new DefaultMultiCauseException("intermediate failure", ultimateCause1)
def branch3 = new DefaultMultiCauseException("third failure", intermediateFailure1)
def ultimateCause2 = new RuntimeException("ultimate cause 2")
def branch4 = new DefaultMultiCauseException("forth failure", ultimateCause2)
def branch5 = new DefaultMultiCauseException("fifth failure", ultimateCause2)
def intermediateFailure2 = new DefaultMultiCauseException("intermediate failure 2", ultimateCause2)
def branch6 = new DefaultMultiCauseException("sixth failure", intermediateFailure2)
Throwable exception = new ContextAwareException(new TypedResolveException("task dependencies", "org:example:1.0", [branch1, branch2, branch3, branch4, branch5, branch6]))
when:
reporter.buildFinished(result(exception))
print(output.value)
then:
output.value == """
{failure}FAILURE: {normal}{failure}Build failed with an exception.{normal}
* What went wrong:
Could not resolve all task dependencies for org:example:1.0.
{info}> {normal}first failure
{info}> {normal}ultimate cause 1
{info}> {normal}forth failure
{info}> {normal}ultimate cause 2
{info}> {normal}There are 4 more failures with identical causes.
* Try:
$STACKTRACE
$INFO_OR_DEBUG
$TRY_SCAN
$GET_HELP
"""
}
// endregion Duplicate Exception Branch Filtering
def result(Throwable failure) {
BuildResult result = Mock()
result.failure >> failure
result
}
abstract class TestException extends GradleException implements FailureResolutionAware {
TestException() {
super(MESSAGE)
}
}
class TestProblemAwareFailure extends Throwable implements CompilationFailedIndicator, ProblemAwareFailure {
List problems
TestProblemAwareFailure(Problem... problems) {
super("")
this.problems = Arrays.asList(problems)
}
@Override
Collection getProblems() {
return problems
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy