【发布时间】:2019-09-27 18:53:48
【问题描述】:
我正在尝试将构建迁移到 gradle,但有一个问题让我无法使用构建缓存。我有一组依赖任务用于:
- 设置嵌入式 MySQL 服务器
- 使用 Flyway 迁移该服务器中的架构
- 从该架构生成 Jooq 代码
- 停止嵌入式 MySQL 服务器
- 用我们的应用程序代码编译生成的代码
整个过程运行良好,但我想了解构建缓存以及它是如何工作的。我的 build.gradle 的相关部分如下所示:
plugins {
id 'java'
id 'nu.studer.jooq' version '2.0.11'
id "org.flywaydb.flyway" version "5.2.4"
id 'com.github.michaelruocco.embedded-mysql-plugin' version '2.1.10'
}
embeddedMysql {
url = 'jdbc:mysql://127.0.0.1:3308/jooqGeneration'
username = 'user'
password = 'password'
version = 'v5_7_latest'
}
startEmbeddedMysql.inputs.files(project.fileTree(dir: "src", include: "**/*.sql"))
startEmbeddedMysql.outputs.dir('build/generated-src/jooq/')
flyway {
url = 'jdbc:mysql://127.0.0.1:3308/jooqGeneration'
user = 'user'
password = 'password'
schemas = ['jooqGeneration']
}
flywayMigrate.dependsOn 'startEmbeddedMysql'
flywayMigrate.outputs.upToDateWhen { !startEmbeddedMysql.didWork }
jooq {
sample(sourceSets.jooqSchemaGeneration) {
jdbc {
driver = 'org.mariadb.jdbc.Driver'
url = 'jdbc:mysql://127.0.0.1:3308'
user = 'user'
password = 'password'
}
generator {
name = 'org.jooq.util.DefaultGenerator'
database {
name = 'org.jooq.util.mysql.MySQLDatabase'
includes = '.*'
excludes = ''
inputSchema = 'jooqGeneration'
}
}
}
}
generateSampleJooqSchemaSource.dependsOn 'flywayMigrate'
generateSampleJooqSchemaSource.finalizedBy 'stopEmbeddedMysql'
generateSampleJooqSchemaSource.outputs.upToDateWhen { !flywayMigrate.didWork }
compileJava.dependsOn 'generateSampleJooqSchemaSource'
我的问题是这里的整个构建链是高度耦合的。对于依赖链中的每一步,我都必须管理该步骤是否是最新的。
这一切似乎源于这样一个事实,即即使一个任务是最新的,该任务所依赖的任务仍然在运行。如果我向generateSampleJooqSchemaSource 任务添加一些输入和输出,它将正确确定它是否是最新的,但即使任务是最新的,dependsOn 任务仍然会运行。这意味着当我们运行时,我需要去每个相关的步骤并处理。我通过查看每个相关步骤是否有任何工作解决了这个问题,然后将正在更新的文件添加到我启动嵌入式 MySQL 服务器的第一个任务中。
有没有办法将所有决策转移到最终的generateSampleJooqSchemaSource,这样dependsOn 任务仅在generateSampleJooqSchemaSource 不是最新的情况下运行?
这将允许我将其中一些步骤重新用于其他用途。例如,我可以将嵌入式 MySQL 服务器重新用于一些数据库集成测试,而无需设置另一个单独的任务。
我主要只是想了解 Gradle 构建缓存,以及它如何决定运行这些任务。
【问题讨论】:
-
最新检查基于每个任务并在任务开始之前执行。所以当一个任务的依赖被执行时,你不知道这个任务以后是否会是最新的。在另一个方向,您可以使用
onlyIf条件来检查之前的任务是否完成了任何工作。