【单元测试】CMocka单元测试框架(支持mock功能)

文章目录

  • 1. 前言
  • 2. CMocka概述
  • 3. CMocka编译安装
  • 4. 编写CMocka测试程序
    • 4.1. 通常用法
    • 4.2. 头文件
    • 4.3. 测试函数
    • 4.4. 初始化CMUnitTest结构体
    • 4.5. 执行测试
  • 5. CMocka API
    • 5.1. 断言
    • 5.2. 模拟函数
      • 5.2.1. 模拟函数返回值
      • 5.2.2. 模拟函数输出参数
      • 5.2.3. 检查模拟函数输入参数
      • 5.2.4. 检查模拟函数调用顺序
    • 5.3. 动态内存检测
    • 5.4. 异常检测
    • 5.5. 输出格式
    • 5.6. 测试用例过滤器
  • 6. 参考资料

1. 前言

本文内容涉及单元测试,需要读者具有单元测试的基础知识,如果没有,请移步至我之前的博客文章:

  • 【单元测试】开篇
  • 【单元测试】Test Double简介

2. CMocka概述

官网:https://cmocka.org/

CMocka 是一款支持 mock 对象、面向C语言的单元测试框架,CMocka 往往是编译成库的形式,供C单元测试程序链接调用。其前身是谷歌开发的 Cmockery,由于后者缺少维护,因此 CMocka 继承了 Cmockery 并继续维护。

CMocka 框架的特性:

  • 支持模拟对象,可设置模拟函数的期望返回值,期望输出参数,可检查模拟函数的输入参数、函数调用顺序。
  • 支持Test fixtures(包括setup和teardown).
  • 不依赖第三方库,只需要一个C库即可.
  • 支持众多平台(Linux, BSD, Solaris, Windows和嵌入式平台)和编译器(GCC, LLVM, MSVC, MinGW等).
  • 提供对异常信号(SIGSEGV, SIGILL, …)的处理。
  • 非fork()执行.
  • 提供基本的内存检测,包括内存泄露,内存溢出检测.
  • 提供丰富的断言宏.
  • 支持多种格式输出 (STDOUT, SUBUNIT, TAP, XML).
  • 开源.

3. CMocka编译安装

CMocka 往往是以库的形式提供给测试程序链接调用的,为此我们需要先将 CMocka 源码编译成库。

从官网下载 CMocka 源码:

git clone git://git.cryptomilk.org/projects/cmocka.git

编译、安装步骤在源码中的 README.md 和 INSTALL.md 文件中有详细说明,我的版本是 cmocka-1.1.5,在linxu系统下编译安装步骤为:

# cd cmocka
# mkdir build && cd build
# cmake ..
# make
# make install

默认是安装在 /usr/local 目录下,安装的文件不多,主要是头文件和库文件,就以下几个:

Install the project...
-- Install configuration: ""
-- Installing: /usr/local/lib/pkgconfig/cmocka.pc
-- Installing: /usr/local/lib/cmake/cmocka/cmocka-config.cmake
-- Installing: /usr/local/lib/cmake/cmocka/cmocka-config-version.cmake
-- Installing: /usr/local/include/cmocka.h
-- Installing: /usr/local/include/cmocka_pbc.h
-- Installing: /usr/local/lib/libcmocka.so.0.7.0
-- Installing: /usr/local/lib/libcmocka.so.0
-- Installing: /usr/local/lib/libcmocka.so

也可以自定义安装目录,比如要安装在 /usr 目录下:

# cmake -DCMAKE_INSTALL_PREFIX=/usr ..
# make
# make install

更多的编译选项,可以见 cmocka/INSTALL.md 文件说明。

4. 编写CMocka测试程序

有了 CMocka 库,就可以开始使用 CMocka 框架编写单元测试程序了。

4.1. 通常用法

源码 cmocka/example/simple_test.c 很好的展示了 CMocka 的通常用法,代码如下所示:

#include 
#include 
#include 
#include 
#include 

/* A test case that does nothing and succeeds. */
static void null_test_success(void **state) {
    (void) state; /* unused */
}

int main(void) {
    const struct CMUnitTest tests[] = {
        cmocka_unit_test(null_test_success),
    };

    return cmocka_run_group_tests(tests, NULL, NULL);
}

