Junit before all kotlin

Котлин и Юнит 5 @ BeforeAll

В Kotlin у классов нет статических методов . Java-эквивалентная семантика может быть предоставлена ​​вызывающим абонентам, используя концепцию сопутствующего объекта . В этом посте будет подробно рассказано о том, что требуется для поддержки аннотаций JUnit 5 @BeforeAll и @AfterAll, которые зависят от прецедента статических методов в тестовых классах.

До и после и все в Java

Junit 5 @BeforeAll аннотированные методы выполняются до всех тестов, а @AfterAll ожидается после всех тестов. Предполагается, что эти аннотации будут применяться к статическим методам:

Грубый поток – платформа JUnit вызывает аннотированные методы @BeforeAll, затем
для каждого теста он создает экземпляр класса теста и вызывает тест. После того, как все тесты выполнены, вызываются аннотированные статические методы «@AfterAll», это подтверждается журналами, посмотрите, как отличаются идентификаторы экземпляра (из toString () объекта):

2018 — 03 — 28 17 : 22 : 03.653 INFO — [ main] c.p.cookbook.Junit5BeforeAllTest : com.pivotalservices.cookbook.Junit5BeforeAllTest @7bc1a03d

2018 — 03 — 28 17 : 22 : 03.664 INFO — [ main] c.p.cookbook.Junit5BeforeAllTest : com.pivotalservices.cookbook.Junit5BeforeAllTest @6591f517

Этот жизненный цикл по умолчанию теста JUnit 5 может быть изменен аннотацией, хотя, если класс теста аннотируется следующим образом:

Преимущество теперь состоит в том, что аннотации @BeforeAll и @AfterAll можно размещать в нестатических методах, поскольку платформа JUnit 5 может гарантировать, что эти методы будут ровно один раз перед ВСЕМИ тестами. Суть в том, что любое состояние уровня экземпляра не будет сбрасываться перед каждым тестом.

Читайте также:  Disabled style css class

До и после и все в Котлине

Так как же это перевести на Kotlin –

Для случая по умолчанию нового экземпляра теста для каждого теста эквивалентный код теста Kotlin выглядит следующим образом:

Сопутствующий объект Kotlin с методами, аннотированными @JvmStatic, выполняет эту работу.

Упрощен тот случай, когда жизненный цикл изменяется:

Лично я предпочитаю подход сопутствующего объекта, поскольку мне нравится идея детерминированного состояния экземпляра теста перед выполнением метода теста. Другое преимущество этого подхода заключается в тестах на основе Spring Boot, в которых вы хотите, чтобы Spring воздействовал на экземпляр теста (вставляет зависимости, разрешает свойства и т. Д.) Только после вызова аннотированного метода @BeforeAll, чтобы конкретнее рассмотреть следующий пример:

Источник

Kotlin and JUnit 5 @BeforeAll

In Kotlin, classes do not have static methods. A Java equivalent semantic can be provided to callers using the concept of a companion object though. This post will go into details of what it takes to support a JUnit 5 @BeforeAll and @AfterAll annotation which depend on the precense of a static methods in test classes.

BeforeAll and AfterAll in Java

Junit 5 @BeforeAll annotated methods are executed before all tests and @AfterAll is exected after all tests. These annotations are expected to be applied to static methods:

import org.junit.jupiter.api.AfterAll; import org.junit.jupiter.api.BeforeAll; import org.junit.jupiter.api.Test; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class Junit5BeforeAllTest < private static final Logger LOGGER = LoggerFactory.getLogger(Junit5BeforeAllTest.class); @BeforeAll static void beforeAll() < LOGGER.info("beforeAll called"); >@Test public void aTest1() < LOGGER.info("aTest1 called"); LOGGER.info(this.toString()); >@Test public void aTest2() < LOGGER.info("aTest2 called"); LOGGER.info(this.toString()); >@AfterAll static void afterAll() < LOGGER.info("afterAll called"); >>

A rough flow is – the JUnit platform calls the “@BeforeAll” annotated methods, then
for each test it creates an instance of the test class and invokes the test. After all tests are executed, the “@AfterAll” annotated static methods are called, this is borne out by the logs, see how the instance ids(from toString() of Object) is different:

2018-03-28 17:22:03.618 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : beforeAll called 2018-03-28 17:22:03.652 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : aTest1 called 2018-03-28 17:22:03.653 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : com.pivotalservices.cookbook.Junit5BeforeAllTest@7bc1a03d 2018-03-28 17:22:03.663 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : aTest2 called 2018-03-28 17:22:03.664 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : com.pivotalservices.cookbook.Junit5BeforeAllTest@6591f517 2018-03-28 17:22:03.669 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : afterAll called

This default lifecycle of a JUnit 5 test can be changed by an annotation though if the test class is annotated the following way:

@TestInstance(TestInstance.Lifecycle.PER_CLASS) public class Junit5BeforeAllTest

