Gradle Release Notes

We are excited to announce Gradle 9.3.0-20251208194200+0000 (released 2025-12-08).

This release brings test reporting improvements, including an improved HTML test report for nested, parameterized, and suite-based tests, with better aggregate reporting.

It also adds error and warning reporting improvements, with Problems API reports now rendered in the console when using --warning-mode=all and clearer explanations for some exit codes.

Finally, build authoring enhancements include a new method on AttributeContainer and a new streaming API in TestKit for efficiently reading build output.

We would like to thank the following community members for their contributions to this release of Gradle: Adam, Adam, Aharnish Solanki, Andrzej Zabost, Björn Kautler, Boris Petrov, Jendrik Johannes, Kamil Krzywanski, KANAKALA SAI KIRAN, Megmeehey, NurmukhametovAlexey, Philip Wedemann, Piotr Kubowicz, Samay Kumar, Shin Minjun, Stefan Oehme, Vincent Potuček, Yongshun Ye.

Be sure to check out the public roadmap for insight into what's planned for future releases.

Table Of Contents

Upgrade instructions

Switch your build to use Gradle 9.3.0-20251208194200+0000 by updating the wrapper in your project:

./gradlew wrapper --gradle-version=9.3.0-20251208194200+0000 && ./gradlew wrapper

See the Gradle 9.x upgrade guide to learn about deprecations, breaking changes, and other considerations when upgrading to Gradle 9.3.0-20251208194200+0000.

For Java, Groovy, Kotlin, and Android compatibility, see the full compatibility notes.

New features and usability improvements

Test reporting improvements

Gradle provides a rich set of features and abstractions for testing JVM code, along with test reports to display results.

Test results reporting

The HTML test report generated by the test task, TestReport, and other AbstractTestTask usages now presents test results in a clearer structure that reflects how tests are defined:

test-results-reporting.png

This results from Gradle adopting the incubating Test Event Reporting API internally.

In the following sections, "Suite" refers specifically to the suite features of JUnit 4, JUnit Jupiter, and TestNG while "Container" refers to a general grouping of tests, such as a class.

Nested test changes

For JUnit 4 and JUnit Jupiter, nested test classes are now shown nested under their enclosing class in the HTML Test Report.

For example, if you had an OuterClass with an InnerClass nested inside it, it would previously be shown as:

+ OuterClass$InnerClass
|-- someTestMethodInInnerClass

It will now be reported in the HTML Test Report as:

+ OuterClass
|-+ InnerClass (or OuterClass$InnerClass for JUnit 4)
  |-- someTestMethodInInnerClass

With multiple inner classes:

+ OuterClass
|-+ InnerClass1 (or OuterClass$InnerClass1 for JUnit 4)
  |-- someTestMethodInInnerClass1
  + InnerClass2 (or OuterClass$InnerClass2 for JUnit 4)
  |-- someTestMethodInInnerClass2

The XML report remains the same, nested classes are still written as TEST-OuterClass$InnerClass.xml.

Parameterized test changes

Parameterized tests now create a suite that contains all the parameterized test cases for a given method.

For example, if you had a class TestClass with two parameterized paramTest1/2 methods, it would previously be shown as:

+ TestClass
|-- paramTest1[0]
|-- paramTest1[1]
|-- paramTest1[2]
|-- paramTest2[a]
|-- paramTest2[b]
|-- paramTest2[c]

It will now be reported in the HTML Test Report as:

+ TestClass
|-+ paramTest1
  |-- paramTest1[0]
  |-- paramTest1[1]
  |-- paramTest1[2]
|-+ paramTest2
  |-- paramTest2[a]
  |-- paramTest2[b]
  |-- paramTest2[c]
Suite changes

Suites now contain the classes they run, rather than those classes being shown as siblings.

For example, if you had a suite AllTests that ran TestClass1 and TestClass2, it would previously be shown as:

+ AllTests
+ TestClass1
|-- someTestMethodInClass1
+ TestClass2
|-- someTestMethodInClass2

It will now be reported in the HTML Test Report as:

+ AllTests
|-+ TestClass1
  |-- someTestMethodInClass1
|-+ TestClass2
  |-- someTestMethodInClass2

In the XML report, only the class report is emitted (TEST-SomeTestClass.xml).

Package suite changes

Packages are no longer represented as containers in the HTML report.

For example, if you had classes org.example.FirstTest and org.example.SecondTest, it would previously be shown as:

+ org.example
  |-- FirstTest
  |-- SecondTest

It will now be reported in the HTML Test Report as:

+ org.example.FirstTest
+ org.example.SecondTest

There are two main reasons for this change:

  1. With support for non-class-based testing, Gradle cannot reliably determine if a container corresponds to a class, so synthesizing a package container can be misleading.
  2. This behavior now aligns more closely with how testing frameworks and IDEs group tests.
Test standard output/error changes

Standard output and standard error (collectively, "output") are no longer combined from individual tests to the container/class level. Instead, the output stays attached to the individual test that produced it:

test-results-test.png

This makes it easier to locate output that is relevant to a specific test.

test-results-standard-output.png