其中 null_test_success 函数是测试用例,CMUnitTest 结构体是测试用例集(可以包含多个测试用例),每个测试用例可以设定可选的 startupteardown 函数,用于负责执行测试前的初始化和测试后的销毁工作。上面示例中使用了 cmocka_unit_test 宏来填充 CMUnitTest 结构体中的测试用例( startupteardown 为 NULL)。cmocka_run_group_tests 函数用于启动测试并展示测试结果,可以为测试集指定全局的 startupteardown(示例中都是NULL)。

编译很简单:

# cd cmocka/example
# gcc -o simple_test simple_test.c -lcmocka

其中 -lcmocka 会链接前面安装的 /usr/local/lib/libcmocka.so 库。执行测试程序看看效果:

# ./simple_test 
[==========] Running 1 test(s).
[ RUN      ] null_test_success
[       OK ] null_test_success
[==========] 1 test(s) run.
[  PASSED  ] 1 test(s).

如果提示找不到 cmocka 动态库,可以先执行以下命令,让新添加的 /usr/local/lib/libcmocka.so 动态库能为系统所识别:

# ldconfig

或者,将 /usr/local/lib/ 目录加入 LD_LIBRARY_PATH 环境变量:

# export LD_LIBRARY_PATH=/usr/local/lib/:$LD_LIBRARY_PATH

4.2. 头文件

#include 之前,必须先 #include 以下四个头文件,这是官网 The CMocka API 中明确要求的,在 cmocka.h 头文件开头部分也有注明。

#include 
#include 
#include 
#include 

4.3. 测试函数

CMocka 的测试用例是一个C函数,其函数签名为:

void test_func(void **state)

其中 void **state 指针是指向 CMUnitTest 结构体中的 void *initial_state 变量的地址,在初始化 CMUnitTest 结构体时会设置 initial_state 变量初始值(如 cmocka_unit_test 宏会设置 initial_state 为 NULL,还有其他初始化的宏,会在后续中介绍),state 指针也会传递给测试用例对应的 setupteardown 函数。

int setup(void **state)
int teardown(void **state)

执行测试用例的过程中,会按先后顺序依次调用 setuptest_functeardown 三个函数,它们都可以访问(读或写) CMUnitTest 结构体中的 initial_state 变量。

源码 cmocka/tests/test_basics.c 中的示例很好的演示了 state 指针的使用,代码如下(有删减):

static int setup(void **state) {
    int *answer = malloc(sizeof(int));
    assert_non_null(answer);
    *answer = 42;
    *state = answer;
    return 0;
}

static int teardown(void **state) {
    free(*state);
    return 0;
}

static void int_test_success(void **state) {
    int *answer = *state;
    assert_int_equal(*answer, 42);
}

int main(void) {
    const struct CMUnitTest tests[] = {
        cmocka_unit_test_setup_teardown(int_test_success, setup, teardown),
    };
    return cmocka_run_group_tests(tests, NULL, NULL);
}

除此之外,源码 cmocka/tests/test_fixtures.c 还演示了有关 state 指针的更多用法,可自行查阅。

4.4. 初始化CMUnitTest结构体

CMUnitTest 结构体变量用于存放测试用例集,该结构体定义如下:

struct CMUnitTest {
    const char *name;                /* 测试用例名 */
    CMUnitTestFunction test_func;    /* 测试用例函数指针 */
    CMFixtureFunction setup_func;    /* 测试用例对应的setup函数指针 */
    CMFixtureFunction teardown_func; /* 测试用例对应的teardown函数指针 */
    void *initial_state;             /* 测试用例私有数据指针 */
};

CMocka 提供了足够丰富的宏用于初始化 CMUnitTest 结构体,官网API手册中的 Running Tests 模块列出了所有的这些宏定义:

#define 	cmocka_unit_test(f)   { #f, f, NULL, NULL, NULL }
#define 	cmocka_unit_test_setup(f, setup)   { #f, f, setup, NULL, NULL }
#define 	cmocka_unit_test_teardown(f, teardown)   { #f, f, NULL, teardown, NULL }
#define 	cmocka_unit_test_setup_teardown(f, setup, teardown)   { #f, f, setup, teardown, NULL }
#define 	cmocka_unit_test_prestate(f, state)   { #f, f, NULL, NULL, state }
#define 	cmocka_unit_test_prestate_setup_teardown(f, setup, teardown, state)   { #f, f, setup, teardown, state }

