-
Notifications
You must be signed in to change notification settings - Fork 1.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
KTOR-7552 Rerun failed tests one more time
- Loading branch information
Showing
8 changed files
with
93 additions
and
13 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,74 @@ | ||
/* | ||
* Copyright 2014-2024 JetBrains s.r.o and contributors. Use of this source code is governed by the Apache 2.0 license. | ||
*/ | ||
|
||
import com.gradle.develocity.agent.gradle.test.* | ||
import org.gradle.api.* | ||
import org.gradle.api.tasks.testing.* | ||
import org.gradle.kotlin.dsl.* | ||
import org.jetbrains.kotlin.gradle.targets.jvm.tasks.* | ||
import org.jetbrains.kotlin.gradle.tasks.* | ||
|
||
val CI = System.getenv("TEAMCITY_VERSION") != null | ||
|
||
/** Applies CI-specific configurations to test tasks. */ | ||
fun Project.configureTestTasksOnCi() { | ||
// Don't fail build on the CI: | ||
// 1. To distinct builds failed because of failed tests and because of compilation errors or anything else. | ||
// TeamCity parses test results to define build status, so the build won't be green. | ||
// 2. To run as many tests as possible while keeping fail-fast behavior locally. | ||
tasks.withType<AbstractTestTask>().configureEach { | ||
ignoreFailures = true | ||
if (this is KotlinTest) ignoreRunFailures = true | ||
} | ||
|
||
tasks.withType<KotlinJvmTest>().configureEach { | ||
testRetry { | ||
maxRetries = 1 | ||
maxFailures = 10 | ||
} | ||
|
||
applyTestRetryCompatibilityWorkaround() | ||
} | ||
} | ||
|
||
/** | ||
* Test retry plugin is incompatible with test tasks that override the `createTestExecuter` method. | ||
* This includes the [KotlinJvmTest] task which wraps the test executor with its own wrapper. | ||
* | ||
* This workaround heavily relies on the internal implementation details of the test-retry plugin and KGP. | ||
* | ||
* The test retry plugin adds a `doFirst` action, which: | ||
* - Retrieves the test executer using `createTestExecuter` (KGP returns wrapped test executer here) | ||
* - Wraps it with `RetryTestExecuter` | ||
* - Sets the executer using `setTestExecuter` | ||
* | ||
* In the `doLast` action, it expects that `createTestExecuter` returns the previously created `RetryTestExecuter` instance. | ||
* However, KGP wraps every result of `createTestExecutor` with its own wrapper, resulting in the following nesting: | ||
* KotlinJvmTarget$Executer(RetryTestExecuter(KotlinJvmTarget$Executer(DefaultTestExecuter))) | ||
* | ||
* KGP wraps the executer only if `targetName` is present, as it is needed to add the target name suffix to the test name. | ||
* The workaround sets `targetName` to `null` after the first KGP wrapper is created, | ||
* so `createTestExecuter` returns the previously created executer: | ||
* RetryTestExecuter(KotlinJvmTarget$Executer(DefaultTestExecuter)) | ||
* | ||
* Issue: https://github.com/gradle/test-retry-gradle-plugin/issues/116 (KT-49155) | ||
*/ | ||
private fun KotlinJvmTest.applyTestRetryCompatibilityWorkaround() { | ||
if (targetName == null) return | ||
|
||
val executeTestsActionIndex = taskActions.indexOfLast { it.displayName == "Execute executeTests" } | ||
check(executeTestsActionIndex != -1) { "Action executeTests not found" } | ||
|
||
// Add the workaround action and then move it to the correct position right before tests execution. | ||
doFirst("workaround for compatibility with testRetry") { | ||
targetName = null | ||
} | ||
val injectedAction = taskActions.removeAt(0) | ||
taskActions.add(executeTestsActionIndex, injectedAction) | ||
} | ||
|
||
// Docs: https://docs.gradle.com/develocity/gradle-plugin/current/#test_retry | ||
private fun Test.testRetry(configure: TestRetryConfiguration.() -> Unit) { | ||
extensions.getByName<DevelocityTestConfiguration>("develocity").testRetry(configure) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters