Tarantool development patches archive
 help / color / mirror / Atom feed
From: Sergey Bronnikov via Tarantool-patches <tarantool-patches@dev.tarantool.org>
To: Sergey Kaplun <skaplun@tarantool.org>,
	Igor Munkin <imun@tarantool.org>,
	Maxim Kokryashkin <m.kokryashkin@tarantool.org>
Cc: tarantool-patches@dev.tarantool.org
Subject: Re: [Tarantool-patches] [PATCH v3 luajit 2/6] test: introduce module for C tests
Date: Mon, 5 Jun 2023 18:08:30 +0300	[thread overview]
Message-ID: <1fcbd8cc-44ab-484f-1cd5-715701282049@tarantool.org> (raw)
In-Reply-To: <97194cc5150bc632c5fd566c9f34033413af1518.1685613304.git.skaplun@tarantool.org>

Sergey,

thanks for patches and fixes. See my comments below.

On 6/5/23 13:41, Sergey Kaplun wrote:
> We need an instrument to write tests in plain C for LuaJIT, to be able:
> * easily test LuaC API
> * test patches without usage of plain Lua
> * write unit tests
> * startup LuaJIT with custom memory allocator, to test some GC issues
> * maybe, in future, use custom hashing function to test a behavior
>    of LuaJIT tables and so on.
>
> The <test.c> module serves to achieve these goals without too fancy
> features.
>
> It's functionality inspired by CMocka API [1], but only TAP14 [2]
> protocol is supported (Version of TAP set to 13 to be compatible with
> old TAP13 harnesses).
>
> The group of unit tests is declared like the following:
>
> | void *t_state = NULL;
> | const struct test_unit tgroup[] = {
> | 	test_unit_new(test_base),

As I proposed in discussion verbally, I suggest to remove postfix "_new" 
because

we have opposite function with destructor ("test_unit_delete").


> | 	test_unit_new(test_subtest),
> | };
> | return test_run_group(tgroup, t_state);
>
> `test_run_group()` runs the whole group of tests, returns
> `TEST_EXIT_SUCCESS` or `TEST_EXIT_FAILURE`.
>
> If a similar group is declared inside unit test, this group will be
> considered as a subtest.
>
> This library provides an API similar to glibc (3) `assert()` to use
> inside unit tests. `assert_[true,false]()` are useful for condition
> checks and `assert_{type}_[not_,]_equal()` are useful for value
> comparisons. If some assertion fails diagnostic is set, all test
> considered as failing and finished via `longjmp()`, so these assertions
> can be used inside custom subroutines.
>
> Also, this module provides ability to skip one test or all tests, mark
> test as todo, bail out all tests. Just use `return skip()`, `skip_all()`
> or `todo()` for early return. They should be used only in the test body
> to make skipping clear. `skip_all()` may be used both for the parent
> test and for a subtest. `bail_out()` prints an error message and exits
> the process.

Nit: replace skip_all() with skip_all(<reason>), bail_out() with 
bail_out(<reason>), todo() with todo(<reason>)

and skip() with skip(<reason>).

I would describe these functions in a test/tarantool-c-tests/README.md too.

BTW, why README.md was added in commit "test: introduce utils.h helper 
for C tests" and not in this patch?

I believe this patch, where test lib added, is more appropriate place 
for API documentation.

