mirror of
https://github.com/ascii-boxes/boxes.git
synced 2024-12-12 18:01:14 +01:00
Extend unit testing capabilities
- Structure unit tests into multiple modules - Add setup/teardown of fixtures - Add ability to capture and check stdout and stderr - Add mock handling to Makefile
This commit is contained in:
parent
729c4e0692
commit
4ae947ff99
@ -55,7 +55,7 @@ pcre2_code *compile_pattern(char *pattern)
|
||||
if (re == NULL) {
|
||||
PCRE2_UCHAR buffer[256];
|
||||
pcre2_get_error_message(errornumber, buffer, sizeof(buffer));
|
||||
fprintf(stderr, "Regular expression pattern \"%s\" failed to compile at offset %d: %s\n",
|
||||
fprintf(stderr, "Regular expression pattern \"%s\" failed to compile at position %d: %s\n",
|
||||
pattern, (int) erroroffset, u32_strconv_to_output(buffer));
|
||||
}
|
||||
return re;
|
||||
|
@ -24,7 +24,8 @@ SRC_DIR = ../src
|
||||
UTEST_DIR = ../utest
|
||||
VPATH = $(SRC_DIR):$(SRC_DIR)/misc:$(UTEST_DIR)
|
||||
|
||||
UTEST_NORM = global_mock.c tools_test.c
|
||||
UTEST_NORM = global_mock.c tools_test.c regulex_test.o main.o
|
||||
MOCKS = fprintf
|
||||
|
||||
.PHONY: check_dir flags_unix flags_win32 flags_ utest
|
||||
|
||||
@ -41,13 +42,13 @@ $(OUT_DIR):
|
||||
|
||||
flags_unix:
|
||||
$(eval CFLAGS := -I. -I$(SRC_DIR) -O -Wall -W $(CFLAGS_ADDTL))
|
||||
$(eval LDFLAGS := $(LDFLAGS) --coverage $(LDFLAGS_ADDTL))
|
||||
$(eval LDFLAGS := $(LDFLAGS) $(foreach MOCK,$(MOCKS),-Wl,--wrap=$(MOCK)) --coverage $(LDFLAGS_ADDTL))
|
||||
$(eval UTEST_EXECUTABLE_NAME := unittest)
|
||||
$(eval UTEST_OBJ := $(UTEST_NORM:.c=.o))
|
||||
|
||||
flags_win32:
|
||||
$(eval CFLAGS := -Os -s -m32 -I. -I$(SRC_DIR) -Wall -W $(CFLAGS_ADDTL))
|
||||
$(eval LDFLAGS := $(LDFLAGS) --coverage -s -m32 $(LDFLAGS_ADDTL))
|
||||
$(eval LDFLAGS := $(LDFLAGS) $(foreach MOCK,$(MOCKS),-Wl,--wrap=$(MOCK)) --coverage -s -m32 $(LDFLAGS_ADDTL))
|
||||
$(eval UTEST_EXECUTABLE_NAME := unittest.exe)
|
||||
$(eval UTEST_OBJ := $(UTEST_NORM:.c=.o))
|
||||
|
||||
@ -70,5 +71,7 @@ unittest.exe: $(UTEST_OBJ) | check_dir
|
||||
-lkernel32 -l:libunistring.a -l:libpcre2-32.a -l:libiconv.a -l:libcmocka.dll.a
|
||||
|
||||
|
||||
global_mock.o: global_mock.c boxes.h
|
||||
tools_test.o: tools_test.c tools.h
|
||||
global_mock.o: global_mock.c global_mock.h boxes.h unicode.h config.h | check_dir
|
||||
tools_test.o: tools_test.c tools_test.h tools.h config.h | check_dir
|
||||
regulex_test.o: regulex_test.c regulex_test.h global_mock.h regulex.h config.h | check_dir
|
||||
main.o: main.c global_mock.h tools_test.h regulex_test.h config.h | check_dir
|
||||
|
@ -22,8 +22,15 @@
|
||||
* Mocks of boxes' global variables.
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
#include <stdarg.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <locale.h>
|
||||
#include <uniconv.h>
|
||||
#include "boxes.h"
|
||||
|
||||
#include "unicode.h"
|
||||
#include "tools.h"
|
||||
|
||||
design_t *designs = NULL;
|
||||
|
||||
@ -32,3 +39,73 @@ int num_designs = 0;
|
||||
opt_t opt;
|
||||
|
||||
input_t input;
|
||||
|
||||
char **collect_out = NULL;
|
||||
int collect_out_size = 0;
|
||||
|
||||
char **collect_err = NULL;
|
||||
int collect_err_size = 0;
|
||||
|
||||
|
||||
void collect_reset()
|
||||
{
|
||||
for (int i = 0; i < collect_out_size; i++) {
|
||||
BFREE(collect_out[i]);
|
||||
}
|
||||
BFREE(collect_out);
|
||||
|
||||
for (int i = 0; i < collect_err_size; i++) {
|
||||
BFREE(collect_err[i]);
|
||||
}
|
||||
BFREE(collect_err);
|
||||
|
||||
collect_out_size = 0;
|
||||
collect_err_size = 0;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Mock of the `fprintf()` function which records its output instead of printing it. Assumes that no output string will
|
||||
* be longer than 512 characters.
|
||||
* @param __stream `stdout` or `stderr`
|
||||
* @param __format the format string, followed by the arguments
|
||||
*/
|
||||
void __wrap_fprintf(FILE *__stream, const char *__format, ...)
|
||||
{
|
||||
char **collect = __stream == stdout ? collect_out : collect_err;
|
||||
int collect_size = __stream == stdout ? collect_out_size : collect_err_size;
|
||||
collect = (char **) realloc(collect, ++collect_size * sizeof(char *));
|
||||
|
||||
char *s = (char *) malloc(512);
|
||||
va_list va;
|
||||
va_start(va, __format);
|
||||
vsprintf(s, __format, va);
|
||||
va_end(va);
|
||||
collect[collect_size - 1] = s;
|
||||
|
||||
if (__stream == stdout) {
|
||||
collect_out = collect;
|
||||
collect_out_size = collect_size;
|
||||
}
|
||||
else {
|
||||
collect_err = collect;
|
||||
collect_err_size = collect_size;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void setup_mocks()
|
||||
{
|
||||
setlocale(LC_ALL, "");
|
||||
encoding = check_encoding("UTF-8", locale_charset());
|
||||
collect_reset();
|
||||
}
|
||||
|
||||
|
||||
void teardown()
|
||||
{
|
||||
collect_reset();
|
||||
}
|
||||
|
||||
|
||||
/*EOF*/ /* vim: set cindent sw=4: */
|
||||
|
44
utest/global_mock.h
Normal file
44
utest/global_mock.h
Normal file
@ -0,0 +1,44 @@
|
||||
/*
|
||||
* boxes - Command line filter to draw/remove ASCII boxes around text
|
||||
* Copyright (c) 1999-2021 Thomas Jensen and the boxes contributors
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License, version 2, as published
|
||||
* by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||
* for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along
|
||||
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||
*
|
||||
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||
*/
|
||||
|
||||
/*
|
||||
* Test fixtures and mocks of global shared state.
|
||||
*/
|
||||
|
||||
#ifndef GLOBAL_MOCK_H
|
||||
#define GLOBAL_MOCK_H
|
||||
|
||||
|
||||
extern char **collect_out;
|
||||
extern int collect_out_size;
|
||||
|
||||
extern char **collect_err;
|
||||
extern int collect_err_size;
|
||||
|
||||
void collect_reset();
|
||||
|
||||
|
||||
void setup_mocks();
|
||||
void teardown();
|
||||
|
||||
|
||||
#endif
|
||||
|
||||
/*EOF*/ /* vim: set cindent sw=4: */
|
61
utest/main.c
Normal file
61
utest/main.c
Normal file
@ -0,0 +1,61 @@
|
||||
/*
|
||||
* boxes - Command line filter to draw/remove ASCII boxes around text
|
||||
* Copyright (c) 1999-2021 Thomas Jensen and the boxes contributors
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License, version 2, as published
|
||||
* by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||
* for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along
|
||||
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||
*
|
||||
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||
*/
|
||||
|
||||
/*
|
||||
* Main module for running the unit tests.
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
#include <stdarg.h>
|
||||
#include <stddef.h>
|
||||
#include <setjmp.h>
|
||||
#include <cmocka.h>
|
||||
|
||||
#include "global_mock.h"
|
||||
#include "tools_test.h"
|
||||
#include "regulex_test.h"
|
||||
|
||||
|
||||
int main(void)
|
||||
{
|
||||
setup_mocks();
|
||||
|
||||
const struct CMUnitTest tools_tests[] = {
|
||||
cmocka_unit_test(test_strisyes_true),
|
||||
cmocka_unit_test(test_strisyes_false),
|
||||
cmocka_unit_test(test_strisno_true),
|
||||
cmocka_unit_test(test_strisno_false)
|
||||
};
|
||||
|
||||
const struct CMUnitTest regulex_tests[] = {
|
||||
cmocka_unit_test(test_compile_pattern_error),
|
||||
cmocka_unit_test(test_compile_pattern_empty)
|
||||
};
|
||||
|
||||
int num_failed = 0;
|
||||
num_failed += cmocka_run_group_tests(tools_tests, NULL, NULL);
|
||||
num_failed += cmocka_run_group_tests(regulex_tests, NULL, NULL);
|
||||
|
||||
teardown();
|
||||
return num_failed;
|
||||
}
|
||||
|
||||
|
||||
/*EOF*/ /* vim: set cindent sw=4: */
|
60
utest/regulex_test.c
Normal file
60
utest/regulex_test.c
Normal file
@ -0,0 +1,60 @@
|
||||
/*
|
||||
* boxes - Command line filter to draw/remove ASCII boxes around text
|
||||
* Copyright (c) 1999-2021 Thomas Jensen and the boxes contributors
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License, version 2, as published
|
||||
* by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||
* for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along
|
||||
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||
*
|
||||
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||
*/
|
||||
|
||||
/*
|
||||
* Unit tests of the 'regulex' module
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
#include <stdarg.h>
|
||||
#include <stddef.h>
|
||||
#include <setjmp.h>
|
||||
#include <cmocka.h>
|
||||
#include "global_mock.h"
|
||||
#include "regulex.h"
|
||||
|
||||
|
||||
void test_compile_pattern_empty(void **state)
|
||||
{
|
||||
(void) state; /* unused */
|
||||
|
||||
assert_null(compile_pattern(NULL));
|
||||
assert_non_null(compile_pattern(""));
|
||||
}
|
||||
|
||||
|
||||
void test_compile_pattern_error(void **state)
|
||||
{
|
||||
(void) state; /* unused */
|
||||
|
||||
assert_null(compile_pattern("incomplete[x"));
|
||||
assert_int_equal(1, collect_err_size);
|
||||
assert_string_equal("Regular expression pattern \"incomplete[x\" failed to compile at position 12: "
|
||||
"missing terminating ] for character class\n", collect_err[0]);
|
||||
|
||||
collect_reset();
|
||||
assert_null(compile_pattern("incomplete\\"));
|
||||
assert_int_equal(1, collect_err_size);
|
||||
assert_string_equal("Regular expression pattern \"incomplete\\\" failed to compile at position 11: "
|
||||
"\\ at end of pattern\n", collect_err[0]);
|
||||
}
|
||||
|
||||
|
||||
/*EOF*/ /* vim: set cindent sw=4: */
|
36
utest/regulex_test.h
Normal file
36
utest/regulex_test.h
Normal file
@ -0,0 +1,36 @@
|
||||
/*
|
||||
* boxes - Command line filter to draw/remove ASCII boxes around text
|
||||
* Copyright (c) 1999-2021 Thomas Jensen and the boxes contributors
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License, version 2, as published
|
||||
* by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||
* for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along
|
||||
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||
*
|
||||
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||
*/
|
||||
|
||||
/*
|
||||
* Unit tests of the 'regulex' module
|
||||
*/
|
||||
|
||||
#ifndef REGULEX_TEST_H
|
||||
#define REGULEX_TEST_H
|
||||
|
||||
|
||||
void test_compile_pattern_empty(void **state);
|
||||
void test_compile_pattern_error(void **state);
|
||||
|
||||
|
||||
#endif
|
||||
|
||||
|
||||
/*EOF*/ /* vim: set cindent sw=4: */
|
@ -22,15 +22,19 @@
|
||||
* Unit tests of the 'tools' module
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
#include <stdarg.h>
|
||||
#include <stddef.h>
|
||||
#include <setjmp.h>
|
||||
#include <cmocka.h>
|
||||
#include "tools.h"
|
||||
#include "tools_test.h"
|
||||
|
||||
|
||||
static void test_strisyes_true()
|
||||
void test_strisyes_true(void **state)
|
||||
{
|
||||
(void) state; /* unused */
|
||||
|
||||
assert_int_equal(1, strisyes("On"));
|
||||
assert_int_equal(1, strisyes("on"));
|
||||
assert_int_equal(1, strisyes("yes"));
|
||||
@ -43,8 +47,10 @@ static void test_strisyes_true()
|
||||
}
|
||||
|
||||
|
||||
static void test_strisyes_false()
|
||||
void test_strisyes_false(void **state)
|
||||
{
|
||||
(void) state; /* unused */
|
||||
|
||||
assert_int_equal(0, strisyes(NULL));
|
||||
assert_int_equal(0, strisyes(""));
|
||||
assert_int_equal(0, strisyes(" "));
|
||||
@ -60,8 +66,10 @@ static void test_strisyes_false()
|
||||
}
|
||||
|
||||
|
||||
static void test_strisno_true()
|
||||
void test_strisno_true(void **state)
|
||||
{
|
||||
(void) state; /* unused */
|
||||
|
||||
assert_int_equal(1, strisno("off"));
|
||||
assert_int_equal(1, strisno("Off"));
|
||||
assert_int_equal(1, strisno("no"));
|
||||
@ -74,8 +82,10 @@ static void test_strisno_true()
|
||||
}
|
||||
|
||||
|
||||
static void test_strisno_false()
|
||||
void test_strisno_false(void **state)
|
||||
{
|
||||
(void) state; /* unused */
|
||||
|
||||
assert_int_equal(0, strisno(NULL));
|
||||
assert_int_equal(0, strisno(""));
|
||||
assert_int_equal(0, strisno(" "));
|
||||
@ -91,14 +101,4 @@ static void test_strisno_false()
|
||||
}
|
||||
|
||||
|
||||
int main(void)
|
||||
{
|
||||
const struct CMUnitTest tests[] = {
|
||||
cmocka_unit_test(test_strisyes_true),
|
||||
cmocka_unit_test(test_strisyes_false),
|
||||
cmocka_unit_test(test_strisno_true),
|
||||
cmocka_unit_test(test_strisno_false)
|
||||
};
|
||||
|
||||
return cmocka_run_group_tests(tests, NULL, NULL);
|
||||
}
|
||||
/*EOF*/ /* vim: set cindent sw=4: */
|
||||
|
38
utest/tools_test.h
Normal file
38
utest/tools_test.h
Normal file
@ -0,0 +1,38 @@
|
||||
/*
|
||||
* boxes - Command line filter to draw/remove ASCII boxes around text
|
||||
* Copyright (c) 1999-2021 Thomas Jensen and the boxes contributors
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License, version 2, as published
|
||||
* by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but
|
||||
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||
* for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along
|
||||
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||
*
|
||||
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
||||
*/
|
||||
|
||||
/*
|
||||
* Unit tests of the 'tools' module
|
||||
*/
|
||||
|
||||
#ifndef TOOLS_TEST_H
|
||||
#define TOOLS_TEST_H
|
||||
|
||||
|
||||
void test_strisyes_true(void **state);
|
||||
void test_strisyes_false(void **state);
|
||||
|
||||
void test_strisno_true(void **state);
|
||||
void test_strisno_false(void **state);
|
||||
|
||||
|
||||
#endif
|
||||
|
||||
/*EOF*/ /* vim: set cindent sw=4: */
|
Loading…
Reference in New Issue
Block a user