I’m having an issue with a plugin I’m developing – sometimes tasks are executed in an order that causes the build to succeed, and sometimes it’s executed in a seemingly different order that causes the build to fail. All I have to do is run ./gradlew clean build over and over, and sometimes it works, sometimes it doesn’t.
gradle.properties has org.gradle.parallel=true. Could this be related?
How can I troubleshoot this? Are there tips on how to ensure that a task runs after different lifecycle tasks have run on its dependencies?
Within each project, the tasks have a deterministic ordering (based on task dependencies) and will always execute in the same order.
When org.gradle.parallel=true this allows tasks in different projects to execute in parallel provided they don’t depend on one another.
So it’s possible that you have tasks in separate projects that logically depend on one another but that dependency hasn’t been configured in your gradle build. If you want to debug the task ordering you could put printlns in a TaskExecutionListener or perhaps a build scan.
Another cause of these sorts of problems is tests where one test “bleeds” state (eg mutable static) into another so TestA will succeed when it runs before TestB but will fail if it runs after. To debug test ordering there’s beforeTest{} and afterTest{} handlers on the Test task
Good to know. In this case, it’s not test cases that are causing the issue. I need to perform validation and codegen from models that depend on other models from within the same project. The way models are discovered is through the classpath and SPI.
So let’s say I have a “producer” model and a “consumer” model. The consumer model uses the model created by the producer model’s JAR. In order to build the consumer model’s JAR, I need the producer model’s JAR. It seems like sometimes producer’s JAR is available before consumer runs, and sometimes it isn’t.
registerSmithyBuildTask has a bit of a code smell to me. Task dependencies are usually set in stone and don’t change based on other tasks being disabled
Consider the following
task copy1(type: Copy) {
from 'dir1'
into 'dir2'
}
task copy2(type: Copy) {
from tasks.copy1
into 'dir3'
}
Gradle will automatically make copy2 depend on copy1 in the example above. Under the hood from tasks.copy1 does two things.
Creates a FileCollection which delegates to copy1’s task outputs.
Uses the FileCollection as TaskInput which causes Gradle to also create a task dependency
I prefer to use this type of configuration where possible, it’s more readable and simpler than manually configuring task dependencies
Unfortunately none of these snippets show the configuration of something being produced by one task (a file of some form) and then being consumed by another task
I ran into a lot of other problems that are unrelated to this original post (like how to dynamically add a project dependency, and that Gradle’s interception of stdout/stderr doesn’t seem to be thread-safe), but those can be raised elsewhere.