DEV Community

Cover image for Intro to Property-Based Testing
Jason Steinhauser
Jason Steinhauser

Posted on • Updated on

Intro to Property-Based Testing

Property-based tests make statements about the output of your code based on the input, and these statements are verified for many different possible inputs. - Jessica Kerr (@jessitron)

Property-based testing is generative testing. You do not supply specific example inputs with expected outputs as with unit tests. Instead, you define properties about the code and use a generative-testing engine (e.g., QuickCheck) to create randomized inputs to ensure the defined properties are correct.

What is the purpose of generative testing?

Generally speaking, property-based tests require only a few lines of code (like unit tests), but unlike unit tests they test a different set of inputs each time. Because of this, you end up covering more domain space with roughly the same amount of test code.

Property-based testing also promotes a more in-depth understanding of the function under test. Sure, we know that for addition 2 + 2 = 4, but how can you say that you truly have implemented an addition function correctly with that one example?

From a business standpoint, a better understanding of the functionality of your code will lead to more accurate determination of requirements being met in your code.

Does it replace unit tests?

While it is tempting to say that unit tests could be completely replaced by high quality property-based tests, unit tests still have their place in the software development cycle. Example-based testing is fantastic for early stages of TDD. These serve as anchor points to ensure that your development efforts are proceeding as desired. For example:

  • You can ensure that your sine function is generating proper values (sin(0) = 0, sin(π/2) = 1, etc.)
  • HTTP POSTs to your /login route without proper authentication headers/cookies results in a 401 response code

However, eventually these example-based tests end up being passive tests; the tests become part of a regression suite and provide no new information about the functionality. Property-based tests, however, are always active tests as they generate new data each time the test suite is run. This can help root out issues that developers overlook. For example, you may expect that the output of your sine function for any floating-point representation to be in the set [-1,-1], but the developer may not think to test for ±∞ or NaN. To this end, when a failing case is identified, this unexpected error can be used as a future unit test to validate fixes were made. This reason alone makes example-based testing an effective practice for enhancing property-based test suites.

"Don't write tests. Generate them!" - John Hughes, co-author of Haskell's original QuickCheck

What are some common properties?

