I'm trying to write a plugin for Gradle that would perform some actions with the file result of performing another task (defined in another plugin - java-library-distribution
). I can easily get the file I need via this.getProject().getTasks().getByName("distZip").getOutputs().getFiles().getSingleFile()
in my task class.
But with this approach, my task will be executed every time, regardless of whether the file has changed in the higher-level task.
I annotated the getter method as an @InputFile
. But unfortunately Gradle still doesn't mark the issue as UP-TO-DATE
.
public class YcfPlugin implements Plugin<Project> {
@Override
public void apply(Project project) {
project.getExtensions().create("ycf", YcfPluginExtension.class);
project.getPluginManager().apply("java-library-distribution");
project.getTasks().register("ycfCreateVersion", YcfTaskCreateVersion.class);
}
}
abstract class YcfTask extends DefaultTask {
public static final String TASK_GROUP = "Some Description";
YcfPluginExtension ycfExtension = this.getProject().getExtensions().getByType(YcfPluginExtension.class);
Logger logger = this.getProject().getLogger();
public YcfTask() {
this.setGroup(TASK_GROUP);
}
}
public class YcfTaskCreateVersion extends YcfTask {
private File fi;
public YcfTaskCreateVersion() {
this.setDescription("Some description");
this.dependsOn(this.getProject().getTasks().getByName("distZip"));
}
@InputFile
public File getFi() {
return this.getProject().getTasks().getByName("distZip").getOutputs().getFiles().getSingleFile();
}
@TaskAction
public void run() {
byte[] bytes = Files.readAllBytes(getFi().toPath());
//..do something with file content
}
}
There are some misconceptions in your current approach:
The
@InputFile
annotation should be used on a getter method. A getter method is not just a method that starts with the prefixget
. Usually it is a method that returns the value of a (private) field. In your example there is a field calledfi
, so the respective getter methodgetFi
should just return the value of this field. By the way, in your current code, the fieldfi
is not used at all.It is good that you know what value should be used as an input of your task (
getProject().getTasks().getByName("distZip").getOutputs().getFiles().getSingleFile()
), but this should not be a part of the task type implementation. Instead, task types should be as configurable as possible. You may then create and configure one instance of this task type in your plugin code:Sadly, the code above won't work, because at the time the task is created (when the plugin is applied), the outputs of the task
distZip
(and maybe even the task) won't be available. But that is not a big problem, because Gradle supports this use case. You can change the type of your field toObject
, so that not only objects of typeFile
may be passed in. Whether a file (or something that may be converted to a file) was passed, will be checked when the task executes.There is a cool thing about this setup: You may pass the task
distZip
directly and Gradle will automatically extract the file outputs. It will even detect that you are using the outputs of a task as inputs of another task, so Gradle will automatically setup a task dependency between the two tasks and you don't have to usedependsOn
anymore:Lets check the Gradle documentation on task outcomes. There is a section on
UP-TO-DATE
:As you can see, your task needs to define outputs to be considered up-to-date. You can define output files in the same way as input files (field, getter with
@OutputFile
and setter). They should be configurable and default to a file inside thebuild
directory. Alternatively you may useonlyIf
to implement a custom check whether the task should be run. If the predicate insideonlyIf
returnsfalse
, the task will beSKIPPED
.