fbpx

Parametrized tests with Spek


Everyone knows the situations when we have to check our system for various input. A good example would be a login/password validation performed both on front and backend.
There are many approaches to create a test code which covers all necessary cases:

1.  copy-paste methods (not recommended),

2. make many assertions in one test method (we can lose some valuable error info)

3. use parametrized test functionality bundled within test framework we use.
Keep in mind that there are more options available.

If you want to have several test cases in jUnit4, you have to use parametrized runner and add some methods with dedicated annotations, and implement method returning a list of parameters and expected output:

@RunWith(Parameterized::class)
class FibonacciTest(private val fInput: Int, private val fExpected: Int) {

    @Test
    fun test() {
        assertEquals(fExpected, Fibonacci.compute(fInput))
    }

    companion object {
        @JvmStatic
        @Parameterized.Parameters
        fun data(): Collection<Array<Any>> {
            return listOf(
                    arrayOf<Any>(0, 0),
                    arrayOf<Any>(1, 1),
                    arrayOf<Any>(2, 1), 
                    arrayOf<Any>(3, 2),
                    arrayOf<Any>(4, 3),
                    arrayOf<Any>(5, 5),
                    arrayOf<Any>(6, 8)))
        }
    }
}

Pretty ugly, huh? A full example in Java can be found here: 
https://github.com/junit-team/junit4/wiki/Parameterized-tests

We can do better

Imagine, if we could write our tests like we write our code. Clear output, no annotation driven development, and no unnecessary static methods. That is were Spek comes to help.

Consider simple distance converter — we parse given distance in meters to desired unit.

value [m]displayed string
753753 m
13371.34 km
1588815.9 km
3127031 km
51000>50 km

It’s a simple case —the implementation is trivial. It only takes one switch or when and a function to round number with desired precision to solve this case.

Despite the fact, that the implementation is simple, we should prepare unit test it. Using Spek we can also write BDD style tests and prepare a specification for one approach.

class DistanceConverterSpecificiation : Spek({
    describe("distance converter") {
        val distanceConverter: DistanceConverter by memoized {
            DistanceConverter()
        }
        on("distance 61888.123m") {
            val testDistance = 61888.123
            it("should return >50km") {
                val convert = distanceConverter.convert(testDistance)
                assertTrue(convert.contentEquals(">50km"))
            }
        }
        on("distance 38777.23m") {
            val testDistance = 38777.23
            it("should return 38.8km") {
                val convert = distanceConverter.convert(testDistance)
                assertTrue(convert.contentEquals("38.8km"))
            }
        }
    }
})

Run this snippet and check results. We tested two cases, however, we are still repeating ourselves. It is as long and ugly as @Parametrized jUnit test or just test with many methods.

Spek DSL is a huge lambda expression, so we can use any Kotlin language features.

Let’s start with defining our test cases and expected outcome using Map<Double, String>:

val testCases = mapOf(
    61888.123 to ">50 km",
    38777.23 to "38.8 km",
    16984.44 to "17.0 km",
    987.98 to "988 m"
)

Now we can use forEach{} to generate desired test cases:

testCases.forEach { value, expectedValue ->
            on("$value"){
                it("should print $expectedValue"){}
            }
        }

Finally, we can make some assertion. A full example will look like this:

class DistanceConverterSpecificiation : Spek({
    describe("distance converter") {
        val testCases = mapOf(
                61888.123 to ">50 km",
                38777.23 to "38.8 km",
                16984.44 to "17.0 km",
                987.98 to "988 m"
        )
        val converter = DistanceConverter()
        testCases.forEach { value, expectedValue ->
            on("$value"){
                it("should print $expectedValue"){
                    assertTrue(converter.convert(value).contentEquals(expectedValue))
                }
            }
        }
    }
})

Conclusion

When creating unit tests we have to be sure that all (reasonable) cases are covered. In provided example (distance converter) we used Map<Double, String> and forEach{k,v -> } to parametrize our test and create on() ActionBody and it() TestBody in a stream. We have performed checks on each value, asserted expected outcome and displayed test result nicely in our IDE. We haven’t lost any valuable info, and we can add another test cases in clear Kotlin syntax.

Using Spek gives you the full power of Kotlin — you can write your tests just like you write your code and make use of language features.

I hope that you find this article useful, and it will help you look at parametrized tests in a different way.


Originally published on: https://blog.kotlin-academy.com


more insights

Uncategorized
Jarosław Michalik

3 things that defined my Android dev career

Building an Android developer career isn’t just about coding—it’s about the experiences that push you to grow. Whether it’s connecting with other Android devs at meetups or mastering the fundamentals, it’s those key moments that shape how you approach the craft.

Read More »

AndroidPro newsletter 🚀

join 3057 developers reading AndroidPro newsletter every week

👉 tips & tricks, articles, videos

👉 every Thursday in your inbox

🎁 bonus on signup – Clean Architecture blueprint

brought to you by Jarek Michalik, Kotlin GDE

You can unsubscribe any time. For details visit our privacy policy.

android pro newsletter tips and tricks how to learn android