>
> As a part of this commit, tarantool-c-tests directory is created with
> the corresponding CMakeLists.txt file to build this test library.
> Tests to be rewritten in C with this library in the next commit and
> placed as unit tests are:
> * lj-49-bad-lightuserdata.test.lua
> * misclib-getmetrics-capi.test.lua
> * misclib-sysprof-capi.test.lua
>
> For now the tarantool-c-tests target just build the test library without
> new tests to run.
>
> The library itself is tested via some primitive tests for `ok` case,
> `skip` and `todo` directives. The TAP13 format is tested via prove, that
> we are using for running our tests. TAP14 format is compatible with
> TAP13, so there are no other tests required.
>
> Also, .c_test suffix is added to the <.gitignore>.
>
> [1]: https://github.com/clibs/cmocka
> [2]: https://testanything.org/tap-version-14-specification.html
>
> Part of tarantool/tarantool#7900
> ---
>   .gitignore                             |   1 +
>   test/CMakeLists.txt                    |   2 +
>   test/tarantool-c-tests/CMakeLists.txt  |  66 +++++++
>   test/tarantool-c-tests/test.c          | 251 +++++++++++++++++++++++++
>   test/tarantool-c-tests/test.h          | 217 +++++++++++++++++++++
>   test/tarantool-c-tests/unit-tap.test.c |  31 +++
>   6 files changed, 568 insertions(+)
>   create mode 100644 test/tarantool-c-tests/CMakeLists.txt
>   create mode 100644 test/tarantool-c-tests/test.c
>   create mode 100644 test/tarantool-c-tests/test.h
>   create mode 100644 test/tarantool-c-tests/unit-tap.test.c
>
> diff --git a/.gitignore b/.gitignore
> index b7908aee..dc5ea5fc 100644
> --- a/.gitignore
> +++ b/.gitignore
> @@ -24,3 +24,4 @@ install_manifest.txt
>   luajit-parse-memprof
>   luajit-parse-sysprof
>   luajit.pc
> +*.c_test
> diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt
> index a8262b12..47296a22 100644
> --- a/test/CMakeLists.txt
> +++ b/test/CMakeLists.txt
> @@ -48,12 +48,14 @@ separate_arguments(LUAJIT_TEST_COMMAND)
>   add_subdirectory(LuaJIT-tests)
>   add_subdirectory(PUC-Rio-Lua-5.1-tests)
>   add_subdirectory(lua-Harness-tests)
> +add_subdirectory(tarantool-c-tests)
>   add_subdirectory(tarantool-tests)
>   
>   add_custom_target(${PROJECT_NAME}-test DEPENDS
>     LuaJIT-tests
>     PUC-Rio-Lua-5.1-tests
>     lua-Harness-tests
> +  tarantool-c-tests
>     tarantool-tests
>   )
>   
> diff --git a/test/tarantool-c-tests/CMakeLists.txt b/test/tarantool-c-tests/CMakeLists.txt
> new file mode 100644
> index 00000000..da128457
> --- /dev/null
> +++ b/test/tarantool-c-tests/CMakeLists.txt
> @@ -0,0 +1,66 @@
> +find_program(PROVE prove)
> +if(NOT PROVE)
> +  message(WARNING "`prove' is not found, so tarantool-c-tests target is not generated")
> +  return()
> +endif()
> +
> +set(C_TEST_SUFFIX .c_test)
> +set(C_TEST_FLAGS --failures --shuffle)
> +
> +if(CMAKE_VERBOSE_MAKEFILE)
> +  list(APPEND C_TEST_FLAGS --verbose)
> +endif()
> +
> +# Build libtest.
> +
> +set(TEST_LIB_NAME "test")
> +add_library(libtest STATIC EXCLUDE_FROM_ALL ${CMAKE_CURRENT_SOURCE_DIR}/test.c)
> +target_include_directories(libtest PRIVATE ${CMAKE_CURRENT_SOURCE_DIR})
> +set_target_properties(libtest PROPERTIES
> +  COMPILE_FLAGS "-Wall -Wextra"
> +  OUTPUT_NAME "${TEST_LIB_NAME}"
> +  LIBRARY_OUTPUT_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}"
> +)
> +
> +# TARGET_C_FLAGS is required here to be sure that headers like
> +# lj_arch.h in compiled test are consistent with the LuaJIT library
> +# to link.
> +AppendFlags(TESTS_C_FLAGS ${TARGET_C_FLAGS})
> +
> +set(CTEST_SRC_SUFFIX ".test.c")
> +file(GLOB tests "${CMAKE_CURRENT_SOURCE_DIR}/*${CTEST_SRC_SUFFIX}")
> +foreach(test_source ${tests})
> +  # Get test name without suffix. Needed to set OUTPUT_NAME.
> +  get_filename_component(exe ${test_source} NAME_WE)
> +  add_executable(${exe} EXCLUDE_FROM_ALL ${test_source})
> +  target_include_directories(${exe} PRIVATE
> +    ${CMAKE_CURRENT_SOURCE_DIR}
> +    ${LUAJIT_SOURCE_DIR}
> +  )
> +  set_target_properties(${exe} PROPERTIES
> +    COMPILE_FLAGS "${TESTS_C_FLAGS}"
> +    OUTPUT_NAME "${exe}${C_TEST_SUFFIX}"
> +    RUNTIME_OUTPUT_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}"
> +  )
> +  target_link_libraries(${exe} libtest ${LUAJIT_LIBRARY})
> +  LIST(APPEND TESTS_COMPILED ${exe})
> +endforeach()
> +
> +add_custom_target(tarantool-c-tests
> +  DEPENDS libluajit libtest ${TESTS_COMPILED}
> +)
> +
> +add_custom_command(TARGET tarantool-c-tests
> +  COMMENT "Running Tarantool C tests"
> +  COMMAND
> +  ${PROVE}
> +    ${CMAKE_CURRENT_BINARY_DIR}
> +    --ext ${C_TEST_SUFFIX}
> +    --jobs ${CMAKE_BUILD_PARALLEL_LEVEL}
> +    # Report any TAP parse errors, if any, since test module is
> +    # maintained by us.
> +    --parse
> +    ${C_TEST_FLAGS}
> +  WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}
> +)
> +
> diff --git a/test/tarantool-c-tests/test.c b/test/tarantool-c-tests/test.c
> new file mode 100644
> index 00000000..74cba3a3
> --- /dev/null
> +++ b/test/tarantool-c-tests/test.c
> @@ -0,0 +1,251 @@
> +#include "test.h"
> +
> +/*
> + * Test module, based on TAP 14 specification [1].
> + * [1]: https://testanything.org/tap-version-14-specification.html
> + */
> +
> +/* Need for `PATH_MAX` in diagnostic definition. */
> +#include <limits.h>
> +#include <setjmp.h>
> +#include <stdarg.h>
> +/* Need for `strchr()` in diagnostic parsing. */
> +#include <string.h>
> +
> +/*
> + * Test level: 0 for the parent test, >0 for any subtests.
> + */
> +static int level = -1;
> +
> +/*
> + * The last diagnostic data to be used in the YAML Diagnostic
> + * block.
> + *
> + * Contains filename, line number and failed expression or assert
> + * name and "got" and "expected" fields. All entries are separated
> + * by \n.
> + * The longest field is filename here, so PATH_MAX * 3 as
> + * the diagnostic string length should be enough.
> + *
> + * The first \0 means the end of diagnostic data.
> + *
> + * As far as `strchr()` searches until \0, all previous entries
> + * are suppressed by the last one. If the first byte is \0 --
> + * diagnostic is empty.
> + */
> +#define TEST_DIAG_DATA_MAX (PATH_MAX * 3)
> +char test_diag_buf[TEST_DIAG_DATA_MAX] = {0};
> +
> +const char *skip_reason = NULL;
> +const char *todo_reason = NULL;
> +
> +/* Indent for the TAP. 4 spaces is default for subtest. */
> +static void indent(void)
> +{
> +	int i;
> +	for (i = 0; i < level; i++)
> +		printf("    ");
> +}
> +
> +void test_message(const char *fmt, ...)
> +{
> +	va_list ap;
> +	indent();
> +	va_start(ap, fmt);
> +	vprintf(fmt, ap);
> +	printf("\n");
> +	va_end(ap);
> +}
> +
> +static void test_print_tap_version(void)
> +{
> +	/*
> +	 * Since several TAP13 parsers in popular usage treat
> +	 * a repeated Version declaration as an error, even if the
> +	 * Version is indented, Subtests _should not_ include a
> +	 * Version, if TAP13 Harness compatibility is
> +	 * desirable [1].
> +	 */
> +	if (level == 0)
> +		test_message("TAP version %d", TAP_VERSION);
> +}
> +
> +static void test_start_comment(const char *t_name)
> +{
> +	if (level > -1)
> +		/*
> +		 * Inform about starting subtest, easier for
> +		 * humans to read.
> +		 * Subtest with a name must be terminated by a
> +		 * Test Point with a matching Description [1].
> +		 */
> +		test_comment("Subtest: %s", t_name);
> +}
> +
> +void _test_print_skip_all(const char *group_name, const char *reason)
> +{
> +	test_start_comment(group_name);
> +	/*
> +	 * XXX: This test isn't started yet, so set indent level
> +	 * manually.
> +	 */
> +	level++;
> +	test_print_tap_version();
> +	/*
> +	 * XXX: `SKIP_DIRECTIVE` is not necessary here according
> +	 * to the TAP14 specification [1], but some harnesses may
> +	 * fail to parse the output without it.
> +	 */
> +	test_message("1..0" SKIP_DIRECTIVE "%s", reason);
> +	level--;
> +}
> +
> +/* Just inform TAP parser how many tests we want to run. */
> +static void test_plan(size_t planned)
> +{
> +	test_message("1..%lu", planned);
> +}
> +
> +/* Human-readable output how many tests/subtests are failed. */
> +static void test_finish(size_t planned, size_t failed)
> +{
> +	const char *t_type = level == 0 ? "tests" : "subtests";
> +	if (failed > 0)
> +		test_comment("Failed %lu %s out of %lu",
> +		     failed, t_type, planned);
> +	fflush(stdout);
> +}
> +
> +void test_set_skip_reason(const char *reason)
> +{
> +	skip_reason = reason;
> +}
> +
> +void test_set_todo_reason(const char *reason)
> +{
> +	todo_reason = reason;
> +}
> +
> +void test_save_diag_data(const char *fmt, ...)
> +{
> +	va_list ap;
> +	va_start(ap, fmt);
> +	vsnprintf(test_diag_buf, TEST_DIAG_DATA_MAX, fmt, ap);
> +	va_end(ap);
> +}
> +
> +static void test_clear_diag_data(void)
> +{
> +	/*
> +	 * Limit buffer with zero byte to show that there is no
> +	 * any entry.
> +	 */
> +	test_diag_buf[0] = '\0';
> +}
> +
> +static int test_diagnostic_is_set(void)
> +{
> +	return test_diag_buf[0] != '\0';
> +}
> +
> +/*
> + * Parse the last diagnostic data entry and print it in YAML
> + * format with the corresponding additional half-indent in TAP
> + * (2 spaces).
> + * Clear diagnostic message to be sure that it's printed once.
> + * XXX: \n separators are changed to \0 during parsing and
> + * printing output for convenience in usage.
> + */
> +static void test_diagnostic(void)
> +{
> +	test_message("  ---");
> +	char *ent = test_diag_buf;
> +	char *ent_end = NULL;
> +	while ((ent_end = strchr(ent, '\n')) != NULL) {
> +		char *next_ent = ent_end + 1;
> +		/*
> +		 * Limit string with with the zero byte for
> +		 * formatted output. Anyway, don't need this \n
> +		 * anymore.
> +		 */
> +		*ent_end = '\0';
> +		test_message("  %s", ent);
> +		ent = next_ent;
> +	}
> +	test_message("  ...");
> +	test_clear_diag_data();
> +}
> +
> +static jmp_buf test_run_env;
> +
> +TEST_NORET void _test_exit(int status)
> +{
> +	longjmp(test_run_env, status);
> +}
> +
> +static int test_run(const struct test_unit *test, size_t test_number,
> +		    void *test_state)
> +{
> +	int status = TEST_EXIT_SUCCESS;
> +	/*
> +	 * Run unit test. Diagnostic in case of failure setup by
> +	 * helpers assert macros defined in the header.
> +	 */
> +	int jmp_status;
> +	if ((jmp_status = setjmp(test_run_env)) == 0) {
> +		if (test->f(test_state) != TEST_EXIT_SUCCESS)
> +			status = TEST_EXIT_FAILURE;
> +	} else {
> +		status = jmp_status - TEST_JMP_STATUS_SHIFT;
> +	}
> +	const char *result = status == TEST_EXIT_SUCCESS ? "ok" : "not ok";
> +
> +	/*
> +	 * Format suffix of the test message for SKIP or TODO
> +	 * directives.
> +	 */
> +#define SUFFIX_SZ 1024
> +	char suffix[SUFFIX_SZ] = {0};
> +	if (skip_reason) {
> +		snprintf(suffix, SUFFIX_SZ, SKIP_DIRECTIVE "%s", skip_reason);
> +		skip_reason = NULL;
> +	} else if (todo_reason) {
> +		/* Prevent count this test as failed. */
> +		status = TEST_EXIT_SUCCESS;
> +		snprintf(suffix, SUFFIX_SZ, TODO_DIRECTIVE "%s", todo_reason);
> +		todo_reason = NULL;
> +	}
> +#undef SUFFIX_SZ
> +
> +	test_message("%s %lu - %s%s", result, test_number, test->name,
> +		     suffix);
> +
> +	if (status && test_diagnostic_is_set())
> +		test_diagnostic();
> +	return status;
> +}
> +
> +int _test_run_group(const char *group_name, const struct test_unit tests[],
> +		    size_t n_tests, void *test_state)
> +{
> +	test_start_comment(group_name);
> +
> +	level++;
> +	test_print_tap_version();
> +
> +	test_plan(n_tests);
> +
> +	size_t n_failed = 0;
> +
> +	size_t i;
> +	for (i = 0; i < n_tests; i++) {
> +		size_t test_number = i + 1;
> +		/* Return 1 on failure, 0 on success. */
> +		n_failed += test_run(&tests[i], test_number, test_state);
> +	}
> +
> +	test_finish(n_tests, n_failed);
> +
> +	level--;
> +	return n_failed > 0 ? TEST_EXIT_FAILURE : TEST_EXIT_SUCCESS;
> +}
> diff --git a/test/tarantool-c-tests/test.h b/test/tarantool-c-tests/test.h
> new file mode 100644
> index 00000000..047f01a5
> --- /dev/null
> +++ b/test/tarantool-c-tests/test.h
> @@ -0,0 +1,217 @@
> +#ifndef TARANTOOL_LUAJIT_TEST_H
> +#define TARANTOOL_LUAJIT_TEST_H
> +
> +#include <stdio.h>
> +#include <stdlib.h>
> +
> +/*
> + * Test module, based on TAP 14 specification [1].
> + * [1]: https://testanything.org/tap-version-14-specification.html
> + * Version 13 is set for better compatibility on old machines.
> + *
> + * TODO:
> + * * Helpers assert macros:
> + *   - assert_uint_equal if needed
> + *   - assert_uint_not_equal if needed
> + *   - assert_str_equal if needed
> + *   - assert_str_not_equal if needed
> + *   - assert_memory_equal if needed
> + *   - assert_memory_not_equal if needed
> + * * Pragmas.
> + */
> +
> +#define TAP_VERSION 13
> +
> +#define TEST_EXIT_SUCCESS 0
> +#define TEST_EXIT_FAILURE 1
> +
> +#define TEST_JMP_STATUS_SHIFT 2
> +#define TEST_LJMP_EXIT_SUCCESS (TEST_EXIT_SUCCESS + TEST_JMP_STATUS_SHIFT)
> +#define TEST_LJMP_EXIT_FAILURE (TEST_EXIT_FAILURE + TEST_JMP_STATUS_SHIFT)
> +
> +#define TEST_NORET __attribute__((noreturn))
> +
> +typedef int (*test_func)(void *test_state);
> +struct test_unit {
> +	const char *name;
> +	test_func f;
> +};
> +
> +/* API declaration. */
> +
> +/*
> + * Print formatted message with the corresponding indent.
> + * If you want to leave a comment, use `test_comment()` instead.
> + */
> +void test_message(const char *fmt, ...);
> +
> +/* Need for `skip_all()`, please, don't use it. */
> +void _test_print_skip_all(const char *group_name, const char *reason);
> +/* End test via `longjmp()`, please, don't use it. */
> +TEST_NORET void _test_exit(int status);
> +
> +void test_set_skip_reason(const char *reason);
> +void test_set_todo_reason(const char *reason);
> +/*
> + * Save formatted diagnostic data. Each entry separated with \n.
> + */
> +void test_save_diag_data(const char *fmt, ...);
> +
> +/* Internal, it is better to use `test_run_group()` instead. */
> +int _test_run_group(const char *group_name, const struct test_unit tests[],
> +		    size_t n_tests, void *test_state);
> +
> +/* Initialize `test_unit` structure. */
> +#define test_unit_new(f) {#f, f}
> +
> +#define lengthof(arr) (sizeof(arr) / sizeof((arr)[0]))
> +
> +/*
> + * __func__ is the name for a test group, "main" for the parent
> + * test.
> + */
> +#define test_run_group(t_arr, t_state) \
> +	_test_run_group(__func__, t_arr, lengthof(t_arr), t_state)
> +
> +#define SKIP_DIRECTIVE " # SKIP "
> +#define TODO_DIRECTIVE " # TODO "
> +
> +#define skip_all(reason) ({						\
> +	_test_print_skip_all(__func__, reason);				\
> +	TEST_EXIT_SUCCESS;						\
> +})
> +
> +static inline int skip(const char *reason)
> +{
> +	test_set_skip_reason(reason);
> +	return TEST_EXIT_SUCCESS;
> +}
> +
> +static inline int todo(const char *reason)
> +{
> +	test_set_todo_reason(reason);
> +	return TEST_EXIT_FAILURE;
> +}
> +
> +#define bail_out(reason) do {						\
> +	/*								\
> +	 * For backwards compatibility with TAP13 Harnesses,		\
> +	 * Producers _should_ emit a "Bail out!" line at the root	\
> +	 * indentation level whenever a Subtest bails out [1].		\
> +	 */								\
> +	printf("Bail out! %s\n", reason);				\
> +	exit(TEST_EXIT_FAILURE);					\
> +} while (0)
> +
> +/* `fmt` should always be a format string here. */
> +#define test_comment(fmt, ...) test_message("# " fmt, __VA_ARGS__)
> +
> +/*
> + * This is a set of useful assert macros like the standard C
> + * libary's assert(3) macro.
> + *
> + * On an assertion failure an assert macro will save the
> + * diagnostic to the special buffer, to be reported via YAML
> + * Diagnostic block and finish a test function with
> + * `return TEST_EXIT_FAILURE`.
> + *
> + * Due to limitations of the C language `assert_true()` and
> + * `assert_false()` macros can only display the expression that
> + * caused the assertion failure. Type specific assert macros,
> + * `assert_{type}_equal()` and `assert_{type}_not_equal()`, save
> + * the data that caused the assertion failure which increases data
> + * visibility aiding debugging of failing test cases.
> + */
> +
> +#define LOCATION_FMT "location:\t%s:%d\n"
> +#define ASSERT_NAME_FMT(name) "failed_assertion:\t" #name "\n"
> +#define ASSERT_EQUAL_FMT(name_type, type_fmt)				\
> +	LOCATION_FMT							\
> +	ASSERT_NAME_FMT(assert_ ## name_type ## _equal)			\
> +	"got: "      type_fmt "\n"					\
> +	"expected: " type_fmt "\n"
> +
> +#define ASSERT_NOT_EQUAL_FMT(type_fmt)					\
> +	LOCATION_FMT							\
> +	ASSERT_NAME_FMT(assert_ ## name_type ## _not_equal)		\
> +	"got:   "      type_fmt "\n"					\
> +	"unexpected: " type_fmt "\n"
> +
> +#define assert_true(cond) do {						\
> +	if (!(cond)) {							\
> +		test_save_diag_data(LOCATION_FMT			\
> +				    "condition_failed:\t'" #cond "'\n",	\
> +				    __FILE__, __LINE__);		\
> +		_test_exit(TEST_LJMP_EXIT_FAILURE);			\
> +	}								\
> +} while (0)
> +
> +#define assert_false(cond) assert_true(!(cond))
> +
> +#define assert_general(cond, fmt, ...) do {				\
> +	if (!(cond)) {							\
> +		test_save_diag_data(fmt, __VA_ARGS__);			\
> +		_test_exit(TEST_LJMP_EXIT_FAILURE);			\
> +	}								\
> +} while (0)
> +
> +#define assert_ptr_equal(got, expected) do {				\
> +	assert_general((got) == (expected),				\
> +		       ASSERT_EQUAL_FMT(ptr, "%p"),			\
> +		       __FILE__, __LINE__, (got), (expected)		\
> +	);								\
> +} while (0)
> +
> +#define assert_ptr_not_equal(got, unexpected) do {			\
> +	assert_general((got) != (unexpected),				\
> +		       ASSERT_NOT_EQUAL_FMT(ptr, "%p"),			\
> +		       __FILE__, __LINE__, (got), (unexpected)		\
> +	);								\
> +} while (0)
> +
> +
> +#define assert_int_equal(got, expected) do {				\
> +	assert_general((got) == (expected),				\
> +		       ASSERT_EQUAL_FMT(int, "%d"),			\
> +		       __FILE__, __LINE__, (got), (expected)		\
> +	);								\
> +} while (0)
> +
> +#define assert_int_not_equal(got, unexpected) do {			\
> +	assert_general((got) != (unexpected),				\
> +		       ASSERT_NOT_EQUAL_FMT(int, "%d"),			\
> +		       __FILE__, __LINE__, (got), (unexpected)		\
> +	);								\
> +} while (0)
> +
> +#define assert_sizet_equal(got, expected) do {				\
> +	assert_general((got) == (expected),				\
> +		       ASSERT_EQUAL_FMT(sizet, "%lu"),			\
> +		       __FILE__, __LINE__, (got), (expected)		\
> +	);								\
> +} while (0)
> +
> +#define assert_sizet_not_equal(got, unexpected) do {			\
> +	assert_general((got) != (unexpected),				\
> +		       ASSERT_NOT_EQUAL_FMT(sizet, "%lu"),		\
> +		       __FILE__, __LINE__, (got), (unexpected)		\
> +	);								\
> +} while (0)
> +
> +/* Check that doubles are __exactly__ the same. */
> +#define assert_double_equal(got, expected) do {				\
> +	assert_general((got) == (expected),				\
> +		       ASSERT_EQUAL_FMT(double, "%lf"),			\
> +		       __FILE__, __LINE__, (got), (expected)		\
> +	);								\
> +} while (0)
> +
> +/* Check that doubles are not __exactly__ the same. */
> +#define assert_double_not_equal(got, unexpected) do {			\
> +	assert_general((got) != (unexpected),				\
> +		       ASSERT_NOT_EQUAL_FMT(double, "%lf"),		\
> +		       __FILE__, __LINE__, (got), (unexpected)		\
> +	);								\
> +} while (0)

