Skip to main content

Property based testing in C#–Part 1

In this multipart blog post I want to introduce you in the world of property-based testing and how to do this in C#.

In the first part(this one), I’ll give you an introduction on what property-based testing is, why it useful and how it can help you write better code. In a second post, I’ll show you a concrete example on how to start writing Property based tests in C#.

Disclaimer: Property-based testing is hard! And although I would strongly recommend it for critical parts of your codebase, I would certainly not use it everywhere.

What is Property-based testing?

Property-based testing was introduced in 2000 by Koen Claessen and John Hughes via the Haskell library QuickCheck.

Mark Seemann uses the following definition in his Pluralsight course on Property based testing:

Property-based Testing is an automated testing technique where you incrementally zero in on the correct behavior of a system by describing its properties or qualities in general terms and then use randomly generated test data to execute deterministic tests.

I have read the definition above multiple times but I still find it hard to understand. So let us try to explain this further…

Most of the test I see developers write are Example-based tests. We try to think about what would be representative input for the functions we want to test and use these examples to create some unit tests. The better developers don't only think about the happy path and try to also think about what could be incorrect input to see if our functions handle failure conditions gracefully.

The problem is that when your functions get at a certain level of complexity, it becomes quite hard to think at all possible variations(examples). And although your test coverage could be 100%, you are still not 100% confident that your tests are covering all edge cases.

Property-based testing can help you deal with the shortcomings of example-based tests. Instead of finding relevant examples, we try to understand and represent the relationship between our inputs and outputs. We search for the rules, specifications, “properties” that allow us to describe this relationship in a general way without specifying exactly what the input or output should be.

Sounds still abstract? Let’s give an example:

 

To test the ‘add’ method above, we can try to come up with representative examples but there is an infinite list of possible combinations. Instead let’s try to describe the add method as a relationship between inputs and outputs:

  • Relationship 1: It doesn’t matter in which order we provide the x and y input, the output is the same.
    • E.g. 1+7 is the same as 7+1
  • Relationship 2: Having zero as either the x or y input is the same as doing nothing (the output doesn’t change).
    • E.g. 7+0 =  7 or 0+7 = 0

These are the obvious ones to come up with, a third property that is also relevant is the following:

  • Relationship 3: If I call the add method multiple times, the order in which I do them doesn’t matter.
    • E.g. 7+(4+5) = (7+4)+5

It is not a coincidence that this perfectly aligns with the mathematical description of an additive operation:

  • It doesn’t matter in which order we provide the x and y parameter, the result is the same. –> Commutativity
  • Having zero as one of the parameters does nothing. –> Identity
  • If I call the add method multiple times, the order in which I do them doesn’t matter –> Associativity

What’s nice about these properties is that they work with all inputs, not just special magic numbers. These properties are in fact the specification of the additive operation.

So far for the theory. In my next post I’ll show you how to write Property-based tests in C#.

If you want to learn more about Property-based testing, here are some interesting resources for you:

Popular posts from this blog

Podman– Command execution failed with exit code 125

After updating WSL on one of the developer machines, Podman failed to work. When we took a look through Podman Desktop, we noticed that Podman had stopped running and returned the following error message: Error: Command execution failed with exit code 125 Here are the steps we tried to fix the issue: We started by running podman info to get some extra details on what could be wrong: >podman info OS: windows/amd64 provider: wsl version: 5.3.1 Cannot connect to Podman. Please verify your connection to the Linux system using `podman system connection list`, or try `podman machine init` and `podman machine start` to manage a new Linux VM Error: unable to connect to Podman socket: failed to connect: dial tcp 127.0.0.1:2655: connectex: No connection could be made because the target machine actively refused it. That makes sense as the podman VM was not running. Let’s check the VM: >podman machine list NAME         ...

Azure DevOps/ GitHub emoji

I’m really bad at remembering emoji’s. So here is cheat sheet with all emoji’s that can be used in tools that support the github emoji markdown markup: All credits go to rcaviers who created this list.

VS Code Planning mode

After the introduction of Plan mode in Visual Studio , it now also found its way into VS Code. Planning mode, or as I like to call it 'Hannibal mode', extends GitHub Copilot's Agent Mode capabilities to handle larger, multi-step coding tasks with a structured approach. Instead of jumping straight into code generation, Planning mode creates a detailed execution plan. If you want more details, have a look at my previous post . Putting plan mode into action VS Code takes a different approach compared to Visual Studio when using plan mode. Instead of a configuration setting that you can activate but have limited control over, planning is available as a separate chat mode/agent: I like this approach better than how Visual Studio does it as you have explicit control when plan mode is activated. Instead of immediately diving into execution, the plan agent creates a plan and asks some follow up questions: You can further edit the plan by clicking on ‘Open in Editor’: ...