id | title | slug | sidebar_label |
---|---|---|---|
testfunctions |
Property Test Functions |
property-test-functions.html |
Test Functions |
There are two variants of functions that are used to execute a property test in Kotest: forAll
and checkAll
.
The first, forAll
, accepts an n-arity function (a, ..., n) -> Boolean
that tests the property.
The test will pass if, for all input values, the function returns true.
class PropertyExample: StringSpec({
"String size" {
forAll<String, String> { a, b ->
(a + b).length == a.length + b.length
}
}
})
Notice that this functions accepts type parameters for the argument types, with arity up to 14. Kotest uses these type parameters to locate a generator which provides (generates) random values of a suitable type.
For example, forAll<String, Int, Boolean> { a, b, c -> }
is a 3-arity property test where
argument a
is a random String, argument b
is a random int, and argument c
is a random boolean.
The second, checkAll
, accepts an n-arity function (a, ..., n) -> Unit
in which you can simply execute assertions against the inputs.
This approach will consider a test valid if no exceptions are thrown.
Here is the same example again written in the equivalent way using checkAll.
class PropertyExample: StringSpec({
"String size" {
checkAll<String, String> { a, b ->
a + b shouldHaveLength a.length + b.length
}
}
})
The second approach is more general purpose than returning a boolean, but the first approach is from the original haskell libraries that inspired this library.
By default, Kotest will run the property test 1000 times. We can easily customize this by specifying the iteration count when invoking the test method.
Let's say we want to run a test 10,000 times.
class PropertyExample: StringSpec({
"a many iterations test" {
checkAll<Double, Double>(10_000) { a, b ->
// test here
}
}
})
You saw in the previous examples that Kotest would provide values automatically based on the type parameter(s). It does this by locating a generator that generates values for the required type.
For example, the automatically provided Integer generator generates random ints from all possible values - negative, positive, infinities, zero and so on.
This is fine for basic tests but often we want more control over the sample space. For example, we may want to test a function for numbers in a certain range only.
Then you would need to specify the generator(s) manually.
class PropertyExample: StringSpec({
"is allowed to drink in Chicago" {
forAll(Arb.int(21..150)) { a ->
isDrinkingAge(a) // assuming some function that calculates if we're old enough to drink
}
}
"is allowed to drink in London" {
forAll(Arb.int(18..150)) { a ->
isDrinkingAge(a) // assuming some function that calculates if we're old enough to drink
}
}
})
You can see we created two tests and in each test passed a generator into the forAll
function with a suitable int range.
See here for a list of the built in generators.