Перевірте очікувані винятки в Котліні


91

У Java програміст може вказати очікувані винятки для тестових випадків JUnit, наприклад:

@Test(expected = ArithmeticException.class)
public void omg()
{
    int blackHole = 1 / 0;
}

Як би я це зробив у Котліні? Я спробував дві синтаксичні варіації, але жодна з них не спрацювала:

import org.junit.Test

// ...

@Test(expected = ArithmeticException) fun omg()
    Please specify constructor invocation;
    classifier 'ArithmeticException' does not have a companion object

@Test(expected = ArithmeticException.class) fun omg()
                            name expected ^
                                            ^ expected ')'

Відповіді:


126

Котлін-переклад прикладу Java для JUnit 4.12 :

@Test(expected = ArithmeticException::class)
fun omg() {
    val blackHole = 1 / 0
}

Однак JUnit 4.13 запровадив два assertThrowsметоди для більш детально визначених областей винятків:

@Test
fun omg() {
    // ...
    assertThrows(ArithmeticException::class.java) {
        val blackHole = 1 / 0
    }
    // ...
}

Обидва assertThrowsметоди повертають очікуваний виняток для додаткових тверджень:

@Test
fun omg() {
    // ...
    val exception = assertThrows(ArithmeticException::class.java) {
        val blackHole = 1 / 0
    }
    assertEquals("/ by zero", exception.message)
    // ...
}

80

Kotlin має власний пакет тестових помічників, який може допомогти зробити такий тип unittest.

Ваш тест може бути дуже виразним завдяки використанню assertFailWith:

@Test
fun test_arithmethic() {
    assertFailsWith<ArithmeticException> {
        omg()
    }
}

1
Якщо ви отримаєте 404 за своїм посиланням, це kotlin.testбуло замінено чимось іншим?
fredoverflow

@fredoverflow Ні, не замінюється, а просто видаляється зі стандартних бібліотек. Я оновив посилання на репозиторій github kotlin, але, на жаль, не можу знайти жодного посилання на документацію. У будь-якому випадку jar поставляється за допомогою плагіна kotlin в intelliJ, або ви можете знайти його в мережі або додати залежність maven / grandle до свого проекту.
Мікеле д'Аміко

7
компілювати "org.jetbrains.kotlin: kotlin-test: $ kotlin_version"
mac229

4
@ mac229 s / compile / testCompile /
Лоуренс Гонсалвес,

@AshishSharma: kotlinlang.org/api/latest/kotlin.test/kotlin.test/… assertFail З поверненням винятку, і ви можете використовувати його для написання власного твердження.
Мікеле д'Аміко,

26

Ви можете використовувати @Test(expected = ArithmeticException::class)або навіть краще один із таких бібліотечних методів Котліна, як failsWith().

Ви можете зробити це ще коротшим, використовуючи реалізовані дженерики та допоміжний метод, такий як:

inline fun <reified T : Throwable> failsWithX(noinline block: () -> Any) {
    kotlin.test.failsWith(javaClass<T>(), block)
}

І приклад використання анотації:

@Test(expected = ArithmeticException::class)
fun omg() {

}

javaClass<T>()зараз застаріло. Використовуйте MyException::class.javaзамість цього.
fasth

failsWith застаріло, замість нього слід використовувати assertFailsWith .
gvlasov

15

Для цього можна використовувати KotlinTest .

У вашому тесті ви можете обернути довільний код блоком shouldThrow:

shouldThrow<ArithmeticException> {
  // code in here that you expect to throw a ArithmeticException
}

здається, лінія не працює належним чином. Я перевіряю 1. shouldThrow <java.lang.AssertionError> {someMethod (). IsOK shouldBe true} - зелений 2. shouldThrow <java.lang.AssertionError> {someMethod (). IsOK shouldBe false} - зелений someMethod () throw "java .lang.AssertionError: повідомлення "коли потрібно, і повернути об'єкт, якщо все в порядку. В обох випадках shouldThrow зелений, коли нормально, а коли НЕ.
Іван Тречокас,

Можливо, погляньте на документи, це могло змінитися з моєї відповіді у 2016 році. Github.com/kotlintest/kotlintest/blob/master/doc/…
sksamuel

13

JUnit5 має вбудовану підтримку kotlin .

import org.junit.jupiter.api.Test
import org.junit.jupiter.api.assertThrows

class MyTests {
    @Test
    fun `division by zero -- should throw ArithmeticException`() {
        assertThrows<ArithmeticException> {  1 / 0 }
    }
}

3
Це моя найкраща відповідь. Якщо ви потрапляєте Cannot inline bytecode built with JVM target 1.8 into bytecode that is being built with JVM target 1.6на assertThrows, переконайтеся, що ваш build.gradle маєcompileTestKotlin { kotlinOptions.jvmTarget = "1.8" }
Big Pumpkin

11

Ви також можете використовувати дженерики з пакетом kotlin.test:

import kotlin.test.assertFailsWith 

@Test
fun testFunction() {
    assertFailsWith<MyException> {
         // The code that will throw MyException
    }
}

1

Затвердити розширення, яке перевіряє клас винятку, а також відповідність повідомлення про помилку.

inline fun <reified T : Exception> assertThrows(runnable: () -> Any?, message: String?) {
try {
    runnable.invoke()
} catch (e: Throwable) {
    if (e is T) {
        message?.let {
            Assert.assertEquals(it, "${e.message}")
        }
        return
    }
    Assert.fail("expected ${T::class.qualifiedName} but caught " +
            "${e::class.qualifiedName} instead")
}
Assert.fail("expected ${T::class.qualifiedName}")

}

наприклад:

assertThrows<IllegalStateException>({
        throw IllegalStateException("fake error message")
    }, "fake error message")

1

Ніхто не згадував, що assertFailsWith () повертає значення, і ви можете перевірити атрибути винятків:

@Test
fun `my test`() {
        val exception = assertFailsWith<MyException> {method()}
        assertThat(exception.message, equalTo("oops!"))
    }
}

0

Інша версія синтаксису з використанням kluent :

@Test
fun `should throw ArithmeticException`() {
    invoking {
        val backHole = 1 / 0
    } `should throw` ArithmeticException::class
}

0

Перші кроки - це додати (expected = YourException::class)до тестової анотації

@Test(expected = YourException::class)

Другим кроком є ​​додавання цієї функції

private fun throwException(): Boolean = throw YourException()

Нарешті у вас вийде щось подібне:

@Test(expected = ArithmeticException::class)
fun `get query error from assets`() {
    //Given
    val error = "ArithmeticException"

    //When
    throwException()
    val result =  omg()

    //Then
    Assert.assertEquals(result, error)
}
private fun throwException(): Boolean = throw ArithmeticException()

0

org.junit.jupiter.api.Assertions.kt

/**
 * Example usage:
 * ```kotlin
 * val exception = assertThrows<IllegalArgumentException>("Should throw an Exception") {
 *     throw IllegalArgumentException("Talk to a duck")
 * }
 * assertEquals("Talk to a duck", exception.message)
 * ```
 * @see Assertions.assertThrows
 */
inline fun <reified T : Throwable> assertThrows(message: String, noinline executable: () -> Unit): T =
        assertThrows({ message }, executable)
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.