其中 cmocka_unit_test 宏最为简单,只设置测试用例函数,其他为NULL。cmocka_unit_test_prestate_setup_teardown 宏最丰富,可设置测试用例的所有信息。

4.5. 执行测试

CMocka 即支持 cmocka_run_group_tests 函数执行所有测试用例集,也支持 run_test 函数执行单个测试用例。

static void null_test_success(void **state) {
    (void) state;
}

int main(void) {
    return run_test(null_test_success);
}

不管是哪种方式执行测试,当执行测试用例遇到错误时,会立刻中断并退出当前测试用例,测试程序将继续执行下一测试用例。

有一些特殊的函数,用于在不执行逻辑测试的情况下,向框架注册通过或失败。这对于测试控制流或其他不需要逻辑测试的情况都非常有用:

  • void fail(void)

    立刻中断当前测试用例的执行,将其标记为失败,并继续执行下一个测试用例会(如果有的话)。

  • void fail_msg(const char *msg,…)

    跟fail()一样,只是多了打印日志信息。

  • void skip(void)

    立刻中断当前测试用例的执行,将其标记为跳过,并继续执行下一个测试用例会(如果有的话)。

源码 cmocka/tests/test_skip.c 演示了 skip() 的用法,可自行查阅,fail() 用法类似。

5. CMocka API

官网API手册 The CMocka API 中有这么一张图,它清晰的描述了 CMocka API 由哪些子模块构成。

【单元测试】CMocka单元测试框架(支持mock功能)_第1张图片

5.1. 断言

CMocka 提供了一组用于测试逻辑条件的断言,其使用方法和标准C中的 assert 差不多。比如要测试 add 函数(两个整形数之和),可以:

static void test_add(void **state) {
    (void) state;

    assert_int_equal(add(1, 2), 3);
}

官网API手册中的 Assert Macros 模块中列出了 CMocka 框架支持的所有断言(从断言的名字就能看出其用途):

Assert Macros Description
assert_true (scalar expression)
assert_false (scalar expression)
Assert that the given expression is true (or false).
assert_int_equal (int a, int b)
assert_int_not_equal (int a, int b)
Assert that the two given integers are (or not) equal.
assert_float_equal (float a, float b, float epsilon)
assert_float_not_equal (float a, float b, float epsilon)
Assert that the two given float are (or not) equal given an epsilon.
Examples:
assert_float_equal(0.5f, 1.f / 2.f, 0.000001f);
assert_float_not_equal(0.5, 0.499f, 0.000001f);
assert_non_null (void *pointer)
assert_null (void *pointer)
Assert that the given pointer is non-NULL (or NULL).
assert_ptr_equal (void *a, void *b)
assert_ptr_not_equal (void *a, void *b)
Assert that the two given pointers are (or not) equal.
assert_string_equal (const char *a, const char *b)
assert_string_not_equal (const char *a, const char *b)
Assert that the two given strings are (or not) equal.
assert_memory_equal (const void *a, const void *b, size_t size)
assert_memory_not_equal (const void *a, const void *b, size_t size)
Assert that the two given areas of memory are (or not) equal.
assert_in_range (LargestIntegralType value, LargestIntegralType minimum, LargestIntegralType maximum)
assert_not_in_range (LargestIntegralType value, LargestIntegralType minimum, LargestIntegralType maximum)
Assert that the specified value is (or not) within the range of [minimum, maximum].
assert_in_set (LargestIntegralType value, LargestIntegralType values[], size_t count)
assert_not_in_set (LargestIntegralType value, LargestIntegralType values[], size_t count)
Assert that the specified value is (or not) within a set.
assert_return_code (int rc, int error) Assert that the return_code is greater than or equal to 0.

5.2. 模拟函数

CMocka 框架带有模拟函数(Mock Functions)的功能,可以为模拟函数设置期望返回值,设置期望输出参数,检查输入参数,检查调用顺序。

5.2.1. 模拟函数返回值

官方API接口列表:见官网API手册中的 Mock Objects 模块。

