fwknop/test/c-unit-tests

Build

C unit library is used to perform c unit testing. Source code associated to those tests must be started with

#ifdef HAVE_C_UNIT_TESTS

and closed with:

#endif /* HAVE_C_UNIT_TESTS */

In order to build the test suite use the following commands with the --enable-c-unit-tests switch

$ ./autogen.sh
$ ./configure --enable-c-unit-tests --prefix=/usr --sysconfdir=/etc --enable-profile-coverage
$ make
$ ./test-fwknop.pl --enable-profile-coverage-check --loopback lo --client-only-mode

Run test suites

Once the build is complete, the three test programs (each in their respective directories) allow the user to run the tests suites:

  • fwknopd_utests: program to run fwknopd c unit test suites (located in the server directory)
  • fwknop_utests: program to run fwknop c unit test suites (located in the client directory)
  • fko_utests: program to run fko c unit test suites (located in the lib directory)
$ ./server/fwknopd_utests

Suite: Access test suite
  Test: check compare_port_list function ...FAILED
    1. access.c:2120  - compare_port_list(acc_pl, in1_pl, 0) == 0

Run Summary:    Type  Total    Ran Passed Failed Inactive
              suites      1      1    n/a      0        0
               tests      1      1      0      1        0
             asserts      6      6      5      1      n/a

Elapsed time =    0.000 seconds
$ ./client/fwknop_utests

Suite: Config init test suite
  Test: Check critcial vars ...passed
  Test: Check var_bitmask functions ...passed

Run Summary:    Type  Total    Ran Passed Failed Inactive
              suites      1      1    n/a      0        0
               tests      2      2      2      0        0
             asserts     12     12     12      0      n/a

Elapsed time =    0.000 seconds
$ ./lib/fko_utests

Suite: FKO decode test suite
  Test: Count the number of SPA fields in a SPA packet ...passed
  Test: Count the number of bytes to the last : ...passed

Run Summary:    Type  Total    Ran Passed Failed Inactive
              suites      1      1    n/a      0        0
               tests      2      2      2      0        0
             asserts     19     19     19      0      n/a

Elapsed time =    0.000 seconds

Manage C unit tests

C unit tests are implemented in source files and registered in a test suite. All test suites are then added to fwknopd, fwknop or fko test programs

In order to add new tests, the user must follow the below steps:

  • Declare a test suite
  • Declare an initialization function
  • Declare a clean up function
  • Create one or more unit tests
  • Create a function to register new tests

Declare a test suite

In source file:

 #ifdef HAVE_C_UNIT_TESTS
 DECLARE_TEST_SUITE(access, "Access test suite");
 #endif

In the above example, we create a test suite using the DECLARE_TEST_SUITE macro:

  • the test suite is named "access".
  • the test suite description is "Access test suite" and is displayed on the console when the test program is executed

Declare an initialization function

Before running the test suite, an init function can be used to initialize the test suite context. To declare such a function use the DECLARE_TEST_SUITE_INIT macro.

In source file:

DECLARE_TEST_SUITE_INIT(access)
{
    log_set_verbosity(LOG_VERBOSITY_ERROR);
    return 0;
}

In the above example, the log message verbosity is decreased to error level to only display error messages since debug messages are too verbose.

In some cases, there is no need for such a function and thus this declaration is not mandatory.

Declare a clean-up function

In order to clean up the context at the end of the test suite, it is possible to declare a clean up function with the DECLARE_TEST_SUITE_CLEANUP macro

In source file:

DECLARE_TEST_SUITE_CLEANUP(access)
{
    return 0;
}

In the above example, the clean up function returns 0 and does strictly nothing.

In some cases, there is no need for such function and thus this declaration is not mandatory.

Create unit tests

In source file:

#ifdef HAVE_C_UNIT_TESTS

DECLARE_UTEST(compare_port_list, "check compare_port_list function")
{
    acc_port_list_t *in1_pl = NULL;
    acc_port_list_t *in2_pl = NULL;
    acc_port_list_t *acc_pl = NULL;

    /* Match any test */
    free_acc_port_list(in1_pl);
    free_acc_port_list(acc_pl);
    add_port_list_ent(&in1_pl, "udp/6002");
    add_port_list_ent(&in2_pl, "udp/6002, udp/6003");
    add_port_list_ent(&acc_pl, "udp/6002, udp/6003");
    CU_ASSERT(compare_port_list(in1_pl, acc_pl, 1) == 1);
    CU_ASSERT(compare_port_list(in2_pl, acc_pl, 1) == 1);
    CU_ASSERT(compare_port_list(in1_pl, acc_pl, 0) == 1);
    CU_ASSERT(compare_port_list(in2_pl, acc_pl, 0) == 1);
    CU_ASSERT(compare_port_list(acc_pl, in1_pl, 0) == 0);
    CU_ASSERT(compare_port_list(acc_pl, in2_pl, 0) == 1);
}

#endif /* HAVE_C_UNIT_TESTS */

In the above example, we create a c-unit test using the DECLARE_UTEST macro:

  • The unit test is named "compare_port_list" ; This id must be unique
  • The unit test description is "check compare_port_list function" and is displayed on the console when the test program is executed

Create a function to register new tests

We have previously declared unit tests, but they have to be registered to a test suite to be executed.

In source file:

#ifdef HAVE_C_UNIT_TESTS

int register_ts_access(void)
{
    ts_init(&TEST_SUITE(access), TEST_SUITE_DESCR(access), TEST_SUITE_INIT(access), TEST_SUITE_CLEANUP(access));
    ts_add_utest(&TEST_SUITE(access), UTEST_FCT(compare_port_list), UTEST_DESCR(compare_port_list));

    return register_ts(&TEST_SUITE(access));
}

#endif /* HAVE_C_UNIT_TESTS */

If no init or cleanup function is defined, they have to be replaced by a NULL pointer at test suite initialization : ts_init

Each unit test must be added using ts_add_utest function.

In header file, add the register function prototype as follows:

#ifdef HAVE_C_UNIT_TESTS
int register_ts_access(void);
#endif

In the unit test program, add the test suite to the current list of existing test suite.

static void register_test_suites(void)
{
        register_ts_access();
}

Check gcov coverage