If you correlate software functions with mathematical functions, then you can apply some standard mathematical properties to some expected functionality. However, some of those properties aren't as apparent in non-mathematical applications. Here are a few example mathematical properties translated to function properties (with updates from Eric Normand), ignoring performance differences:

  • Associative – a + (b + c) = (a + b) + c

    • hashmap1.merge(hashmap2.merge(hashmap3)) = (hashmap.merge(hashmap2)).merge(hashmap3)
    • list1.append(list2.append(list3)) = (list1.append(list2).append(list3)
    • Math.max(Math.max(a, b), c) = Math.max(a, Math.max(b, c))
    • (bool1 && bool2) && bool3 = bool1 && (bool2 && bool3)
  • Commutative – a + b = b + a

    • users.Sort().Filter(x => !x.IsAdmin) = (users.Filter(x => !x.IsAdmin)).Sort() (relocated from Associative)
    • image.flipX().flipY() = image.flipY().flipX()
    • Math.max(a,b) = Math.max(b,a)
    • bool1 && bool2 = bool2 && bool1
    • average(a,b) = average(b, a)
  • Distributive – a(b + c) = ab + ac

    • title.ToUpper() + author.ToUpper() = (title + author).ToUpper()
  • Idempotent – f(a) = f(f(a))

    • Math.abs(x) = Math.abs(Math.abs(x))
    • hashmap.merge({a:1}) = hashmap.merge({a:1}).merge({a:1})
    • title.Trim() = title.Trim().Trim()
    • list.Sort() = list.Sort().Sort()
  • Identity – f(a, i) where i is identity value of f = a

    • a + 0 = a
    • a * 1 = a
    • userNames.append([]) = userNames
    • hashmap.merge({}) = hashmap
    • bool1 && true = bool1
  • Zero – f(a, z) where z is zero value of f = z

    • a * 0 = 0
    • intersect(valueSet, emptySet) = emptySet
    • bool1 && false = false

There are additional, commonly-tested properties that are not necessarily rooted in math, but are equally as useful:

  • Bilbo Testing (aka, There And Back Again)
    • list = list.Reverse().Reverse()
    • obj = JSON.parse(JSON.stringify(obj))
  • No Unexpected Changes
    • list.Length = list.Sort().Length
  • Hard to Prove, Easy to Verify
    • Sorting: each element should be greater than or equal to the previous element
    • Tokenizing: concatenating tokens with separator interposed should equal the original string, number of tokens should equal number of separators in original string - 1

Is PBT actually used in the "real world"?

In short, yes. Property-based testing is absolutely used to solve real-world problems.

Volvo

Volvo has employed QuviQ's QuickCheck to validate third-party components conforming to communication bus standards. John Hughes and his team analyzed 3k pages of requirements, wrote 20k lines of QuickCheck tests, and applied those properties to 1MM+ lines of vendor code. By checking their properties generated around the specifications, they found 200 issues that had slipped through the test fixtures that the vendors had generated... and 100 of those issues were in the actual specifications themselves. By refining the code analyzed (as well as the defining specifications), property-based testing has saved lives.

Clojure

Clojure (a LISP variant on the JVM) uses immutable data structures by default. However, mutable datatypes are available for performance increases. Using property-based testing, an issue was found when converting to/from transient (mutable) and persistent (immutable) structures that none of the example-based tests (re: unit tests). However, property-based testing was able to readily reproduce the issue. A diff patch was provided to Cognitect (the developers/maintainers of Clojure), and it was incorporated in the Clojure 1.6 release.

Personal experience

I've also used property-based testing on a project where we were calculating network performance by various metrics (QoS, packet type, packet size, etc.). I had to prove that my packet classification and matching algorithm wouldn't provide false matches, so I set my CI server up to generate 100 million different packets, distributed between the different packet types we were interested in. As of now, we have randomly generated over 10 billion packets and have not had a single false matches.

What are some awesome parts to look for in PBT suites?

Shrinking

While large inputs may produce the errors, several property-based test suites (most QuickCheck variants) will attempt to shrink the input sequence to the smallest possible that will reproduce the error. The smaller the input, the easier it is to reproduce and fix.

Race Conditions

Race conditions are notoriously hard to set up via example-based testing. Some PBT suites (notably, QuviQ's QuickCheck in Erlang) can perform many actions in parallel in order to test if the actions, performed in any serial combination, would produce the same output. If the parallelized version can not be matched to a serialized version, then the race condition is identified and shrunk to the smallest possible set of operations to reproduce the error.

Custom generators

You may want to limit the input domain you are testing, or have data structures constructed in a particular fashion. Most PBT suites will allow you to create custom generators, and many common custom ones are generally included as well – limiting strings to printable characters, constraining floating point values, etc.
If you're migrating from one algorithm to another, a PBT test suite that passes the same input to both the old and new implementations can be used to ensure that the new implementation produces the same output.

Where can I learn more?

Thankfully, there are several good learning resources for property-based testing lurking around on the Internet. Here are some of the ones I've found to be extremely helpful.

Top comments (11)

Collapse
 
ericnormand profile image
Eric Normand

Hey Jason,

Very nice article!

I really like that you focus on the mathematical properties. That's where property-based testing really shines. Just like unit tests help us write testable code, property-based tests help us write operations with simple and useful properties.

A couple of comments:

a = a * 1 = a * 1 * 1 is idempotent. 1 is also the identity of *, which is another property.

Idempotence (function f is idempotent): f(a) = f(f(a))

Examples:

Math.abs(x) = Math.abs(Math.abs(x))
hashmap.merge({a:1}) = hashmap.merge({a:1}).merge({a:1})

Identity (i is the identity of f): f(a, i) = a

Examples:

a + 0 = a
a * 1 = a
usernames.append([]) = usernames
hashmap.merge({}) = hashmap
Bool1 && True = Bool1

There's also the mathematical idea of zero. a * 0 = 0. It's another cool property.

Zero (z is the zero of f): f(a, z) = z
intersect(set1, EmptySet) = EmptySet
Bool1 && False = False

I think the example you gave for associativity (with sorting and filtering) is actually an example of commutativity since it shows that order doesn't matter between sorting and filtering.

Associativity examples:

hashmap1.merge(hashmap2.merge(hashmap3)) = (hasmap1.merge(hashmap2)).merge(hashmap3)
list1.append(list2.append(list3)) = (list1.append(list2)).append(list3)

Commutativity examples:

Math.max(a, b) = Math.max(b, a) (though this one is also associative :)
Bool1 && Bool2 = Bool2 && Bool1 (this one is associative too)
average(a, b) = average(b, a)

Again, this was an awesome article!

Rock on!
Eric

Collapse
 
jdsteinhauser profile image
Jason Steinhauser

Hey Eric,

First off, I geeked out again when I saw I had a comment from you. I've loved several of your articles on Clojure!

Secondly, I think your critiques are correct. I misused idempotency slightly, and my associative example is actually commutative. I definitely appreciate the clarification and additional examples? Would you mind if I added those to the article?

Thanks again for the complements and clarifications, and keep up the good work at PurelyFunctional.tv!

Collapse
 
ericnormand profile image
Eric Normand

Hey Jason,

Sorry I didn't reply to this sooner. Be my guest and reuse the examples.

Rock on!
Eric

Collapse
 
quii profile image
Chris James

This is such a better writeup than the one I just published! Haha, amazing work

Collapse
 
jdsteinhauser profile image
Jason Steinhauser

Well thank you! I have your article on my reading list, so I'll be sure to check it out 😃

Collapse
 
Sloan, the sloth mascot
Comment deleted
Collapse
 
jdsteinhauser profile image
Jason Steinhauser

Thanks! I'm glad you were able to find value in it :-)

Collapse
 
ben profile image
Ben Halpern

Absolutely superb writeup 🙌

Collapse
 
jdsteinhauser profile image
Jason Steinhauser

Why thank you! When I saw this email notification I may have geeked out a bit

Collapse
 
akotek profile image
Aviv Kotek • Edited

Finally an article who doesn't copy/paste clojure-docs :p
liked the property-patterns, Thanks!

Collapse
 
dotnettec profile image
Ravi Kumar

Very nice article!

Thanks.
Ravi (DotNetTec)