为了简化模拟函数的实现,CMocka 为模拟函数提供了存储返回值的功能。CMocka 框架内部维护着每个模拟函数所特有的「返回值队列」,will_return() 将期望返回值 push 到相应队列中,mock() 再从相应队列 pop 出期望返回值。

  • will_return(function, value)

    该宏用于给模拟函数function设置期望返回值value,一般是在测试用例中调用该宏。

  • mock()

    该宏用于返回will_return预设的期望值,它只能在模拟函数中调用。mock() 会根据自身所在的当前函数名(即调用 mock() 的函数),去框架内部队列中寻找匹配的函数名(即寻找will_return指定的函数名),然后返回对应的期望返回值。一句话就是:返回当前函数的期望返回值。所以, mock() 必须在模拟函数 function 中调用,才能返回期望返回值 value,在其他地方调用 mock() 将无法返回期望值。

需要注意的是 will_return()mock() 要成对出现,执行测试用例的过程中,如果发现 will_return()mock() 没有成对出现,就会将测试用例标记为失败。

举个例子:

int mock_function(void)
{
    return (int) mock();
}

static void test_function(void **state)
{
    will_return(mock_function, 42);
    assert_int_equal(mock_function(), 42);
}

从官网API手册中的 Mock Objects 模块中可以看出,除了 will_return() 之外,设置期望返回值的AIP接口家族有好几个。

will_return(function, value)
will_return_count(function, value, count)
will_return_always(function, value)
will_return_maybe(function, value)

其实,这些接口最终都是以 will_return_count 作为基础,理解了 will_return_count,就能掌握所有接口。will_return_count 的第三个参数 count 表明应该通过 mock() 返回的次数,即应该执行 mock() 的次数,如果不匹配,就会将测试用例标记为失败。如果 count 设置为-1,表明可以无数次执行 mock(),但至少得执行一次。如果 count 设置为-2,表明可以无数次执行 mock(),甚至一次都不执行也可以。

  • will_return(function, value) 相当于 will_return_count(function, value, 1)
  • will_return_always(function, value) 相当于 will_return_count(function, value, -1)
  • will_return_maybe(function, value) 相当于 will_return_count(function, value, -2)

源码 cmocka/tests/test_returns.ccmocka/tests/test_returns_fail.c 演示了有关 Mock Objects 的更多用法,可自行查阅。

5.2.2. 模拟函数输出参数

will_return() 除了可以预先设定模拟函数的期望返回值之外,还可以用于预先设定模拟函数的输出参数期望值。函数的返回值和输出参数都可以统称为return,结合代码解释:

int mock_function(char **out)
{
    *out = (char *) mock();
    return (int) mock();
}

static void test_function(void **state)
{
    char *out = NULL;
    const char * const hel = "hello";

    will_return(mock_function, hel);
    will_return(mock_function, 42);

    assert_int_equal(mock_function(&out), 42);
    assert_ptr_equal(out, hel);
}

在测试用例函数 test_function 中,调用了两次 will_return() 往框架内部队列push了两个期望值,在模拟函数 mock_function 中,相应地,调用两次 mock() 从队列中pop出期望值,一个用于模拟函数输出参数,一个用于模拟函数返回值。

5.2.3. 检查模拟函数输入参数

官方API接口列表:见官网API手册中的 Checking Parameters 模块。

除了存储模拟函数的期望返回值之外,CMocka 框架内部还可以存储模拟函数的输入参数的期望值,以便测试用例能够检查模拟函数的输入参数的正确性。在 CMocka 框架内部,维护着每个模拟函数所特有的Key-Value数据结构队列,用于存储模拟函数的形参字符串(作为Key)及其输入参数期望值(作为Value)。

预先设置模拟函数输入参数期望值的宏有很多,以下统称为 expect_*() 宏。下文以 expect_value 为例进行说明,其他 expect_*() 宏的工作原理是类似的。

  • expect_value(function, parameter, value)

    该宏用于设定模拟函数的参数期望值,一般是在测试用例中调用该宏。通过该宏可将模拟函数function的形参名parameter及其参数期望值value组合成Key-Value数据,并push进CMocka框架内部的队列中。

  • check_expected(parameter)

    该宏用于验证模拟函数输入参数值是否符合期望值,它只能在模拟函数中调用。该宏首先在CMocka框架内部模拟函数function所属的Key-Value队列中寻找parameter,找不到会将测试用例标记为失败,找到了则从队列中pop出数据提取期望值Value,并跟实际传入模拟函数的参数值进行对比,相同则测试通过,不同则测试失败(测试用例标记为失败)。

