Package
Ponycheck is a library for property based testing with tight integration into ponytest.
Property Based Testing¶
In traditional unit testing the developer specifies one or more input examples manually for the class or system under test and assert on certain output conditions. The difficulty here is to find enough examples to cover all branches and cases of the class or system under test.
In property bases testing the developer defines a property, a kind of predicate for the class or system under test that should hold for all kinds or just a subset of possible input values. The property based testing engine then generates a big number of random input values and checks if the property holds for all of them. The developer only needs to specify the possible set of input values using a Generator.
This testing technique is great for finding edge cases that would easily go unnoticed with manually constructed test samples. In general it can lead to much higher coverage than traditional unit-testing, with much less code to write.
How Ponycheck implements Property Based Testing¶
A property based test in ponycheck consists of the following:
- A name (for integration into ponytest mostly)
- One or more generators, depending on how your property is layed out. There are tons of them defined on primitive Generators.
- A
property
method that asserts a certain property for each sample generated by the Generator(s) with the help of PropertyHelper which tries to expose a similar API as TestHelper. - Optionally, the method
params()
can be used to configure how ponycheck executes the property by specifying a custom PropertyParams object.
The classical list-reverse example:
use "collections"
use "ponycheck"
class ListReverseProperty is Property1[List[USize]]
fun name(): String => "list/reverse"
fun gen(): Generator[List[USize]] =>
Generators.list_of[USize](Generators.usize())
fun property(arg1: List[USize], ph: PropertyHelper) =>
ph.array_eq[USize](arg1, arg1.reverse().reverse())
Integration into Ponytest¶
There are two ways of integrating a Property into ponytest:
- In order to pass your Property to the ponytest engine, you need to wrap it inside a Property1UnitTest.
actor Main is TestList
new create(env: Env) => PonyTest(env, this)
fun tag tests(test: PonyTest) =>
test(Property1UnitTest[String](MyStringProperty))
- Run as much Properties as you wish inside one ponytest
UnitTest using the convenience function
Ponycheck.for_all providing a
Generator, the TestHelper and the
actual property function. (Note that the property function is supplied in a
second application of the result to
for_all
.)
class ListReversePropertyWithinAUnitTest is UnitTest
fun name(): String => "list/reverse/forall"
fun apply(h: TestHelper) =>
let gen = recover val Generators.list_of[USize](Generators.usize()) end
Ponycheck.for_all[List[USize]](gen, h)(
{(sample, ph) =>
ph.array_eq[Usize](arg1, arg1.reverse().reverse())
})
// ... possibly more properties, using ``Ponycheck.for_all``
Independent of how you integrate with ponytest, the ponycheck machinery will instantiate the provided Generator, and will execute it for a configurable number of samples.
If the property fails using an assertion method of PropertyHelper, the failed example will be shrunken by the generator to obtain a smaller and more informative, still failing, sample for reporting.
Public Types¶
- class Randomness
- interface PropertyLogger
- interface PropertyResultNotify
- actor PropertyRunner
- class PropertyHelper
- class PropertyParams
- trait Property1
- trait Property2
- trait Property3
- trait Property4
- class Poperator
- primitive Ponycheck
- class IntPropertySample
- trait IntProperty
- class IntPairPropertySample
- trait IntPairProperty
- type ValueAndShrink
- type GenerateResult
- class CountdownIter
- trait GenObj
- class Generator
- type WeightedGenerator
- primitive Generators
- class ForAll
- class ForAll2
- class ForAll3
- class ForAll4
- primitive ASCIINUL
- primitive ASCIIDigits
- primitive ASCIIWhiteSpace
- primitive ASCIIPunctuation
- primitive ASCIILettersLower
- primitive ASCIILettersUpper
- primitive ASCIILetters
- primitive ASCIIPrintable
- primitive ASCIINonPrintable
- primitive ASCIIAll
- primitive ASCIIAllWithNUL
- type ASCIIRange