Overview of Property-Based Testing Functionality

November 12, 2025 · View on GitHub

Property-based testing (PBT) frameworks come with a number of different features, but which library supports which features? For a PBT newcomer, it can be hard to tell. Strictly speaking you don't even need a PBT framework. Property-based tests can be written from scratch on a case-by-case basis using a random number generator. One can even test stateful code without a state-machine framework, e.g., as outlined here. However a framework provides reusable parts and infrastructure, thus paving the way for bigger developments, such as property-based testing automotive software against the AUTOSAR specification.

  • When finding a counterexample shrinking is nice to get to the essence of an issue
  • To test an imperative API a framework with state machine support would be nice.
  • Integrated shrinking can be a nice feature for bigger developments where writing custom shrinkers may be out of the question.
  • ...

This overview is to help myself keep track. It has been compiled over a number of years teaching PBT. As features are gradually added to a framework the table's entries may unfortunately become outdated. YMMV.

I'll be happy to accept PRs for updating entries and adding new frameworks.

Framework functionality

FrameworkLanguageGen. EDSLShrinkingInt. shr.State machinePar. st. mach.Cov. guidance
theftC( :heavy_check_mark: ):heavy_check_mark:( :heavy_check_mark: )( :heavy_check_mark: )
DeepStateC / C++( :heavy_check_mark: ):heavy_check_mark::heavy_check_mark::heavy_check_mark:
RapidCheckC++:heavy_check_mark::heavy_check_mark::heavy_check_mark:( :heavy_check_mark: )
CsCheckC# / .Net:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
test.checkClojure:heavy_check_mark::heavy_check_mark::heavy_check_mark:??
check-itCommon Lisp:heavy_check_mark::heavy_check_mark:
cl-quickcheckCommon Lisp:heavy_check_mark:
QuickChickCoq / Rocq:heavy_check_mark::heavy_check_mark::heavy_check_mark:
PropCheckElixir:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
StreamDataElixir:heavy_check_mark::heavy_check_mark::heavy_check_mark:
elm testElm:heavy_check_mark::heavy_check_mark::heavy_check_mark:
propcheckEmacs Lisp:heavy_check_mark::heavy_check_mark::heavy_check_mark:
QuickCheckErlang:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
PropErErlang:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
FsCheckF# / .Net:heavy_check_mark::heavy_check_mark::heavy_check_mark:
FSharp-HedgehogF# / .Net:heavy_check_mark::heavy_check_mark::heavy_check_mark:
gopterGo:heavy_check_mark::heavy_check_mark::heavy_check_mark:
RapidGo:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
QuickCheckHaskell:heavy_check_mark::heavy_check_mark:( :heavy_check_mark: )( :heavy_check_mark: )
SmallCheckHaskell:heavy_check_mark:11
HedgehogHaskell:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
AmericiumJava / Scala:heavy_check_mark::heavy_check_mark::heavy_check_mark:
junit-quickcheckJava:heavy_check_mark::heavy_check_mark:( :heavy_check_mark: )2
QuickTheoriesJava:heavy_check_mark::heavy_check_mark::heavy_check_mark:( :heavy_check_mark: )( :heavy_check_mark: ):heavy_check_mark:
jqwikJava:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
fast-checkJS / TS:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:( :heavy_check_mark: )3
propCheckKotlin:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
Lua-QuickCheckLua:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
FoxObj.C / Swift:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:( :heavy_check_mark: )
QCheckOCaml:heavy_check_mark::heavy_check_mark::heavy_check_mark:4:heavy_check_mark::heavy_check_mark:
CrowbarOCaml:heavy_check_mark:( :heavy_check_mark: )5:heavy_check_mark:
MonolithOCaml:heavy_check_mark:( :heavy_check_mark: )5:heavy_check_mark:
Base_quickcheckOCaml:heavy_check_mark::heavy_check_mark:
PopperOCaml:heavy_check_mark::heavy_check_mark::heavy_check_mark:
quickcheckProlog:heavy_check_mark::heavy_check_mark:
HypothesisPython:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:( :heavy_check_mark: )3( :heavy_check_mark: ), ( :heavy_check_mark: )
TSTLPython6:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
R-HedgehogR:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
racket-quickcheckRacket:heavy_check_mark:
PropCheckRuby:heavy_check_mark::heavy_check_mark::heavy_check_mark:
PBTRuby:heavy_check_mark::heavy_check_mark:
RantlyRuby:heavy_check_mark::heavy_check_mark:
quickcheckRust:heavy_check_mark::heavy_check_mark:
proptestRust:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
Scala-HedgehogScala / JVM:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
ScalaCheckScala / JVM:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
scalapropsScala / JVM:heavy_check_mark::heavy_check_mark:
EchidnaSolidity / EVM:heavy_check_mark::heavy_check_mark::heavy_check_mark::heavy_check_mark:
qcheckStandard ML:heavy_check_mark::heavy_check_mark:
SwiftCheckSwift:heavy_check_mark::heavy_check_mark:
...

Legend:

  • Language denotes the frameworks's language or platform
  • Generator EDSL denotes whether the framework has an expressive, embedded domain-specific language for programming custom generators (int, list, map, ...)
  • Shrinking denotes whether the framework has built-in support for reducing counterexamples.
  • Integrated shrinking denotes whether a shrinker automatically preserves any invariants of an EDSL-based custom generator (sorted lists, non-empty array, valid JSON, ...)
  • State machine denotes whether the framework has a state-machine library for model-based testing.
  • Parallel state machine denotes whether the framework supports parallel state-machine testing for race conditons, etc.
  • Coverage guidance denotes whether the framework's generators (and shrinkers) can be guided by code coverage information obtained via instrumentation.

Footnotes

Background:

The term property-based testing seems to originate from 'Towards a Property-based Testing Environment with Applications to Security-Critical Software' by Fink, Ko, Archer, and Levitt (Irvine Software Symposium, 1994). It was later the topic of Fink's 1995 UC Davis PhD dissertation 'Discovering security and safety flaws using property-based testing' and the paper 'Property-Based Testing; A New Approach to Testing for Assurance' by Fink and Bishop (SE Notes 1997). The approach was popularized as an embedded domain-specific language in 'QuickCheck: A Lightweight Tool for Random Testing of Haskell Programs' by Claessen and Hughes (ICFP 2000) which inspired ports to many other languages.

Integrated shrinking is explained in more detail in

State machines to test protocols and systems with state are described in

Parallel state-machine tests for race conditions were later introduced in

Additional resources: