paint-brush
Testing in C#: Property-Based Testing With Input Generatorsby@miguel-bernard
550 reads
550 reads

Testing in C#: Property-Based Testing With Input Generators

by Miguel BernardJuly 6th, 2020
Read on Terminal Reader
Read this story w/o Javascript
tldt arrow

Too Long; Didn't Read

Testing in C#: Property-Based Testing With Input Generators. Miguel Bernard is passionate about teaching, developers' communities and everything related to.Net. The following post will try to define properties for the first and last lines of our diamond diamond test. The first property-based test reads like this:Call the DiamondGenerate() method with the provided input char c. Use the ToProperty() method to wrap it into a property object. The next property test will be written using a generator that will do just that.

Company Mentioned

Mention Thumbnail
featured image - Testing in C#: Property-Based Testing With Input Generators
Miguel Bernard HackerNoon profile picture

Intro

Now that we have all the tools installed, we are finally ready to write our first Property-Based test.

FsCheck

Creating a property-based test is as simple as tagging a method with the [Property] attribute, similarly as you would do in xUnit with the [Fact] or [Theory] attributes. If you dig a little, you will realize that there's a lot of configurable options on the [Property] attribute. You can control how many random tests will be run each time, timeouts, etc.

One of them is essential; the Arbitrary property. That property will define which method to use to generate random inputs. In our specific case, we are not interested in all the possible values of char. We only care about the 26 letters of the alphabet. To scope the input, we'll need to configure a generator that will do just that.

Generators

Generators are relatively simple.

  1. Create a static class
  2. Add a static method that returns Arbitrary<TypeYouWantToGenerateValueFor>
  3. Voilà
You can write your randomization logic, but it's usually better to start from the built-in methods as they are already well balanced.
public static class LetterGenerator
{
    public static Arbitrary<char> Generate() =>
            Arb.Default.Char().Filter(c => c >= 'A' && c <= 'Z');
}

Not Empty Test

It may seem basic, but it's quite essential. We want to make sure we always have a non-empty result.

[Property(Arbitrary = new[] { typeof(LetterGenerator) })]
public Property NotEmpty(char c)
{
    return Diamond.Generate(c).All(s => s != string.Empty).ToProperty();
}

One thing I forgot to mention earlier, the test method also needs to return a property object. Fortunately for us, the library contains an extension method that can transform any boolean expression into a property object by calling ToProperty().

So our test reads like this:

  • Call the Diamond.Generate() method with the provided input char c.
  • Make sure all the lines of the returned diamond are not empty.
  • Use the ToProperty() method to wrap it into a property object.

Stay tuned

Stay tuned for my next posts on this series about solving the diamond kata with property-based testing. The following post will try to define properties for the first and last lines of our diamond.

Previously published at https://blog.miguelbernard.com/input-generators-in-property-based-tests-with-fscheck/