Skip to main content

Generator Operations


If you want to use an Arb to just return a value (even outside of a property test), then you can call next on it.

val arbA: Arb<A> = ...
val a = // use Random.Default
val a2 = // pass in Random


If you have an arb and you want to create a new arb that provides a subset of values, you can call filter on the source arb. For example, one way of generating even numbers is to take the integer arb, and filter out odd values. Viz:

val evens = { it.value % 2 == 0 }
val odds = { it.value % 2 == 1 }


If you have an arb and you want to transform the value generated, you can use map.

val integerStrings: Arb<String> = { it.toString() }


If you have an arb whose emission or edge cases depends on the emission of the previous arbitraries, you can use flatMap.

val dependentArbs: Arb<String> = Arb.of("foo", "bar").flatMap { prefix -> { integer ->


Two generators can be merged together, so that elements 0, 2, 4, ... are taken from the first generator, and elements 1, 3, 5, ... are taken from the second generator.

val merged = arbA.merge(arbB)

So with the following example:

val a = arbitrary { "a" }
val b = arbitrary { "b" }
val ab = a.merge(b)
ab.take(10).forEach { println(it) }

Would ouput ababababab


Bind is useful if you want to apply multiple arbitraries. We can take a look at how we might construct values for a data class using bind.

data class Person(val name: String, val age: Int)

val personArb: Arb<Person> = Arb.bind(
) { name, age -> Person(name, age) }