The advantage now is that the @BeforeAll and @AfterAll annotations can be placed on non-static methods, as the JUnit 5 platform can guarantee that these methods are exactly once before ALL tests. The catch though is that any instance-level state will not be reset before each test.

BeforeAll and AfterAll in Kotlin

So how does this translate to Kotlin –

For the default case of a new test instance per test, an equivalent Kotlin test code looks like this:

import org.junit.jupiter.api.AfterAll import org.junit.jupiter.api.BeforeAll import org.junit.jupiter.api.Test import org.slf4j.LoggerFactory class Junit5BeforeAllKotlinTest < @Test fun aTest1() < LOGGER.info("aTest1 called") LOGGER.info(this.toString()) >@Test fun aTest2() < LOGGER.info("aTest2 called") LOGGER.info(this.toString()) >companion object < private val LOGGER = LoggerFactory.getLogger(Junit5BeforeAllTest::class.java) @BeforeAll @JvmStatic internal fun beforeAll() < LOGGER.info("beforeAll called") >@AfterAll @JvmStatic internal fun afterAll() < LOGGER.info("afterAll called") >> >

A Kotlin companion object with methods annotated with @JvmStatic does the job.

Simpler is the case where the lifecycle is modified:

import org.junit.jupiter.api.AfterAll import org.junit.jupiter.api.BeforeAll import org.junit.jupiter.api.Test import org.junit.jupiter.api.TestInstance import org.slf4j.LoggerFactory @TestInstance(TestInstance.Lifecycle.PER_CLASS) class Junit5BeforeAllKotlinTest < private val LOGGER = LoggerFactory.getLogger(Junit5BeforeAllTest::class.java) @BeforeAll internal fun beforeAll() < LOGGER.info("beforeAll called") >@Test fun aTest1() < LOGGER.info("aTest1 called") LOGGER.info(this.toString()) >@Test fun aTest2() < LOGGER.info("aTest2 called") LOGGER.info(this.toString()) >@AfterAll internal fun afterAll() < LOGGER.info("afterAll called") >>

My personal preference is for the companion object approach as I like the idea of a deterministic state of the test instance before the test method is executed. Another advantage of the approach is with Spring Boot based tests where you want Spring to act on the test instance (inject dependencies, resolve properties etc) only after @BeforeAll annotated method is called, to make this more concrete consider the following example:

import org.assertj.core.api.Assertions.assertThat import org.junit.jupiter.api.AfterAll import org.junit.jupiter.api.BeforeAll import org.junit.jupiter.api.Test import org.junit.jupiter.api.extension.ExtendWith import org.springframework.beans.factory.annotation.Value import org.springframework.boot.test.context.SpringBootTest import org.springframework.context.annotation.Configuration import org.springframework.test.context.junit.jupiter.SpringExtension @ExtendWith(SpringExtension::class) @SpringBootTest class BeforeAllSampleTest < @Value("\$") private lateinit var someKey: String companion object < @BeforeAll @JvmStatic fun beforeClass() < System.setProperty("some.key", "some-value") >@AfterAll @JvmStatic fun afterClass() < System.clearProperty("some.key") >> @Test fun testValidateProperties() < assertThat(someKey).isEqualTo("some-value") >@Configuration class SpringConfig >

This kind of a test will not work at all if the lifecycle were changed to “@TestInstance(TestInstance.Lifecycle.PER_CLASS)”

Reference

This stackoverflow answer was instrumental in my understanding of the nuances of JUnit 5 with Kotlin.

Источник

all and sundry

In Kotlin, classes do not have static methods. A Java equivalent semantic can be provided using the concept of a companion object though. This post will go into details of what it takes to support a JUnit 5 @BeforeAll and @AfterAll annotation which depend on the presence of a static methods in test classes.

BeforeAll and AfterAll in Java

Junit 5 @BeforeAll annotated methods are executed before all tests and @AfterAll is exected after all tests. These annotations are expected to be applied to static methods:

import org.junit.jupiter.api.AfterAll; import org.junit.jupiter.api.BeforeAll; import org.junit.jupiter.api.Test; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class Junit5BeforeAllTest < private static final Logger LOGGER = LoggerFactory.getLogger(Junit5BeforeAllTest.class); @BeforeAll static void beforeAll() < LOGGER.info("beforeAll called"); >@Test public void aTest1() < LOGGER.info("aTest1 called"); LOGGER.info(this.toString()); >@Test public void aTest2() < LOGGER.info("aTest2 called"); LOGGER.info(this.toString()); >@AfterAll static void afterAll() < LOGGER.info("afterAll called"); >>

A rough flow is — the JUnit platform calls the «@BeforeAll» annotated methods, then for each test it creates an instance of the test class and invokes the test. After all tests are executed, the «@AfterAll» annotated static methods are called, this is borne out by the logs, see how the instance ids(from toString() of Object) is different:

