Skip to content

Latest commit

 

History

History
497 lines (342 loc) · 19 KB

mocking_manual.markdown

File metadata and controls

497 lines (342 loc) · 19 KB
layout title
default
CppUTest Mocking Manual

CppUTest has support for building mocks. This document describes the mocking support. A couple of design goals for the mocking support were:

  • Same design goals as CppuTest -- limited C++ set to make it well suitable for embedded soft.
  • No code generation
  • No or very few magic hiding macros
  • Very simple to use
  • The developer stays in control

The main idea is to make manual mocking easier, rather than to make automated mocks. If manual mocking is easier, then it could also be automated in the future, but that isn't a goal by itself.

Table of Content

Simple Scenario

About the simplest scenario is to check that one particular function call has happened. The below code is an example of this:

{% highlight c++ %} #include "CppUTest/TestHarness.h" #include "CppUTestExt/MockSupport.h"

TEST_GROUP(MockDocumentation) { void teardown() { mock().clear(); } };

void productionCode() { mock().actualCall("productionCode"); }

TEST(MockDocumentation, SimpleScenario) { mock().expectOneCall("productionCode"); productionCode(); mock().checkExpectations(); } {% endhighlight %}

The only additional include for mocking is CppUTestExt/MockSupport.h which is the main include for anything CppUTest Mocking. The declaration of the TEST_GROUP is the same as always, there is no difference.

The TEST(MockDocumentation, SimpleScenario) contains the recording of the expectations as:

{% highlight c++ %} mock().expectOneCall("productionCode"); {% endhighlight %}

The call to mock() returns the global MockSupport (more about that later) on which we can record our expectations. In this example, we'll call expectOneCall("productionCode") which... records an expectation for one call to a function called productionCode.

The productionCode call is to show a call to whatever your real code is. The test ends with checking whether the expectations have been met. This is done with the:

{% highlight c++ %} mock().checkExpectations(); {% endhighlight %}

This call is needed when not using the MockSupportPlugin, otherwise this is done automatically for every single test. Also, the call to mock().clear() in the teardown is not needed when using the MockSupportPlugin, otherwise it is needed to clear the MockSupport. Without the clear, the memory leak detector will report the mock calls as leaks.

The actual mocked function call looks like:

{% highlight c++ %} void productionCode() { mock().actualCall("productionCode"); } {% endhighlight %}

where we use MockSupport by calling mock() and then record the actual call to the productionCode function.

This scenario is quite common when using linker stubbing of e.g. a 3rd party library.

If the call to productionCode wouldn't happen, then the test would fail with the following error message:

ApplicationLib/MockDocumentationTest.cpp:41: error: Failure in TEST(MockDocumentation, SimpleScenario)
    Mock Failure: Expected call did not happen.
    EXPECTED calls that did NOT happen:
        productionCode -> no parameters
    ACTUAL calls that did happen:
        <none>

Using Objects

Simple scenario using objects There is no difference between mocking functions and objects. Below code shows a mock using run-time mocking:

{% highlight c++ %} class ClassFromProductionCodeMock : public ClassFromProductionCode { public: virtual void importantFunction() { mock().actualCall("importantFunction"); } };

TEST(MockDocumentation, SimpleScenarioObject) { mock().expectOneCall("importantFunction");

ClassFromProductionCode* object = new ClassFromProductionCodeMock; /* create mock instead of real thing */
object->importantFunction();
mock().checkExpectations();

delete object;

} {% endhighlight %}

The code is self-explanatory. The real object is replaced by a hand-made mock object. The call to the mock then records the actual call via the MockSupport.

When using objects, we can also check whether the call was done on the right object, via this:

{% highlight c++ %} mock().expectOneCall("importantFunction").onObject(object); {% endhighlight %}

and the actual call would then be:

{% highlight c++ %} mock().actualCall("importantFunction").onObject(this); {% endhighlight %}

If the call to a wrong object happens, it would give the following error message:

MockFailure: Function called on a unexpected object: importantFunction
    Actual object for call has address: <0x1001003e8>
    EXPECTED calls that DID NOT happen related to function: importantFunction
        (object address: 0x1001003e0)::importantFunction -> no parameters
    ACTUAL calls that DID happen related to function: importantFunction
        <none>