I propose to describe all types of assertions in the 
test/tarantool-c-tests/README.md.


> +
> +#endif /* TARANTOOL_LUAJIT_TEST_H */
> diff --git a/test/tarantool-c-tests/unit-tap.test.c b/test/tarantool-c-tests/unit-tap.test.c
> new file mode 100644
> index 00000000..27dc84ee
> --- /dev/null
> +++ b/test/tarantool-c-tests/unit-tap.test.c
> @@ -0,0 +1,31 @@
> +#include "test.h"
> +
> +#define UNUSED(x) ((void)(x))
> +
> +static int test_ok(void *test_state)
> +{
> +	UNUSED(test_state);
> +	return TEST_EXIT_SUCCESS;
> +}
> +
> +static int test_skip(void *test_state)
> +{
> +	UNUSED(test_state);
> +	return skip("test skip");
> +}
> +
> +static int test_todo(void *test_state)
> +{
> +	UNUSED(test_state);
> +	return todo("test todo");
> +}
> +
> +int main(void)
> +{
> +	const struct test_unit tgroup[] = {
> +		test_unit_new(test_ok),
> +		test_unit_new(test_skip),
> +		test_unit_new(test_todo)
> +	};
> +	return test_run_group(tgroup, NULL);
> +}

  reply	other threads:[~2023-06-05 15:08 UTC|newest]

