c++ TDD quest
Moderator: Coders of Rage
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
c++ TDD quest
Hi. After a short programming-absence on the last semester (academic purposes), I started working on a company which applies TDD on its development process (ironically, the same university). They use the python language, so I decided to look for an alternative for c++, to use on my own projects. So I thought: "Hey, I could post my research on the forum, so people can use and learn from it too.". Then I thought there probably was already some posts about it. But I was wrong. If you're curious, search for "unit test": you'll find two unanswered posts. But I decided to do it anyway.
I'm not going to explain what TDD is and how it works. First, there's google. And I think the basic idea can be learned from the examples. I'm going to do my research and post it here, from time to time, so you can check it out, maybe learn something new and maybe help me with something or add new material. I'll leave this introduction on a post of its own.
Index:
I'm not going to explain what TDD is and how it works. First, there's google. And I think the basic idea can be learned from the examples. I'm going to do my research and post it here, from time to time, so you can check it out, maybe learn something new and maybe help me with something or add new material. I'll leave this introduction on a post of its own.
Index:
Last edited by bbguimaraes on Fri Jul 08, 2016 5:31 am, edited 4 times in total.
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part I: Finding alternatives
Here are the options I found after a short google search:
- CppUnit: c++ port of JUnit, one of the first testing frameworks. This will probably be the last one tested, because I have preconcieved (bad) ideas about anything slightly related to Java.
- boost::test: part of the famous boost library. If it's as good as the other boost libraries, I already have a favorite.
- QtTest: part of the famouse Qt framework. If it's as good as the other Qt libraries, I already have a second favorite.
- googletest: this one surprised me. Google has its own c++ test framework!
- dandymcgee
- ES Beta Backer
- Posts: 4709
- Joined: Tue Apr 29, 2008 3:24 pm
- Current Project: https://github.com/dbechrd/RicoTech
- Favorite Gaming Platforms: NES, Sega Genesis, PS2, PC
- Programming Language of Choice: C
- Location: San Francisco
- Contact:
Re: c++ TDD quest
That's fine, but you could have at least said what the acronym stands for..bbguimaraes wrote:I'm not going to explain what TDD is and how it works.
http://en.wikipedia.org/wiki/Test-driven_development
Falco Girgis wrote:It is imperative that I can broadcast my narcissistic commit strings to the Twitter! Tweet Tweet, bitches!
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part II: Hello world
Following the short introduction, here is the basic setup. I've covered installation on my Ubuntu test machine. I'm using apt to install the packages, so much of the complexity of setting up paths is hidden. For all the libraries though, installation basically meant putting headers and libraries on /usr/include and /usr/lib.
If you're wondering where is the CppUnit section: I'm sorry. I just couldn't find a simple program to run the tests. They all require a huge amount of code to setup the simplest of tests. I have a working example, that I'll post on the next sections.
Update Aug 09, 2012: I decided to include cxxtest too. I really liked the idea of parsing the headers to create the test runners. Let's see how it scales to more complex tests.
I've also decided to give CppUnit another chance. No, I'm kidding. I just looked at the code for the "minimal problem" and thought there were only two possibilities: (a) I clearly missed something, so someone can correct me (b) the code is so ridiculously long they deserve to be mocked.
This sections show:
If you're wondering where is the CppUnit section: I'm sorry. I just couldn't find a simple program to run the tests. They all require a huge amount of code to setup the simplest of tests. I have a working example, that I'll post on the next sections.
Update Aug 09, 2012: I decided to include cxxtest too. I really liked the idea of parsing the headers to create the test runners. Let's see how it scales to more complex tests.
I've also decided to give CppUnit another chance. No, I'm kidding. I just looked at the code for the "minimal problem" and thought there were only two possibilities: (a) I clearly missed something, so someone can correct me (b) the code is so ridiculously long they deserve to be mocked.
This sections show:
- installation of the library and compilation of a simple program
- a simple program and its output
Last edited by bbguimaraes on Thu Aug 09, 2012 2:42 pm, edited 3 times in total.
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part II: Hello world
boost::test
Install libraries
Compilation
Minimal program
Output
Install libraries
Code: Select all
sudo apt-get install libboost-test-dev
Code: Select all
g++ main.cpp -o boost-test
Code: Select all
#define BOOST_TEST_MAIN
#include <boost/test/included/unit_test.hpp>
BOOST_AUTO_TEST_CASE(test) {}
Code: Select all
Running 1 test case...
*** No errors detected
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part II: Hello world
QtTest
Install libraries
Compilation
Minimal program
Output
Install libraries
Code: Select all
sudo apt-get install libqt4-test
Code: Select all
qmake -project "CONFIG += qtestlib"
qmake
make
Code: Select all
#include <QtTest/QtTest>
class Test : public QObject {};
QTEST_MAIN(Test)
#include <main.moc>
Code: Select all
********* Start testing of QObject *********
Config: Using QTest library 4.8.1, Qt 4.8.1
PASS : QObject::initTestCase()
PASS : QObject::cleanupTestCase()
Totals: 2 passed, 0 failed, 0 skipped
********* Finished testing of QObject *********
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part II: Hello world
googletest
Installation
Compilation
Minimal program
Output
Installation
Code: Select all
sudo apt-get install libgtest-dev
cd /usr/src/gtest
sudo cmake CMakeLists.txt
sudo make
cd /usr/lib
sudo ln -s /usr/src/gtest/libgtest.a libgtest.a
sudo ln -s /usr/src/gtest/libgtest_main.a libgtest_main.a
Code: Select all
g++ main.cpp -lgtest_main -lgtest -pthread -o googletest
Code: Select all
#include <gtest/gtest.h>
TEST(TestCase, Test) {}
Code: Select all
Running main() from gtest_main.cc
[==========] Running 1 test from 1 test case.
[----------] Global test environment set-up.
[----------] 1 test from TestCase
[ RUN ] TestCase.Test
[ OK ] TestCase.Test (0 ms)
[----------] 1 test from TestCase (0 ms total)
[----------] Global test environment tear-down
[==========] 1 test from 1 test case ran. (0 ms total)
[ PASSED ] 1 test.
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part II: Hello World
cxxtest
Installation
Compilation
Minimal program
Output
Installation
Code: Select all
add-apt-repository ppa:dhart/ppa
sudo apt-get install cxxtest
Code: Select all
cxxtestgen --error-printer -o main.cpp Test.h
g++ main.cpp -o cxxtest
Code: Select all
include <cxxtest/TestSuite.h>
using CxxTest::TestSuite;
class Test : public CxxTest::TestSuite {
public:
void testCxxtest() {}
};
Code: Select all
Running 1 test.OK!
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part II: Hello World
CppUnit
Installation
Compilation
Minimal program
Output (no whitespaces added)
Installation
Code: Select all
sudo apt-get install libcppunit-dev
Code: Select all
g++ main.cpp -lcppunit -o cppunit
Code: Select all
#include <cppunit/extensions/HelperMacros.h>
#include <cppunit/TestFixture.h>
using CPPUNIT_NS::TestFixture;
#include <cppunit/ui/text/TestRunner.h>
using CPPUNIT_NS::TextTestRunner;
#include <cppunit/extensions/TestFactoryRegistry.h>
using CPPUNIT_NS::TestFactoryRegistry;
class Test : public TestFixture {
virtual void runTest() {}
};
int main() {
TextTestRunner runner;
TestFactoryRegistry & registry = TestFactoryRegistry::getRegistry();
runner.addTest(registry.makeTest());
runner.run();
return 0;
}
Code: Select all
OK (0 tests)
- Falco Girgis
- Elysian Shadows Team
- Posts: 10294
- Joined: Thu May 20, 2004 2:04 pm
- Current Project: Elysian Shadows
- Favorite Gaming Platforms: Dreamcast, SNES, NES
- Programming Language of Choice: C/++
- Location: Studio Vorbis, AL
- Contact:
Re: c++ TDD quest
So what are you liking about boost::test better than QT's testing framework? I have been a long-time QT loyalist, and you have gotten me curious. I have never done much looking into a testing framework.
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Re: c++ TDD quest
Actually, I'm still in a very early stage of comparison. What I meant is that, if boost::test and QtTest were in an equal level, I'd choose boost, for the single fact that it's more modularized than Qt (smaller footprint if you want to get fancy). That's inherent to the purpose of both of them. If the project already has Qt dependencies, I think I'd choose Qt. It probably has easier ways do test GUIs (at least its own). So far (I haven't tested cxxtest properly, which seems pretty promising), they seem like the two best alternatives.
Yes, I'm biased. Sue me.
Yes, I'm biased. Sue me.
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part III: First tests
Now we're going to de some simple tests, just to learn how each library structures each test and combines all of them on execution, including the output of running for many tests. The function used on the tests is the following:
That is not much useful for everyday use, but it will serve in our tests to show success and failure messages.
Index:
Code: Select all
int sum(int n1, int n2) {
return 3;
}
Index:
Last edited by bbguimaraes on Mon Oct 15, 2012 12:55 am, edited 4 times in total.
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part III: First tests
boost::test
Code
Output
Note
Since we're moving out of compiling the test runner with our executable to only linking to the pre-compiled library, we need to alter our compilation command to:
Code
Code: Select all
#define BOOST_TEST_MODULE Test
#include <boost/test/unit_test.hpp>
int sum(int n1, int n2) {
return 3;
}
BOOST_AUTO_TEST_CASE(test1plus2) {
BOOST_CHECK_EQUAL(sum(1, 2), 3);
}
BOOST_AUTO_TEST_CASE(test2plus2) {
BOOST_CHECK_EQUAL(sum(2, 2), 4);
}
Code: Select all
Running 2 test cases...
main.cpp(13): error in "test2plus2": check sum(2, 2) == 4 failed [3 != 4]
*** 1 failure detected in test suite "Test"
Since we're moving out of compiling the test runner with our executable to only linking to the pre-compiled library, we need to alter our compilation command to:
Code: Select all
g++ main.cpp -lboost_test_exec_monitor -o boost-test
Last edited by bbguimaraes on Mon Oct 15, 2012 12:42 am, edited 1 time in total.
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part III: First tests
QtTest
Code
Output
Note
I'm just going to get out of the way here to say that it took me two minutes to learn how to write this program. Once again, Qt proved to have an excelent documentation. You can also see how complete the output is, showing the expressions and values of both parameters of QCOMPARE, and the line on the file where it's being called.
Code
Code: Select all
int sum(int n1, int n2) {
return 3;
}
class Test : public QObject {
Q_OBJECT
private slots:
void test1plus2() {
QCOMPARE(sum(1, 2), 3);
}
void test2plus2() {
QCOMPARE(sum(2, 2), 4);
}
};
QTEST_MAIN(Test)
#include <main.moc>
Code: Select all
********* Start testing of Test *********
Config: Using QTest library 4.8.1, Qt 4.8.1
PASS : Test::initTestCase()
PASS : Test::test1plus2()
FAIL! : Test::test2plus2() Compared values are not the same
Actual (sum(2, 2)): 3
Expected (4): 4
Loc: [main.cpp(16)]
PASS : Test::cleanupTestCase()
Totals: 3 passed, 1 failed, 0 skipped
********* Finished testing of Test *********
I'm just going to get out of the way here to say that it took me two minutes to learn how to write this program. Once again, Qt proved to have an excelent documentation. You can also see how complete the output is, showing the expressions and values of both parameters of QCOMPARE, and the line on the file where it's being called.
- bbguimaraes
- Chaos Rift Junior
- Posts: 294
- Joined: Wed Apr 11, 2012 4:34 pm
- Programming Language of Choice: c++
- Location: Brazil
- Contact:
Part III: First tests
googletest
Code
Output
(Yet another) Note
I'm attaching a screenshot, because you can't see here that the output is colored.
Code
Code: Select all
#include <gtest/gtest.h>
int sum(int n1, int n2) {
return 3;
}
TEST(test1plus2, Test) {
ASSERT_EQ(3, sum(1, 2));
}
TEST(test2plus2, Test) {
ASSERT_EQ(4, sum(1, 2));
}
Code: Select all
Running main() from gtest_main.cc
[==========] Running 2 tests from 2 test cases.
[----------] Global test environment set-up.
[----------] 1 test from test1plus2
[ RUN ] test1plus2.Test
[ OK ] test1plus2.Test (0 ms)
[----------] 1 test from test1plus2 (0 ms total)
[----------] 1 test from test2plus2
[ RUN ] test2plus2.Test
main.cpp:12: Failure
Value of: sum(1, 2)
Actual: 3
Expected: 4
[ FAILED ] test2plus2.Test (0 ms)
[----------] 1 test from test2plus2 (0 ms total)
[----------] Global test environment tear-down
[==========] 2 tests from 2 test cases ran. (1 ms total)
[ PASSED ] 1 test.
[ FAILED ] 1 test, listed below:
[ FAILED ] test2plus2.Test
1 FAILED TEST
I'm attaching a screenshot, because you can't see here that the output is colored.
- Attachments
-
- googletest.png (111.24 KiB) Viewed 6567 times