DEV Community

Mario Fernández
Mario Fernández

Posted on • Originally published at hceris.com on

Mock verification in Kotlin using MockK and Atrium

I have been working with Kotlin a lot lately. It is a really awesome language. Elegant, powerful and succinct, it fixes most of the annoyances that I had with Java, yet it keeps a certain amount of familiarity that allows the transition from it to be very manageable.

Anyhow, I found myself recently having to build a filter in SpringBoot that I wanted to test. For that I needed to use both a mock and verify that behavior at the same time. Kotlin is evolving quite fast and there are plenty of alternatives to choose from. I will show how to do this with two excellent libraries, MockK, and Atrium.

MockK and Atrium, a powerful combo

In the short time that I have been developing Kotlin, I’ve noticed a pattern. Whenever you need something not provided in the standard library, you tend to start by using the existing Java library that you are familiar with. Then, at some point, you figure out there is a native Kotlin library that leverages the features from the language better.

MockK seems to be on its way to become the defacto mocking library for Kotlin apps. With a syntax based heavily around lambdas, it just looks like a DSL, as you can see in this example taken directly from their page:

val car = mockk<Car>()

every { car.drive(Direction.NORTH) } returns Outcome.OK
car.drive(Direction.NORTH) // returns OK

verify { car.drive(Direction.NORTH) }
Enter fullscreen mode Exit fullscreen mode

Meanwhile, Atrium is less established, but after getting the recommendation from a colleague, I gave it a try. It uses expect, so for somebody like me who is used to RSpec it is already a win. Anyhow, the syntax takes some time to get used to, but it can be quite expressive. I particularly like combining it with data classes to have exactly one assertion per test instead of many.

The problem at hand

What I was trying to build was not particularly complex. I wanted to write a filter for a SpringBoot application that would inject some headers into the request based on some logic. All controllers would use these values transparently, without having to care about the computation. The filter looks like this.

@Component
class Filter : OncePerRequestFilter() {
    @Autowired
    lateinit var processor: Processor

    override fun doFilterInternal(request: HttpServletRequest, response: HttpServletResponse, filterChain: FilterChain) {
        val wrappedRequest = HttpServletRequestWrapper(request).apply {
            addHeader(Headers.EXTRA_HEADER, processor.process(request))
        }

        filterChain.doFilter(wrappedRequest, response)
    }
}
Enter fullscreen mode Exit fullscreen mode

I want to test two things:

  • The filterChain should be called with my wrappedRequest
  • The wrappedRequest should have the correct header in it

Setting up the test

I am using JUnit 5 for the test (Speaking of native libraries, I haven’t tried something like Spock yet). The basic setup of the test requires to set up the filter and the mocks that I need.

@ExtendWith(MockKExtension::class)
internal class FilterTest {
    val request = MockHttpServletRequest()

    @MockK
    lateinit var response: HttpServletResponse

    @RelaxedMockK
    lateinit var filterChain: FilterChain

    val filter = Filter().apply {
        processor = ProcessorImpl()
    }
}
Enter fullscreen mode Exit fullscreen mode

I am using annotations to initialize the mocks (which requires annotating the test with MockKExtension). My filterChain is a RelaxedMockK, which means that its methods will return a default value unless otherwise specified.

A very simple test

If I just want to check that the method is being called, I don’t really need much

@Test
fun `calls the next step in the filter`() {
    filter.doFilter(request, response, filterChain)
    verify { filterChain.doFilter(any(), response) }
}
Enter fullscreen mode Exit fullscreen mode

Testing the wrapped request

The previous test is OK, but it is a bit bland for my taste. I want to make sure that that the filterChain is being called with my wrappedRequest, and that it contains the header I injected. This test becomes much more interesting

@Test
fun `injects header into the request and passes it to the filter`() {
    filter.doFilter(request, response, filterChain)

    slot<ServletRequest>().let { slot ->
        verify { filterChain.doFilter(capture(slot), response) }

        expect(slot.captured).isA<HttpServletRequestWrapper> {
            expect(subject.getHeader(Headers.EXTRA_HEADER)).toBe("value")
        }
    }
}
Enter fullscreen mode Exit fullscreen mode

Not so simple anymore! Let’s break it down.

First we are capturing the first argument for doFilter (i.e: the wrapped request). We are creating a new slot by doing slot<ServletRequest>, and capturing it by passing it in the verify block by doing capture(slot). The let block wrapping everything is there so that we don’t need an extra local variable (and to feel more kotlin-y inside).

After all this slot.captured contains the wrappedRequest that we created in the filter. Here is where Atrium can shine. We use isA first to check that the request is of the right type. Then inside the block subject is the casted type, where we finally check that our header is there.

Summary

With this our small filter can be tested properly and with very little overhead. I am no Kotlin connoisseur, but the syntax of both MockK and Atrium feels quite elegant to me, once you wrap your head around it. I think this is a good starting point to build better and better tests without a ton of boilerplate code.

Top comments (0)