More readable data-driven tests
When the logic of a test method remains constant but the data varies, data-driven testing is a great tool. It allows you, the test author, to write compact code and to add new test cases rapidly. Unfortunately, data-driven tests have a disadvantage: The inputs are often less readable.
A simple example
Let’s take an example; testing Rob Conery’s PagedList implementation. A page is basically a slice of the data returned by a linq query. If more data exists beyond the ‘slice’ represented by the PagedList
Here’s a quick data-driven test for HasNextPage:
As you can see, the method itself is readable, but the parametrised values fed in via the [TestCase] attribute are not. It’s really hard to keep everything in your head and remember what each number maps to in the function, especially when the method parameter types are all identical. If you have a list of 20 [TestCase] attributes, you start wondering what’s for your tea and forget that the second value is the (checks image) page size. Mince. Mince for tea.
Hmm, if only we could those TestCases more readable; something like object initializers would be ideal.
A simple trick: Subclass TestCase
My friend Hughel helped me come up with this one and it works quite well. Simply subclass the TestCaseAttribute class and add your own properties to represent the test parameters. It gets a little bit hairy when you have to access the Arguments array directly (especially since Attributes can be weird), but in practice, it works fine. In most of the tests, we’re only interested in parametrising three things, so it’s simple to add them as properties.
The end result
Finally, we apply these attributes to our data-driven test, significantly improving the readability!
I would hasten to add that I don’t recommend using this approach willy-nilly - only when you have a large amount of tests that are parametrised by the same data types, causing the test cases to become hard to follow. I’ve used the [TestCaseSource] attribute and the [TestCase] attribute a lot in the past and most of the time it’s not a problem.