@Before and @After class output is now associated with the correct class in JUnit 4, JUnit Jupiter, and TestNG (starting with TestNG 6.9.13.3). @Before and @After suite or container output is now associated with the correct suite or container for JUnit 4 (starting with JUnit 4.13), JUnit Jupiter, and TestNG (starting with TestNG 6.19.13.3):

test-results-standard-error.png

Aggregate report changes

The Test Report Aggregation Plugin provides tasks and configurations used to aggregate the results of multiple Test task invocations into a single HTML report.
This report, which can also be generated manually with TestReport, now supports overlapping test structures.

For example, when a suite with the same name exists in multiple subprojects:

Previously, the report looked like this when a failure occurred:

old-aggregate-report.png

Now, each individual report source is represented as a separate tab. To see the tests from a specific source, select its corresponding tab:

new-aggregate-report.png

Error and warning reporting improvements

Gradle provides a rich set of error and warning messages to help you understand and resolve problems in your build.

Simple console rendering for Problem Reports

The Problems API provides structured feedback on build issues, helping developers and tools like IDEs identify and resolve warnings, errors, or deprecations during configuration or runtime.

Previously, a limitation was that the problem report was linked to in the console output, but the problems themselves were not displayed:

$ ./gradlew :test --warning-mode=all

> Configure project :
[Incubating] Problems report is available at: file:///Users/user/test-report-aggregation-sample/build/reports/problems/problems-report.html

In this release, we've added basic console integration. Relevant problems in the report are now rendered in the console output when you use --warning-mode=all:

$ ./gradlew :test --warning-mode=all

> Configure project :
Build file '/Users/lkassovic/Downloads/test-report-aggregation-sample/build.gradle': line 16
The Wrapper.getAvailableDistributionTypes method has been deprecated...
        at build_5teuix0v7qf7ou93kgnmvnicp.run(/Users/user/test-report-aggregation-sample/build.gradle:16)
        (Run with --stacktrace to get the full stack trace of this deprecation warning.)
[Incubating] Problems report is available at: file:///Users/user/test-report-aggregation-sample/build/reports/problems/problems-report.html

Clearer explanation for worker process exit codes

Gradle now provides a short explanation when a worker process exits with a code that typically indicates it was killed by the operating system.

Previously, you would see:

> Process 'Gradle Worker Daemon 87' finished with non-zero exit value 137

Now, Gradle adds a helpful hint:

> Process 'Gradle Worker Daemon 87' finished with non-zero exit value 137 
(this value may indicate that the process was terminated with the SIGKILL 
signal, which is often caused by the system running out of memory)

This makes it easier to understand failures caused by conditions such as running out of memory.

Build authoring improvements

Gradle provides rich APIs for plugin authors and build engineers to develop custom build logic.

New AttributeContainer.named() method

An AttributeContainer stores attributes that Gradle uses to pick the right variant of a dependency during resolution.
This release introduces the new convenience method on AttributeContainer.named().
This method can create attribute values directly from the container without requiring the use of the ObjectFactory.

This method makes attribute assignment more concise while preserving the same semantics as creating a named value via the ObjectFactory:

configurations.resolvable("foo") {
    attributes {
        // Before: 
        attribute(Usage.USAGE_ATTRIBUTE, objects.named("red"))
        
        // After:
        attribute(Usage.USAGE_ATTRIBUTE, named("red"))
    }
}

Stream TestKit output

In Gradle’s TestKit, BuildResult is the object that represents the outcome of a test build you ran with GradleRunner.
BuildResult now offers a new method for accessing the build console output efficiently, especially for builds that produce a large volume of logs.

BuildResult.getOutput() returns a String with the full build console output.
This can use large amounts of memory for builds with extensive logs.

A new BuildResult.getOutputReader() method is available, returning a BufferedReader for streaming the build output incrementally. This can help reduce memory pressure in TestKit tests.

Ensure you close the BufferedReader after use.
We recommend the standard Java try-with-resources pattern for this:

void testProject() {
    BuildResult buildResult = GradleRunner.create()
        .withProjectDir(File("test-project"))
        .withArguments(":build", "--info")
        .build();

    try (BufferedReader outputReader = buildResult.getOutputReader()) {
        List<String> logLines = outputReader.lines()
            .filter(line -> line.contains("example build message"))
            .collect(Collectors.toList());
        // do something with the log lines...
    }
}

Promoted features are features that were incubating in previous versions of Gradle but are now supported and subject to backward compatibility. See the User Manual section on the "Feature Lifecycle" for more information.

The following are the features that have been promoted in this Gradle release.

Documentation and training

User Manual

The Plugin Development section has been updated with clearer explanations and coded examples of init scripts and init plugins, including when to use each.

Best Practices

The following best practices have been added in this Gradle release:

Training

The following course is now available:

The following learning path is now available:

Fixed issues

Known issues

Known issues are problems that were discovered post-release that are directly related to changes made in this release.

External contributions

We love getting contributions from the Gradle community. For information on contributing, please see gradle.org/contribute.

Reporting problems

If you find a problem with this release, please file a bug on GitHub Issues adhering to our issue guidelines. If you're not sure if you're encountering a bug, please use the forum.

We hope you will build happiness with Gradle, and we look forward to your feedback via Twitter or on GitHub.