结合例子进行说明:

void mock_function(int a)
{
    check_expected(a);
}

static void test_check_parameter(void **state)
{
    expect_value(mock_function, a, 42);
    mock_function(42);
}

在测试用例 test_check_parameter 中,expect_value 宏为模拟函数 mock_function 的形参 a 设定了参数期望值42,紧接着调用模拟函数 mock_function 进行测试。而在模拟函数 mock_function 中,使用 check_expected 宏对输入参数 a 做了检查,检查结果有以下几种情况:

  1. 在CMocka框架内部的队列中,没有找到模拟函数 mock_function 参数a的预设期望值,则将测试标记为失败。没有调用 expect_*() 就是这种情况。

  2. 找到模拟函数 mock_function 参数 a 的预设期望值(例子中期望值为42),那就跟拿输入参数的实际值和期望值进行对比,相同则测试成功,不同则测试失败。例子中测试用例调用 mock_function 模拟函数传入是42,所以测试通过。

expect_value() 宏相关还有一个叫 expect_value_count 的宏:

expect_value_count (function, parameter, value, count)

该宏干么用的呢?就某个模拟函数的某个参数而言,执行 expect_*() 宏的次数和执行 check_expected() 宏的次数要一致,必须成对被执行,否则 CMocka 框架会将测试用例标记为失败。在上面的例子中,如果测试用例要多次执行模拟函数 mock_function(内部执行 check_expected() 宏),就要预先多次执行 expect_value() 宏,使用 expect_value_count() 宏可以简化这个操作,其中 count 指定运行 check_expected() 的次数。如果 count 设置为-1,表明可以无数次执行 check_expected(),但至少得执行一次。如果 count 设置为-2,表明可以无数次执行 check_expected(),甚至一次都不执行也可以。

expect_value() 宏相关还有一个叫 expect_not_value 的宏:

expect_not_value (function, parameter, value)

该宏干么用的呢?expect_value() 宏用于期望参数等于某个值,而 expect_not_value 宏用于期望参数不等于某个值。

官网API手册中的 Checking Parameters 模块中列出了 expect_*() 宏家族所有接口,它们的工作原理跟 expect_value 是一样的。

expect_*() Macros Description
expect_check (function, parameter, check_function, check_data) 用于自定义
expect_in_set (function, parameter, value_array[])
expect_in_set_count (function, parameter, value_array[], count)
expect_not_in_set (function, parameter, value_array[])
expect_not_in_set_count (function, parameter, value_array[], count)
验证参数值(整形)是否在期望值数组value_array[]之内(或之外)
expect_in_range (function, parameter, minimum, maximum)
expect_in_range_count (function, parameter, minimum, maximum, count)
expect_not_in_range (function, parameter, minimum, maximum)
expect_not_in_range_count (function, parameter, minimum, maximum, count)
验证参数值(整形)是否在期望值[minimum , maximum]范围之内(或之外)
expect_value (function, parameter, value)
expect_value_count (function, parameter, value, count)
expect_not_value (function, parameter, value)
expect_not_value_count (function, parameter, value, count)
验证参数值(整形)是否等于(或不等于)期望值value
expect_string (function, parameter, string)
expect_string_count (function, parameter, string, count)
expect_not_string (function, parameter, string)
expect_not_string_count (function, parameter, string, count)
验证参数值(字符串)是否等于(或不等于)期望值string
expect_memory (function, parameter, memory, size)
expect_memory_count (function, parameter, memory, size, count)
expect_not_memory (function, parameter, memory, size)
expect_not_memory_count (function, parameter, memory, size, count)
验证参数值(内存块)是否等于(或不等于)期望值内存块memory
expect_any (function, parameter)
expect_any_count (function, parameter, count)
expect_any_always (function, parameter)
不管参数值是什么,都判定通过。
expect_any_always就是expect_any_count (function, parameter, -1)
check_expected (parameter)
check_expected_ptr (parameter)
check_expected用于检查模拟函数的参数值是否跟期望值一致,check_expected_ptr用于检查指针是否跟期望的一致。

5.2.4. 检查模拟函数调用顺序

官方API接口列表:见官网API手册中的 Call Ordering 模块。

对模拟函数的检查,除了检查模拟函数的输入参数和返回值之外,有时候也会检查模拟函数的调用顺序。为了简化实现,CMocka 框架内部维护了一个先进先出(FIFO)的「期望调用函数名队列」。

  • expect_function_call(function)

    该宏用于设定模拟函数的期望调用顺序,一般是在测试用例中调用该宏。通过该宏可以往队列中 push 期望调用的模拟函数名 function,多次调用该宏的顺序代表了期望调用模拟函数的顺序。

  • function_called()

    该宏用于验证模拟函数调用顺序是否符合期望顺序,它只能在模拟函数中调用。该宏从队列中 pop 出期望调用的模拟函数名,并与当前调用 function_called() 的函数名进行对比,不一致就将测试用例标记为失败。

static void mock_test_a_called(void)
{
    function_called();
}

static void mock_test_b_called(void)
{
    function_called();
}

static void test_does_succeed_for_expected(void **state)
{
    (void)state;
    expect_function_call(mock_test_a_called);
    expect_function_call(mock_test_b_called);

    mock_test_a_called();
    mock_test_b_called();
}

通常情况下 expect_function_call()function_called() 是成对出现的,否则会导致测试用例失败。特殊情况可以通过 ignore_function_calls() 绕过该限制。ignore_function_calls() 用于忽略某个期望函数的调用。

static void test_ordering_does_ignore_calls(void **state)
{
(void)state;
    expect_function_call(mock_test_a_called);
    ignore_function_calls(mock_test_b_called);

    mock_test_a_called();
    mock_test_b_called();
    mock_test_b_called();
}

源码 cmocka/tests/test_ordering.ccmocka/tests/test_ordering_fail.c 演示了有关 Call Ordering 的更多用法,可自行查阅。

5.3. 动态内存检测

官方API接口列表:见官网API手册中的 Dynamic Memory Allocation 模块。

CMocka 框架提供了动态内存检测功能,用于测试待测模块的内存溢出(越界访问)、内存泄漏(未释放内存)问题。CMocka 框架会跟踪所有使用 test_*() 接口分配的内存块,每次使用 test_free() 释放内存块,都会检查内存块是否有内存溢出,一旦发现就会将测试用例标记为失败,并终止当前测试用例。当测试用例执行完成时,CMocka 框架内部会检测是否有内存块还未释放(内存泄漏),一旦发现也会将测试用例标记为失败。

因此,你需要做的就是将待测模块代码中的 mallocrealloccallocfree 分别替换成CMocka框架提供的 test_malloctest_realloctest_calloctest_free 接口。可以使用以下代码替换:

#ifdef UNIT_TESTING
#define malloc test_malloc
#define realloc test_realloc
#define calloc test_calloc
#define free test_free
#endif

编译测试用例时(包括编译待测模块代码),开启 UNIT_TESTING 宏定义,动态内存分配和释放就会使用 CMocka 框架的 test_*() 接口。编译产品发布版本是,关闭 UNIT_TESTING 宏定义,动态内存分配和释放就会使用默认的接口。举个例子:

#include 
#include 
#include 
#include 
#include 
#include 

#define UNIT_TESTING

#ifdef UNIT_TESTING
#define malloc test_malloc
#define realloc test_realloc
#define calloc test_calloc
#define free test_free
#endif

void leak_memory(void **state) {
    int * const temporary = (int*)malloc(sizeof(int));
    *temporary = 0;
}

void buffer_overflow(void **state) {
    char * const memory = (char*)malloc(sizeof(int));
    memory[sizeof(int)] = '!';
    free(memory);
}

void buffer_underflow(void **state) {
    char * const memory = (char*)malloc(sizeof(int));
    memory[-1] = '!';
    free(memory);
}

int main(void) {
    const struct CMUnitTest tests[] = {
        cmocka_unit_test(leak_memory),
        cmocka_unit_test(buffer_overflow),
        cmocka_unit_test(buffer_underflow),
    };

    return cmocka_run_group_tests(tests, NULL, NULL);
}

5.4. 异常检测

程序在执行过程中,如果遇到异常,系统会按默认方式处理系统,很经常的就是终止(杀死)进程,这种情况很难定位问题出在哪里,无法快速定位哪些函数引起的异常。

CMocka 框架为了解决该问题,在执行测试用例之前,CMocka 框架内部会先覆盖默认的异常/信号处理器,在拦截到异常信号后,只是打印出错误信息并退出当前测试用例,不会终止进程。以关键字 signal 全字匹配搜索 cmocka/src/cmocka.c 源码就能看出端倪。

举个例子:

static void null_test_success(void **state) {
    char *ch = NULL;
    *ch = 'Y';
}

int main(void) {
    const struct CMUnitTest tests[] = {
        cmocka_unit_test(null_test_success),
    };

    return cmocka_run_group_tests(tests, NULL, NULL);
}

执行结果如下:

[==========] Running 1 test(s).
[ RUN      ] test_segfault
[  ERROR   ] --- Test failed with exception: Segmentation fault(11)
[  FAILED  ] test_segfault
[==========] 1 test(s) run.
[  PASSED  ] 0 test(s).
[  FAILED  ] 1 test(s), listed below:
[  FAILED  ] test_segfault

 1 FAILED TEST(S)

源码 cmocka/tests/test_exception_handler.c 演示了有关异常检测的更多用法,可自行查阅。

5.5. 输出格式

CMocka框架支持多种格式输出 (STDOUT, SUBUNIT, TAP, XML),默认是输出到STDOUT,有两种方式可以设置输出格式。

  • 通过 cmocka_set_message_output 函数接口进行配置。

    enum cm_message_output {
        CM_OUTPUT_STDOUT,
        CM_OUTPUT_SUBUNIT,
        CM_OUTPUT_TAP,
        CM_OUTPUT_XML,
    };
    void cmocka_set_message_output(enum cm_message_output output)
    

    示例代码:

    int main(void) {
        const struct CMUnitTest tests[] = {
            cmocka_unit_test(null_test_success),
        };
    
        cmocka_set_message_output(CM_OUTPUT_SUBUNIT);
        return cmocka_run_group_tests(tests, NULL, NULL);
    }
    
  • 通过环境变量 CMOCKA_MESSAGE_OUTPUT 配置,环境变量可以设置为STDOUT、SUBUIT、TAP或XML。

    export CMOCKA_MESSAGE_OUTPUT=XML
    

    注意:如果同时使用了以上两种方式设置输出格式,环境变量 CMOCKA_MESSAGE_OUTPUT 配置优先级更高。

    默认情况下,输出格式设置成XML,输出内容也是打印到标准错误输出(stderr)。可以通过环境变量 CMOCKA_XML_FILE 将输出内容重定向到文件中,如果有多个用户组,可以将CMOCKA_XML_FILE设置为:

    CMOCKA_XML_FILE=/path/cm_%g.xml
    

    此时%g将被测试的group_name替换,并为每个用户组创建一个文件,否则所有用户组都将打印到同一文件中。注意:环境变量 CMOCKA_XML_FILE 只能针对XML格式,对其他格式不起作用。

5.6. 测试用例过滤器

CMocka 框架还提供了两个测试用例过滤器,用于在开始执行测试之前,过滤掉不想运行的测试用例:

void cmocka_set_test_filter(const char *pattern); /* 只运行匹配的测试用例 */
void cmocka_set_skip_filter(const char *pattern); /* 不运行匹配的测试用例 */

pattern 参数支持两个通配符,一个是 *,表示匹配零个或多个字符,另一个是 ?,表示匹配一个字符。举个例子:

int main(void) {
    const struct CMUnitTest tests[] = {
        cmocka_unit_test(test_skip1),
        cmocka_unit_test(test_skip2),
        cmocka_unit_test(test_fail),
    };

    cmocka_set_test_filter("test_skip*");             /* 只运行匹配test_skip*的测试用例 */
    cmocka_set_skip_filter("test_skip2");             /* 不运行匹配test_skip2的测试用例 */

    return cmocka_run_group_tests(tests, NULL, NULL); /* 综上,最后只有test_skip1会被执行 */
}

源码 cmocka/tests/test_skip_filter.ccmocka/tests/test_wildcard.c 演示了 cmocka_set_test_filter()cmocka_set_skip_filter() 的用法,可自行查阅。

6. 参考资料

撰写本文时用到的 CMocka 版本是 cmocka-1.1.5,参考了以下资料:

参考资料 说明
源码 cmocka/doc/index.html 附带的文档手册
源码 cmocka/example/ 示例代码
源码 cmocka/tests/ 示例代码
https://api.cmocka.org 在线官网手册

你可能感兴趣的:(单元测试)