| ![[ICO]](/icons/blank.gif) | Name | Last modified | Size | Description | 
|---|---|---|---|---|
| ![[PARENTDIR]](/icons/back.gif) | 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/calloccelix_properties_createHave 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.