How do I unit test a protected method in C++?
Asked Answered
O

6

24

How do I unit test a protected method in C++?

In Java, I'd either create the test class in the same package as the class under test or create an anonymous subclass that exposes the method I need in my test class, but neither of those methods are available to me in C++.

I am testing an unmanaged C++ class using NUnit.

Oregano answered 14/7, 2009 at 19:57 Comment(1)
Are you using GenTestAsm-codeproject.com/KB/applications/GenTestAsmBase.aspx? Or how else are you running unmanaged C++ code from NUNit (.NET)?Dashboard
S
34

Assuming you mean a protected method of a publicly-accessible class:

In the test code, define a derived class of the class under test (either directly, or from one of its derived classes). Add accessors for the protected members, or perform tests within your derived class . "protected" access control really isn't very scary in C++: it requires no co-operation from the base class to "crack into" it. So it's best not to introduce any "test code" into the base class, not even a friend declaration:

// in realclass.h
class RealClass {
    protected:
    int foo(int a) { return a+1; }
};

// in test code
#include "realclass.h"
class Test : public RealClass {
    public:
    int wrapfoo(int a) { return foo(a); }
    void testfoo(int input, int expected) {
        assert(foo(input) == expected);
    }
};

Test blah;
assert(blah.wrapfoo(1) == 2);
blah.testfoo(E_TO_THE_I_PI, 0);
Suribachi answered 14/7, 2009 at 23:49 Comment(1)
This is the true object oriented solution to this problem. Inheritance exists for a reason!Irate
S
22

You can also use using keyword to expose public block (using .

// in realclass.h
class RealClass {
    protected:
    int foo(int a) { return a+1; }
    int foo(string a) { return a.length(); } // Overload works too
    template<class T> int foo(const T& a) { return 1; } // Templates work too
};

// in test code
#include "realclass.h"
class RealClassExposed : public RealClass {
    public:
        using RealClass::foo;
};

RealClassExposed blah;
assert(blah.foo(1) == 2);
assert(blah.foo("test") == 4);
assert(blah.foo(blah) == 1);

See: http://en.cppreference.com/w/cpp/language/using_declaration

Subtangent answered 21/7, 2015 at 15:59 Comment(0)
M
2

I use CxxTest and have the CxxTest derive from the class that contains the protected member function. If you're still searching around for your favorite C++ Unit Testing framework, take a look at this article.

Maxon answered 14/7, 2009 at 23:53 Comment(0)
A
1

Declare a friend class MyClass_UnitTest; in your MyClass. You can then define MyClass_UnitTest elsewhere in your unit test program that has full access to MyClass internals, but you don't have to provide an implementation in your release application. See CppUnit documentation for a good example of how this is done.

Appendicitis answered 14/7, 2009 at 20:36 Comment(3)
Friend classes are not your friend.Irate
Then nothing is your friend. Every feature of every language can be misused and abused. Just because idiots misuse the feature doesn't mean I should never use it for its proper purpose.Appendicitis
Agreed, but some features are easier to abuse than others. More often than not, I see friend functions used to circumvent encapsulation.Irate
B
1

There's a simple solution in C++ using #define. Just wrap the include of your "ClassUnderTest" like this:

#define protected public
 #define private   public
    #include <ClassUnderTest.hpp>
 #undef protected
#undef private

Credit goes to this article and RonFox

Booma answered 16/12, 2015 at 17:43 Comment(1)
In response to the -1... I know in "theory" you should be testing the public interface. However, in practice during the development cycle this can be an invaluable trick. My advice... use it judiciously.Booma
C
-1

Consider a public, possibly static 'unit test' function.

Ugly, but better than the alternatives I can think of using macros or friends or such.

Commemorate answered 14/7, 2009 at 20:11 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.