Dead simple unit testing for C
To use CCheck, your project needs to provide three components:
- Subjects, shared objects that implement the interface to be tested
- Providers, subroutines that produce typed data to be fed into tests
- Tests, subroutines that check compliance of an interface
Usually, a C project already produces subjects. You then need to write unit tests, and providers for any custom data types in those tests. These are compiled as normal shared objects, where one object may contain any number of providers or tests.
ccheck [subjects...] -- [providers/tests...]
Every argument names a shared object.
Unless a path explicitly contains /
, it is searched on the standard system include path.
Providers are executed sequentially before all tests. Individual test objects are run in parallel, and the tests inside each object are run sequentially.
You can build ccheck with a make rule
ccheck/ccheck ccheck/integer-provider.so ccheck/interface.h:
if [ ! -d ccheck ]; then git clone https://github.com/loglob/ccheck; fi
make -C ccheck
Then invoke the tests with a rule like
.PHONY: test --
test: ccheck/ccheck out/my-lib.so -- ccheck/integer-prodiver.so $(MY_PROVIDERS) $(MY_TESTS)
./$^
All tests must include interface.h
, which provides the TEST()
macro.
This macro is used to mark a function definition as a unit test, like this:
TEST(myTest, uint32_t, x, uint32_t, y, struct Foo, z)
{ /* ... */ }
This wraps a function called myTest
, accepting the arguments x
, y
and z
that doesn't return a value.
To indicate failure, use the testFailure()
function, which accepts a printf-style formatted string with parameters describing the reason the test case was rejected.
For convenience, assertTrue()
is macro for testFailure()
that ensures its argument is true and generates corresponding error messages.
When the function returns without calling testFailure()
the test run is considered successful.
Calls to exit()
and assert()
failures in test code are also caught and considered failures.
The exit syscall itself cannot be caught so it may cause false positives in very specific situations.
Providers must include interface.h
, which provides the PROVIDER()
macro.
This macro is used to create the interface for a provider:
PROVIDER((struct, Foo), randomizedFoo)
Note: For this macro, multi-word type names (such as
struct Foo
) must be given as parenthesized, comma-separated lists
This invocation creates two function stubs like
/**
@param data An array to write the test values to
@return The number of elements provided. May be larger than cap, in which case the function is called again with larger capacity.
*/
size_t randomizedFoo(size_t cap, struct Foo data[restrict static cap]);
/**
@param to A string buffer to write to
@param n The capacity of `to`, including the NUL terminator (exactly like `snprintf`)
@param data A pointer to the value to print
@return The number of characters written, or that would have been written if there was not enough space in `to` (exactly like `snprintf`)
*/
size_t format_struct_Foo(char *to, size_t n, const struct Foo data[restrict static 1]);
This repo also contains a provider for integer types.
It provides the types uint*_t
and int*_t
from inttypes.h
, excluding uint8_t
and int8_t
.
Note: This uses most 3/4 of available bits so that addition definitely won't overflow