Provides a simple, opinionated structure for writing TestKit-based tests for Gradle plugins and collecting code coverage from them. Contains the following components:
- The main plugin,
com.toasttab.testkit
is applied to the plugin project. - The junit5 extension injects the test project model into JUnit 5 tests.
- The coverage plugin,
com.toasttab.testkit.coverage
is applied to TestKit fixture projects.
In the plugin project, apply the main plugin and bring in the junit5 extension dependency.
plugins {
kotlin("jvm")
jacoco
id("com.toasttab.testkit") version <<version>>
}
dependencies {
testImplementation("com.toasttab.gradle.testkit:junit5:<<version>>")
}
Each test method is expected to have a corresponding test project located, by default, at src/test/projects/<<Test>>/<<method>>
.
src/test/projects/MyTest/sometest:
build.gradle.kts
settings.gradle.kts # optional, but IntelliJ will complain
The location of the test project is can be customized by specifying a custom ProjectLocator
implementation
via the @TestKit
annotation on the test class (e.g. FullyQualifiedNameProjectLocator
).
Test project files may contain Ant-style placeholders. The predefined placeholders are:
@TESTKIT_PLUGIN_VERSION@
- the version of this project@TESTKIT_INTEGRATION_REPO@
- the location of the integration repository, see below@VERSION@
- the version of the plugin under test
In the test project's build.gradle.kts
, make sure to apply the coverage plugin, in addition to the plugin under test.
plugins {
id("com.toasttab.testkit.coverage")
id("my.plugin.under.test")
}
Now, write the actual test. Note that a TestProject
instance will be automatically injected into the test method.
@TestKit
class MyTest {
@Test
fun sometest(project: TestProject) {
project.build("check")
}
}
To run a test against multiple versions of Gradle, use the @ParameterizedWithGradleVersions
annotation.
Gradle versions can be specified per class in the @TestKit
annotation or per method in the
@ParameterizedWithGradleVersions
annotation. Each gradle version argument will be automatically
injected into the runner created via TestProject.createRunner
.
@TestKit(gradleVersions = ["8.6", "8.7"])
class ParameterizedTest {
@ParameterizedWithGradleVersions
fun sometest(project: TestProject) {
project.createRunner()
.withArguments("check")
.build()
}
}
This plugin does not use the TestKit's plugin classpath injection mechanism because the mechanism breaks in certain scenarios, e.g. when plugins depend on other plugins. Instead, this plugin installs the plugin under test and its sibling dependencies, optionally preinstrumented for Jacoco code coverage, into an integration repository on disk, a technique borrowed from DAGP.
The integration repository is then injected into the plugin management repositories via a custom init script which is generated on the fly.
It is notoriously difficult to collect code coverage data from TestKit tests. By default, TestKit tests launch in a separate Gradle daemon JVM, which lingers after the tests finish. Gradle attaches an agent to the daemon JVM which instruments all classes from the plugin classpath and makes it impossible for the Jacoco agent to instrument classes on the fly.
- The main plugin pre-instruments the plugin classes and other project classes that the plugin under test depends on using Jacoco offline instrumentation.
- The junit5 extension configures the TestKit build to use pre-instrumented classes, starts a Jacoco coverage TCP server, and points the jacoco runtime to the TCP server.
- The TCP server writes coverage data into a separate file and stops writing the file when the tests finish running. This allows the main Gradle process to collect task outputs even though the TestKit process might still be lingering.
- The jacoco plugin applied to the TestKit project ensures that the coverage is fully flushed after the test finishes. This ensures that the coverage is recorded even though the TestKit process might still be lingering.
For more context, see