2018-03-28 17:22:03.618 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : beforeAll called 2018-03-28 17:22:03.652 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : aTest1 called 2018-03-28 17:22:03.653 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : com.pivotalservices.cookbook.Junit5BeforeAllTest@7bc1a03d 2018-03-28 17:22:03.663 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : aTest2 called 2018-03-28 17:22:03.664 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : com.pivotalservices.cookbook.Junit5BeforeAllTest@6591f517 2018-03-28 17:22:03.669 INFO --- [ main] c.p.cookbook.Junit5BeforeAllTest : afterAll called

This default lifecycle of a JUnit 5 test can be changed by an annotation though if the test class is annotated the following way:

@TestInstance(TestInstance.Lifecycle.PER_CLASS) public class Junit5BeforeAllTest

The advantage now is that the @BeforeAll and @AfterAll annotations can be placed on non-static methods, as the JUnit 5 platform can guarantee that these methods are exactly once before ALL tests. The catch though is that any instance-level state will not be reset before each test.

BeforeAll and AfterAll in Kotlin

So how does this translate to Kotlin —
For the default case of a new test instance per test, an equivalent Kotlin test code looks like this:

import org.junit.jupiter.api.AfterAll import org.junit.jupiter.api.BeforeAll import org.junit.jupiter.api.Test import org.slf4j.LoggerFactory class Junit5BeforeAllKotlinTest < @Test fun aTest1() < LOGGER.info("aTest1 called") LOGGER.info(this.toString()) >@Test fun aTest2() < LOGGER.info("aTest2 called") LOGGER.info(this.toString()) >companion object < private val LOGGER = LoggerFactory.getLogger(Junit5BeforeAllTest::class.java) @BeforeAll @JvmStatic internal fun beforeAll() < LOGGER.info("beforeAll called") >@AfterAll @JvmStatic internal fun afterAll() < LOGGER.info("afterAll called") >> >

A Kotlin companion object with methods annotated with @JvmStatic does the job.

Simpler is the case where the lifecycle is modified:

import org.junit.jupiter.api.AfterAll import org.junit.jupiter.api.BeforeAll import org.junit.jupiter.api.Test import org.junit.jupiter.api.TestInstance import org.slf4j.LoggerFactory @TestInstance(TestInstance.Lifecycle.PER_CLASS) class Junit5BeforeAllKotlinTest < private val LOGGER = LoggerFactory.getLogger(Junit5BeforeAllTest::class.java) @BeforeAll internal fun beforeAll() < LOGGER.info("beforeAll called") >@Test fun aTest1() < LOGGER.info("aTest1 called") LOGGER.info(this.toString()) >@Test fun aTest2() < LOGGER.info("aTest2 called") LOGGER.info(this.toString()) >@AfterAll internal fun afterAll() < LOGGER.info("afterAll called") >>

My personal preference is for the companion object approach as I like the idea of a deterministic state of the test instance before the test method is executed. Another advantage of the approach is with Spring Boot based tests where you want Spring to act on the test instance (inject dependencies, resolve properties etc) only after @BeforeAll annotated method is called, to make this more concrete consider the following example:

import org.assertj.core.api.Assertions.assertThat import org.junit.jupiter.api.AfterAll import org.junit.jupiter.api.BeforeAll import org.junit.jupiter.api.Test import org.junit.jupiter.api.extension.ExtendWith import org.springframework.beans.factory.annotation.Value import org.springframework.boot.test.context.SpringBootTest import org.springframework.context.annotation.Configuration import org.springframework.test.context.junit.jupiter.SpringExtension @ExtendWith(SpringExtension::class) @SpringBootTest class BeforeAllSampleTest < @Value("\$") private lateinit var someKey: String companion object < @BeforeAll @JvmStatic fun beforeClass() < System.setProperty("some.key", "some-value") >@AfterAll @JvmStatic fun afterClass() < System.clearProperty("some.key") >> @Test fun testValidateProperties() < assertThat(someKey).isEqualTo("some-value") >@Configuration class SpringConfig >

This kind of a test will not work at all if the lifecycle were changed to «@TestInstance(TestInstance.Lifecycle.PER_CLASS)»

Correction
Per comments from the one and only Sébastien Deleuze, the previous test can be simplified by injecting in dependencies and properties via constructor injection, so the test can be re-written as:

@ExtendWith(SpringExtension::class) @SpringBootTest class BeforeAllSampleTest(@Value("\$") val someKey: String) < companion object < @BeforeAll @JvmStatic fun beforeClass() < System.setProperty("some.key", "some-value") >@AfterAll @JvmStatic fun afterClass() < System.clearProperty("some.key") >> @Test fun testValidateProperties() < assertThat(someKey).isEqualTo("some-value") >@Configuration class SpringConfig >

Reference

This stackoverflow answer was instrumental in my understanding of the nuances of JUnit 5 with Kotlin.

Источник

Оцените статью