2023-11-06 13:59:19 -05:00
|
|
|
/*
|
|
|
|
|
|
|
|
9unit
|
2023-11-07 14:48:41 -05:00
|
|
|
Copyright (C) Jonathan Lamothe <jonathan@jlamothe.net>
|
2023-11-06 13:59:19 -05:00
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
2023-11-09 18:24:48 -05:00
|
|
|
it under the terms of the GNU Lesser General Public License as
|
|
|
|
published by the Free Software Foundation, either version 3 of the
|
|
|
|
License, or (at your option) any later version.
|
2023-11-06 13:59:19 -05:00
|
|
|
|
|
|
|
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
|
2023-11-09 18:24:48 -05:00
|
|
|
Lesser General Public License for more details.
|
2023-11-06 13:59:19 -05:00
|
|
|
|
2023-11-09 18:24:48 -05:00
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
|
|
License along with this program. If not, see
|
|
|
|
<http://www.gnu.org/licenses/>.
|
2023-11-06 13:59:19 -05:00
|
|
|
|
|
|
|
*/
|
|
|
|
|
2023-11-15 19:52:01 -05:00
|
|
|
// Types & Structs
|
|
|
|
|
2023-11-06 18:23:18 -05:00
|
|
|
// The following structures will typically be maintained by the
|
|
|
|
// testing framework. You shouldn't need to concern yourself with
|
|
|
|
// them.
|
2023-11-19 16:44:43 -05:00
|
|
|
|
2023-11-19 18:48:11 -05:00
|
|
|
// Possible results of running a single test
|
|
|
|
typedef enum TestResult
|
|
|
|
{
|
|
|
|
test_success, // the test succeeded
|
|
|
|
test_failure, // the test failed
|
|
|
|
test_pending // the test is pending
|
|
|
|
} TestResult;
|
|
|
|
|
2023-11-19 16:44:43 -05:00
|
|
|
typedef struct TestState TestState;
|
|
|
|
typedef struct TestLogEntry TestLogEntry;
|
|
|
|
|
|
|
|
// Tracks information about the tests being run.
|
2023-11-06 18:23:18 -05:00
|
|
|
struct TestState
|
2023-11-06 13:59:19 -05:00
|
|
|
{
|
|
|
|
int run; // number of tests run
|
2023-11-06 15:20:09 -05:00
|
|
|
int passed; // number of successful tests
|
|
|
|
int failed; // number of failed tests
|
2023-11-07 18:44:53 -05:00
|
|
|
int pending; // number of pending tests
|
2023-11-06 18:23:18 -05:00
|
|
|
TestLogEntry *first_log; // the first log entry
|
|
|
|
TestLogEntry *last_log; //the last log entry
|
2023-11-07 18:44:53 -05:00
|
|
|
void *ptr; // used for passing data between tests
|
2023-11-14 18:08:19 -05:00
|
|
|
const char *context; // immediate context of current test
|
|
|
|
const char *full_context; // full context of current test
|
|
|
|
int depth; // how many tests "deep" are we?
|
2023-11-14 18:53:43 -05:00
|
|
|
void (*report)(const char *); // prints a string immediately
|
2023-11-06 18:23:18 -05:00
|
|
|
};
|
|
|
|
|
2023-11-19 16:44:43 -05:00
|
|
|
// Defines a log entry in a TestState struct
|
2023-11-06 18:23:18 -05:00
|
|
|
struct TestLogEntry
|
|
|
|
{
|
|
|
|
char *text; // the entry text
|
|
|
|
TestLogEntry *next; // points to the next entry
|
|
|
|
};
|
2023-11-06 13:59:19 -05:00
|
|
|
|
2023-11-15 19:52:01 -05:00
|
|
|
// Functions
|
|
|
|
|
2023-11-19 18:48:11 -05:00
|
|
|
// Creates an initial TestState, passes it to the supplied function,
|
|
|
|
// and displays the resulting log and summary
|
|
|
|
extern void run_tests(void (*)(TestState *));
|
|
|
|
|
|
|
|
// Adds an entry to the log that is displayed after the tests have
|
|
|
|
// completed
|
|
|
|
extern void append_test_log(
|
|
|
|
TestState *, // the current state
|
|
|
|
const char * // the message to append
|
|
|
|
);
|
|
|
|
|
2023-11-20 00:03:33 -05:00
|
|
|
// notes the current full context in the log
|
|
|
|
extern void log_test_context(TestState *);
|
|
|
|
|
2023-11-06 15:20:09 -05:00
|
|
|
// Runs a single test
|
2023-11-06 18:23:18 -05:00
|
|
|
extern void run_test(
|
2023-11-06 15:20:09 -05:00
|
|
|
TestState *, // the TestState data
|
2023-11-06 18:23:18 -05:00
|
|
|
TestResult (*)(TestState *) // the test to run
|
2023-11-06 15:20:09 -05:00
|
|
|
);
|
|
|
|
|
2023-11-17 18:35:02 -05:00
|
|
|
// Runs a single test with an arbitrary input
|
|
|
|
extern void run_test_with(
|
|
|
|
TestState *, // the state
|
|
|
|
TestResult (*)(TestState *, void *), // the test
|
|
|
|
void * // the value to pass in
|
|
|
|
);
|
|
|
|
|
2023-11-19 17:14:21 -05:00
|
|
|
// Runs a single test passing in two values to be compared
|
|
|
|
extern void run_test_compare(
|
|
|
|
TestState *, // the current state
|
|
|
|
|
|
|
|
// the test to be run
|
|
|
|
TestResult (*test)(
|
|
|
|
TestState *, // the current state
|
|
|
|
void *, // the first value
|
|
|
|
void * // the second value
|
|
|
|
),
|
|
|
|
|
|
|
|
void *, // the first value
|
|
|
|
void * // the second value
|
|
|
|
);
|
|
|
|
|
2023-11-15 18:46:25 -05:00
|
|
|
// Gives additional context for a test
|
|
|
|
extern void test_context(
|
|
|
|
TestState *, // the current state
|
|
|
|
const char *, // a description of the context
|
|
|
|
void (*)(TestState *) // the actual test
|
2023-11-15 19:52:01 -05:00
|
|
|
);
|
|
|
|
|
2023-11-17 16:30:52 -05:00
|
|
|
// Runs a test with a context and an additional input
|
|
|
|
extern void test_context_with(
|
|
|
|
TestState *, // the current state
|
|
|
|
const char *, // a description of the context
|
|
|
|
void (*)(TestState *, void *), // the test function
|
|
|
|
void * // the value being passed in
|
|
|
|
);
|
|
|
|
|
2023-11-19 20:16:44 -05:00
|
|
|
// Passes two values into a test context
|
|
|
|
void test_context_compare(
|
|
|
|
TestState *, // the current state
|
|
|
|
const char *, // the context label
|
|
|
|
|
|
|
|
// test function
|
|
|
|
void (*)(
|
|
|
|
TestState *, // the current state
|
|
|
|
void *, // the first value
|
|
|
|
void * // the second value
|
|
|
|
),
|
|
|
|
|
|
|
|
void *, // the first value
|
|
|
|
void * // the second value
|
|
|
|
);
|
|
|
|
|
2023-11-18 23:40:19 -05:00
|
|
|
// Runs a single test with a context label
|
|
|
|
extern void single_test_context(
|
|
|
|
TestState *, // the current state
|
|
|
|
const char *, // a description of the context
|
|
|
|
TestResult (*)(TestState *) // the actual test
|
|
|
|
);
|
|
|
|
|
2023-11-18 11:38:00 -05:00
|
|
|
// Runs a single test with a context and input
|
|
|
|
extern void single_test_context_with(
|
|
|
|
TestState *, // the state
|
|
|
|
const char *, // the context
|
|
|
|
|
|
|
|
// the test
|
|
|
|
TestResult (*)(
|
|
|
|
TestState *, // the state
|
|
|
|
void * // the value being passed
|
|
|
|
),
|
|
|
|
|
|
|
|
void * // the value being passed
|
|
|
|
);
|
|
|
|
|
2023-11-19 22:58:21 -05:00
|
|
|
// Runs a single test with a context, passing it two values to be
|
|
|
|
// compared
|
|
|
|
extern void single_test_context_compare(
|
|
|
|
TestState *, // the state
|
|
|
|
const char *, // the context
|
|
|
|
|
|
|
|
// the test function
|
|
|
|
TestResult (*)(
|
|
|
|
TestState *, // the state
|
|
|
|
void *, // the first value
|
|
|
|
void * // the second value
|
|
|
|
),
|
|
|
|
|
|
|
|
void *, // the first value
|
|
|
|
void * // the second value
|
|
|
|
);
|
|
|
|
|
2023-11-06 13:59:19 -05:00
|
|
|
//jl
|