Some assorted unit-testing related utilities.
There project contains two targets for building Mac and iOS static libraries.
It also contains a target with some example unit tests in it, which illustrate how to use ECTestCase and ECParameterisedTestCase.
The ECTest macros are much like the STTest ones, except that they don't take a description parameter. Instead, they generate a suitable description from the context.
In practise I find that most of the time, the description is redundant, and ends up repeating the logic of the test itself. The knowledge that an assertion has failed is usually enough, and the descriptions just add clutter to the code.
This class contains a few utility methods which:
- support the macros
- support using classes that need run loops from unit tests
See this blog post for more details of the run loop support.
This class allows you to define unit tests that have a series of test data items applied in turn.
The normal tests work great if you want to run each test once, but what if you have a set of test data and you want to run each test multiple times, applying each item of test data to it in turn?
The naive approach is to define lots of test methods that just call onto another helper method supplying a different argument each time. Something like this:
- (void)testXWithDataA { [self helperX:@"A"]; }
- (void)testXWithDataB { [self helperX:@"B"]; }
That gets tired quickly, and it doesn't allow for a dynamic amount of test data determined at runtime.
What you really want in this case is to add the following abilities to SenTest:
- the ability to define parameterised test methods using a similar naming convention to the normal ones
- the ability to define a class method which returns a dictionary of test data
- have SenTest make a sub-suite for each parameterised method we found
- have the sub-suite contain a test for each data item
- iterate the suites and tests as usual, applying the relevant data item to each test in turn
This is what ECParameterisedTest gives you.
- inherit from ECParameterisedTest instead of SenTestCase
- define test methods which are named parameterisedTestXYZ instead of testXYZ (they still take no parameters)
- either: define a class method called parameterizedTestData which returns a dictionary containing data
- or: create a plist with the name of your test class, which will contain the data
- or: create a directory with the extension .testsuite, containing your test data (see below)
At runtime the data method will be called to obtain the data. The names of each key should be single words which describe the data. The values can be anything you like - whatever the test methods are expecting.
To simplify the amount of modification to SenTest, the test methods still take no parameters. Instead, to obtain the test data, each test method uses the parameterisedTestDataItem property.
To make its test suites, SenTestKit calls a class method called defaultTestSuite on each SenTestCase subclass that it finds.
The default version of this makes a suite based on finding methods called testXYZ, but it's easy enough to do something else.
Our version calls parameterisedTestData, which is a class method which returns a dictionary containing the data For each parameterised test method we find, we make a suite. For each item in the dictionary, we add a test to this suite. Finally we make a master suite, add the other suites to it, and return that.
To make things simple, we use the existing SenTestKit mechanism to invoke the test methods. Since SenTestKit expects test methods not to have any parameters, we need another way of passing the test data to each method. Each test invocation creates an instance of a our class, and we do this creation at the point we build the test suite, so the simple answer is just to add a property to the test class. We can set this property value when we make the test instance, and the test method can extract the data from the instance when it runs.
To obtain the test data, we've added a method parameterisedTestData that we expect the test class to implement.
This method returns a dictionary rather than an array, so that we can use the keys as test names, and the values as the actual data. Having names for the data is useful because of the way SenTestKit reports the results.
Typically it reports each test as [SuiteName testName], taking these names from the class and method. Since we're going to use the name of the test method for each of our suites, we really need another name to use for each test. This is where the dictionary key comes in.
Where the test data comes from is of course up to you and the kind of tests you are trying to perform.
There is are a couple of simple scenarios though, which are that we want to load it from a plist that we provide along with the test class, or that we want to load it from a directory of files.
Since we need a default implementation of the method anyway, we cater for these simple case automatically. We look for a plist with the same name as the test class, or a folder with the name of the test class and the file extension "testsuite".
The plist can have one of two formats.
In the simple case, it's just a dictionary, where each entry is a test data item.
In the slightly more complex case, it contains two keys: ECTestItems and ECTestSuites. The items key contains the data items for this suite. The suites item contains data for any sub-suites (using the same data layout recursively for these sub-suites).
Using this complex case it's possible to create hierarchies of test suites, to group the results more clearly.
Instead of a plist, you can store your test data in a recursive directory structure.
If you provide a directory with the name of the test class, and the extension "testsuite", it will get parsed into a dictionary of test data in the following way:
Any .testsuite folders inside it will be added as sub-suites
Any other folders in it will be treated as test data items
The contents of any files inside each test data item folder will be added to the dictionary for that test as text, if possible. If not, the URL will be added instead.
A file called testdata.plist in any item will be read as a dictionary, and the keys in it added to the dictionary for the item.
A file called testdata.plist in any suite folder will be read and added to the data properties for that suite. This can be used to define some items on disk, and others from plists.
This framework depends on the ECConfig framework for some standard xcconfig and pch files.
Annoyingly, Xcode doesn't look in the build products folder when searching for files included in xcconfig files. This means that we can't include things from the XCConfig framework in our xcconfig files in a totally portable way, by doing
#include <ECConfig/ECMacDebug.xcconfig>
Instead, we have to assume that the framework's source folder is actually at a known relative location. If this isn't the case, you'll get a build warning:
ECMacDebug.xcconfig line 8: Unable to find included file...
You can fix this by adding a symbolic link called ECConfig that points to the right place.