Update (Apr 2010): It’s been quite a few years since I originally did this comparison. Since then, Charles Nicholson and I created Unit Test++, a C/C++ unit-testing framework that addresses most of my requirements and wish-list items. It’s designed to be a light-weight, high-performance testing framework, particularly aimed at games and odd platforms with limited functionality. It’s definitely my framework of choice and I haven’t looked at new ones in several years because it fits my needs so well. I definitely encourage you to check it out.
———-
One of the topics I’ve been meaning to get to for quite a while is the applicability of test-driven development in games. Every time the topic comes up in conversations or mailing lists, everybody is always very curious about it and they immediately want to know more. I will get to that soon. I promise!
In the meanwhile I’m now in the situation that I need to choose a unit-testing framework to roll out for my team at work. So, before I get to talk about how to use test-driven development in games, or the value of unit testing, or anything like that, we dive deep into a detailed comparison of existing C++ unit-testing frameworks. Hang on tight. It’s going to be a long and bumpy ride with a plot twist at the end.
If you just want to read about a specific framework, you can go directly there:
Overview
How do we choose a unit-testing framework? It depends on what we’re going to do with it and how we’re going to use it. If I used Java for most of my work, the choice would be easy since JUnit seems to be the framework of choice for those working with Java. I don’t hear them arguing over frameworks or proposing new ones very frequently, so it must be pretty good.
Unfortunately that’s not the case with C++. We have our XUnit family member, CppUnit, but we’re clearly not happy with that. We have more unit-testing frameworks than you can shake a stick at. And a lot of teams end up writing their own from scratch. Why is that? Is C++ so inadequate for unit testing that we have trouble fitting the XUnit approach in the language? Not like it’s a bad thing, mind you. Diversity is good. Otherwise I would be stuck writing this under Windows and you would be stuck reading it with Internet Explorer. In any case, I’m clearly not the first one who’s asked this question. This page tries to answer the question, and comes up with some very plausible answers: differences in compilers, platforms, and programming styles. C++ is not exactly a clean, fully supported language, with one coding standard.
A good way to start is to create a list of features that are important given the type of work I expect to be doing. In particular, I want to be doing test-driven development (TDD), which means I’m going to be constantly writing and running many small tests. It’s going to be used for game development, so I’d like to run the tests in a variety of different platforms (PC, Xbox, PS2, next-generation consoles, etc). It should also fit my own personal TDD style (many tests, heavy use of fixtures, etc).
The following list summarizes the features I would like in a unit-testing framework in order of importance. I’ll evaluate each framework on the basis of these features. Thanks to Tom Plunket for providing a slightly different view on the topic that helped me to re-evaluate the relative importance of the different features.
- Minimal amount of work needed to add new tests. I’m going to be doing this all the time, so I don’t want to do a lot of typing, and I especially don’t want to do any duplicated typing. The shorter and easier it is to write, the easier it’ll be to refactor, which is crucial when you’re doing TDD.
- Easy to modify and port. It should have no dependencies with non-standard libraries, and it shouldn’t rely on “exotic†C++ features if possible (RTTI, exception handling, etc). Some of the compilers we have to use for console development are not exactly cutting edge. To verify this one, I created a set of unit tests using each library under Linux with g++. Since most of the tests are written with Windows and Visual Studio in mind, it’s not a bad initial test.
- Supports setup/teardown steps (fixtures). I’ve adopted the style recommended by David Astels in his book Test Driven Development: A Practical Guide about using only one assertion per test. It really makes tests a lot easier to understand and maintain, but it requires heavy use of fixtures. A framework without them is ruled out immediately. Bonus points for frameworks that let me declare objects used in the fixture on the stack (and still get created right before the test) as opposed to having to allocate them dynamically.
- Handles exceptions and crashes well. We don’t want the tests to stop just because some code that was executed accessed some invalid memory location or had a division by zero. The unit-testing framework should report the exception and as much information about it as possible. It should also be possible to run it again and have the debugger break at the place where the exception was triggered.
- Good assert functionality. Failing assert statements should print the content of the variables that were compared. It should also provide a good set of assert statements for doing “almost equality†(absolutely necessary for floats), less than, more than, etc. Bonus points for providing ways to check whether exceptions were or were not thrown.
- Supports different outputs. By default, I’d like to have a format that can be understood and parsed by IDEs like Visual Studio or KDevelop, so it’s easy to navigate to any test failures as if they were syntax errors. But I’d also like to have ways to display different outputs (more detailed ones, shorter ones, parsing-friendly ones, etc).
- Supports suites. It’s kind of funny that this is so low in my priority list when it’s usually listed as a prominent feature in most frameworks. Frankly, I’ve had very little need for this in the past. It’s nice, yes, but I end up having many libraries, each of them with its own set of tests, so I hardly ever need this. Still, it certainly would be nice to have around in case it starts getting slow to run the unit tests at some point.
Bonus: Timing support. Both for total running time of tests, and for individual ones. I like to keep an eye on my running times. Not for performance reasons, but to prevent them from getting out of hand. I prefer to keep running time to under 3-4 seconds (it’s the only way to be able to run them very frequently). Ideally, I’d also like to see a warning printed if any single test goes over a certain amount of time.
Easy of installation was not considered a priority; after all, I only have to go through that once—it’s creating new tests that I’m going to be doing all day long. Non-commercial-friendly licenses (like GPL or LGPL) are also not much of an issue because the unit test framework is not something we’re going to link to the executable we ship, so they don’t really impose any restrictions on the final product.
Incidentally, during my research for this article, I found that other people have compiled lists of what they wish for in C++ unit-testing frameworks. It’s interesting to contrast that article with this one and make a note of the differences and similarities between what we’d like to see in a unit test framework.
Ideal Framework
Before I start going over each of the major (and a few minor) C++ unit-testing frameworks, I decided I would apply the philosophy behind test-driven development to this analysis and start by thinking what I would like to have. So I decided to write the set of sample tests in some ideal unit-testing framework without regard for language constrains or anything. In the ideal world, this is what I would like my unit tests to be like.
The simplest possible test should be trivial to create. Just one line to declare the test and then the test body itself:
TEST (SimplestTest) { Â float someNum = 2.00001f; Â ASSERT_CLOSE (someNum, 2.0f); }
A test with fixtures is going to be a bit more complicated, but it should still be really easy to set up:
SETUP (FixtureTestSuite) { float someNum = 2.0f; std::string str = "Hello"; MyClass someObject("somename"); someObject.doSomethng(); } TEARDOWN (FixtureTestSuite) { someObject.doSomethingElse(); } TEST (FixtureTestSuite, Test1) { ASSERT_CLOSE (someNum, 2.0f); someNum = 0.0f; } TEST (FixtureTestSuite, Test2) { ASSERT_CLOSE (someNum, 2.0f); } TEST (FixtureTestSuite, Test3) { ASSERT_EQUAL(str, "Hello"); }
The first thing to point out about this set of tests is that there is a minimum amount of code spent in anything other than the tests themselves. The simplest possible test takes a couple of lines and needs no support other than a main file that runs all the tests. Setting up a fixture with setup/teardown calls should also be totally trivial. I don’t want to inherit from any classes, override any functions, or anything. Just write the setup step and move on.
Look at the setup function again. The variables that are going to be used in the tests are not dynamically created. Instead, they appear to be declared on the stack and used directly there. Additionally, I should point out that those objects should only be created right before each test, and not before all tests start. How exactly are the tests going to use them? I don’t know, but that’s what I would like to use. That’s why this is an ideal framework.
Now let’s contrast it to six real unit-testing frameworks that have to worry about actually compiling and running. For each of the frameworks I look at the list of wanted features and I try to implement the two tests I implemented with this ideal framework. Here is the source code for all the examples.
CppUnit is probably the most widely used unit-testing framework in C++, so it’s going to be a good reference to compare other unit tests against. I had used CppUnit three or four of years ago and my impressions back then were less than favorable. I remember the code being a mess laced with MFC, the examples all tangled up with the framework, and the silly GUI bar tightly coupled with the software. I even ended up creating a patch to provide console-only output and removed MFC dependencies. So this time I approached it with a bit of apprehension to say the least.
I have to admit that CppUnit has come a long way since then. I was expecting the worst, but this time I found it much easier to use and configure. It’s still not perfect, but it’s much, much better than it used to. The documentation is pretty decent, but you’ll have to end up digging deep into the module descriptions to even find out that some functionality is available.
- Minimal amount of work needed to add new tests. This is one of the major downfalls of CppUnit, and, ironically, it’s the highest-rated feature I was looking for. CppUnit requires quite a bit of work for the simplest possible test.
- Easy to modify and port. It gets mixed marks on this one. On one hand, it runs under Windows and Linux, and the functionality is reasonably well modularized (results, runners, outputs, etc). On the other hand, CppUnit still requires RTTI, the STL, and (I think) exception handling. It’s not the end of the world to require that, but it could be problematic if you want to link against libraries that have no RTTI enabled, or if you don’t want to pull in the STL.
- Supports fixtures. Yes. If you want the objects to be created before each test, they need to be dynamically allocated in the setup() function though, so no bonus there.
- Handles exceptions and crashes well. Yes. It uses the concept of “protectors†which are wrappers around tests. The default one attempts to catch all exceptions (and identify some of them). You can write your own custom protectors and push them on the stack to combine them with the ones already there. It didn’t catch system exceptions under Linux, but it would have been trivial to add with a new protector. I don’t think it had a way to easily turn off exception handling and let the debugger break where the exception happened though (no define or command-line parameter).
- Good assert functionality. Pretty decent. It has the minimum set of of assert statements, including one for comparing floating-point numbers. It’s missing asserts for less than, greater than, etc. The contents of the variables compared are printed to a stream if the assert fails, giving you as much information as possible about the failed test.
- Supports different outputs. Yes. Has very-well defined functionality for “outputters†(which display the results of the tests), as well as “listeners†(which get notified while the tests are happening). It comes with an IDE-friendly output that is perfect for integrating with Visual Studio. Also supports GUI progress bars and the like.
- Supports suites. Yes.
// Simplest possible test with CppUnit #include <cppunit/extensions/HelperMacros.h> class SimplestCase : public CPPUNIT_NS::TestFixture { CPPUNIT_TEST_SUITE( SimplestCase ); CPPUNIT_TEST( MyTest ); CPPUNIT_TEST_SUITE_END(); protected: void MyTest(); }; CPPUNIT_TEST_SUITE_REGISTRATION( SimplestCase ); void SimplestCase::MyTest() { float fnum = 2.00001f; CPPUNIT_ASSERT_DOUBLES_EQUAL( fnum, 2.0f, 0.0005 ); }
#include <cppunit/extensions/HelperMacros.h> #include "MyTestClass.h" class FixtureTest : public CPPUNIT_NS::TestFixture { CPPUNIT_TEST_SUITE( FixtureTest ); CPPUNIT_TEST( Test1 ); CPPUNIT_TEST( Test2 ); CPPUNIT_TEST( Test3 ); CPPUNIT_TEST_SUITE_END(); protected: float someValue; std::string str; MyTestClass myObject; public: void setUp(); protected: void Test1(); void Test2(); void Test3(); }; CPPUNIT_TEST_SUITE_REGISTRATION( FixtureTest ); void FixtureTest::setUp() { someValue = 2.0; str = "Hello"; } void FixtureTest::Test1() { CPPUNIT_ASSERT_DOUBLES_EQUAL( someValue, 2.0f, 0.005f ); someValue = 0; //System exceptions cause CppUnit to stop dead on its tracks //myObject.UseBadPointer(); // A regular exception works nicely though myObject.ThrowException(); } void FixtureTest::Test2() { CPPUNIT_ASSERT_DOUBLES_EQUAL( someValue, 2.0f, 0.005f ); CPPUNIT_ASSERT_EQUAL (str, std::string("Hello")); } void FixtureTest::Test3() { // This also causes it to stop completely //myObject.DivideByZero(); // Unfortunately, it looks like the framework creates 3 instances of MyTestClass // right at the beginning instead of doing it on demand for each test. We would // have to do it dynamically in the setup/teardown steps ourselves. CPPUNIT_ASSERT_EQUAL (1, myObject.s_currentInstances); CPPUNIT_ASSERT_EQUAL (3, myObject.s_instancesCreated); CPPUNIT_ASSERT_EQUAL (1, myObject.s_maxSimultaneousInstances); }
Overall, CppUnit is frustrating because it’s almost exactly what I want, except for my most wanted feature. I really can’t believe that it takes so much typing (and duplicated typing at that) to add new tests. Other than that, the main complaint is the need for RTTI or exceptions, and the relative complexity of the source code, which could make it challenging to port to different platforms.
Update: I’ve revised my comments and ratings of the Boost.Test framework in light of the comments from Gennadiy Rozental pointing out how easy it is to add fixtures in boost.
I’m a big fan of Boost, but I have to admit that it wasn’t until about a year ago that I even learned that Boost was providing a unit testing library. Clearly, I had to check it out.
The first surprise is that Boost.Test isn’t exclusively a unit-testing framework. It also pretends to be a bunch of other things related to testing. Nothing terribly wrong with that, but to me is the first sign of a “smell.†The other surprise is that it wasn’t really based on the XUnit family. Hmmm… In that case, it had better provide some outstanding functionality.
The documentation was top notch. Some of the best I saw for any testing framework. The concepts were clearly explained, and had lots of simple examples to demonstrate different features. Interestingly, from the docs I saw that Boost.Test was designed to support some things that I would consider bad practices such as dependencies between tests, or long tests.
- Minimal amount of work needed to add new tests. Almost. Boost.Test requires really minimal work to add new tests. It’s very much like the ideal testing framework described earlier. Unfortunately, adding tests that are part of a suite requires more typing and explicit registration of each test.
- Easy to modify and port. It gets mixed marks on this one, for the same reasons as CppUnit. Being part of the Boost libraries, portability is something that they take very seriously. It worked flawlessly under Linux (better than most frameworks). But I question how easy it is to actually get inside the source code and start making modifications. It also happens to pull into quite a few supporting headers from other Boost libraries, so it’s not exactly small and self-contained.
- Supports fixtures. Boost.Test eschews the setup/teardown structure of NUnit tests in favor of plain C++ constructors/destructors. At first this threw me off for a loop. After years of being used to setup/teardown, and a fairly complex suite setup, I didn’t see the obvious ways of using fixtures with composition.Now that I’ve tried it this way I’ve come to like it almost better than setup/teardown fixtures. One of the great advantages of this approach is that you don’t need to create fixture objects dynamically, and instead you can put the whole fixture on the stack.On the downside, it’s annoying to have to refer to the variables in the fixture through the object name. It would be great if they could somehow magically appear in the same scope as the test case itself. Also, it would have been a bit cleaner if the fixture could have been setup on the stack by the BOOST_AUTO_UNIT_TEST macro instead of having to explicitely put it on the stack for every test case.
- Handles exceptions and crashes well. This is one of the aspects where Boost.Test is head and shoulders above all the competition. Not only does it handle exceptions correctly, but it prints some information about them, it catches Linux system exceptions, and it even has a command-line argument that disables exception handling, which allows you to catch the problem in your debugger on a second run. I really couldn’t ask for much more.
- Good assert functionality. Yes. Has assert statements for just about any operation you want (equality, closeness, less than, greater than, bitwise equal, etc). It even has support for checking whether exceptions were thrown. The assert statements correctly print out the contents of the variables being checked. Top marks on this one.
- Supports different outputs. Probably, but it’s not exactly trivial to change. At least the default output is IDE friendly. I suspect I would need to dig deeper into the unit_test_log_formatter, but I certainly didn’t see a variety of preset output types that I could just plug in.
- Supports suites. Yes, but with a big catch. Unless I’m missing something (which is very possible at this point–if so make sure to let me know), creating a suite requires a bunch of fairly verbose statements and also requires modifying the test runner itself in main. Have a look at the example below. Couldn’t that have been simplified to the extreme? It’s not a big deal as this is my least-wanted requirement, but I wish I could label all the test cases in one file as part of a suite with a simple macro at the beginning of the file. Another minor shortcoming is the lack of setup/teardown steps for whole suites, which could come in really handy (especially if suite creation were streamlined).
#include <boost/test/auto_unit_test.hpp> #include <boost/test/floating_point_comparison.hpp> BOOST_AUTO_UNIT_TEST (MyFirstTest) { float fnum = 2.00001f; BOOST_CHECK_CLOSE(fnum, 2.f, 1e-3); }
#include <boost/test/auto_unit_test.hpp> #include <boost/test/floating_point_comparison.hpp> #include "MyTestClass.h" struct MyFixture { MyFixture() { someValue = 2.0; str = "Hello"; } float someValue; std::string str; MyTestClass myObject; }; BOOST_AUTO_UNIT_TEST (TestCase1) { MyFixture f; BOOST_CHECK_CLOSE (f.someValue, 2.0f, 0.005f); f.someValue = 13; } BOOST_AUTO_UNIT_TEST (TestCase2) { MyFixture f; BOOST_CHECK_EQUAL (f.str, std::string("Hello")); BOOST_CHECK_CLOSE (f.someValue, 2.0f, 0.005f); // Boost deals with this OK and reports the problem //f.myObject.UseBadPointer(); // Same with this //myObject.DivideByZero(); } BOOST_AUTO_UNIT_TEST (TestCase3) { MyFixture f; BOOST_CHECK_EQUAL (1, f.myObject.s_currentInstances); BOOST_CHECK_EQUAL (3, f.myObject.s_instancesCreated); BOOST_CHECK_EQUAL (1, f.myObject.s_maxSimultaneousInstances); }
#include <boost/test/unit_test.hpp> #include <boost/test/floating_point_comparison.hpp> using boost::unit_test::test_suite; struct MyTest { void TestCase1() { float fnum = 2.00001f; BOOST_CHECK_CLOSE(fnum, 2.f, 1e-3); } void TestCase2() {} }; test_suite * GetSuite1() { test_suite * suite = BOOST_TEST_SUITE("my_test_suite"); boost::shared_ptr instance( new MyTest() ); suite->add (BOOST_CLASS_TEST_CASE( &MyTest::TestCase1, instance )); suite->add (BOOST_CLASS_TEST_CASE( &MyTest::TestCase2, instance )); return suite; }
#include <boost/test/auto_unit_test.hpp> using boost::unit_test::test_suite; extern test_suite * GetSuite1(); boost::unit_test::test_suite* init_unit_test_suite( int /* argc */, char* /* argv */ [] ) { test_suite * test = BOOST_TEST_SUITE("Master test suite"); test->add( boost::unit_test::ut_detail::auto_unit_test_suite() ); test->add(GetSuite1()); return test; }
Boost.Test is a library with a huge amount of potential. It has great support for exception handling and advanced assert statements. It also has other fairly unique functionality such as support for checking for infinite loops, and different levels of logging. On the other hand, it’s very verbose to add new tests that are part of a suite, and it might be a bit heavy weight for game console environments.
CppUnitLite has a funny story behind it. Michael Feathers, the original author of CppUnit, got fed up with the complexity of CppUnit and how it didn’t fit everyone’s needs, so we wrote the ultra-light weight framework CppUnitLite. It is as light on the features as it is on complexity and size, but his philosophy was to let people customize it to deal with whatever they need.
Indeed, CppUnitLite is only a handful of files and it probably adds up to about 200 lines of very clear, easy to understand and modify code. To be fair, in this comparison I actually used a version of CppUnitLite I modified a couple of years ago (download it along with all the sample code) to add some features I needed (fixtures, exception handling, different outputs). I figured it was definitely in the spirit that CppUnitLite was intended, and if nothing else, it can show what can be accomplished by just a few minutes of work with the source code.
On the other hand, CppUnitLite doesn’t have any documentation to speak of. Heck, it doesn’t even have a web site of its own, which I’m sure is not helping the adoption rate by other developers.
-
Minimal amount of work needed to add new tests. Absolutely! Of all the unit-test frameworks, this is the one that comes closest to the ideal. On the other hand, it could be the fact that I’ve used CppUnitLite the most and I’m biased. In any way, it really fits my idea of minimum amount of work required to set up a simple test or even one with a fixture (although that could be made even better).
-
Easy to modify and port. Definitely. Again, it gets best of the class award in this category. No other unit-test framework comes close to being this simple, easy to modify and port, and at the same time having reasonably well separated functionality. The original version of CppUnitLite even had a special lightweight string class to avoid dependencies on STL. In my modified version I changed it to use std::string since that’s what I use in most of my projects, but the change took under one minute to do. Also, using it under Linux was absolutely trivial, even though I had only used it under Windows before.
-
Supports fixtures. This is where the original CppUnitLite starts running into trouble. It’s so lightweight that it doesn’t have room for many features. This was an absolute must for me, so I went ahead and added it. I’m sure it could be improved to make it so adding a fixture requires even less typing, but it’s functional as it stands. Unfortunately, it suffers from the problem that objects need to be created dynamically if we want them to be created right before each test. To be fair though, every single unit-test framework in this evaluation has that requirement. Oh well.
-
Handles exceptions and crashes well. The original CppUnitLite didn’t handle them at all. I added minor support for this (just an optional try/catch). To run the tests without exception support it requires recompiling the tests with a special define turned on, so it’s not at slick as the command-line argument that Boost.Test features.
-
Good assert functionality. Here is where CppUnitLite really shows its age. The assert macros are definitely the worst of the lot. They don’t use a stream to print out the contents of their variables, so we need custom macros for each object type you want to use. It comes with support for doubles, longs, and strings, but anything else you need to add by hand. Also, it doesn’t have any checks for anything other than equality (or closeness in the case of floating-point numbers).
-
Supports different outputs. Again, the original only had one type of output. But it was very well isolated and it was trivial to add more.
-
Supports suites. Probably the only framework that doesn’t support suites. I never really needed them, but they would probably be very easy to add on a per-file basis.
#include "lib/TestHarness.h" TEST (Whatever,MyTest) { float fnum = 2.00001f; CHECK_DOUBLES_EQUAL (fnum, 2.0f); }
#include "lib/TestHarness.h" #include "MyTestClass.h" class MyFixtureSetup : public TestSetup { public: void setup() { someValue = 2.0; str = "Hello"; } void teardown() {} protected: float someValue; std::string str; MyTestClass myObject; }; TESTWITHSETUP (MyFixture,Test1) { CHECK_DOUBLES_EQUAL (someValue, 2.0f); someValue = 0; // CppUnitLite doesn't handle system exceptions very well either //myObject.UseBadPointer(); // A regular exception works nicely though myObject.ThrowException(); } TESTWITHSETUP (MyFixture,Test2) { CHECK_DOUBLES_EQUAL (someValue, 2.0f); CHECK_STRINGS_EQUAL (str, std::string("Hello")); } TESTWITHSETUP (MyFixture,Test3) { // Unfortunately, it looks like the framework creates 3 instances of MyTestClass // right at the beginning instead of doing it on demand for each test. We would // have to do it dynamically in the setup/teardown steps ourselves. CHECK_LONGS_EQUAL (1, myObject.s_currentInstances); CHECK_LONGS_EQUAL (3, myObject.s_instancesCreated); CHECK_LONGS_EQUAL (1, myObject.s_maxSimultaneousInstances); }
CppUnitLite is as barebones as it gets, but with a few modifications it hits the mark in all the important categories. If it had better support for assert statements, it would come very close to my ideal framework. Still, it’s a worthy candidate for the final crown.
I had never heard of NanoCppUnit until Phlip brought it up. From reading the feature list, it really appeared to be everything that I wanted CppUnitLite to be, except that it was better and ready to work out of the box.
The first point against NanoCppUnit is the awful “packaging†of the framework. If you thought that CppUnitLite was bad (not having a web page of its own), well, at least you could download it as a zip file. For NanoCppUnit you actually have to copy and paste the five files that make up the framework from a web page. I’m not kidding. That makes for some “lovely†formatting issues I might add. The documentation found in the web page wasn’t exactly very useful either.
In any case, I continued my quest to get a simple test program up and running with NanoCppUnit. Out of the box (or out of the web page, rather) it’s clearly aimed only at Windows platforms. I thought it would be trivial to fix, but changing it required more time than I thought at first (I personally gave up when I started getting errors buried three macros deep into some assert statement). Unlike CppUnitLite, the source code is not very well structured at all, full of ugly macros everywhere, making it not trivial to add new features like new output types. Unless I’m totally mistaken, it even looks like it has sample code inside the test framework itself. Eventually I had to give up on running it under Linux, so my comments here are just best guesses by looking at the source code.
- Minimal amount of work needed to add new tests. I think so. I’m not sure it’s possible to create a standalone test that is part of a global suite, but at least creating a suite doesn’t require manual registration of every test. This is (probably) the simplest possible test with NanoCppUnit.
- Easy to modify and port. Not really. Windows dependencies run deeper than it seems on the surface. The code is small, but it’s messy enough that it’s a pain to work with. I’m sure it can be ported with a bit of effort though since it’s so small.
- Supports fixtures. Yes. Setup and teardown calls very similar to the modified version of CppUnitLite.
- Handles exceptions and crashes well. No idea since I wasn’t able to run it. I see some try/catch statements in the code, but no way to turn them on or off. Probably no better than CppUnitLite.
- Supports different outputs. Not really. Everything is hardwired to use a stream that sends its contents to OutputDebugString() in Windows. I think the default output text is formatted to match the Visual Studio error format.
- Good assert functionality. Yes. Good range of assert statements, including floating point closeness, greater than, less than, etc.
- Supports suites. Yes. I don’t know what’s involved in just running a single suite though. Not a big deal either way.
struct MySuite: TestCase { }; TEST_(MySuite, MyTest) { float fnum = 2.00001f; CPPUNIT_ASSERT_DOUBLES_EQUAL(fnum, 2.0f, 0.0001); }
One of NanoCppUnit’s unique features is regular expression support as part of its assert tests. That’s very unusual, but I can see how it could come in handy. A few times in the past, I’ve had to check that a certain line of code has some particular format, so I had to sscanf it, and then check on some of the contents. A regular expression check would have done the job nicely.
Unfortunately, NanoCppUnit doesn’t really live up to the standards of other frameworks. Right now it feels too much as a work in progress, with too much missing functionality and not clearly structured code.
The further along we get in this evaluation, the less Xunit-like the frameworks become. Unit++’s unique feature is that it pretends to be more C++-like than CppUnit. Wait a second, did I hear that right? More C++ like? Is that supposed to be a good thing? Looking back at my ideal test framework, it really isn’t very much like C++ at all. Once I started thinking about that topic I realized that there really is no reason at all why the tests framework itself needs to be in C++. The tests you write need to be in the language of the code being tested, but all the wrapper code doesn’t. That’s a point that the next, and final, testing framework will drive home.
So, what does it mean to be more C++ like? No macros for a start. You create suites of tests by creating classes that derive from suite. That’s the same thing we were doing in most other frameworks, really, but it was just happening behind the scenes. It really doesn’t help me any to know that that is what I’m doing, and I would certainly not call it a “feature.†As a result, tests are more verbose than they could be.
The documentation is simply middle-of-the-road. It’s there, but it’s not particularly detailed and it doesn’t come loaded with examples.
- Minimal amount of work needed to add new tests. I’m afraid it gets failing marks for this. It requires manual registration of tests, and every test needs to be part of a suite. This makes adding new tests tedious and error prone (by writing a new test and forgetting to register it). I don’t know about you, but with all the C++ cruft, I look at the code below and it’s not immediately obvious what it does until I’ve scanned it a couple of times. The signal to noise ratio is pretty poor.
- Easy to modify and port. So-so. It needs the STL and it pulls in some stuff like iostreams (which I remember having distinct problems with when I was working with STLPort). On the other hand the source code is relatively small and self-contained so it’s certainly doable to port and modify if you’re willing to put in some time.
- Supports fixtures. Another framework that I just can’t see how to do fixtures with. Like Boost.Test, it seems to think that using the constructor and destructor for each class is all you need. A quick search for fixture or setup or teardown in the documentation doesn’t reveal anything. I don’t know if I’m totally missing something or if other people just write very different tests from me. I suppose I could create a new class for every fixture I want, put the setup in the constructor and the teardown in the destructor and inherit from it for every test case (and somehow figure out how to create an instance of that class and use it for each test run). It’s probably possible, but it’s not exactly trivial, is it? Again, the lack of fixtures puts this framework out of the running.
- Handles exceptions and crashes well. Average. It manages to catch regular exceptions without crashing, but that’s about it. No system exceptions in Linux. No way to turn it off for debugging.
- Supports different outputs. I couldn’t figure out how to do it from the documentation. There is probably a way to do it since it even supports GUI functionality , but it’s not obvious (and there are no examples). Besides, by this point, having failed points 1 and 3, I wasn’t really motivated to spend a while learning the framework. Incidentally, this is one of the few frameworks whose default text output is not formatted correctly for IDEs like KDevelop.
- Good assert functionality. It scrapes by with the minimum in this department. It provides equality and condition checks, but that’s it. It doesn’t even provide a float version of assert to check for “close enough.†At least it prints the contents of the variables to a stream correctly.
- Supports suites. Yes, like most of them.
#define __UNITPP #include "unit++.h" using namespace unitpp; namespace { class Test : public suite { void test1() { float fnum = 2.00001f; assert_eq("Checking floats", fnum, 2.0f); } public: Test() : suite("MySuite") { add("test1", testcase(this, "Simplest test", &Test::test1)); suite::main().add("demo", this); } }; Test* theTest = new Test(); }
Overall, Unit++ is not really a candidate. Perhaps it’s because it’s not intended for the type of testing I intend to use it for, but it doesn’t offer anything new over other frameworks and it has a lot of drawbacks of its own. The lack of fixtures is simply unforgivable.
After looking into a framework that tried to be different from XUnit (Unit++), I wasn’t particularly looking forward to evaluating possibly the wackiest one of them all, CxxTest. I had never heard of it until a few days ago, but I knew that it required using Perl along the way to generate some C++ code. My spider senses were tingling.
Boy was I wrong!! Within minutes of using CxxTest and reading through its great documentation (the best by far), I was completely convinced this was the way to go. This came as a complete surprise to me since I was ready to leave somewhat dissatisfied and pronounce a victor between CppUnit and CppUnitLite.
Let’s start from the beginning. What’s with the use of Perl and why is it different from CppUnit? Erez Volk, the author of CxxTest, had the unique insight that just because we’re testing a C++ program, we don’t need to rely on C++ for everything. Other languages, such as Java, are better suited to what we want to do in a unit-testing framework because they have good introspection (reflection) capabilities. C++ is quite lacking in that category, so we’re forced to use kludges like manual registration of tests, ugly macros, etc. CxxTest gets around that by parsing our simple tests and generating a C++ test runner that calls directly into our tests. The result is simply brilliant. We get all the flexibility we need without the need for any ugly macros, exotic libraries, or fancy language features. As a matter of fact, CxxTest’s requirements are as plain vanilla as you can get (other than being able to run Perl).
The code-generation step is also trivial to integrate into the regular build system. The wonderful documentation gives explicit step-by-step instructions on how to integrate it with make files, Visual Studio projects files, or Cons. Once you have it set up, you won’t even remember there’s anything out of the ordinary going on.
Let’s see how it stacks up against the competition.
- Minimal amount of work needed to add new tests. Very good. It’s almost as simple as the best of them. If I could nit-pick, I would have wished for an even simpler way to create tests without the need to declare the class explicitly. Since we’re doing processing with a Perl script, there’s no reason we couldn’t have taken it a step beyond that and used a syntax even closer to my ideal test framework.
- Easy to modify and port. CxxUnit requires the simplest set of language features (no RTTI, no exception handling, no template functions, etc). It also doesn’t require any external libraries. It is also distributed simply as a set of header files, so there’s no need to compile into a separate library or anything like that. Functionality is pretty well broken down and separated in the original source code, so making modifications should be fairly straightforward.
- Supports fixtures. CxxUnit gets the “top of its class†label in this category. Not only does it support setup/teardown steps on a per-test level, but it also supports them at the suite and at the world (global) level. Creating fixtures is pretty straightforward and just requires inheriting from a class and creating as many functions as you want starting with the letters “test.†To be really picky, I would have loved it if they had taken it a step further and, apart from simplifying the code a bit more, also inserted the setup and teardown code around the code for each test. That would have allowed us to work with those objects directly on the stack and their lifetime would have been managed correctly around each test. Oh well. Can’t have everything.
- Handles exceptions and crashes well. Great support. It catches all exceptions and prints information about them formatted like any other error (no system exceptions under Linux though). You can easily re-run the tests with a command-line argument to the Perl script to avoid catching exceptions and catch them in the debugger instead. It also gives you a custom version of every assert macro that lets you catch the exceptions yourself in case you ever need to do that.
- Supports different outputs. Different outputs are supported by passing a parameter indicating which type of output you want to the Perl processing step. The default one (error-printer) was formatted correctly for IDE parsing, and you can use several others (including GUIs for those of you addicted to progress bars, a yes/no report, or a stdio one). Adding new output formatting sounds very straightforward and it’s even covered in the documentation.
- Good assert functionality. Again, it gets “top of its class†for this one. It has a whole suite of very comprehensive assert functions, including ones for exception handling, checking predicates, and arbitrary relations. It even has a way to print out warnings which can be used to differentiate between two parts of the code calling the same test, or to print reminder “TODO†messages to yourself.
- Supports suites. Yes. All tests are part of a suite.
class SimplestTestSuite : public CxxTest::TestSuite { public: void testMyTest() { float fnum = 2.00001f; TS_ASSERT_DELTA (fnum, 2.0f, 0.0001f); } };
#include "MyTestClass.h" class FixtureSuite : public CxxTest::TestSuite { public: void setUp() { someValue = 2.0; str = "Hello"; } void tearDown() {} void test1() { TS_ASSERT_DELTA (someValue, 2.0f, 0.0001f); someValue = 13.0f; // A regular exception works nicely though myObject.ThrowException(); } void test2() { TS_ASSERT_DELTA (someValue, 2.0f, 0.0001f); TS_ASSERT_EQUALS (str, std::string("Hello")); } void test3() { //myObject.UseBadPointer(); TS_ASSERT_EQUALS (1, myObject.s_currentInstances); TS_ASSERT_EQUALS (3, myObject.s_instancesCreated); TS_ASSERT_EQUALS (1, myObject.s_maxSimultaneousInstances); } float someValue; std::string str; MyTestClass myObject; };
Another feature supported by CxxUnit that I haven’t had time to look into is some support for mock objects. Anybody doing TDD knows the value of mock objects when it comes to testing the interactions between a set of objects. Apparently CxxUnit allows you to override global functions with specific mock functions (it gives an example of overriding fopen()). I don’t think it helps any with regular classes; for those you’re on your own.
So, what’s not to like in CxxTest? Not much, really. Other than wishing that the test syntax were a bit tighter, the only thing to watch out for is what happens with large projects. If you follow the examples in the documentation, it will create a single runner for all the tests you give it. This can be problematic if you’re going to be having thousands of tests, and then making one small change in one of them causes a full recompilation of all your code.
Update: After talking with Erez and re-checking the documentation, I realized this is already fully supported in CxxUnit. By default, when you generate a test runner, it adds a main function and some global variables, so linking with other similar runners gives all sorts of problems. However, it turns out you can generate a test runner with the –part argument, and it will leave out the main function and any other globals. You can then link together all the runners and have a single executable. I wonder if it would be worth going as far as creating a runner for every suite, or if it would be best to cluster suites together. Worth investigating at some point whenever I get enough tests to make a difference.
Conclusion
After going through all six C++ unit-testing frameworks, four stand out as reasonable candidates: CppUnit, Boost.Test, a modified CppUnitLite, and CxxTest.
Of the four, CxxTest is my new personal favorite. It fits very closely the requirements of my ideal framework by leveraging the power of an external scripting language. It’s very usable straight out of the “box†and it provides some nifty advanced features and great assert functionality. It does require the use of a scripting language as part of the build process, so those unconfortable with that requirement, might want to look at one of the other three frameworks.
CppUnit is a solid, complete framework. It has come a long, long way in the last few years. The major drawbacks are the relative verbosity for adding new tests and fixtures, as well as the reliance on STL and some advanced language issues.
If what you need is absolute simplicity, you can do no wrong starting with CppUnitLite (or a modified version), and tweaking it to fit your needs. It’s a well-structured, ultra-light framework with no external dependencies, so modifying it is extremely easy. Its main drawback is the lack of features and the primitive assert functionality.
If you’re going to be working mostly on the PC, you don’t expect to have to modify the framework itself, and you don’t mind pulling in some additional Boost libraries, Boost.Test could be an excellent choice.
Should you roll your own unit-test framework? I know that Kent Beck recommends it in his book Test-Driven Development: By Example, and it might be a great learning experience, but I just can’t recommend it. Just as it’s probably good to write a linked-list and a stack data structure a few times but I wouldn’t recommend actually doing that in production code instead of using the ones provided in the STL, I strongly recommend starting with one of the three unit-testing frameworks mentioned above. If you really feel the need to roll your own, grab CppUnitLite and get hacking.
Whichever one you choose, you can really do no wrong with one of those three frameworks. The most important thing is that you are writing unit tests, or, even better, doing test-driven development. To paraphrase Michael Feathers, code without unit tests is legacy code, and you don’t want to be writing legacy code, do you?
So far I haven’t used the comments feature of this blog, but I figured this might be a good time to give it a try. I suspect people are going to be quick to point out how you can do things faster/better/cleaner with their favorite framework, or even bring up some framework I completely overlooked that is close to my ideal. Let’s give it a try and hope the spammers stay away 🙂
Hey Noel!
It’s great to see you’ve enabled comments for your site!
I’ve never actually tried using TDD but, where I can, I have been using unit testing for a few years now and it has really improved my productivity and the quality of my code. With regard to minimizing the amount of typing required to implement tests, something that really interests me is mock objects:
http://www.martinfowler.com/articles/mocksArentStubs.html
I have yet to see a C++ unit testing framework that supports mocking. I think it is because mocking requires a language that supports reflection and/or runtime code generation. Do you consider mocking to be a useful technique for game development?
Al
Hi Al,
Yes, mock objects are really invaluable. The only framework that has some support in that area is CxxUnit, but (I think) it only applies to global functions.
I’ll definitely cover mock objects in a later article (since I’ll be writing a lot more about TDD and related techniques), but they’re really essential to test object interactions.
Up until now, I’ve created really simple mock objects by hand: derive from the same interface as the object you want, and override the functions you care about to keep interesting information. That’s pretty easy to do as long as the classes you’re mocking are simple and the interaction not too complex.
Java seems to have a bunch of tools to help with the mock process. There is also at least one for C++ that I”m aware of: http://mockpp.sourceforge.net/index-en.html I’m sure I’ll be talking about it in the near future 🙂
–Noel
Mockpp is really interesting. It achieves the expectation specification features of mock object frameworks like http://www.easymock.org/ and http://dotnetmock.sourceforge.net/tikiwiki/tiki-index.php through an ingenious use of C++ macros. This is really useful because it means you can reuse the same mock object class for several different tests, just by specifying different expectations for each test.
The code you have to write to set up a mock object is really verbose though. I would only use this framework to define mock objects that could be reused in a lot of different tests. Otherwise it would be easier just to implement simple mock objects by hand.
CxxUnit mentions mock objects in its documentation but has no support for specifying expectations. Rather it allows global functions to be overriden with mock objects for testing purposes.
There must be a clean way of specifying expectations for mock objects in C++. I might see if I can improve on mockpp.
Comparing C unit testing frameworks
BinaryShift.com’s Guide to Developer Testing
Reviews of C++ Unit Testing Frameworks
Noel Llopis has a survey of CppUnit, Boost.Test, CppUnitLight, NanoCppUnit, Unit++, CxxTest. He has some examples and links to each of those frameworks.
C++ Unit-Testing Frameworks
CxxTest comes out top
CxxTest came out top in this comparison of C++ unit testing tools.
The choice of test framework depends on what you want out of it. I personally prefer and use Boost.Test, especially in conjunction with their build system (bjam + Boost.Build v2), as this integrates with the test framework allowing you to automate regression tests.
> The choice of test framework depends on what you want out of it.
Yes, of course. Still, I’m curious about how you use Boost. You don’t feel the need for setup/teardown calls for your tests? Are you using TDD, or are you writing a small amount of unit tests after the program is written? Just curious.
I don’t understand why the author believes that setup()/teardown() is better than ctor()/dtor(). It seems to me that teardown() is a hack that Java needs because one cannot force invocation of finalize().
I’ve written a less-than-120-line perl script that generates CppUnit registration code from xml generated by swig from a header file like:
// AoeuTest.H
class AoeuTest:
public CppUnit::TestFixture
{
public: void testAoeu();
};
Currently, we implement the tests in the corresponding .C file, but the script also works for a header file with the contents like:
// AoeuTest.H
class AoeuTest:
public CppUnit::TestFixture
{
public: void testAoeu()
{
CPPUNIT_ASSERT( true );
}
};
which is extremely close to what Noel wants.
Apologies, but without jumping through hoops, I can’t contribute the script. The bright side is that it wasn’t difficult to create (if you’re handy with XML/XPath).
> I don’t understand why the author believes that setup()/teardown() is
> better than ctor()/dtor(). It seems to me that teardown() is a hack that
> Java needs because one cannot force invocation of finalize().
Using constructor/destructor calls for fixture setup would be perfectly fine as long as they are called immediately before and after each test call. Maybe I don’t know how to use Boost.Test or Unit++ properly, but I wasn’t able to figure out how to do it in a reasonable way. If you look at my source code, this was my best attempt (look at the source code included with the article for the details):
struct MyTest
{
MyTest()
{
someValue = 2.0;
}
void TestCase1()
{
BOOST_CHECK_CLOSE (someValue, 2.0f, 0.005f);
someValue = 13;
}
void TestCase2()
{
BOOST_CHECK_CLOSE (someValue, 2.0f, 0.005f);
}
float someValue;
};
The problem is that one object of the type MyTest is created at initialization time, and then each of the test cases is called in turn, making the second test case fail completely because someValue is not set to the “expected” value.
What’s the cleanest/easiest way to set up fixtures with Boost.Test or Unit++? Any suggestions?
Hi,
My name is Gennadiy Rozental. I developer and maintainer for Boost.Test library. I was pointed to this arcticle for comments. Here is copy of my comments to your article in Boost mailing list:
Regards,
Gennadiy
—————————–
Hi,
I don’t have much to say about this article, other then some statements
seems questionable and many people have different priorities and
requirements for testing framework. One thing needs to be spelled out
though:
Boost.Test *does* support fixtures. At least in a sense author using this
term. Noel found proper thread in mailing list to refer, but made completely
wrong conclusion from it. To implement fixtures one needs to use constructor
and destructor *for fixture*. Here an example:
struct my_fixture {
my_fixture() { /* do initialization here */ }
~my_fixture() { /* do deinitilaization here */ }
T1 obj1;
T2 obj2;
….
};
…
TEST( test1 ) {
my_fixture fx;
// test body here
}
TEST( test2 ) {
my_fixture fx;
// test body here
}
TEST( test3 ) {
my_fixture fx;
// test body here
}
That’s all. You even got a “bonus” ( in author terms ) that it lets you
declare objects used in the fixture on the stack. It may not be convinient tod ofor every test case but this is as good/bad as an author “Ideal Framework”.
I admit I may had to explain it in docs/FAQ. But seems obvious in modern
C++. (Actually in prerelease version of the library there were additional
interfaces to support fixtures. But I dropped them later since I realized we
don’t really need them in C++).
One thing though may worth considering: add support for fixtures on
test_suite level, so that one needs to specify them once during test_suite
initialization in a form of two setup/teardown methods or fixture type. I
may add it next release.
Gennadiy
P.S. I would like to see author to apply “one assertion per test case”
policy in real life. In my experience single unit test program may contain
hundreds of them.
I’ve made some modifications to our version of CppUnitLite since you left Day 1 that you might be interested in. I’ve wrapped our calls to, for example, runAllTests so they look like:
if (IsDebuggerPresent())
{
TestRegistry::runAllTests(result);
}
else
{
try
{
TestRegistry::runAllTests(result);
}
catch (…)
{
result.addFailure (Failure (“Unhandled exception”, “Uncaught”, “”, 0));
}
}
Having the debugger check means that when you’re working on a test, you can hit compile and get an error if it fails. If it fails with an exception, then you can just hit run and the program will run and stop in the debugger where the exception is raised instead of catching the exception. It’s sped up my iteration time and (IMHO) is even slicker than having a command-line option to turn off exception catching.
Noel LlopisÀÇ C++ ´ÜÀ§ °Ë»ç Ç÷¹ÀÓ¿÷ 6þ ºñ±³
GPG ½Ã¸®Ãî¿¡µµ ±ÛÀ» ±â°ÃÇß´ø C++ for Game ProgrammersÀÇ ÀúÀÚ Noel Llopis°¡ ¸çÄ¥ Àü ¿©¼¸ °¡Ãö C++ ´ÜÀ§ °Ë»ç(unit testing) Ç÷¹ÀÓ¿÷µéÀ» ºñ±³ÇÑ ±ÛÀ» ¿Ã·È½À´Ã´Ù.
Exploring the C++ Unit Testing Framework Jungle
±Û¿¡¼ Noel˼ ¹Ù¶÷Ã÷ÇÑ UT Ç÷¹ÀÓ¿÷ÀÇ ¿…
Thanks for the clarification, Gennadiy. I guess after so many years of XUnit, I missed something as obvious as how to use fixtures as pure objects without support from the testing framework. That makes a lot of sense. I’ll update the sample code and my comments in the article.
Definitely adding fixtures at the suite level and global level like CxxUnit would be a very welcome addition.
As for the “one assertion per test case”, I have applied that in real life. In practice a few test cases end up with two assertions, but the majority of them only have one. When I started using unit tests, I would cram a lot of asserts in a single test case, but that lead to tests that were difficult to refactor. When you’re doing TDD, you need to refactor your tests as much as your production code, and simplicity is essential. I suspect it might be different whether you’re doing TDD or writing unit tests after the code. I’ll write in detail about it soon here.
That’s a great idea, Kyle (at least for platforms where you can detect if you have an attached debugger or not, which are most of them for us). You can even take it further and change what test output you select depending on whether the debugger is attached or not. So if it is attached, you can use the DebugOutputString() output, otherwise you use std::out (or a file log if you’re running it on a game console w/o debugger).
That’s an argument for easily being able to modify the test runner itself instead of generating it on every build. Hmmm…. I’m going to have to look at CxxUnit and see how easy it would be to add that.
Thnks for making this public – it’s very useful – I just need a similar review of commercial unit test tools.
> As for the “one assertion per test case”, I
> have applied that in real life. In practice a
> few test cases end up with two assertions, but
> the majority of them only have one. When I
> started using unit tests, I would cram a lot
> of asserts in a single test case, but that
> lead to tests that were difficult to refactor.
> When you’re doing TDD, you need to refactor
> your tests as much as your production code,
> and simplicity is essential.
After you mention this principle I read a couple of articles and discussions about it and it still doesn’t make too much sense to me. In general I am in favor in TDD ideas but I never like extremes in any aspects of life. Solid unit tests written pre/during/post class implementation is a cornerstone of reliable development. But what you advocate is just a matter of style IMO. I prefer test cases organization by feature/area: test constructors. test destructors, test access method, test comparison, test search, validations e.t.c. With proper test framework you will get pointed exactly to the error location whether or not you use one assertion per test case. One thing I am sure though is that one assertion per test case will definitely lead to longer run time ( with couple hundred assertions in test – 10 times and more), more time spend on tests development (just compare how many lines more you will need to type) and more time spend on fixures preparation and managing (imagine simple case where you just refactor common code into fixure. But then you found that one assertion require slightly different one, the same with another. Now you write second fixure that reuse first one e.t.c). All in all I donÂ’t see how writing several related assertions together make my unit test code less “requirement like”.
As a side note I would like to mention that in design and implementation on test framework the biggest challenge I found is not code portability (even though Boost.Test works on several dozens of configurations – essentially most that I know about – which should say something about portability) or support some specific features (like test case dependency – which BTW could be useful, believe it or not, specifically in a cases when you know that one test case will run very long and fail if another test case is failing). The biggest challenge is huge diversity of requirements to/expectations from unit test framework. And in many case they contradict each other. You yourself managed to state both requirements independency from exception handling and complete exception handling support for your ideal framework. For this reason Boost.Test is presented not as one monolithic framework, but rather as a collection of many components used in different environments/circumstances: from simplest one header minimal testing component to the full featured Unit Test Framework. I did not try to enforce any specific testing style and I am trying to support features that could be useful.
> With proper test framework you will get pointed exactly
> to the error location whether or not you use one
> assertion per test case. One thing I am sure though is
> that one assertion per test case will definitely lead to
> longer run time ( with couple hundred assertions in
> test – 10 times and more), more time spend on tests
> development
It not so much a matter of style, but of maintainability. Doing TDD, I expect to write a lot of unit tests. Usually about as many lines of code as production code, or even more. I also expect to refactor my code heavily, which means refactoring my unit tests as well.
When I used to write “large” unit tests (>50 lines containing >5 assert statements), I found it very difficult to refactor anything. The causes for a failure were not clear, and it often required me to jump in the debugger and step through the code.
Having really simple test cases (I rather use that term than “one assertion per test case”), wich means test cases <10 lines and at most 1-2 asserts, I find that I can refactor any test case trivially because it’s immediatelly obvious what they’re trying to accomplish.
Yes, it’ll be slower to run the tests, but hopefully not much slower (I usually want a unit test to take less than 1ms to execute). When the running time goes over 3-4 seconds then it’s when I have to start using suites. Also, that’s when having per-suite setup/teardown can really help with some of those performance issues.
But you hit in the nail when you said that the main problem of working in a completely generic library is trying to support everybody. I’m sure there are people who want exactly the opposite.
I’m sure you found some of these links already, but here is some reading on the whole “one assert per test approach”: http://www.artima.com/weblogs/viewpost.jsp?thread=35578 and http://www.theserverside.com/news/thread.tss?thread_id=24136.
I will definitely write about it in the near future 🙂
You could read my comments here:
http://article.gmane.org/gmane.comp.lib.boost.devel/116244
Boost.Testを含めたテストフレームワークのまとめ
Boost.Test
Check out:
http://arrizza.com/unittesters/jutasserter/jutasserter.html
for a very simple unittester
# Small:
* UtAsserter.h: 55 lines
* UtAsserter.cpp: 343 lines
* main.cpp: 2 lines
# Simple:
* Test cases are created by declaring them: TEST(x)
* main() requires only one line TestSuite::Run()
# don’t have to maintain the test suite, done automatically
# easy to add more assert types
# format works with MSVCs next_error (F4)
# works with VC6 & VC7; partially works with g++
Wow, this is a great article Noel. I’d love to see a step by step article about how you do TDD on a real project, say adding a dialog to a config screen or something. I can’t help thinking that TDD doesn’t work well for apps that are mostly GUI in nature and games might be the most GUI. I’m about to start a big new project at my day job that involves a lot of back end coding that seems suitable to TDD so I thought I’d try the same thing with my own stuff. However, since a lot of the code is GUI I’m having trouble applying it.
Thanks, Bill. An article on doing TDD with games is the next one I’ll write. I promise! It’s just that other things keep coming up 🙂
GUI applications are the most difficult kind to do with TDD (and, to a certain extent, the ones you get least amount of benefit from, especially if they don’t have much real “logic” inside).
Some resources for TDD with GUIs are the Yahoo mailing list (very little traffic) http://groups.yahoo.com/group/TestFirstUserInterfaces/, and Phlip’s book (http://www.amazon.com/exec/obidos/tg/detail/-/0321227328), which should be available in a few months.
The main problem with GUI apps is that you often end up having to twist the framework around to be able to do TDD comfortably.
Games on the other hand, are *not* mostly GUIs. There’s a layer that deals with rendering graphics or creating sound, but most of it is the game logic inside, which can easily be done in a TDD way (especially for anything below game scripts, I’m not convinced that TDD is the right way to go for really high level gameplay code which is changing all the time and nothing depends on it).
More about all this soon. Very soon 🙂
I’m guessing that if I keep the minimum amount of code in the gui and move all the logic to a separate class it will help a lot with unit testing. That said I’m only halfway through TDD and I haven’t used it in practice at all. I love seeing how other people have put these kinds of methods into practice, i.e. do they really work for projects. Do the tests really stick around throughout the project, or do they get tossed like most class diagrams or project plans once you get started.
I also find book reviews by people who actually do a lot of programming very useful. Sometimes it’s really hard to find books that don’t talk about the basics for half the book. So I’m really happy to see your reviews, keep it up 🙂
BTW, would you mind mentioning what MT plugin you are using for the currently reading section? I’m new to the MT world.
Thanks
Test Driven Development
Hi, Noel!
Great review. I did spent some time comparing unit test frameworks and asking people for something simple.
For now i’m using TUT http://tut-framework.sourceforge.net . Its favourite feature for me – fixtures. Once you wrote struct containing data setup(ctor) and teardown(dtor) for test it gets constructed and destructed for each test in a group. And fixture members are referenced implicitly through “this” pointer. I choosed it over a boost because i dont like libraries that need to be built in some special way only. Major drawback of TUT is in its compiler requirements – i just cant use it in the old project built by MSVC6.
Comparing C++ unit testing frameworks
If you’re shopping around for a C++ unit test framework, make sure to check out this analysis of C++ unit test frameworks. It’s a good idea to pay attention to the conclusions made by the author of such a…
2005-05-16 links
2005-05-16 links
* Games from Within: Exploring the C++ Unit Testing Framework Jungle
* SCons: A software construction tool
Taken from JimÂ’s del.icio.us links
Feet Up!
Test driven development of distributed systems
Years of work on various large-scale software systems made me realize a few simple things: Software testing is hard. Great advances in Test Driven Development made testing more productive. Still, to achieve full test coverage for a system,…
Exploring the C++ Unit Testing Framework Jungle
Exploring the C++ Unit Testing Framework Jungle over on Games from Within is a really good look at all of the C++ unit testing frameworks out there. It compares the following frameworks: * CppUnit * Boost.Test * CppUnitLite * NanoCppUnit…
what about google test? is it better than Boost UTF?
Concerning Boost: My tests with boost.unittest look very different. Therefore I wondered why you “complained” that you have to “access the fixtures via the objects name”. All _my_ TestCases are proper classes with methods like “testXyz()”. All test-methods are inserted (manually) inside a Test-Suite function. Thus, my fixtures can be in the constructor of the TestCase-Class More like shown here http://www.boost.org/doc/libs/1_35_0/libs/test/doc/components/utf/components/test_case/class_tc.html. Yes, adding one new test requires changes in 2-3 places (test-case-class implementation, test-suite add test-function and probably also test-case-class definition). I do not use any AUTO-stuff — when I read it, I felt the explicit approach was more appealing. Maybe its “slower” then using AUTO. I have not tried it, so I can not really compare. But I a quite happy with the TestCase-Class approach, and there is not much “typing if fixture objects names”, as I can see it.
Just to update this excellent article: CxxTest now requires Python, it no longer uses Perl.
Very indepth! One thing to note, obviously the “++” was dropped of the URL because they are not supported, and so your left with “exploring-the-c-unit-testing-framework-jungle”, which suggests it’s about C, not C++. You can use cpp or cplusplus to fix that, but I don’t know if it’s worth changing now.
Boost test suite definition is now much
simpler (I tried it with v 1.54.0). To define a test suite just use these macro
BOOST_AUTO_TEST_SUITE(
SPI_Utilities_Tests )
(define
your test here)
BOOST_AUTO_TEST_SUITE_END()
It
is now simple to the extreme! 😉
As
a side note, I also find interesting that you can both use the automated
macro AND the manual definition (the hard way) at the same time in the same
code.
On my previous post, the “SPI_Utilities_Tests” in the BOOST macro was the name I gave to the test suite. In case someone was wondering what it was
I’d like to see an Evaluation of Google Test as well! It looks quite close to your ideal framework, including with fixtures!
https://github.com/google/googletest/blob/master/googletest/docs/Primer.md