This article provides guidance on writing unittests for Qt4 and KDE4, based on the QtTestLib framework provided starting with Qt4.1. It provides an introduction to the ideas behind unit testing, tutorial material on the QtTestLib framework, and suggestions for getting the most value for your effort.

+

This article provides guidance on writing unittests for Qt4 and KDE4, based on the [http://doc.qt.nokia.com/latest/qtestlib-manual.html QtTestLib framework] provided starting with Qt4.1. It provides an introduction to the ideas behind unit testing, tutorial material on the [http://doc.qt.nokia.com/latest/qtestlib-manual.html QtTestLib framework], and suggestions for getting the most value for your effort.

This document is matched to Qt 4.1 and KDE 4.

This document is matched to Qt 4.1 and KDE 4.

Line 22:

Line 24:

==About QtTestLib==

==About QtTestLib==

−

QtTestlib is a lightweight testing library developed by Trolltech and released under the GPL (a commercial version is also available, for those who need alternative licensing). It is written in C++, and is cross-platform. It is provided as part of the tools included in Qt 4.1, and earlier versions were provided separately. Note that earlier versions had a different API, and the examples will not work with that earlier version.

+

QtTestlib is a lightweight testing library developed by Trolltech and released under the LGPL (a commercial version is also available, for those who need alternative licensing). It is written in C++, and is cross-platform. It is provided as part of the tools included in Qt 4.1, and earlier versions were provided separately. Note that earlier versions had a different API, and the examples will not work with that earlier version.

In addition to normal unit testing capabilities, QtTestLib also offers basic GUI testing, based on sending QEvents. This allows you to test GUI widgets, but is not generally suitable for testing full applications.

In addition to normal unit testing capabilities, QtTestLib also offers basic GUI testing, based on sending QEvents. This allows you to test GUI widgets, but is not generally suitable for testing full applications.

Line 36:

Line 38:

'''Example 1. QDate test code'''

'''Example 1. QDate test code'''

−

<code cppqt n>

+

<syntaxhighlight lang="cpp-qt" line="GESHI_NORMAL_LINE_NUMBERS">

#include <QtTest>

#include <QtTest>

#include <QtCore>

#include <QtCore>

Line 68:

Line 70:

QTEST_MAIN(testDate)

QTEST_MAIN(testDate)

#include "tutorial1.moc"

#include "tutorial1.moc"

−

</code>

+

</syntaxhighlight>

+

Save as tutorial1.cpp

Stepping through the code, the first line imports the header files for the QtTest namespace. The second line imports the headers for the QtCore namespace (not strictly necessary, since QtTest also imports it, but it is robust and safe). Lines 4 to 10 give us the test class, testData. Note that testDate inherits from QObject and has the Q_OBJECT macro - QtTestLib requires specific Qt functionality that is present in QObject.

Stepping through the code, the first line imports the header files for the QtTest namespace. The second line imports the headers for the QtCore namespace (not strictly necessary, since QtTest also imports it, but it is robust and safe). Lines 4 to 10 give us the test class, testData. Note that testDate inherits from QObject and has the Q_OBJECT macro - QtTestLib requires specific Qt functionality that is present in QObject.

−

Lines 12 to 17 provide our first test, which checks that a date is valid. Note the use of the QVERIFY macro, which checks that the condition is true. So if date.isValid() returns true, then the test will pass, otherwise the test will fail. QVERIFY is similar to ASSERT in other test suites.

+

Lines 12 to 17 provide our first test, which checks that a date is valid. Note the use of the QVERIFY macro, which checks that the condition is true. So if <tt>date.isValid()</tt> returns true, then the test will pass, otherwise the test will fail. QVERIFY is similar to ASSERT in other test suites.

Similarly, lines 19 to 27 provide another test, which checks a setter, and a couple of accessor routines. In this case, we are using QCOMPARE, which checks that the conditions are equal. So if date.month() returns 3, then that part of that test will pass, otherwise the test will fail.

Similarly, lines 19 to 27 provide another test, which checks a setter, and a couple of accessor routines. In this case, we are using QCOMPARE, which checks that the conditions are equal. So if date.month() returns 3, then that part of that test will pass, otherwise the test will fail.

−

{{Warning|As soon as a QVERIFY evaluates to false or a QCOMPARE does not have two equal values, the rest of that test will be skipped over. So in the example above, if the check at line 24 fails, then the check at lines 25 and 26 will not be run.}}

+

{{Warning|As soon as a QVERIFY evaluates to false or a QCOMPARE does not have two equal values, the whole test is marked as failed and the next test will be started. So in the example above, if the check at line 24 fails, then the check at lines 25 and 26 will not be run.}}

In a later tutorial we will see how to work around problems that this behaviour can cause.

In a later tutorial we will see how to work around problems that this behaviour can cause.

Line 82:

Line 85:

Line 30 uses the QTEST_MAIN which creates an entry point routine for us, with appropriate calls to invoke the testDate unit test class.

Line 30 uses the QTEST_MAIN which creates an entry point routine for us, with appropriate calls to invoke the testDate unit test class.

−

Line 31 includes the Meta-Object compiler output, so we can make use of our QObject functionality.

+

Line 31 includes the Meta-Object compiler output, so we can make use of our QObject functionality.

The qmake project file that corresponds to that code is shown below. You would then use qmake to turn this into a Makefile and then compile it with make.

The qmake project file that corresponds to that code is shown below. You would then use qmake to turn this into a Makefile and then compile it with make.

'''Example 2. QDate unit test project'''

'''Example 2. QDate unit test project'''

−

<code>

+

<pre>

CONFIG += qtestlib

CONFIG += qtestlib

TEMPLATE = app

TEMPLATE = app

Line 96:

Line 99:

# Input

# Input

SOURCES += tutorial1.cpp

SOURCES += tutorial1.cpp

−

</code>

+

</pre>

+

Save as tutorial1.pro

−

This is a fairly normal project file, except for the addition of the <code>CONFIG += qtestlib</code>. This adds the right header and library setup to the Makefile.

+

This is a fairly normal project file, except for the addition of the <tt>CONFIG += qtestlib</tt>. This adds the right header and library setup to the Makefile.

−

This will produce an application that can then be run on the command line. The output looks like the following:

+

Create an empty file called tutorial1.h and compile with <tt>qmake; make</tt> The output looks like the following:

'''Example 3. QDate unit test output'''

'''Example 3. QDate unit test output'''

−

<code>

+

<pre>

$ ./tutorial1

$ ./tutorial1

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 113:

Line 117:

Totals: 4 passed, 0 failed, 0 skipped

Totals: 4 passed, 0 failed, 0 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

Looking at the output above, you can see that the output includes the version of the test library and Qt itself, and then the status of each test that is run. In addition to the testValidity and testMonth tests that we defined, there is also a setup routine (initTestCase) and a teardown routine (cleanupTestCase) that can be used to do additional configuration if required.

Looking at the output above, you can see that the output includes the version of the test library and Qt itself, and then the status of each test that is run. In addition to the testValidity and testMonth tests that we defined, there is also a setup routine (initTestCase) and a teardown routine (cleanupTestCase) that can be used to do additional configuration if required.

Line 122:

Line 126:

'''Example 4. QDate unit test output showing failure'''

'''Example 4. QDate unit test output showing failure'''

−

<code>

+

<pre>

$ ./tutorial1

$ ./tutorial1

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 135:

Line 139:

Totals: 3 passed, 1 failed, 0 skipped

Totals: 3 passed, 1 failed, 0 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

===Running selected tests===

===Running selected tests===

Line 143:

Line 147:

'''Example 5. QDate unit test output - selected function'''

'''Example 5. QDate unit test output - selected function'''

−

<code>

+

<pre>

$ ./tutorial1 testValidity

$ ./tutorial1 testValidity

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 152:

Line 156:

Totals: 3 passed, 0 failed, 0 skipped

Totals: 3 passed, 0 failed, 0 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

Note that the initTestCase and cleanupTestCase routines are always run, so that any necessary setup and cleanup will still be done.

Note that the initTestCase and cleanupTestCase routines are always run, so that any necessary setup and cleanup will still be done.

Line 159:

Line 163:

'''Example 6. QDate unit test output - listing functions'''

'''Example 6. QDate unit test output - listing functions'''

−

<code>

+

<pre>

$ ./tutorial1 -functions

$ ./tutorial1 -functions

testValidity()

testValidity()

testMonth()

testMonth()

−

</code>

+

</pre>

===Verbose output options===

===Verbose output options===

Line 171:

Line 175:

'''Example 7. QDate unit test output - verbose output'''

'''Example 7. QDate unit test output - verbose output'''

−

<code>

+

<pre>

$ ./tutorial1 -v1

$ ./tutorial1 -v1

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 185:

Line 189:

Totals: 4 passed, 0 failed, 0 skipped

Totals: 4 passed, 0 failed, 0 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

The -v2 option shows each QVERIFY, QCOMPARE and QTEST, as well as the message on entering each test function. I found this useful for verifying that a particular step is being run. This is shown below:

The -v2 option shows each QVERIFY, QCOMPARE and QTEST, as well as the message on entering each test function. I found this useful for verifying that a particular step is being run. This is shown below:

Line 191:

Line 195:

''''Example 8. QDate unit test output - more verbose output'''

''''Example 8. QDate unit test output - more verbose output'''

−

<code>

+

<pre>

$ ./tutorial1 -v2

$ ./tutorial1 -v2

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 211:

Line 215:

Totals: 4 passed, 0 failed, 0 skipped

Totals: 4 passed, 0 failed, 0 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

The -vs option shows each signal that is emitted. In our example, there are no signals, so -vs has no effect. Getting a list of signals is useful for debugging failing tests, especially GUI tests which we will see in the third tutorial.

The -vs option shows each signal that is emitted. In our example, there are no signals, so -vs has no effect. Getting a list of signals is useful for debugging failing tests, especially GUI tests which we will see in the third tutorial.

Line 226:

Line 230:

'''Example 9. QDate test code, data driven version'''

'''Example 9. QDate test code, data driven version'''

−

<code cppqt n>

+

<syntaxhighlight lang="cpp-qt" line="GESHI_NORMAL_LINE_NUMBERS">

#include <QtTest>

#include <QtTest>

#include <QtCore>

#include <QtCore>

Line 276:

Line 280:

QTEST_MAIN(testDate)

QTEST_MAIN(testDate)

#include "tutorial2.moc"

#include "tutorial2.moc"

−

</code>

+

</syntaxhighlight>

As you can see, we've introduced a new method - testMonth_data, and moved the specific test date out of testMonth. We've had to add some more code (which will be explained soon), but the result is a separation of the data we are testing, and the code we are using to test it.

As you can see, we've introduced a new method - testMonth_data, and moved the specific test date out of testMonth. We've had to add some more code (which will be explained soon), but the result is a separation of the data we are testing, and the code we are using to test it.

Line 287:

Line 291:

'''Example 10. Results of data driven testing, showing QFETCH'''

'''Example 10. Results of data driven testing, showing QFETCH'''

−

<code>

+

<pre>

$ ./tutorial2 -v2

$ ./tutorial2 -v2

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 315:

Line 319:

Totals: 4 passed, 0 failed, 0 skipped

Totals: 4 passed, 0 failed, 0 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

=== The QTEST macro ===

=== The QTEST macro ===

Line 323:

Line 327:

'''Example 11. QDate test code, data driven version using QTEST'''

'''Example 11. QDate test code, data driven version using QTEST'''

−

<code cppqt n>

+

<syntaxhighlight lang="cpp-qt" line="GESHI_NORMAL_LINE_NUMBERS">

void testDate::testMonth()

void testDate::testMonth()

{

{

Line 335:

Line 339:

QTEST( QDate::longMonthName(date.month()), "monthName" );

QTEST( QDate::longMonthName(date.month()), "monthName" );

}

}

−

</code>

+

</syntaxhighlight>

−

In the example above, note that monthname is enclosed in quotes, and we no longer have a QFETCH call for monthname.

+

In the example above, note that monthName is enclosed in quotes, and we no longer have a QFETCH call for monthName.

−

The other QCOMPARE could also have been converted to use QTEST, however this would be less efficient, because we already needed to use QFETCH to get month for the setYMD in the line above.

+

The other QCOMPARE could also have been converted to use QTEST, however this would be less efficient, because we already needed to use QFETCH to get month for the setYMD in the line above.

===Running selected tests with selected data===

===Running selected tests with selected data===

Line 346:

Line 350:

'''Example 12. QDate unit test output - selected function and data'''

'''Example 12. QDate unit test output - selected function and data'''

−

<code>

+

<pre>

$ ./tutorial2 -v2 testMonth:1967/3/11

$ ./tutorial2 -v2 testMonth:1967/3/11

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 362:

Line 366:

Totals: 3 passed, 0 failed, 0 skipped

Totals: 3 passed, 0 failed, 0 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

==Tutorial 3: Testing Graphical User Interfaces==

==Tutorial 3: Testing Graphical User Interfaces==

Line 378:

Line 382:

'''Example 13. QDateEdit test code'''

'''Example 13. QDateEdit test code'''

−

<code cppqt n>

+

<syntaxhighlight lang="cpp-qt" line="GESHI_NORMAL_LINE_NUMBERS">

#include <QtTest>

#include <QtTest>

#include <QtCore>

#include <QtCore>

Line 454:

Line 458:

QTEST_MAIN(testDateEdit)

QTEST_MAIN(testDateEdit)

#include "tutorial3.moc"

#include "tutorial3.moc"

−

</code>

+

</syntaxhighlight>

Much of this code is common with previous examples, so I'll focus on the new elements and the more important changes as we work through the code line-by-line.

Much of this code is common with previous examples, so I'll focus on the new elements and the more important changes as we work through the code line-by-line.

Line 486:

Line 490:

Lines 50 to 60 show a data-driven test - in this case we are checking that the validator on QDateEdit is performing as expected. This is a case where data-driven testing can really help to ensure that things are working the way they should.

Lines 50 to 60 show a data-driven test - in this case we are checking that the validator on QDateEdit is performing as expected. This is a case where data-driven testing can really help to ensure that things are working the way they should.

−

At lines 52 to 54, we fetch in an initial value, a series of key-clicks, and an expected result. These are the columns that are set up in lines 39 to 41. However note that we are now pulling in a QDate, where in previous examples we used three integers and then build the QDate from those. However QDate isn't a registered type for {{qt|QMetaType}, and so we need to register it before we can use it in our data-driven testing. This is done using the Q_DECLARE_METATYPE macro in line 4 and the qRegisterMetaType function in line 38.

+

At lines 52 to 54, we fetch in an initial value, a series of key-clicks, and an expected result. These are the columns that are set up in lines 39 to 41. However note that we are now pulling in a QDate, where in previous examples we used three integers and then build the QDate from those. However QDate isn't a registered type for {{qt|QMetaType}}, and so we need to register it before we can use it in our data-driven testing. This is done using the Q_DECLARE_METATYPE macro in line 4 and the qRegisterMetaType function in line 38.

Lines 42 to 47 add in a couple of sample rows. Lines 42 to 44 represent a case where the input is valid, and lines 45 to 47 are a case where the input is only partly valid (the day part). A real test will obviously contain far more combinations than this.

Lines 42 to 47 add in a couple of sample rows. Lines 42 to 44 represent a case where the input is valid, and lines 45 to 47 are a case where the input is only partly valid (the day part). A real test will obviously contain far more combinations than this.

Line 503:

Line 507:

'''Example 14. QDateEdit test code, using QTestEventList'''

'''Example 14. QDateEdit test code, using QTestEventList'''

−

<code cppqt n>

+

<syntaxhighlight lang="cpp-qt" line="GESHI_NORMAL_LINE_NUMBERS">

#include <QtTest>

#include <QtTest>

#include <QtCore>

#include <QtCore>

Line 593:

Line 597:

QTEST_MAIN(testDateEdit)

QTEST_MAIN(testDateEdit)

#include "tutorial3a.moc"

#include "tutorial3a.moc"

−

</code>

+

</syntaxhighlight>

This example is pretty much the same as the previous version, up to line 25. In line 26, we create a QTestEventList. We add events to the list in lines 27 and 28 - note that we don't specify the widget we are calling them on at this stage. In line 30, we simulate each event on the widget. If we had multiple widgets, we could call simulate using the same set of events.

This example is pretty much the same as the previous version, up to line 25. In line 26, we create a QTestEventList. We add events to the list in lines 27 and 28 - note that we don't specify the widget we are calling them on at this stage. In line 30, we simulate each event on the widget. If we had multiple widgets, we could call simulate using the same set of events.

Line 618:

Line 622:

'''Example 15. Unit test showing skipped tests'''

'''Example 15. Unit test showing skipped tests'''

−

<code>

+

<syntaxhighlight lang="cpp-qt" line="GESHI_NORMAL_LINE_NUMBERS">

void testDate::testSkip_data()

void testDate::testSkip_data()

{

{

Line 642:

Line 646:

QCOMPARE( val1, val2 );

QCOMPARE( val1, val2 );

}

}

−

</code>

+

</syntaxhighlight>

'''Example 16. Output of unit test showing skipped tests'''

'''Example 16. Output of unit test showing skipped tests'''

−

<code>

+

<pre>

$ ./tutorial4 testSkip -v2

$ ./tutorial4 testSkip -v2

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 665:

Line 669:

Totals: 3 passed, 0 failed, 2 skipped

Totals: 3 passed, 0 failed, 2 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

from the verbose output, you can see that the test was run on the first and third rows. The second row wasn't run because of the QSKIP call with a SkipSingle argument. Similarly, the fourth and fifth rows weren't run because the fourth row triggered a QSKIP call with a SkipAll argument.

from the verbose output, you can see that the test was run on the first and third rows. The second row wasn't run because of the QSKIP call with a SkipSingle argument. Similarly, the fourth and fifth rows weren't run because the fourth row triggered a QSKIP call with a SkipAll argument.

Line 677:

Line 681:

'''Example 17. Unit test showing expected failures'''

'''Example 17. Unit test showing expected failures'''

−

<code cppqt n>

+

<syntaxhighlight lang="cpp-qt" line="GESHI_NORMAL_LINE_NUMBERS">

void testDate::testExpectedFail()

void testDate::testExpectedFail()

{

{

Line 689:

Line 693:

QCOMPARE( 3, 3 );

QCOMPARE( 3, 3 );

}

}

−

</code>

+

</syntaxhighlight>

'''Example 18. Output of unit test showing expected failures'''

'''Example 18. Output of unit test showing expected failures'''

−

<code>

+

<pre>

$ ./tutorial4/tutorial4 testExpectedFail -v2

$ ./tutorial4/tutorial4 testExpectedFail -v2

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 719:

Line 723:

Totals: 3 passed, 0 failed, 0 skipped

Totals: 3 passed, 0 failed, 0 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

As you can see from the verbose output, we expect a failure from each time we do a QCOMPARE( 1, 2);. In the first call to QEXPECT_FAIL, we use the Continue argument, so the rest of the tests will still be run. However in the second call to QEXPECT_FAILwe use the Abort and the test bails at this point. Generally it is better to use Continue unless you have a lot of closely related tests that would each need a QEXPECT_FAIL entry.

As you can see from the verbose output, we expect a failure from each time we do a QCOMPARE( 1, 2);. In the first call to QEXPECT_FAIL, we use the Continue argument, so the rest of the tests will still be run. However in the second call to QEXPECT_FAILwe use the Abort and the test bails at this point. Generally it is better to use Continue unless you have a lot of closely related tests that would each need a QEXPECT_FAIL entry.

Line 728:

Line 732:

'''Example 19. Unit test showing unexpected pass'''

'''Example 19. Unit test showing unexpected pass'''

−

<code cppqt n>

+

<syntaxhighlight lang="cpp-qt" line="GESHI_NORMAL_LINE_NUMBERS">

void testDate::testUnexpectedPass()

void testDate::testUnexpectedPass()

{

{

Line 739:

Line 743:

QCOMPARE( 3, 3 );

QCOMPARE( 3, 3 );

}

}

−

</code>

+

</syntaxhighlight>

'''Example 20. Output of unit test showing unexpected pass'''

'''Example 20. Output of unit test showing unexpected pass'''

−

<code>

+

<pre>

$ ./tutorial4/tutorial4 testUnexpectedPass -v2

$ ./tutorial4/tutorial4 testUnexpectedPass -v2

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 764:

Line 768:

Totals: 2 passed, 2 failed, 0 skipped

Totals: 2 passed, 2 failed, 0 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

−

+

−

As you can see from the verbose output, we expect a failure from each time we do a QCOMPARE( 1, 2);. In the first call to QEXPECT_FAIL, we use the Continue argument, so the rest of the tests will still be run. However in the second call to QEXPECT_FAILwe use the Abort and the test bails at this point. Generally it is better to use Continue unless you have a lot of closely related tests that would each need a QEXPECT_FAIL entry.

+

−

+

−

Also note that tests that are marked as expected failures are not considered

+

−

to be failures, so the test function above is considered to be a pass.

+

−

+

−

If a test that is marked to be an expected failure, and it unexpectedly passes, then that is flagged as an error, as shown below:

+

−

+

−

'''Example 19. Unit test showing unexpected pass'''

+

−

+

−

<code cppqt n>

+

−

void testDate::testUnexpectedPass()

+

−

{

+

−

QEXPECT_FAIL("", "1 is not 2, even for very large 1", Continue);

+

−

QCOMPARE( 1, 1 );

+

−

QCOMPARE( 2, 2 );

+

−

+

−

QEXPECT_FAIL("", "1 is not 2, even for very small 2", Abort);

+

−

QCOMPARE( 1, 1 );

+

−

QCOMPARE( 3, 3 );

+

−

}

+

−

</code>

+

−

+

−

'''Example 20. Output of unit test showing unexpected pass'''

+

−

+

−

<code>

+

−

$ ./tutorial4/tutorial4 testUnexpectedPass -v2

+

−

********* Start testing of testDate *********

+

−

Config: Using QTest library 4.1.0, Qt 4.1.0-snapshot-20051107

+

−

INFO : testDate::initTestCase() entering

+

−

PASS : testDate::initTestCase()

+

−

INFO : testDate::testUnexpectedPass() entering

+

−

INFO : testDate::testUnexpectedPass() COMPARE()

+

−

Loc: [tutorial4.cpp(53)]

+

−

XPASS : testDate::testUnexpectedPass() COMPARE()

+

−

Loc: [tutorial4.cpp(53)]

+

−

INFO : testDate::testUnexpectedPass() COMPARE()

+

−

Loc: [tutorial4.cpp(54)]

+

−

INFO : testDate::testUnexpectedPass() COMPARE()

+

−

Loc: [tutorial4.cpp(57)]

+

−

XPASS : testDate::testUnexpectedPass() COMPARE()

+

−

Loc: [tutorial4.cpp(57)]

+

−

INFO : testDate::cleanupTestCase() entering

+

−

PASS : testDate::cleanupTestCase()

+

−

Totals: 2 passed, 2 failed, 0 skipped

+

−

********* Finished testing of testDate *********

+

−

</code>

+

The effect of unexpected passes on the running of the test is controlled by the second argument to QEXPECT_FAIL. If the argument is Continue and the test unexpectedly passes, then the rest of the test function will be run. If the argument is Abort, then the test will stop.

The effect of unexpected passes on the running of the test is controlled by the second argument to QEXPECT_FAIL. If the argument is Continue and the test unexpectedly passes, then the rest of the test function will be run. If the argument is Abort, then the test will stop.

Note that while this example produces the debug and warning messages within the test function (testQdebug), those messages would normally be propagated up from the code being tested. However the source of the messages does not make any difference to how they are handled.

Note that while this example produces the debug and warning messages within the test function (testQdebug), those messages would normally be propagated up from the code being tested. However the source of the messages does not make any difference to how they are handled.

Line 853:

Line 810:

'''Example 23. Example of using ignoreMessage'''

'''Example 23. Example of using ignoreMessage'''

−

<code cppqt n>

+

<syntaxhighlight lang="cpp-qt">

void testDate::testValidity()

void testDate::testValidity()

{

{

Line 865:

Line 822:

qWarning("validiti warning");

qWarning("validiti warning");

}

}

−

</code>

+

</syntaxhighlight>

+

−

'''Example 24. Output of ignoreMessage example'''

'''Example 24. Output of ignoreMessage example'''

−

<code>

+

<pre>

$ ./tutorial4 testValidity testValiditi

$ ./tutorial4 testValidity testValiditi

********* Start testing of testDate *********

********* Start testing of testDate *********

Line 881:

Line 838:

Totals: 3 passed, 1 failed, 0 skipped

Totals: 3 passed, 1 failed, 0 skipped

********* Finished testing of testDate *********

********* Finished testing of testDate *********

−

</code>

+

</pre>

Note that the warning message in testDate::testValidity has been "swallowed" by the the call to ignoreMessage.

Note that the warning message in testDate::testValidity has been "swallowed" by the the call to ignoreMessage.

Line 891:

Line 848:

An important part of Qt programming is the use of signals and slots. This section covers the support for testing of these features.

An important part of Qt programming is the use of signals and slots. This section covers the support for testing of these features.

−

{{Tip|'''Note:''' If you are not familiar with Qt signals and slots, you probably should review the introduction to this feature provided in your Qt documentation. It is also available at http://doc.trolltech.com/latest/signalsandslots.html. You also might like to review Tutorial 5 in your Qt documentation. This tutorial is also available at http://doc.trolltech.com/4.1/tutorial-t5.html.}}

+

{{Tip|'''Note:''' If you are not familiar with Qt signals and slots, you probably should review the introduction to this feature provided in your Qt documentation. It is also available at http://doc.trolltech.com/latest/signalsandslots.html. You also might like to review Tutorial 5 in your Qt documentation. This tutorial is also available at http://doc.trolltech.com/latest/tutorial-t5.html.}}

The first 11 lines are essentially unchanged from previous examples that we've seen. Line 15 creates the object that will be tested - as noted in the comments in lines 12-14, the only reason that I'm creating it with new is because I need to delete it in line 45 to cause the destroyed() signal to be emitted.

The first 11 lines are essentially unchanged from previous examples that we've seen. Line 15 creates the object that will be tested - as noted in the comments in lines 12-14, the only reason that I'm creating it with new is because I need to delete it in line 45 to cause the destroyed() signal to be emitted.

Line 1,049:

Line 1,006:

Tests are not built by default - you have to enable the test system, and build the tests.

Tests are not built by default - you have to enable the test system, and build the tests.

−

You enable tests by adding an "ENABLE_TESTING()" line to the top of your CMakeLists.txt file.

+

You enable tests by adding an '''ENABLE_TESTING()''' line to the top of your CMakeLists.txt file.

−

In some configurations, there may be a build system option to turn on (or off) the compilation of tests. At this stage, you have to enable the BUILD_TESTING option in KDE4 modules, however this may go away in the near future, as later version of CMake can build the test applications on demand.

+

In some configurations, there may be a build system option to turn on (or off) the compilation of tests. At this stage, you have to enable the '''BUILD_TESTING''' option in KDE4 modules, however this may go away in the near future, as later version of CMake can build the test applications on demand.

+

+

If the tests are still not building, you might want to issue make buildtests in tests directory.

=== Adding Tests ===

=== Adding Tests ===

−

You add a single test to the list of all tests that can be run by using "ADD_TEST()", which takes a number of arguments.

+

You add a single test to the list of all tests that can be run by using

+

'''ADD_TEST''', which takes a number of arguments:

+

* The first argument is a display name for the test. This is required.

+

* The second argument is the name of the executable to be run (including the path to that executable, if required). This is required.

+

* Any following arguments are passed to the executable. These are optional, and are usually omitted for QTestLib tests.

+

+

Note that '''ADD_TEST''' does nothing if '''ENABLE_TESTING()''' has not been run.

+

+

When choosing a display name, you should adopt a similar convention to the existing tests (e.g. in kdelibs/kdecore, all the display names start with kdecore-, which makes it easy to find the failing test if you run all the tests in kdelibs). If there are no existing tests, using a submodule prefix is probably a good idea.

+

+

The executable doesn't need to be something you build, but it will be for QTestLib, and many other tests. Remember that you need to specify the path, so this normally looks something like <tt>${EXECUTABLE_OUTPUT_PATH}/testapp</tt>.

+

+

=== KDE4 CMake Recipe for QTestLib ===

+

+

If you are working in a KDE4 environment, then it is pretty easy to get CMake set up to build and run a test on demand.

+

+

<pre>

+

cmake_minimum_required(VERSION 2.8)

+

+

FIND_PACKAGE ( KDE4 REQUIRED )

+

FIND_PACKAGE ( Qt4 REQUIRED QT_USE_QT* )

+

+

INCLUDE( ${QT_USE_FILE} )

+

include(KDE4Defaults)

+

+

set( kwhatevertest_SRCS kwhatevertest.cpp )

+

+

kde4_add_unit_test( kwhatevertest

+

TESTNAME ksubmodule-kwhatevertest

+

${kwhatevertest_SRCS}

+

)

+

+

target_link_libraries( kwhatevertest

+

${KDE4_KDECORE_LIBS}

+

${QT_QTTEST_LIBRARY}

+

${KDE4_KDEUI_LIBS}

+

)

+

+

</pre>

+

+

You are meant to replace "kwhatevertest" with the name of your test application. The target_link_libraries() line will need to contain whatever libraries are needed for the feature you are testing, so if it is a GUI feature, you'll likely need to use "${KDE4_KDEUI_LIBS}.

+

=== Running the Tests ===

=== Running the Tests ===

+

To run all tests, you can just "make test". This will work through each of the tests that have been added (at any lower level) using '''kde4_add_unit_test''', provided that you have '''include(KDE4Defaults)''' in your CMakeLists.txt.

+

+

This is equivalent to running the "ctest" executable with no arguments. If you want finer grained control over which tests are run or the output format, you can use additional arguments. These are explained in the ctest man page ("man ctest" on a *nix system, or run "ctest --help-full").

+

+

To run a single test, use '''./tutorial1.shell''' rather than just '''./tutorial1''', this will make it use the locally-built version of the shared libraries you're testing, rather than the installed ones.

=== Further Reading ===

=== Further Reading ===

+

+

Chapter 8 of the [http://www.kitware.com/products/cmakebook.html CMake Book] provides a detailed description of how to do testing with CMake. Also see Appendix B for more on CTest and the special commands you can use.

+

+

Various sections of the CMake Wiki, especially [http://www.cmake.org/Wiki/CMake_Testing_With_CTest CTest testing]

Abstract

This article provides guidance on writing unittests for Qt4 and KDE4, based on the QtTestLib framework provided starting with Qt4.1. It provides an introduction to the ideas behind unit testing, tutorial material on the QtTestLib framework, and suggestions for getting the most value for your effort.
This document is matched to Qt 4.1 and KDE 4.

About Unit Testing

A unit test is a test that checks the functionality, behaviour and correctness of a single software component. In Qt4 code (including KDE4 code) unit tests are almost always used to test a single C++ class (although testing a macro or C function is also possible).

Unit tests are a key part of Test Driven Development, however they are useful for all software development processes. It is not essential that all of the code is covered by unit tests (although that is obviously very desirable!). Even a single test is a useful step to improving code quality.

Note that unit tests are dynamic tests (i.e. they run, using the compiled code) rather than static analysis tests (which operate on the source or some intermediate representation).

Even if they don't call them "unit tests", most programmers have written some "throwaway" code that they use to check an implementation. If that code was cleaned up a little, and built into the development system, then it could be used over and over to check that the implementation is still OK. To make that work a little easier, we can use test frameworks.

Note that it is sometimes tempting to treat the unit test as a pure verification tool. While it is true that unit tests do help to ensure correct functionality and behaviour, they also assist with other aspects of code quality. Writing a unit test requires a slightly different approach to coding up a class, and thinking about what inputs need to be tested can help to identify logic flaws in the code (even before the tests get run). In addition, the need to make the code testable is a very useful driver to ensure that classes do not suffer from close coupling.

Anyway, enough of the conceptual stuff - lets talk about a specific tool that can reduce some of the effort and let us get on with the job.

About QtTestLib

QtTestlib is a lightweight testing library developed by Trolltech and released under the LGPL (a commercial version is also available, for those who need alternative licensing). It is written in C++, and is cross-platform. It is provided as part of the tools included in Qt 4.1, and earlier versions were provided separately. Note that earlier versions had a different API, and the examples will not work with that earlier version.

In addition to normal unit testing capabilities, QtTestLib also offers basic GUI testing, based on sending QEvents. This allows you to test GUI widgets, but is not generally suitable for testing full applications.

Each testcase is a standalone test application. Unlike CppUnit or JUnit, there is no Runner type class. Instead, each testcase is an executable which is simply run.

Tutorial 1: A simple test of a date class

In this tutorial, we will build a simple test for a class that represents a date, using QtTestLib as the test framework. To avoid too much detail on how the date class works, we'll just use the QDate. class that comes with Qt. In a normal unittest, you would more likely be testing code that you've written yourself.

Stepping through the code, the first line imports the header files for the QtTest namespace. The second line imports the headers for the QtCore namespace (not strictly necessary, since QtTest also imports it, but it is robust and safe). Lines 4 to 10 give us the test class, testData. Note that testDate inherits from QObject and has the Q_OBJECT macro - QtTestLib requires specific Qt functionality that is present in QObject.

Lines 12 to 17 provide our first test, which checks that a date is valid. Note the use of the QVERIFY macro, which checks that the condition is true. So if date.isValid() returns true, then the test will pass, otherwise the test will fail. QVERIFY is similar to ASSERT in other test suites.

Similarly, lines 19 to 27 provide another test, which checks a setter, and a couple of accessor routines. In this case, we are using QCOMPARE, which checks that the conditions are equal. So if date.month() returns 3, then that part of that test will pass, otherwise the test will fail.

Warning

As soon as a QVERIFY evaluates to false or a QCOMPARE does not have two equal values, the whole test is marked as failed and the next test will be started. So in the example above, if the check at line 24 fails, then the check at lines 25 and 26 will not be run.

In a later tutorial we will see how to work around problems that this behaviour can cause.

Line 30 uses the QTEST_MAIN which creates an entry point routine for us, with appropriate calls to invoke the testDate unit test class.

Line 31 includes the Meta-Object compiler output, so we can make use of our QObject functionality.

The qmake project file that corresponds to that code is shown below. You would then use qmake to turn this into a Makefile and then compile it with make.

Looking at the output above, you can see that the output includes the version of the test library and Qt itself, and then the status of each test that is run. In addition to the testValidity and testMonth tests that we defined, there is also a setup routine (initTestCase) and a teardown routine (cleanupTestCase) that can be used to do additional configuration if required.

Failing tests

If we had made an error in either the production code or the unit test code, then the results would show an error. An example is shown below:

Running selected tests

When the number of test functions increases, and some of the functions take a long time to run, it can be useful to only run a selected function. For example, if you only want to run the testMonth function, then you just specify that on the command line, as shown below:

Note that the initTestCase and cleanupTestCase routines are always run, so that any necessary setup and cleanup will still be done.

You can get a list of the available functions by passing the -functions option, as shown below:

Example 6. QDate unit test output - listing functions

$ ./tutorial1 -functions
testValidity()
testMonth()

Verbose output options

You can get more verbose output by using the -v1, -v2 and -vs options. -v1 produces a message on entering each test function. I found this is useful when it looks like a test is hanging. This is shown below:

The -v2 option shows each QVERIFY, QCOMPARE and QTEST, as well as the message on entering each test function. I found this useful for verifying that a particular step is being run. This is shown below:

The -vs option shows each signal that is emitted. In our example, there are no signals, so -vs has no effect. Getting a list of signals is useful for debugging failing tests, especially GUI tests which we will see in the third tutorial.

Output to a file

If you want to output the results of your testing to a file, you can use the -o filename, where you replace filename with the name of the file you want to save output to.

Tutorial 2: Data driven testing of a date class

In the previous example, we looked at how we can test a date class. If we decided that we really needed to test a lot more dates, then we'd be cutting and pasting a lot of code. If we subsequently changed the name of a function, then it has to be changed in a lot of places. As an alternative to introducing these types of maintenance problems into our tests, QtTestLib offers support for data driven testing.

The easiest way to understand data driven testing is by an example, as shown below:

As you can see, we've introduced a new method - testMonth_data, and moved the specific test date out of testMonth. We've had to add some more code (which will be explained soon), but the result is a separation of the data we are testing, and the code we are using to test it.
The names of the functions are important - you must use the _data suffix for the data setup routine, and the first part of the data setup routine must match the name of the driver routine.

It is useful to visualise the data as being a table, where the columns are the various data values required for a single run through the driver, and the rows are different runs. In our example, there are four columns (three integers, one for each part of the date; and one QString ), added in lines 19 through 22. The addColumn template obviously requires the type of variable to be added, and also requires a variable name argument. We then add as many rows as required using the newRow function, as shown in lines 23 through 26. The string argument to newRow is a label, which is handy for determining what is going on with failing tests, but doesn't have any effect on the test itself.

To use the data, we simply use QFETCH to obtain the appropriate data from each row. The arguments to QFETCH are the type of the variable to fetch, and the name of the column (which is also the local name of the variable it gets fetched into). You can then use this data in a QCOMPARE or QVERIFY check. The code is run for each row, which you can see below:

The QTEST macro

As an alternative to using QFETCH and QCOMPARE, you may be able to use the QTEST macro instead. QTEST takes two arguments, and if one is a string, it looks up that string as an argument in the current row. You can see how this can be used below, which is equivalent to the testMonth() code in the previous example.

In the example above, note that monthName is enclosed in quotes, and we no longer have a QFETCH call for monthName.

The other QCOMPARE could also have been converted to use QTEST, however this would be less efficient, because we already needed to use QFETCH to get month for the setYMD in the line above.

Running selected tests with selected data

In the previous tutorial, we saw how to run a specific test by specifying the name of the test as a command line argument. In data driven testing, you can select which data you want the test run with, by adding a colon and the label for the data row. For example, if we just want to run the testMonth test for the first row, we would use

Tutorial 3: Testing Graphical User Interfaces

In the previous two tutorials, we've tested a date management class. This is an pretty typical use of unit testing. However Qt and KDE applications will make use graphical classes that take user input (typically from a keyboard and mouse). QtTestLib offers support for testing these classes, which we'll see in this tutorial.

Again, we'll use an existing class as our test environment, and again it will be date related - the standard Qt QDateEdit class. For those not familiar with this class, it is a simple date entry widget (although with some powerful back end capabilities). A picture of the widget is shown below.

Figure 1. QDateEdit widget screenshot

The way QtTestLib provides GUI testing is by injecting QInputEvent events. To the application, these input events appear the same as normal key press/release and mouse clicks/drags. However the mouse and keyboard are unaffected, so that you can continue to use the machine normally while tests are being run.

An example of how you can use the GUI functionality of QtTestLib is shown below.

Much of this code is common with previous examples, so I'll focus on the new elements and the more important changes as we work through the code line-by-line.

Lines 1 to 3 import the various Qt declarations, as before.

Line 4 is a macro that is required for the data-driven part of this test, which I'll come to soon.

Lines 5 to 12 declare the test class - while the names have changed, it is pretty similar to the previous example. Note the testValidator and testValidator_data functions - we will be using data driven testing again in this example.

Our first real test starts in line 13. Line 16 creates a QDate, and line 17 uses that date as the initial value for a QDateEdit widget.

Lines 19 and 20 show how we can test what happens when we press the up-arrow key. The QTest::keyClick function takes a pointer to a widget, and a symbolic key name (a char or a Qt::Key). At line 20, we check that the effect of that event was to increment the date by a day. The QTest:keyClick function also takes an optional keyboard modifier (such as Qt::ShiftModifier for the shift key) and an optional delay value (in milliseconds). As an alternative to using QTest::keyClick, you can use QTest::keyPress and QTest::keyRelease to construct more complex keyboard sequences.

Lines 23 to 29 show a similar test to the previous one, but in this case we are simulating a mouse click. We need to click in the lower right hand part of the widget (to hit the decrement arrow - see Figure 1), and that requires knowing how large the widget is. So lines 23 and 24 calculate the correct point based off the size of the widget. Line 26 (and the identical line 27) simulates clicking with the left-hand mouse button at the calculated point. The arguments to Qt::mouseClick are:

a pointer to the widget that the click event should be sent to.

the mouse button that is being clicked.

an optional keyboard modifier (such a Qt::ShiftModifier), or 0 for no modifiers.

an optional click point - this defaults to the middle of the widget if not specified.

an optional mouse delay.

In addition to QTest::mouseClick, there is also QTest::mousePress, QTest::mouseRelease, QTest::mouseDClick (providing double-click) and QTest::mouseMove. The first three are used in the same way as QTest::mouseClick. The last takes a point to move the mouse to. You can use these functions in combination to simulate dragging with the mouse.

Lines 30 and 31 show another approach to keyboard entry, using the QTest::keyClicks. Where QTest::keyClick sends a single key press, QTest::keyClicks takes a QString (or something equivalent, in line 30 a character array) that represents a sequence of key clicks to send. The other arguments are the same.

Lines 32 to 34 show how you may need to use a combination of functions. After we've entered a new date in line 30, the cursor is at the end of the widget. At line 32, we use a Shift-Tab combination to move the cursor back to the month value. Then at line 33 we enter a new month value. Of course we could have used individual calls to QTest::keyClick, however that wouldn't have been as clear, and would also have required more code.

Data-driven GUI testing

Lines 50 to 60 show a data-driven test - in this case we are checking that the validator on QDateEdit is performing as expected. This is a case where data-driven testing can really help to ensure that things are working the way they should.

At lines 52 to 54, we fetch in an initial value, a series of key-clicks, and an expected result. These are the columns that are set up in lines 39 to 41. However note that we are now pulling in a QDate, where in previous examples we used three integers and then build the QDate from those. However QDate isn't a registered type for QMetaType, and so we need to register it before we can use it in our data-driven testing. This is done using the Q_DECLARE_METATYPE macro in line 4 and the qRegisterMetaType function in line 38.

Lines 42 to 47 add in a couple of sample rows. Lines 42 to 44 represent a case where the input is valid, and lines 45 to 47 are a case where the input is only partly valid (the day part). A real test will obviously contain far more combinations than this.

Those test rows are actually tested in lines 55 to 59. We construct the QDateEdit widget in line 55, using the initial value. We then send an Enter key click in line 57, which is required to get the widget into edit mode. At line 58 we simulate the data entry, and at line 59 we check whether the results are what was expected.

Lines 61 and 62 are the same as we've seen in previous examples.

Re-using test elements

If you are re-using a set of events a number of times, then it may be an advantage to build a list of events, and then just replay them. This can improve maintainability and clarity of a set of tests, especially for mouse movements.

The key class for building a list of test events is imaginatively known as QTestEventList. It is a QList of QTestEvents. The normal approach is to create the list, and then use various member functions to add key and mouse events. The normal functions that you'll need are addKeyClick and addMouseClick, which are very similar to the QTest::keyClick and QTest::mouseClick functions we used earlier in this tutorial. For finer grained operations, you can also use addKeyPress, addKeyRelease, addKeyEvent, addMousePress, addMouseRelease, addMouseDClick and addMouseMove to build up more complex event lists. You can also use addDelay to add a specified delay between events. When the list has been built up, you just call simulate on each widget.

You can see how this works in the example below, which is the QDateEdit example (from above) converted to use QTestEventList.

This example is pretty much the same as the previous version, up to line 25. In line 26, we create a QTestEventList. We add events to the list in lines 27 and 28 - note that we don't specify the widget we are calling them on at this stage. In line 30, we simulate each event on the widget. If we had multiple widgets, we could call simulate using the same set of events.

Lines 31 to 34 are as per the previous example.

We create another list in lines 35 to 37, although this time we are using addKeyClick and addKeyClicks instead of adding mouse events. Note that an event list can contain combinations of mouse and keyboard events - it just didn't make sense in this test to have such a combination. We use the second list at line 38, and check the results in line 39.

You can also build lists of events in data driven testing as well, as shown in lines 41 to 70. The key difference is that instead of fetching a QString in each row, we are fetching a QTestEventList. This requires that we add a column of QTestEventList, rather than QString (see line 45). At lines 47 to 50, we create a list of events. At line 52 we add those events to the applicable row. We create a second list at lines 54 to 56, and add that second list to the applicable row in line 58.

We fetch the events in line 65, and use them in line 68. If we had multiple widgets, then we could use the same event list several times.

Tutorial 4 - Testing for failure and avoiding tests

Under some conditions, it is impossible to avoid tests failing. In this section, we'll see how to deal with these cases.

Skipping tests

Where a test doesn't make sense to run (for example, if the required test files aren't available, or the feature is architecture or operating system dependent), the cleanest solution is to skip the test.

Tests are skipped using the QSKIP macro. QSKIP takes two arguments - a label string that should be used to describe why the test is being skipped, and a enumerated constant that controls how much of the test is skipped. If you pass SkipSingle, and the test is data driven, then only the current row is skipped. If you pass SkipAll and the test is data driven, then all following rows are skipped. If the test is not data driven, then it doesn't matter which one is used.

from the verbose output, you can see that the test was run on the first and third rows. The second row wasn't run because of the QSKIP call with a SkipSingle argument. Similarly, the fourth and fifth rows weren't run because the fourth row triggered a QSKIP call with a SkipAll argument.

Also note that the test didn't fail, even though there were two calls to QSKIP. Conceptually, a skipped test is a test that didn't make sense to run for test validity reasons, rather than a test that is valid but will fail because of bugs or lack of features in the code being tested.

Handling expected failures

If you have valid tests, but the code that you are testing doesn't pass them, then ideally you fix the code you are testing. However sometimes that isn't possible in the time that you have available, or because of a need to avoid binary incompatible changes. In this case, it is undesirable to delete or modify the unit tests - it is better to flag the test as "expected to fail", using the QEXPECT_FAIL macro. An example of this is shown below:

Example 17. Unit test showing expected failures

1 voidtestDate::testExpectedFail() 2 { 3 QEXPECT_FAIL("","1 is not 2, even for very large 1",Continue); 4 QCOMPARE(1,2); 5 QCOMPARE(2,2); 6 7 QEXPECT_FAIL("","1 is not 2, even for very small 2",Abort); 8 QCOMPARE(1,2); 9 // The next line will not be run, because we Abort on previous failure10 QCOMPARE(3,3);11 }

As you can see from the verbose output, we expect a failure from each time we do a QCOMPARE( 1, 2);. In the first call to QEXPECT_FAIL, we use the Continue argument, so the rest of the tests will still be run. However in the second call to QEXPECT_FAILwe use the Abort and the test bails at this point. Generally it is better to use Continue unless you have a lot of closely related tests that would each need a QEXPECT_FAIL entry.

Also note that tests that are marked as expected failures are not considered to be failures, so the test function above is considered to be a pass.

If a test that is marked to be an expected failure, and it unexpectedly passes, then that is flagged as an error, as shown below:

Example 19. Unit test showing unexpected pass

1 voidtestDate::testUnexpectedPass() 2 { 3 QEXPECT_FAIL("","1 is not 2, even for very large 1",Continue); 4 QCOMPARE(1,1); 5 QCOMPARE(2,2); 6 7 QEXPECT_FAIL("","1 is not 2, even for very small 2",Abort); 8 QCOMPARE(1,1); 9 QCOMPARE(3,3);10 }

The effect of unexpected passes on the running of the test is controlled by the second argument to QEXPECT_FAIL. If the argument is Continue and the test unexpectedly passes, then the rest of the test function will be run. If the argument is Abort, then the test will stop.

Checking debug messages and warnings

If you are testing border cases, you will likely run across the case where some kind of message will be produced using the qDebug or qWarning functions. Where a test produces a debug or warning message, that message will be logged in the test output (although it will still be considered a pass unless some other check fails), as shown in the example below:

Note that while this example produces the debug and warning messages within the test function (testQdebug), those messages would normally be propagated up from the code being tested. However the source of the messages does not make any difference to how they are handled.

If your test needs include either a clean output, or verification that appropriate messages are generated, then you will probably need the QtTest::ignoreMessage function.

Tip

Note: The ignoreMessage function can be used to ignore a message, however it might be clearer to think of this function as checking for the presence of an expected message. In particular, it is a test failure if you call ignoreMessage and the message is not generated.

Note that the warning message in testDate::testValidity has been "swallowed" by the the call to ignoreMessage.

By contrast, the warning message in testDate::testValiditi still causes a warning to be logged, because the ignoreMessage call does not match the text in the warning message. In addition, because a we expected a particular warning message and it wasn't received, the testDate::testValiditi test function fails.

Tutorial 5: Testing Qt slots and signals

An important part of Qt programming is the use of signals and slots. This section covers the support for testing of these features.

Testing signals

Testing of signals is a little more difficult than testing of slots, however Qt offers a very useful class called QSignalSpy that helps a lot.

QSignalSpy is a class provided with Qt that allows you to record the signals that have been emitted from a particular QObject subclass object. You can then check that the right number of signals have been emitted, and that the right kind of signals were emitted. You can find more information on the QSignalSpy class in your Qt documentation.

An example of how you can use QSignalSpy to test a class that has signals is shown below.

Example 26. QCheckBox test code, showing testing of signals

1 #include <QtTest> 2 #include <QtCore> 3 #include <QtGui> 4 5 classtestCheckBox:publicQObject 6 { 7 Q_OBJECT 8 privateslots: 9 voidtestSignals();10 };11 12 voidtestCheckBox::testSignals()13 {14 // You don't need to use an object created with "new" for15 // QSignalSpy, I just needed it in this case to test the emission16 // of a destroyed() signal.17 QCheckBox*xbox=newQCheckBox;18 19 // We are going to have two signal monitoring classes in use for20 // this test.21 // The first monitors the stateChanged() signal. 22 // Also note that QSignalSpy takes a pointer to the object.23 QSignalSpystateSpy(xbox,SIGNAL(stateChanged(int)));24 25 // Not strictly necessary, but I like to check that I have set up26 // my QSignalSpy correctly.27 QVERIFY(stateSpy.isValid());28 29 // Now we check to make sure we don't have any signals already30 QCOMPARE(stateSpy.count(),0);31 32 // Here is a second monitoring class - this one for the33 // destroyed() signal.34 QSignalSpydestroyedSpy(xbox,SIGNAL(destroyed()));35 QVERIFY(destroyedSpy.isValid());36 37 // A sanity check to verify the initial state38 // This also shows that you can mix normal method checks with39 // signal checks.40 QCOMPARE(xbox->checkState(),Qt::Unchecked);41 42 // Shouldn't already have any signals43 QCOMPARE(destroyedSpy.count(),0);44 45 // If we change the state, we should get a signal.46 xbox->setCheckState(Qt::Checked);47 QCOMPARE(stateSpy.count(),1);48 49 xbox->setCheckState(Qt::Unchecked);50 QCOMPARE(stateSpy.count(),2);51 52 xbox->setCheckState(Qt::PartiallyChecked);53 QCOMPARE(stateSpy.count(),3);54 55 // If we destroy the object, the signal should be emitted.56 deletexbox;57 58 // So the count of objects should increase.59 QCOMPARE(destroyedSpy.count(),1);60 61 // We can also review the signals that we collected62 // QSignalSpy is really a QList of QLists, so we take the first63 // list, which corresponds to the arguments for the first signal64 // we caught.65 QList<QVariant>firstSignalArgs=stateSpy.takeFirst();66 // stateChanged() only has one argument - an enumerated type (int)67 // So we take that argument from the list, and turn it into an integer.68 intfirstSignalState=firstSignalArgs.at(0).toInt();69 // We can then check we got the right kind of signal.70 QCOMPARE(firstSignalState,static_cast<int>(Qt::Checked));71 72 // check the next signal - note that takeFirst() removes from the list73 QList<QVariant>nextSignalArgs=stateSpy.takeFirst();74 // this shows another way of fudging the argument types75 Qt::CheckStatenextSignalState=(Qt::CheckState)nextSignalArgs.at(0).toInt();76 QCOMPARE(nextSignalState,Qt::Unchecked);77 78 // and again for the third signal79 nextSignalArgs=stateSpy.takeFirst();80 nextSignalState=(Qt::CheckState)nextSignalArgs.at(0).toInt();81 QCOMPARE(nextSignalState,Qt::PartiallyChecked);82 }83 84 QTEST_MAIN(testCheckBox)85 #include "tutorial5a.moc"

The first 11 lines are essentially unchanged from previous examples that we've seen. Line 15 creates the object that will be tested - as noted in the comments in lines 12-14, the only reason that I'm creating it with new is because I need to delete it in line 45 to cause the destroyed() signal to be emitted.

Line 20 sets up the first of our two QSignalSpy instances. The one in line 20 monitors the stateChanged(int) signal, and the one in line 29 monitors the destroyed() signal. If you get the name or signature of the signal wrong (for example, if you use stateChanged() instead of stateChanged(int)), then this will not be caught at compile time, but will result in a runtime failure. You can test if things were set up correctly using the isValid(), as shown in lines 24 and 30.

As shown in line 34, there is no reason why you cannot test normal methods, signals and slots in the same test.

Line 38 changes the state of the object under test, which is supposed to result in a stateChanged(int) signal being emitted. Line 39 checks that the number of signals increases from zero to one. Lines 40 and 41 repeat the process, and again in lines 42 and 43.

Line 45 deletes the object under test, and line 47 tests that the destroyed() signal has been emitted.

For signals that have arguments (such as our stateChanged(int) signal), you may also wish to check that the arguments were correct. You can do this by looking at the list of signal arguments. Exactly how you do this is fairly flexible, however for simple tests like the one in the example, you can manually work through the list using takeFirst() and check that each argument is correct. This is shown in line 52, 55 and 57 for the first signal. The same approach is shown in lines 59, 61 and 62 for the second signal, and the in lines 64 to 66 for the third signal. For a more complex set of tests, you may wish to apply some data driven techniques.

Tip

Note: You should be aware that, for some class implementations, you may need to return control to the event loop to have signals emitted. If you need this, try using the QTest::qWait() function.

Tutorial 6: Integrating with CMake

The KDE4 build tool is CMake, and I assume that you are familiar with the use of CMake. If not, you should review the CMake Tutorial first.

CMake offers quite good support for unit testing, and QTestLib tests can be easily integrated into any CMake build system.

Configuring for Testing

Tests are not built by default - you have to enable the test system, and build the tests.

You enable tests by adding an ENABLE_TESTING() line to the top of your CMakeLists.txt file.

In some configurations, there may be a build system option to turn on (or off) the compilation of tests. At this stage, you have to enable the BUILD_TESTING option in KDE4 modules, however this may go away in the near future, as later version of CMake can build the test applications on demand.

If the tests are still not building, you might want to issue make buildtests in tests directory.

Adding Tests

You add a single test to the list of all tests that can be run by using
ADD_TEST, which takes a number of arguments:

The first argument is a display name for the test. This is required.

The second argument is the name of the executable to be run (including the path to that executable, if required). This is required.

Any following arguments are passed to the executable. These are optional, and are usually omitted for QTestLib tests.

Note that ADD_TEST does nothing if ENABLE_TESTING() has not been run.

When choosing a display name, you should adopt a similar convention to the existing tests (e.g. in kdelibs/kdecore, all the display names start with kdecore-, which makes it easy to find the failing test if you run all the tests in kdelibs). If there are no existing tests, using a submodule prefix is probably a good idea.

The executable doesn't need to be something you build, but it will be for QTestLib, and many other tests. Remember that you need to specify the path, so this normally looks something like ${EXECUTABLE_OUTPUT_PATH}/testapp.

KDE4 CMake Recipe for QTestLib

If you are working in a KDE4 environment, then it is pretty easy to get CMake set up to build and run a test on demand.

You are meant to replace "kwhatevertest" with the name of your test application. The target_link_libraries() line will need to contain whatever libraries are needed for the feature you are testing, so if it is a GUI feature, you'll likely need to use "${KDE4_KDEUI_LIBS}.

Running the Tests

To run all tests, you can just "make test". This will work through each of the tests that have been added (at any lower level) using kde4_add_unit_test, provided that you have include(KDE4Defaults) in your CMakeLists.txt.

This is equivalent to running the "ctest" executable with no arguments. If you want finer grained control over which tests are run or the output format, you can use additional arguments. These are explained in the ctest man page ("man ctest" on a *nix system, or run "ctest --help-full").

To run a single test, use ./tutorial1.shell rather than just ./tutorial1, this will make it use the locally-built version of the shared libraries you're testing, rather than the installed ones.

Further Reading

Chapter 8 of the CMake Book provides a detailed description of how to do testing with CMake. Also see Appendix B for more on CTest and the special commands you can use.

Tutorial 7: Integrating with qmake

Tutorial 8: XML output

Tutorial 9: KDE specifics

Alternative tools for testing

There are a range of alternative testing approaches that can
be used either with unit tests, or as an addition to the unit tests.

Static tests

As noted in the introduction, unit tests are dynamic tests - they exercise the compiled code. Static tests are slightly different - they look for problems in source code, rather than making sure that the object code runs correctly.

Static test tools tend to identify completely different types of problems to unit tests, and you should seek to use them both.