gradle - 如何将自定义任务用作工件

标签 gradle gradle-dependencies gradle-task

我在项目(我们称之为 projectA)构建脚本中有一个简单的自定义任务,它创建一个文件。任务看起来像这样:

task createFile() {
  def outputFile = rootProject.file("${buildDir}/tmp/outputFile.txt")
  outputs.file(outputFile)
  outputs.upToDateWhen { false }
  doLast {
    outputFile.parentFile.mkdirs()
    outputFile.text = "Hello World"
  }
}

该项目是多项目构建的一部分,我需要此任务生成的文件作为 projectB 中复制任务的输入。为此,我在 projectB 和相应的依赖项中创建了一个配置:

configurations {
  mySourceConfig
}

dependencies {
  mySourceConfig project(path: ':projectA', configuration: 'myDistConfig')
}

task copySourceDependencies(type: Copy) {
   from configurations.mySourceConfig 
   into contextDir
}

projectA 中,我得到了一个 myDistConfig 配置,并希望将 createFile 任务的输出文件用作此配置的工件:

configurations {
  myDistConfig
}

artifacts {
  myDistConfig createFile
}

如果我这样做,gradle 会告诉我任务无法转换为 ConfigurablePublishArtifact:

> Cannot convert the provided notation to an object of type ConfigurablePublishArtifact: task ':projectA:createFile'.   The
  following types/formats are supported:
     - Instances of ConfigurablePublishArtifact.
     - Instances of PublishArtifact.
     - Instances of AbstractArchiveTask, for example jar.
     - Instances of Provider<RegularFile>.
     - Instances of Provider<Directory>.
     - Instances of Provider<File>.
     - Instances of RegularFile.
     - Instances of Directory.
     - Instances of File.
     - Maps with 'file' key

所以我尝试将 createFile.outputs.files.singleFile 作为工件。这会阻止 gradle 提示,但无法设置 :projectB:copySourceDependencies:projectA:createFile 之间的依赖关系,如果 projectB:copySourceDependencies 被执行在干净的工作区中,它会被简单地跳过,状态为 NO-SOURCE

是否有可能以某种方式使用自定义任务作为类似于 Zip 任务的工件,让 gradle 意识到依赖关系?!


2020-02-28更新:

基于@BjørnVester 的非常好的回答,我在一个单独的 gradle 文件 util.gradle 中实现了以下任务:

class CreateFile extends DefaultTask {

  @OutputFile
  RegularFileProperty outputFile = project.objects.fileProperty()

  @TaskAction
  void createFile() {
    def tOutFile = outputFile.get().asFile
    tOutFile.parentFile.mkdirs()
    tOutFile.text = "Hello World"
  }
}

rootProject.ext.CreateFile = CreateFile

在项目 A 的 build.gradle 中,它看起来像这样:

apply from 'util.gradle'

task createFile(type: CreateFile) {
  outputFile = rootProject.file("${buildDir}/tmp/outputFile.txt")
}

artifacts {
  myDistConfig createFile.outputFile
}

现在 gradle 知道正确的依赖关系并且它的工作就像一个魅力!

最佳答案

当使用任务作为工件符号时,它必须是 AbstractArchiveTask 类型.因此,对于其他类型的任务,您将不得不做其他事情。

使用createFile.outputs.files.singleFile的方法很好,因为它是受支持的类型(正如您从错误输出中看到的那样)。但是 Gradle 不创建对生产任务的依赖的原因是因为 File不携带该信息。为此,它必须是 Provider .但修复它的一种快速方法是明确配置生成工件的任务:

artifacts.add("myDistConfig", createFile.outputs.files.singleFile) {
    builtBy("createFile")
}

或者,您也可以将 outputFile 更改为 Provider<File>RegularFileProperty使用 ObjectFactory 作为任务生产者信息。我从未将它们用作 DSL 的一部分,它可能更适合使用类。 (虽然这看起来需要更多工作,但一旦代码增长到一定程度,它就会使代码更具可读性。)下面是 Groovy 中的一个示例:

class MyFileCreator extends DefaultTask {
  @OutputFile
  RegularFileProperty outputFile = project.objects.fileProperty().convention(project.layout.buildDirectory.file("tmp/outputFile.txt"))

  @TaskAction
  void createFile() {
    File outFile = outputFile.get().asFile
    outFile.parentFile.mkdirs()
    outFile.text = "Hello World"
  }
}

MyFileCreator createFileTask = tasks.create("createFile", MyFileCreator)

configurations {
  myDistConfig
}

artifacts {
  myDistConfig createFileTask.outputFile
}

关于gradle - 如何将自定义任务用作工件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60449365/

相关文章:

android - android支持库必须使用完全相同的版本吗?

android - 具有 ABI 拆分的特定架构的 Gradle 依赖性

android-studio - 我的flutter项目无法在Android设备上运行,并且提到了一些许可问题

gradle - 自定义增量 Gradle 编译任务的依赖项

java - sdkman gradle 3.4无法确定java版本

gradle - 在Android Studio中的gradle项目中添加commons-io依赖

android - Gradle:无法解析从非 build.gradle 文件导入的任务

c - Gradle C 插件示例

Android Studio依赖冲突(程序类型已经存在)

class - 在Gradle中创建便宜的新任务类