cpp-mock-testing
Mock Testing
Instructions for AI coding agents on automating mock test creation using Google Mock (GMock) with consistent software testing patterns in this C++ project.
1. Benefits
-
Isolation
Isolates the unit under test from external dependencies, ensuring tests focus on the specific component's behavior.
-
Control
Provides precise control over dependency behavior through expectations and return values, enabling thorough testing of edge cases and error conditions.
-
Verification
Automatically verifies that dependencies are called correctly with expected parameters and call counts.
-
Flexibility
Supports various testing scenarios including strict mocks, nice mocks, and sequence verification for complex interactions.
2. Patterns
-
Mock Objects
Simulated objects that mimic the behavior of real objects in controlled ways. They verify interactions between the unit under test and its dependencies.
-
Interface Mocking
Creating mock implementations of abstract interfaces or base classes to isolate the unit under test from concrete implementations.
-
Behavior Verification
Verifying that methods are called with expected arguments and in the correct order, rather than just checking return values.
-
Return Value Stubbing
Configuring mock objects to return specific values when their methods are called, allowing control over dependency behavior during tests.
-
Exception Injection
Using mocks to simulate error conditions by throwing exceptions, enabling tests to verify error handling logic.
3. Workflow
-
Identify Dependencies
Identify interfaces or classes that need to be mocked (e.g., database connections, file systems, network services, external APIs).
-
Create Mock Classes
Create mock classes for interfaces under
test(s)/unit/<module>/using GMock'sMOCK_METHODmacro. -
Register with CMake
Add the test file to
test(s)/unit/<module>/CMakeLists.txtusingmeta_gtest()withWITH_GMOCKoption.meta_gtest( WITH_GMOCK TARGET ${PROJECT_NAME}-test SOURCES <header>_test.cpp ) -
Define Expectations
Set up expectations using
EXPECT_CALLto specify:- Which methods should be called
- Expected arguments (using matchers)
- Call frequency (Times, AtLeast, AtMost, etc.)
- Return values or actions
-
Test Coverage Requirements
Include comprehensive scenarios:
- Normal operation with mocked dependencies
- Error conditions (exceptions, null returns, invalid data)
- Boundary conditions in dependency interactions
- Sequence of calls to multiple dependencies
- Concurrent access scenarios when applicable
-
Apply Templates
Structure all tests using the template pattern below.
4. Commands
| Command | Description |
|---|---|
make cmake-gcc-test-unit-build |
CMake preset configuration with GMock support and Compile with Ninja |
make cmake-gcc-test-unit-run |
Execute tests via ctest (mock tests are part of unit tests) |
make cmake-gcc-test-unit-coverage |
Execute tests via ctest and generate coverage reports including mock test coverage |
5. Style Guide
-
Test Framework
Use Google Mock (GMock) framework via
#include <gmock/gmock.h>and#include <gtest/gtest.h>. -
Mock Class Definition
Define mock classes inheriting from the interface to be mocked. Use
MOCK_METHODmacro with proper method signature, including const qualifiers and override specifiers. -
Include Headers
GMock/GTest headers are listed first in mock test files as a convention to clearly identify the file as a test file using the GMock framework.
Include necessary headers in this order:
- GMock/GTest headers (
<gmock/gmock.h>,<gtest/gtest.h>) - Standard library headers (
<memory>,<string>, etc.) - Project interface headers
- Project implementation headers
- GMock/GTest headers (
-
Namespace
Use
using namespace <namespace>;andusing namespace ::testing;for convenience within test functions to access GMock matchers and actions. -
Test Organization
Use table-driven testing for multiple scenarios with the same mock setup. Each
TESTorTEST_Fshould focus on one aspect of the interaction with mocked dependencies. -
Mock Types
- NiceMock
Ignores unexpected calls (use for non-critical dependencies)
- StrictMock
Fails on any unexpected calls (use for strict verification)
- Default Mock
Warns on unexpected calls (balanced approach)
- NiceMock
-
Expectations
- Use
EXPECT_CALLto set up expectations before exercising the unit under test - Chain matchers with
.With(),.WillOnce(),.WillRepeatedly(),.Times() - Prefer specific matchers (
Eq(),Gt(),_) over generic ones when possible
- Use
-
Matchers and Actions
- Use built-in matchers:
_(anything),Eq(),Ne(),Lt(),Gt(),Le(),Ge(),IsNull(),NotNull() - Container matchers:
IsEmpty(),SizeIs(),Contains(),ElementsAre() - String matchers:
StartsWith(),EndsWith(),HasSubstr(),MatchesRegex() - Use
Return(),ReturnRef(),Throw(),DoAll(),Invoke()for actions
- Use built-in matchers:
-
Sequence Verification
Use
InSequenceorSequenceobjects when call order matters. -
Traceability
Employ
SCOPED_TRACE(tc.label)for traceable failures in table-driven mock tests. -
Assertions
Use
EXPECT_*macros to allow all test cases to run. Mock expectations are automatically verified at the end of each test.
6. Template
Use these templates for new mock tests. Replace placeholders with actual values.
6.1. File Header Template
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include <memory>
#include <string>
#include <vector>
#include "<module>/<interface>.hpp"
#include "<module>/<implementation>.hpp"
using namespace <namespace>;
using namespace ::testing;
6.2. Mock Class Template
/**
* @brief Mock implementation of <Interface> for testing.
*/
class Mock<Interface> : public <Interface>
{
public:
MOCK_METHOD(<return_type>, <method_name>, (<param_types>), (override));
MOCK_METHOD(<return_type>, <method_name2>, (<param_types>), (const, override));
};
6.3. Table-Driven Mock Test Template
TEST(<Module>Test, <FunctionName>WithMock)
{
// In-Got-Want
struct Tests
{
std::string label;
struct In
{
/* input types and names */
} in;
struct Want
{
<output_type> expected; // expected output type(s) and name(s)
/* expected mock call parameters and behavior */
<size_t> call_count; // number of times method should be called
<return_type> return_value; // value mock should return
<param_type> param; // expected parameter value(s)
} want;
};
// Table-Driven Testing
const std::vector<Tests> tests = {
{
"case-description-1",
/* in */ {/* input values */},
/* want */ {/* expected */, /* call_count */ 1, /* return_value */ {}, /* param */ {}}
},
{
"case-description-2",
/* in */ {/* input values */},
/* want */ {/* expected */, /* call_count */ 1, /* return_value */ {}, /* param */ {}}
},
// add more cases as needed
};
for (const auto &tc : tests)
{
SCOPED_TRACE(tc.label);
// Arrange
auto mock_dependency = std::make_shared<Mock<Interface>>();
EXPECT_CALL(*mock_dependency, <method_name>(tc.want.param))
.Times(tc.want.call_count)
.WillOnce(Return(tc.want.return_value));
<Implementation> object(mock_dependency);
// Act
auto got = object.<function>(tc.in.<input>);
// Assert
EXPECT_EQ(got, tc.want.expected);
}
}
6.4. Sequence Verification Template
TEST(<Module>Test, <FunctionName>WithSequence)
{
// Arrange
auto mock_dependency = std::make_shared<StrictMock<Mock<Interface>>>();
InSequence seq;
EXPECT_CALL(*mock_dependency, <method1>(_)).WillOnce(Return(<value1>));
EXPECT_CALL(*mock_dependency, <method2>(_)).WillOnce(Return(<value2>));
<Implementation> object(mock_dependency);
// Act
auto got = object.<function>();
// Assert
EXPECT_EQ(got, <expected>);
}
6.5. Exception Testing Template
TEST(<Module>Test, <FunctionName>ThrowsOnError)
{
// Arrange
auto mock_dependency = std::make_shared<Mock<Interface>>();
EXPECT_CALL(*mock_dependency, <method_name>(_))
.WillOnce(Throw(std::runtime_error("error message")));
<Implementation> object(mock_dependency);
// Act & Assert
EXPECT_THROW(object.<function>(), std::runtime_error);
}
6.6. NiceMock Template
TEST(<Module>Test, <FunctionName>WithNiceMock)
{
// Arrange
auto mock_dependency = std::make_shared<NiceMock<Mock<Interface>>>();
ON_CALL(*mock_dependency, <method_name>(_))
.WillByDefault(Return(<default_value>));
EXPECT_CALL(*mock_dependency, <critical_method>(_))
.Times(1)
.WillOnce(Return(<value>));
<Implementation> object(mock_dependency);
// Act
auto got = object.<function>();
// Assert
EXPECT_EQ(got, <expected>);
}
7. References
- GoogleTest Mocking for Dummies guide.
- GoogleTest Mocking Cookbook guide.
- GoogleTest Mocking Cheat Sheet guide.