mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-03-30 13:43:26 +08:00
7.5 KiB
7.5 KiB
description
| description |
|---|
| 为Kotlin强制执行TDD工作流程。首先编写Kotest测试,然后实施。使用Kover验证80%以上的覆盖率。 |
Kotlin TDD 命令
此命令使用 Kotest、MockK 和 Kover 为 Kotlin 代码强制执行测试驱动开发方法。
此命令的作用
- 定义类型/接口:首先搭建函数签名
- 编写 Kotest 测试:创建全面的测试规范(RED)
- 运行测试:验证测试因正确原因而失败
- 实现代码:编写最少的代码以通过测试(GREEN)
- 重构:在保持测试通过的同时进行改进
- 检查覆盖率:确保使用 Kover 达到 80% 以上的覆盖率
何时使用
在以下情况下使用 /kotlin-test:
- 实现新的 Kotlin 函数或类
- 为现有的 Kotlin 代码添加测试覆盖率
- 修复错误(首先编写失败的测试)
- 构建关键业务逻辑
- 学习 Kotlin 中的 TDD 工作流程
TDD 循环
RED -> 编写失败的 Kotest 测试
GREEN -> 实现最小化代码以通过测试
REFACTOR -> 改进代码,测试保持通过状态
REPEAT -> 进入下一个测试用例
示例会话
用户:/kotlin-test 我需要一个验证用户注册请求的函数
代理:
# TDD 会话:注册验证器
## 第一步:定义接口
```kotlin
// validator/RegistrationValidator.kt
package com.example.validator
data class RegistrationRequest(
val name: String,
val email: String,
val password: String,
)
sealed class ValidationResult {
data object Valid : ValidationResult()
data class Invalid(val errors: List<String>) : ValidationResult()
}
fun validateRegistration(request: RegistrationRequest): ValidationResult {
TODO("not implemented")
}
```
## 第二步:编写 Kotest 测试(RED)
```kotlin
// validator/RegistrationValidatorTest.kt
package com.example.validator
import io.kotest.core.spec.style.FunSpec
import io.kotest.matchers.shouldBe
import io.kotest.matchers.types.shouldBeInstanceOf
class RegistrationValidatorTest : FunSpec({
test("valid registration returns Valid") {
val request = RegistrationRequest(
name = "Alice",
email = "alice@example.com",
password = "SecureP@ss1",
)
val result = validateRegistration(request)
result.shouldBeInstanceOf<ValidationResult.Valid>()
}
test("blank name returns Invalid") {
val request = RegistrationRequest(
name = "",
email = "alice@example.com",
password = "SecureP@ss1",
)
val result = validateRegistration(request)
val invalid = result.shouldBeInstanceOf<ValidationResult.Invalid>()
invalid.errors shouldBe listOf("Name is required")
}
test("invalid email returns Invalid") {
val request = RegistrationRequest(
name = "Alice",
email = "not-an-email",
password = "SecureP@ss1",
)
val result = validateRegistration(request)
val invalid = result.shouldBeInstanceOf<ValidationResult.Invalid>()
invalid.errors shouldBe listOf("Invalid email format")
}
test("short password returns Invalid") {
val request = RegistrationRequest(
name = "Alice",
email = "alice@example.com",
password = "short",
)
val result = validateRegistration(request)
val invalid = result.shouldBeInstanceOf<ValidationResult.Invalid>()
invalid.errors shouldBe listOf("Password must be at least 8 characters")
}
test("multiple errors returns all errors") {
val request = RegistrationRequest(
name = "",
email = "bad",
password = "short",
)
val result = validateRegistration(request)
val invalid = result.shouldBeInstanceOf<ValidationResult.Invalid>()
invalid.errors.size shouldBe 3
}
})
```
## 第三步:运行测试 - 验证失败
```bash
$ ./gradlew test
RegistrationValidatorTest > valid registration returns Valid FAILED
kotlin.NotImplementedError: An operation is not implemented
FAILED (5 tests, 0 passed, 5 failed)
```
✓ 测试按预期失败(NotImplementedError)。
## 第四步:实现最小代码(GREEN)
```kotlin
// validator/RegistrationValidator.kt
package com.example.validator
private val EMAIL_REGEX = Regex("^[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\\.[A-Za-z]{2,}$")
private const val MIN_PASSWORD_LENGTH = 8
fun validateRegistration(request: RegistrationRequest): ValidationResult {
val errors = buildList {
if (request.name.isBlank()) add("Name is required")
if (!EMAIL_REGEX.matches(request.email)) add("Invalid email format")
if (request.password.length < MIN_PASSWORD_LENGTH) add("Password must be at least $MIN_PASSWORD_LENGTH characters")
}
return if (errors.isEmpty()) ValidationResult.Valid
else ValidationResult.Invalid(errors)
}
```
## 第五步:运行测试 - 验证通过
```bash
$ ./gradlew test
RegistrationValidatorTest > valid registration returns Valid PASSED
RegistrationValidatorTest > blank name returns Invalid PASSED
RegistrationValidatorTest > invalid email returns Invalid PASSED
RegistrationValidatorTest > short password returns Invalid PASSED
RegistrationValidatorTest > multiple errors returns all errors PASSED
PASSED (5 tests, 5 passed, 0 failed)
```
✓ 所有测试通过!
## 第六步:检查覆盖率
```bash
$ ./gradlew koverHtmlReport
Coverage: 100.0% of statements
```
✓ 覆盖率:100%
## TDD 完成!
测试模式
StringSpec(最简单)
class CalculatorTest : StringSpec({
"add two positive numbers" {
Calculator.add(2, 3) shouldBe 5
}
})
BehaviorSpec(BDD)
class OrderServiceTest : BehaviorSpec({
Given("a valid order") {
When("placed") {
Then("should be confirmed") { /* ... */ }
}
}
})
数据驱动测试
class ParserTest : FunSpec({
context("valid inputs") {
withData("2026-01-15", "2026-12-31", "2000-01-01") { input ->
parseDate(input).shouldNotBeNull()
}
}
})
协程测试
class AsyncServiceTest : FunSpec({
test("concurrent fetch completes") {
runTest {
val result = service.fetchAll()
result.shouldNotBeEmpty()
}
}
})
覆盖率命令
# Run tests with coverage
./gradlew koverHtmlReport
# Verify coverage thresholds
./gradlew koverVerify
# XML report for CI
./gradlew koverXmlReport
# Open HTML report
open build/reports/kover/html/index.html
# Run specific test class
./gradlew test --tests "com.example.UserServiceTest"
# Run with verbose output
./gradlew test --info
覆盖率目标
| 代码类型 | 目标 |
|---|---|
| 关键业务逻辑 | 100% |
| 公共 API | 90%+ |
| 通用代码 | 80%+ |
| 生成的代码 | 排除 |
TDD 最佳实践
应做:
- 首先编写测试,在任何实现之前
- 每次更改后运行测试
- 使用 Kotest 匹配器进行表达性断言
- 使用 MockK 的
coEvery/coVerify来处理挂起函数 - 测试行为,而非实现细节
- 包含边界情况(空值、null、最大值)
不应做:
- 在测试之前编写实现
- 跳过 RED 阶段
- 直接测试私有函数
- 在协程测试中使用
Thread.sleep() - 忽略不稳定的测试
相关命令
/kotlin-build- 修复构建错误/kotlin-review- 在实现后审查代码/verify- 运行完整的验证循环
相关
- 技能:
skills/kotlin-testing/ - 技能:
skills/tdd-workflow/