Thread overview: 14+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2023-06-05 10:41 [Tarantool-patches] [PATCH v3 luajit 0/6] Reworking " Sergey Kaplun via Tarantool-patches
2023-06-05 10:41 ` [Tarantool-patches] [PATCH v3 luajit 1/6] test: fix setting of {DY}LD_LIBRARY_PATH variables Sergey Kaplun via Tarantool-patches
2023-06-05 14:21   ` Sergey Bronnikov via Tarantool-patches
2023-06-05 10:41 ` [Tarantool-patches] [PATCH v3 luajit 2/6] test: introduce module for C tests Sergey Kaplun via Tarantool-patches
2023-06-05 15:08   ` Sergey Bronnikov via Tarantool-patches [this message]
2023-06-07 15:51     ` Sergey Bronnikov via Tarantool-patches
2023-06-05 10:41 ` [Tarantool-patches] [PATCH v3 luajit 3/6] test: introduce utils.h helper " Sergey Kaplun via Tarantool-patches
2023-06-05 15:11   ` Sergey Bronnikov via Tarantool-patches
2023-06-05 10:41 ` [Tarantool-patches] [PATCH v3 luajit 4/6] test: rewrite misclib-getmetrics-capi test in C Sergey Kaplun via Tarantool-patches
2023-06-07 16:25   ` Sergey Bronnikov via Tarantool-patches
2023-06-05 10:41 ` [Tarantool-patches] [PATCH v3 luajit 5/6] test: rewrite misclib-sysprof-capi " Sergey Kaplun via Tarantool-patches
2023-06-07 16:18   ` Sergey Bronnikov via Tarantool-patches
2023-06-05 10:41 ` [Tarantool-patches] [PATCH v3 luajit 6/6] test: rewrite lj-49-bad-lightuserdata " Sergey Kaplun via Tarantool-patches
2023-06-07 16:14   ` Sergey Bronnikov via Tarantool-patches

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=1fcbd8cc-44ab-484f-1cd5-715701282049@tarantool.org \
    --to=tarantool-patches@dev.tarantool.org \
    --cc=imun@tarantool.org \
    --cc=m.kokryashkin@tarantool.org \
    --cc=sergeyb@tarantool.org \
    --cc=skaplun@tarantool.org \
    --subject='Re: [Tarantool-patches] [PATCH v3 luajit 2/6] test: introduce module for C tests' \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox