org.gradle.process.internal.CancellationBuildOperationIntegrationTest.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.
The newest version!
/*
* Copyright 2024 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.process.internal
import org.gradle.integtests.fixtures.AbstractIntegrationSpec
import org.gradle.integtests.fixtures.BuildOperationsFixture
import org.gradle.integtests.fixtures.UnsupportedWithConfigurationCache
import org.gradle.test.fixtures.server.http.BlockingHttpServer
import org.gradle.test.precondition.Requires
import org.gradle.test.preconditions.IntegTestPreconditions
import org.junit.Rule
import static org.hamcrest.CoreMatchers.anyOf
import static org.hamcrest.CoreMatchers.equalTo
class CancellationBuildOperationIntegrationTest extends AbstractIntegrationSpec {
@Rule
BlockingHttpServer server = new BlockingHttpServer()
BuildOperationsFixture operations = new BuildOperationsFixture(executer, temporaryFolder)
@Requires(IntegTestPreconditions.NotEmbeddedExecutor)
@UnsupportedWithConfigurationCache(because = "captures worker threads in shared state")
def "task operations are closed even when interrupting the execution workers"() {
server.start()
executer.withStackTraceChecksDisabled()
def parallelTaskCount = 5
buildFile << """
ext.workerThreads = new HashSet<>()
ext.latch = new java.util.concurrent.CountDownLatch(${parallelTaskCount} )
tasks.register('interrupting') {
doFirst {
${server.callFromBuild("before-interrupt")}
rootProject.ext.workerThreads.each {
if (it.stackTrace.any { it.toString().contains("DefaultResourceLockCoordinationService.withStateLock") }) {
it.interrupt()
}
}
// We fail this task so the build fails consistently
throw new RuntimeException("Interrupting done")
}
}
"""
parallelTaskCount.times { project ->
settingsFile << """include 'a$project'\n"""
file("a$project/build.gradle") << """
tasks.register('parallelTask') {
doFirst {
println 'executing a parallelTask in thread ' + Thread.currentThread()
rootProject.ext.workerThreads << Thread.currentThread()
${server.callFromBuild("parallel-task-done")}
}
}
"""
}
def numWorkers = (parallelTaskCount / 2) + 2 as int
when:
server.expectConcurrent(numWorkers, ["parallel-task-done"] * parallelTaskCount + ["before-interrupt"] as String[])
fails('parallelTask', '--parallel', ':interrupting', "--console=plain", "--max-workers=${numWorkers}", '--continue')
then:
operations.danglingChildren.empty
failure.assertThatAllDescriptions(anyOf(
equalTo("Execution failed for task ':interrupting'."),
equalTo("Execution failed for task ':a0:parallelTask'."),
equalTo("Execution failed for task ':a1:parallelTask'."),
equalTo("Execution failed for task ':a2:parallelTask'."),
equalTo("Execution failed for task ':a3:parallelTask'."),
equalTo("Execution failed for task ':a4:parallelTask'.")
))
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy