Skip to content

Power-assert 编译器插件

DANGER

Power-assert 编译器插件是 实验性 的。

它可能随时更改。仅用于评估目的。

欢迎在 YouTrack 中提供反馈。

Kotlin Power-assert 编译器插件通过提供带有上下文信息的详细失败消息来改善调试体验。 它通过自动生成失败消息中的中间值来简化编写测试的过程。 它有助于你理解测试失败的原因,而无需复杂的断言库。

这是插件提供的一个示例消息:

text
Incorrect length
assert(hello.length == world.substring(1, 4).length) { "Incorrect length" }
       |     |      |  |     |               |
       |     |      |  |     |               3
       |     |      |  |     orl
       |     |      |  world!
       |     |      false
       |     5
       Hello

Power-assert 插件的主要特性:

  • 增强的错误消息:插件捕获并显示断言中变量和子表达式的值,以清楚地识别失败原因。
  • 简化的测试:自动生成信息丰富的失败消息,减少对复杂断言库的需求。
  • 支持多种函数:默认情况下,它转换 assert() 函数调用,但也可以转换其他函数, 例如 require()check()assertTrue()

应用插件

要启用 Power-assert 插件,请按以下方式配置你的 build.gradle(.kts) 文件:

kotlin
// build.gradle.kts
plugins {
    kotlin("multiplatform") version "2.0.0"
    kotlin("plugin.power-assert") version "2.0.0"
}
groovy
// build.gradle
plugins {
    id 'org.jetbrains.kotlin.multiplatform' version '2.0.0'
    id 'org.jetbrains.kotlin.plugin.power-assert' version '2.0.0'
}

配置插件

Power-assert 插件提供了几个选项来自定义其行为:

  • functions:一个包含完全限定函数路径的列表。Power-assert 插件将转换对这些函数的调用。如果未指定,默认只转换 kotlin.assert() 调用。
  • includedSourceSets:一个 Gradle 源集 (source sets) 列表,Power-assert 插件将对其进行转换。如果未指定,默认情况下将转换所有 测试源集

要自定义行为,请将 powerAssert {} 块添加到你的构建脚本文件:

kotlin
// build.gradle.kts
powerAssert {
    functions = listOf("kotlin.assert", "kotlin.test.assertTrue", "kotlin.test.assertEquals", "kotlin.test.assertNull")
    includedSourceSets = listOf("commonMain", "jvmMain", "jsMain", "nativeMain")
}
groovy
// build.gradle
powerAssert {
    functions = ["kotlin.assert", "kotlin.test.assertTrue", "kotlin.test.assertEquals", "kotlin.test.assertNull"]
    includedSourceSets = ["commonMain", "jvmMain", "jsMain", "nativeMain"]
}

由于该插件是实验性的,每次构建应用程序时都会看到警告。 要排除这些警告,请在声明 powerAssert {} 块之前添加此 @OptIn 注解:

kotlin
import org.jetbrains.kotlin.gradle.ExperimentalKotlinGradlePluginApi

@OptIn(ExperimentalKotlinGradlePluginApi::class)
powerAssert {
    ...
}

使用插件

本节提供使用 Power-assert 编译器插件的示例。

请参阅所有这些示例的构建脚本文件 build.gradle.kts 的完整代码:

kotlin
import org.jetbrains.kotlin.gradle.ExperimentalKotlinGradlePluginApi

plugins {
    kotlin("jvm") version "2.1.21"
    kotlin("plugin.power-assert") version "2.1.21"
}

group = "org.example"
version = "1.0-SNAPSHOT"

repositories {
    mavenCentral()
}

dependencies {
    testImplementation(kotlin("test"))
}

tasks.test {
    useJUnitPlatform()
}

@OptIn(ExperimentalKotlinGradlePluginApi::class)
powerAssert {
    functions = listOf("kotlin.assert", "kotlin.test.assertEquals", "kotlin.test.assertTrue", "kotlin.test.assertNull", "kotlin.require", "org.example.AssertScope.assert")
}

assert 函数

考虑以下使用 assert() 函数的测试:

kotlin
import kotlin.test.Test

class SampleTest {

    @Test
    fun testFunction() {
        val hello = "Hello"
        val world = "world!"
        assert(hello.length == world.substring(1, 4).length) { "Incorrect length" }
    }
}

如果启用 Power-assert 插件并运行 testFunction() 测试,你将获得明确的失败消息:

text
Incorrect length
assert(hello.length == world.substring(1, 4).length) { "Incorrect length" }
       |     |      |  |     |               |
       |     |      |  |     |               3
       |     |      |  |     orl
       |     |      |  world!
       |     |      false
       |     5
       Hello

要获得更完整的错误消息,请始终将变量内联到测试函数参数中。 考虑以下测试函数:

kotlin
class ComplexExampleTest {

    data class Person(val name: String, val age: Int)
 
    @Test
    fun testComplexAssertion() {
        val person = Person("Alice", 10)
        val isValidName = person.name.startsWith("A") && person.name.length > 3
        val isValidAge = person.age in 21..28
        assert(isValidName && isValidAge)
    }
}

执行后的代码输出没有提供足够的信息来找出问题的原因:

text
Assertion failed
assert(isValidName && isValidAge)
       |              |
       |              false
       true

将变量内联到 assert() 函数中:

kotlin
class ComplexExampleTest {

    data class Person(val name: String, val age: Int)

    @Test
    fun testComplexAssertion() {
        val person = Person("Alice", 10)
        assert(person.name.startsWith("A") && person.name.length > 3 && person.age > 20 && person.age < 29)
    }
}

执行后,你将获得更多关于出错情况的明确信息:

text
Assertion failed
assert(person.name.startsWith("A") && person.name.length > 3 && person.age > 20 && person.age < 29)
       |      |    |                  |      |    |      |      |      |   |
       |      |    |                  |      |    |      |      |      |   false
       |      |    |                  |      |    |      |      |      10
       |      |    |                  |      |    |      |      Person(name=Alice, age=10)
       |      |    |                  |      |    |      true
       |      |    |                  |      |    5
       |      |    |                  |      Alice
       |      |    |                  Person(name=Alice, age=10)
       |      |    true
       |      Alice
       Person(name=Alice, age=10)

除了 assert 函数之外

Power-assert 插件除了 assert 函数(默认转换)之外,还可以转换各种函数。 像 require()check()assertTrue()assertEqual() 等函数也可以被转换, 如果它们的形式允许将 String() -> String 值作为最后一个参数。

在测试中使用新函数之前,请在构建脚本文件的 powerAssert {} 块中指定该函数。 例如,require() 函数:

kotlin
// build.gradle.kts
import org.jetbrains.kotlin.gradle.ExperimentalKotlinGradlePluginApi

@OptIn(ExperimentalKotlinGradlePluginApi::class)
powerAssert {
    functions = listOf("kotlin.assert", "kotlin.require")
}

添加函数后,你可以在测试中使用它:

kotlin
class RequireExampleTest {

    @Test
    fun testRequireFunction() {
        val value = ""
        require(value.isNotEmpty()) { "Value should not be empty" }
    }
}

此示例的输出使用 Power-assert 插件提供了关于失败测试的详细信息:

text
Value should not be empty
require(value.isNotEmpty()) { "Value should not be empty" }
        |     |
        |     false

该消息显示了导致失败的中间值,使调试更容易。

软断言

Power-assert 插件支持软断言,它不会立即使测试失败,而是收集断言失败并在测试运行结束时报告它们。 当你想在单次运行中查看所有断言失败而不在第一次失败时停止时,这会很有用。

要启用软断言,请实现你收集错误消息的方式:

kotlin
fun <R> assertSoftly(block: AssertScope.() -> R): R {
    val scope = AssertScopeImpl()
    val result = scope.block()
    if (scope.errors.isNotEmpty()) {
        throw AssertionError(scope.errors.joinToString("
"))
    }
    return result
}

interface AssertScope {
    fun assert(assertion: Boolean, message: (() -> String)? = null)
}

class AssertScopeImpl : AssertScope {
    val errors = mutableListOf<String>()
    override fun assert(assertion: Boolean, message: (() -> String)?) {
        if (!assertion) {
            errors.add(message?.invoke() ?: "Assertion failed")
        }
    }
}

将这些函数添加到 powerAssert {} 块中,以使其可供 Power-assert 插件使用:

kotlin
@OptIn(ExperimentalKotlinGradlePluginApi::class)
powerAssert {
    functions = listOf("kotlin.assert", "kotlin.test.assert", "org.example.AssertScope.assert")
}

TIP

您应该指定声明 AssertScope.assert() 函数的包的完整名称。

之后,你可以在测试代码中使用它:

kotlin
// Import the assertSoftly() function
import org.example.assertSoftly
        
class SoftAssertExampleTest1 {

    data class Employee(val name: String, val age: Int, val salary: Int)

    @Test
    fun `test employees data`() {
        val employees = listOf(
            Employee("Alice", 30, 60000),
            Employee("Bob", 45, 80000),
            Employee("Charlie", 55, 40000),
            Employee("Dave", 150, 70000)
        )

        assertSoftly {
            for (employee in employees) {
                assert(employee.age < 100) { "${employee.name} has an invalid age: ${employee.age}" }
                assert(employee.salary > 50000) { "${employee.name} has an invalid salary: ${employee.salary}" }
            }
        }
    }
}

在输出中,所有 assert() 函数的错误消息将按顺序打印:

text
Charlie has an invalid salary: 40000
assert(employee.salary > 50000) { "${employee.name} has an invalid salary: ${employee.salary}" }
       |        |      |
       |        |      false
       |        40000
       Employee(name=Charlie, age=55, salary=40000)
Dave has an invalid age: 150
assert(employee.age < 100) { "${employee.name} has an invalid age: ${employee.age}" }
       |        |   |
       |        |   false
       |        150
       Employee(name=Dave, age=150, salary=70000)

接下来