Parameters

Of course, just checking whether a function is called is not particularly useful when we cannot check the parameters. Recording parameters on a function is done like this:

{% highlight c++ %} mock().expectOneCall("function").onObject(object).withParameter("p1", 2).withParameter("p2", "hah"); {% endhighlight %}

And the actual call is like:

{% highlight c++ %} mock().actualCall("function").onObject(this).withParameter("p1", p1).withParameter("p2", p2); {% endhighlight %}

If a parameter isn't passed, it will give the following error:

Mock Failure: Expected parameter for function "function" did not happen.
    EXPECTED calls that DID NOT happen related to function: function
        (object address: 0x1)::function -> int p1: <2>, char* p2: <hah>
    ACTUAL calls that DID happen related to function: function
        <none>
    MISSING parameters that didn't happen:
        int p1, char* p2

Objects as Parameters

withParameters can only use int, double, const char* or void* . However, parameters are often objects of other types and not of the basic types. How to handle objects as parameters? Below is an example:

{% highlight c++ %} mock().expectOneCall("function").withParameterOfType("myType", "parameterName", object); {% endhighlight %}

When using withParameterOfType, the mocking framework needs to know how to compare the type and therefore a Comparator has to be installed before using parameters of this type. This is done using installComparator, as below:

{% highlight c++ %} MyTypeComparator comparator; mock().installComparator("myType", comparator); {% endhighlight %}

MyTypeComparator is a custom comparator, which implements the MockNamedValueComparator interface. For example:

{% highlight c++ %} class MyTypeComparator : public MockNamedValueComparator { public: virtual bool isEqual(const void* object1, const void* object2) { return object1 == object2; } virtual SimpleString valueToString(const void* object) { return StringFrom(object); } }; {% endhighlight %}

The isEqual is called to compare the two parameters. The valueToString is called when an error message is printed and it needs to print the actual and expected values. If you want to use normal C functions, you can use the MockFunctionComparator which accepts pointers to functions in the constructor.

To remove the comparators, all you needs to do is call removeAllComparators, like:

{% highlight c++ %} mock().removeAllComparators(); {% endhighlight %}

Comparators sometimes lead to surprises, so a couple of warnings on its usage:

Warning 1:

  • Pay attention to the scope of your comparator variable!

Comparators are not copied, instead it uses the exact instance as passed to the installComparator function. So make sure it is still in-scope when the framework tries to use it! For example, if you installComparator inside the TEST, but do the checkExpectations in the teardown, then it is likely to cause a crash since the comparator has been destroyed.

Warning 2:

  • Pay extra attention to scope when using the MockPlugin

When using the MockPlugin (recommended), then it's best to install the comparators via the MockPlugin or put them in global space. The checkExpectations will be called after teardown and if your comparator was destroyed in the teardown then this will cause a crash.

Output Parameters

Some parameters do not represent data passed to the called function, but are passed by reference so that the function can 'return' a value by modifying the pointed-to data.

CppUMock allows the value of these output parameters to be specified in the expected call:

{% highlight c++ %} int outputValue = 4; mock().expectOneCall("Foo").withOutputParameterReturning("bar", &outputValue, sizeof(outputValue)); {% endhighlight %}

...and written during the actual call:

{% highlight c++ %} void Foo(int *bar) { mock().actualCall("foo").withOutputParameter("bar", bar); } {% endhighlight %}

After the actual call, the bar parameter passed to function Foo will have the value specified in the expected call (4, in this case).

Warning 1:

  • CppUMock does not and cannot prevent invalid memory accesses when using output parameters. It will memcpy exactly the number of bytes specified in the withOutputParameterReturning call. A segmentation fault may occur if this is larger than the data pointed to by the output parameter provided in the actual call.

Function overloads of withOutputParameterReturning are provided for char, int, unsigned, long, unsigned long, and double types so that the size parameter may be omitted:

{% highlight c++ %} char charOutputValue = 'a'; mock().expectOneCall("Foo").withOutputParameterReturning("bar", &charOutputValue);

int intOutputValue = 4; mock().expectOneCall("Foo").withOutputParameterReturning("bar", &intOutputValue);

unsigned unsignedOutputValue = 4; mock().expectOneCall("Foo").withOutputParameterReturning("bar", &unsignedOutputValue);

long longOutputValue = 4; mock().expectOneCall("Foo").withOutputParameterReturning("bar", &longOutputValue);

unsigned long unsignedLongOutputValue = 4; mock().expectOneCall("Foo").withOutputParameterReturning("bar", &unsignedLongOutputValue);

double doubleOutputValue = 4; mock().expectOneCall("Foo").withOutputParameterReturning("bar", &doubleOutputValue); {% endhighlight %}

Warning 2:

  • When a char, int, etc. array is passed to withOutputParameter, you must use the generic withOutputParameterReturning and provide the actual size of the array or only one element will be copied.

Return Values

Sometimes it is needed to let a mock function return a value which can then be used in production code. The test code would look like this:

{% highlight c++ %} mock().expectOneCall("function").andReturnValue(10); {% endhighlight %}

The mock function would look like:

{% highlight c++ %} int function () { return mock().actualCall("function").returnIntValue(); } {% endhighlight %}

or we could separate returnIntValue from the actualCall (below it!) like:

{% highlight c++ %} int function () { mock().actualCall("function"); return mock().intReturnValue(); } {% endhighlight %}

The return value options are used to transfer data between the test and the mock object, they themselves do not cause the tests to fail.

Passing other data

Sometimes a test wants to pass more data to the mock object to, for example, vary only a couple of parameters in a calculation. This can be done like this:

{% highlight c++ %} ClassFromProductionCode object; mock().setData("importantValue", 10); mock().setDataObject("importantObject", "ClassFromProductionCode", &object); {% endhighlight %}

And it can be used in the mock object like:

{% highlight c++ %} ClassFromProductionCode * pobject; int value = mock().getData("importantValue").getIntValue(); pobject = (ClassFromProductionCode*) mock().getData("importantObject").getObjectPointer(); {% endhighlight %}

Like return values, setting data will not ever make a test fail but it provides support in building mock objects.

Other MockSupport - ignoring, enabling, clearing, crashing

MockSupport offers a couple of other useful functions, which will be covered in this section.

Frequently, you only want to check a couple of calls in your test and ignore all the other calls. If you add expectOneCall for each of these calls, your tests might become too large (though, it might be a smell that your test is indeed too large). One way to prevent this is the ignoreOtherCalls, like:

{% highlight c++ %} mock().expectOneCall("foo"); mock().ignoreOtherCalls(); {% endhighlight %}

This will check that one call of foo happens (and only one call!), but all other calls will be ignored (such as "bar").

Sometimes, you don't want to just ignore calls, but instead disable the whole mocking framework for a while (to do something). This happens sometimes in initialization where you might want to do something without the mocking framework checking calls. You can do this by enabling/disabling such as:

{% highlight c++ %} mock().disable(); doSomethingThatWouldOtherwiseBlowUpTheMockingFramework(); mock().enable(); {% endhighlight %}

When you are ignoring calls to mocked functions that have a return value there is a catch. You will probably get a runtime error on your test when you try to ignore it.

As almost all things in life it is easier to explain this with an example, lets say you have this mock function:

{% highlight c++ %} int function () { return mock().actualCall("function").returnIntValue(); } {% endhighlight %}

If you try to ignore it or disable the framework, it will explode. Why ? The return value is not defined so there is no way to define the return value of the mocked function.

To cases like these there is a series of return value functions that allows you to define a default return value that will be returned when the mock function is ignored.

The example above could be written as:

{% highlight c++ %} int function () { return mock().actualCall("function").returnIntValueOrDefault(5); } {% endhighlight %}

If the function is ignored or the framework is disabled, a 5 will be returned, otherwise the expected return value will be returned (according to the expectations set on the test case).

You can also use mock support directly to do this (instead of using the actual call object):

{% highlight c++ %} int function () { mock().actualCall("function"); return mock().returnIntValueOrDefault(5); } {% endhighlight %}

If you want to clear all the expectations, settings, and comparators, call clear:

{% highlight c++ %} mock().clear(); {% endhighlight %}

Clear won't do checkExpectations, but just erase everything and start over. Usually clear() is called after a checkExpectations.

Sometimes, a mock actual call happens, but you cannot figure out from where it is called. If you only had a call stack, then it you could track it. Well, unfortunately, the mocking framework doesn't print stack traces, but it can crash! If you call the crashOnFailure on the MockSupport, then it will crash so that you can use the debugger to get a stack trace. like:

{% highlight c++ %} mock().crashOnFailure(); {% endhighlight %}

When using gdb, get a stack trace using:

gdb examples/CppUTestExamples_tests
r
bt

(r is run, it will run until crashes. bt is back trace which will produce a stack)

MockSupport Scope

MockSupport can be used hierarchically using MockSupport scope. This sounds really complex, but in reality it is very simple. When getting a mock support using the mock function, you can pass a namespace or scope and record the expectations (or do other things) inside this scope. For example:

{% highlight c++ %} mock("xmlparser").expectOneCall("open"); {% endhighlight %}

The actual call then has to look like this:

{% highlight c++ %} mock("xmlparser").actualCall("open"); {% endhighlight %}

A call on another namespace won't work, for example this won't match the call to xmlparser open:

{% highlight c++ %} mock("").actualCall("open"); {% endhighlight %}

Keeping calls in namespaces makes it easy to ignore one type of call and focus on another, for example:

{% highlight c++ %} mock("xmlparser").expectOneCall("open"); mock("filesystem").ignoreOtherCalls(); {% endhighlight %}

MockPlugin

CppUTest plugins can be installed in the main and 'extent' the unit test framework. It is a place where you can put work that needs to be done in all unit tests. There is a MockPlugin to make the work with mocks easier. It does the following work:

  • checkExpectations at the end of every test (on global scope, which goes recursive over all scopes)
  • clear all expectations at the end of every test
  • install all comparators that were configured in the plugin at the beginning of every test
  • remove all comparators at the end of every test

Installing the MockPlugin means you'll have to add to main something like:

{% highlight c++ %} #include "CppUTest/TestRegistry.h" #include "CppUTestExt/MockSupportPlugin.h"

MyDummyComparator dummyComparator; MockSupportPlugin mockPlugin;

mockPlugin.installComparator("MyDummyType", dummyComparator); TestRegistry::getCurrentRegistry()->installPlugin(&mockPlugin); {% endhighlight %}

This code creates a comparator for MyDummy and installs it at the plugin. This means the comparator is available for all test cases. It creates the plugin and installs it at the current test registry. After installing the plugin, you don't have to worry too much anymore about calling checkExpectations or cleaning your MockSupport.

C Interface

Sometimes it is useful to access the mocking framework from a .c file rather than a .cpp file. For example, perhaps, for some reason, the stubs are implemented in a .c file rather than a .cpp file. Instead of changing over all to .cpp, it would be easier if the mocking framework can be called via C. The C interface is exactly meant for this. The interface is based on the C++ one, so below is some code and it ought to be easy to figure out what it does (if you've read all that was written earlier):

{% highlight c++ %} #include "CppUTestExt/MockSupport_c.h"

mock_c()->expectOneCall("foo")->withIntParameters("integer", 10)->andReturnDoubleValue(1.11); mock_c()->actualCall("foo")->withIntParameters("integer", 10)->returnValue().value.doubleValue;

mock_c()->installComparator("type", equalMethod, toStringMethod); mock_scope_c("scope")->expectOneCall("bar")->withParameterOfType("type", "name", object); mock_scope_c("scope")->actualCall("bar")->withParameterOfType("type", "name", object); mock_c()->removeAllComparators();

mock_c()->setIntData("important", 10);

mock_c()->checkExpectations(); mock_c()->clear(); {% endhighlight %}

The C interface uses a similar builder structure as the C++ interface. It is far less common in C, but it works the same.