org.gradle.integtests.composite.CompositeBuildOperationsIntegrationTest.groovy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* Copyright 2017 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.gradle.integtests.composite
import org.gradle.api.internal.tasks.execution.ExecuteTaskBuildOperationType
import org.gradle.execution.taskgraph.NotifyTaskGraphWhenReadyBuildOperationType
import org.gradle.initialization.ConfigureBuildBuildOperationType
import org.gradle.initialization.LoadBuildBuildOperationType
import org.gradle.integtests.fixtures.build.BuildTestFile
import org.gradle.internal.operations.trace.BuildOperationRecord
import org.gradle.internal.taskgraph.CalculateTaskGraphBuildOperationType
import org.gradle.internal.taskgraph.CalculateTreeTaskGraphBuildOperationType
import org.gradle.launcher.exec.RunBuildBuildOperationType
import java.util.regex.Pattern
class CompositeBuildOperationsIntegrationTest extends AbstractCompositeBuildIntegrationTest {
BuildTestFile buildB
def setup() {
buildB = multiProjectBuild("buildB", ['b1', 'b2']) {
buildFile << """
allprojects {
apply plugin: 'java'
}
"""
}
includedBuilds << buildB
}
def "generates build operations for tasks in included builds"() {
given:
dependency 'org.test:buildB:1.0'
when:
execute(buildA, ":jar", [])
then:
executed ":buildB:jar"
and:
List allOps = operations.all(ExecuteTaskBuildOperationType)
allOps.find { it.details.buildPath == ":buildB" && it.details.taskPath == ":jar" }
allOps.find { it.details.buildPath == ":" && it.details.taskPath == ":jar" }
for (BuildOperationRecord operationRecord : allOps) {
assertChildrenNotIn(operationRecord, operationRecord, allOps)
}
}
def "generates build lifecycle operations for included builds with #display"() {
given:
dependency "org.test:${dependencyName}:1.0"
buildB.settingsFile << settings << "\n"
when:
execute(buildA, ":jar", [])
then:
executed ":${buildName}:jar"
and:
def root = operations.root(RunBuildBuildOperationType)
def loadOps = operations.all(LoadBuildBuildOperationType)
loadOps.size() == 2
loadOps[0].displayName == "Load build"
loadOps[0].details.buildPath == ":"
loadOps[0].parentId == root.id
loadOps[1].displayName == "Load build (:buildB)"
loadOps[1].details.buildPath == ":${buildName}"
loadOps[1].parentId == loadOps[0].id
def configureOps = operations.all(ConfigureBuildBuildOperationType)
configureOps.size() == 2
configureOps[0].displayName == "Configure build"
configureOps[0].details.buildPath == ":"
configureOps[0].parentId == root.id
configureOps[1].displayName == "Configure build (:${buildName})"
configureOps[1].details.buildPath == ":${buildName}"
configureOps[1].parentId == configureOps[0].id
def treeTaskGraphOps = operations.all(CalculateTreeTaskGraphBuildOperationType)
treeTaskGraphOps.size() == 1
treeTaskGraphOps[0].displayName == "Calculate build tree task graph"
treeTaskGraphOps[0].parentId == root.id
def taskGraphOps = operations.all(CalculateTaskGraphBuildOperationType)
taskGraphOps.size() == 2
taskGraphOps[0].displayName == "Calculate task graph"
taskGraphOps[0].details.buildPath == ":"
taskGraphOps[0].parentId == treeTaskGraphOps[0].id
taskGraphOps[1].displayName == "Calculate task graph (:${buildName})"
taskGraphOps[1].details.buildPath == ":${buildName}"
taskGraphOps[1].parentId == treeTaskGraphOps[0].id
def runMainTasks = operations.first(Pattern.compile("Run main tasks"))
runMainTasks.parentId == root.id
def runTasksOps = operations.all(Pattern.compile("Run tasks.*"))
runTasksOps.size() == 2
// Build operations are run in parallel, so can appear in either order
[runTasksOps[0].displayName, runTasksOps[1].displayName].sort() == ["Run tasks", "Run tasks (:${buildName})"]
runTasksOps[0].parentId == runMainTasks.id
runTasksOps[1].parentId == runMainTasks.id
def graphNotifyOps = operations.all(NotifyTaskGraphWhenReadyBuildOperationType)
graphNotifyOps.size() == 2
graphNotifyOps[0].displayName == "Notify task graph whenReady listeners (:${buildName})"
graphNotifyOps[0].details.buildPath == ":${buildName}"
graphNotifyOps[0].parentId == treeTaskGraphOps[0].id
graphNotifyOps[1].displayName == 'Notify task graph whenReady listeners'
graphNotifyOps[1].details.buildPath == ':'
graphNotifyOps[1].parentId == treeTaskGraphOps[0].id
where:
settings | buildName | dependencyName | display
"" | "buildB" | "buildB" | "default root project name"
"rootProject.name='someLib'" | "buildB" | "someLib" | "configured root project name"
}
def "generates build lifecycle operations for multiple included builds"() {
given:
def buildC = multiProjectBuild("buildC", ["someLib"]) {
buildFile << """
allprojects {
apply plugin: 'java'
}
"""
}
includedBuilds << buildC
dependency "org.test:buildB:1.0"
dependency "org.test:buildC:1.0"
dependency buildB, "org.test:buildC:1.0"
when:
execute(buildA, ":jar", [])
then:
executed ":buildB:jar", ":buildC:jar"
and:
def root = operations.root(RunBuildBuildOperationType)
def treeTaskGraphOps = operations.all(CalculateTreeTaskGraphBuildOperationType)
treeTaskGraphOps.size() == 1
treeTaskGraphOps[0].displayName == "Calculate build tree task graph"
treeTaskGraphOps[0].parentId == root.id
def taskGraphOps = operations.all(CalculateTaskGraphBuildOperationType)
taskGraphOps.size() == 3
taskGraphOps[0].displayName == "Calculate task graph"
taskGraphOps[0].details.buildPath == ":"
taskGraphOps[0].parentId == treeTaskGraphOps[0].id
taskGraphOps[1].displayName == "Calculate task graph (:buildB)"
taskGraphOps[1].details.buildPath == ":buildB"
taskGraphOps[1].parentId == treeTaskGraphOps[0].id
taskGraphOps[2].displayName == "Calculate task graph (:buildC)"
taskGraphOps[2].details.buildPath == ":buildC"
taskGraphOps[2].parentId == treeTaskGraphOps[0].id
def graphNotifyOps = operations.all(NotifyTaskGraphWhenReadyBuildOperationType)
graphNotifyOps.size() == 3
graphNotifyOps[0].displayName == "Notify task graph whenReady listeners (:buildB)"
graphNotifyOps[0].details.buildPath == ":buildB"
graphNotifyOps[0].parentId == treeTaskGraphOps[0].id
graphNotifyOps[1].displayName == "Notify task graph whenReady listeners (:buildC)"
graphNotifyOps[1].details.buildPath == ":buildC"
graphNotifyOps[1].parentId == treeTaskGraphOps[0].id
graphNotifyOps[2].displayName == 'Notify task graph whenReady listeners'
graphNotifyOps[2].details.buildPath == ':'
graphNotifyOps[2].parentId == treeTaskGraphOps[0].id
}
def "generates build lifecycle operations for multiple included builds used as buildscript dependencies"() {
given:
def buildC = multiProjectBuild("buildC", ["someLib"]) {
buildFile << """
allprojects {
apply plugin: 'java'
}
"""
}
includedBuilds << buildC
buildA.buildFile.text = """
buildscript {
dependencies {
classpath 'org.test:buildB:1.0'
classpath 'org.test:buildC:1.0'
}
}
""" + buildA.buildFile.text
dependency buildB, "org.test:buildC:1.0"
when:
execute(buildA, ":jar", [])
then:
executed ":buildB:jar", ":buildC:jar"
and:
def root = operations.root(RunBuildBuildOperationType)
def applyRootProjectBuildScript = operations.first(Pattern.compile("Apply build file 'build.gradle' to root project 'buildA'"))
def treeTaskGraphOps = operations.all(CalculateTreeTaskGraphBuildOperationType)
treeTaskGraphOps.size() == 2
treeTaskGraphOps[0].displayName == "Calculate build tree task graph"
treeTaskGraphOps[0].parentId == applyRootProjectBuildScript.id
treeTaskGraphOps[1].displayName == "Calculate build tree task graph"
treeTaskGraphOps[1].parentId == root.id
def taskGraphOps = operations.all(CalculateTaskGraphBuildOperationType)
taskGraphOps.size() == 3
taskGraphOps[0].displayName == "Calculate task graph (:buildB)"
taskGraphOps[0].details.buildPath == ":buildB"
taskGraphOps[0].parentId == treeTaskGraphOps[0].id
taskGraphOps[1].displayName == "Calculate task graph (:buildC)"
taskGraphOps[1].details.buildPath == ":buildC"
taskGraphOps[1].parentId == treeTaskGraphOps[0].id
taskGraphOps[2].displayName == "Calculate task graph"
taskGraphOps[2].details.buildPath == ":"
taskGraphOps[2].parentId == treeTaskGraphOps[1].id
def graphNotifyOps = operations.all(NotifyTaskGraphWhenReadyBuildOperationType)
graphNotifyOps.size() == 3
graphNotifyOps[0].displayName == "Notify task graph whenReady listeners (:buildB)"
graphNotifyOps[0].details.buildPath == ":buildB"
graphNotifyOps[0].parentId == treeTaskGraphOps[0].id
graphNotifyOps[1].displayName == "Notify task graph whenReady listeners (:buildC)"
graphNotifyOps[1].details.buildPath == ":buildC"
graphNotifyOps[1].parentId == treeTaskGraphOps[0].id
graphNotifyOps[2].displayName == 'Notify task graph whenReady listeners'
graphNotifyOps[2].details.buildPath == ':'
graphNotifyOps[2].parentId == treeTaskGraphOps[1].id
}
def "generates build lifecycle operations for included build used as buildscript and production dependency"() {
given:
buildA.buildFile.text = """
buildscript {
dependencies {
classpath 'org.test:b1:1.0'
}
}
""" + buildA.buildFile.text
dependency "org.test:b2:1.0"
when:
execute(buildA, ":jar", [])
then:
executed ":buildB:b1:jar", ":buildB:b2:jar"
and:
def root = operations.root(RunBuildBuildOperationType)
def loadOps = operations.all(LoadBuildBuildOperationType)
loadOps.size() == 2
loadOps[0].displayName == "Load build"
loadOps[0].details.buildPath == ":"
loadOps[0].parentId == root.id
loadOps[1].displayName == "Load build (:buildB)"
loadOps[1].details.buildPath == ":buildB"
loadOps[1].parentId == loadOps[0].id
def configureOps = operations.all(ConfigureBuildBuildOperationType)
configureOps.size() == 2
configureOps[0].displayName == "Configure build"
configureOps[0].details.buildPath == ":"
configureOps[0].parentId == root.id
configureOps[1].displayName == "Configure build (:buildB)"
configureOps[1].details.buildPath == ":buildB"
configureOps[1].parentId == configureOps[0].id
def applyRootProjectBuildScript = operations.first(Pattern.compile("Apply build file 'build.gradle' to root project 'buildA'"))
def treeTaskGraphOps = operations.all(CalculateTreeTaskGraphBuildOperationType)
treeTaskGraphOps.size() == 2
treeTaskGraphOps[0].displayName == "Calculate build tree task graph"
treeTaskGraphOps[0].parentId == applyRootProjectBuildScript.id
treeTaskGraphOps[1].displayName == "Calculate build tree task graph"
treeTaskGraphOps[1].parentId == root.id
// The task graph for buildB is calculated multiple times, once for the buildscript dependency and again for the production dependency
def taskGraphOps = operations.all(CalculateTaskGraphBuildOperationType)
taskGraphOps.size() == 3
taskGraphOps[0].displayName == "Calculate task graph (:buildB)"
taskGraphOps[0].details.buildPath == ":buildB"
taskGraphOps[0].parentId == treeTaskGraphOps[0].id
taskGraphOps[1].displayName == "Calculate task graph"
taskGraphOps[1].details.buildPath == ":"
taskGraphOps[1].parentId == treeTaskGraphOps[1].id
taskGraphOps[2].displayName == "Calculate task graph (:buildB)"
taskGraphOps[2].details.buildPath == ":buildB"
taskGraphOps[2].parentId == treeTaskGraphOps[1].id
def runMainTasks = operations.first(Pattern.compile("Run main tasks"))
runMainTasks.parentId == root.id
// Tasks are run for buildB multiple times, once for buildscript dependency and again for production dependency
def runTasksOps = operations.all(Pattern.compile("Run tasks.*"))
runTasksOps.size() == 3
runTasksOps[0].displayName == "Run tasks (:buildB)"
runTasksOps[0].parentId == applyRootProjectBuildScript.id
// Build operations are run in parallel, so can appear in either order
[runTasksOps[1].displayName, runTasksOps[2].displayName].sort() == ["Run tasks", "Run tasks (:buildB)"]
runTasksOps[1].parentId == runMainTasks.id
runTasksOps[2].parentId == runMainTasks.id
// Task graph ready event sent only once
def graphNotifyOps = operations.all(NotifyTaskGraphWhenReadyBuildOperationType)
graphNotifyOps.size() == 2
graphNotifyOps[0].displayName == 'Notify task graph whenReady listeners (:buildB)'
graphNotifyOps[0].details.buildPath == ':buildB'
graphNotifyOps[0].parentId == treeTaskGraphOps[0].id
graphNotifyOps[1].displayName == "Notify task graph whenReady listeners"
graphNotifyOps[1].details.buildPath == ":"
graphNotifyOps[1].parentId == treeTaskGraphOps[1].id
}
def assertChildrenNotIn(BuildOperationRecord origin, BuildOperationRecord op, List allOps) {
for (BuildOperationRecord child : op.children) {
assert !allOps.contains(child): "Task operation $origin has child $child which is also a task operation"
assertChildrenNotIn(origin, child, allOps)
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy