Name | Last modified | Size | Description | |
---|---|---|---|---|
Parent Directory | - | |||
Error handling path is infamously difficult to test. To achieve high testing coverage, some extra mechanism is generally needed to emulate various error conditions in the testing environment. Such mechanism, as SQLite’s test VFS, tends to be heavy-weight and requires a lot of efforts to implement.
Celix Error Injector provides a lightweight alternative way. As its name suggests, Celix Error Injector enables you to inject arbitrary errors into target function call very easily. You only have to:
test_framework_with_ei
for a way of doing this with minimal CMake duplication.We have implemented several injectors.
malloc
/realloc
/calloc
celix_properties_create
Have a look at them before implementing your own.
Celix Error injector only works properly in debug build.
#include <gtest/gtest.h>
#include <iostream>
#include <cstring>
#include "malloc_ei.h"
#include "pubsub_wire_protocol_common.h"
class WireProtocolCommonEiTest : public ::testing::Test {
public:
WireProtocolCommonEiTest() = default;
~WireProtocolCommonEiTest() override {
celix_ei_expect_calloc(nullptr, 0, nullptr);
};
};
TEST_F(WireProtocolCommonEiTest, WireProtocolCommonTest_NotEnoughMemoryForMultipleEntries) {
pubsub_protocol_message_t message;
message.header.convertEndianess = 1;
message.metadata.metadata = nullptr;
char* data = strdup("ABCD4:key1,6:value1,4:key2,6:value2,6:key111,8:value111,"); //note 3 entries
auto len = strlen(data);
pubsubProtocol_writeInt((unsigned char*)data, 0, message.header.convertEndianess, 3);
for (int i = 0; i < 6; ++i) {
celix_ei_expect_calloc((void *)pubsubProtocol_decodeMetadata/* caller */, 0, nullptr, i+1/* ordinal */);
auto status = pubsubProtocol_decodeMetadata((void*)data, len, &message);
EXPECT_EQ(status, CELIX_ENOMEM);
EXPECT_EQ(nullptr, message.metadata.metadata);
}
free(data);
}
In the above test, pubsubProtocol_decodeMetadata
makes six calls to calloc
.
Failure in any of the six calls should return CELIX_ENOMEM
.
celix_ei_expect_calloc((void *)pubsubProtocol_decodeMetadata, 0, nullptr, i+1)
specifies only the (i+1)-th call made by pubsubProtocol_decodeMetadata
should fail.
Note that caller
does not always work, especially if the caller is a static function (not visible outside the translation unit).
In this case, you should specify CELIX_EI_UNKNOWN_CALLER
for caller
, and rely solely on the ordinal
parameter to specify which call to injector error into.
Note also that celix_ei_expect_calloc(nullptr, 0, nullptr)
is called to disable error injector for calloc
in the destructor.