forked from github/server
added a simple unit testing framework (CuTest)
added unit tests for base36 added eressea-only lua bindings fixed flying ships
This commit is contained in:
parent
00e34fa972
commit
c1f4b2e6ce
24
src/all.sln
24
src/all.sln
|
@ -1,17 +1,20 @@
|
|||
Microsoft Visual Studio Solution File, Format Version 9.00
|
||||
# Visual Studio 2005
|
||||
Project("{228B8781-480F-4608-BA14-1EB82581EADE}") = "kernel", "common\kernel.vcproj", "{6F104C0A-DDF5-A34B-A89C-0DC278DCEF6D}"
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "kernel", "common\kernel.vcproj", "{6F104C0A-DDF5-A34B-A89C-0DC278DCEF6D}"
|
||||
EndProject
|
||||
Project("{228B8781-480F-4608-BA14-1EB82581EADE}") = "gamecode", "common\gamecode.vcproj", "{1E8BFF9E-3044-0742-992F-C5765B80FE65}"
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "gamecode", "common\gamecode.vcproj", "{1E8BFF9E-3044-0742-992F-C5765B80FE65}"
|
||||
EndProject
|
||||
Project("{228B8781-480F-4608-BA14-1EB82581EADE}") = "util", "common\util.vcproj", "{F70CFB27-8A2F-E447-B452-4E1C590EDA6D}"
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "util", "common\util.vcproj", "{F70CFB27-8A2F-E447-B452-4E1C590EDA6D}"
|
||||
EndProject
|
||||
Project("{228B8781-480F-4608-BA14-1EB82581EADE}") = "lua-bindings", "common\lua-bindings.vcproj", "{75501170-51C2-E641-BA8B-EDC008184192}"
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "lua-bindings", "common\lua-bindings.vcproj", "{75501170-51C2-E641-BA8B-EDC008184192}"
|
||||
EndProject
|
||||
Project("{228B8781-480F-4608-BA14-1EB82581EADE}") = "external", "external\external.vcproj", "{F9AE4586-8F65-486B-9666-744839E40A54}"
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "external", "external\external.vcproj", "{F9AE4586-8F65-486B-9666-744839E40A54}"
|
||||
EndProject
|
||||
Project("{228B8781-480F-4608-BA14-1EB82581EADE}") = "amalgamation-eressea", "eressea\eressea.vcproj", "{AD80EB0B-7CB4-42F2-9C95-8CCEF68DB387}"
|
||||
Project("{228B8781-480F-4608-BA14-1EB82581EADE}") = "amalgamation-basic", "basic\basic.vcproj", "{4A17DAEE-2261-4E2C-96F6-BA4132A09551}"
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "amalgamation-eressea", "eressea\eressea.vcproj", "{AD80EB0B-7CB4-42F2-9C95-8CCEF68DB387}"
|
||||
EndProject
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "amalgamation-basic", "basic\basic.vcproj", "{4A17DAEE-2261-4E2C-96F6-BA4132A09551}"
|
||||
EndProject
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "tests", "common\tests.vcproj", "{228B8781-480F-4608-BA14-1EB82581EADE}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SubversionScc) = preSolution
|
||||
|
@ -29,9 +32,6 @@ Global
|
|||
{1E8BFF9E-3044-0742-992F-C5765B80FE65}.Debug|Win32.ActiveCfg = Debug|Win32
|
||||
{1E8BFF9E-3044-0742-992F-C5765B80FE65}.Release|Win32.ActiveCfg = Release|Win32
|
||||
{1E8BFF9E-3044-0742-992F-C5765B80FE65}.Release|Win32.Build.0 = Release|Win32
|
||||
{D893D6B3-805D-9848-8EA4-CDA1B79151F6}.Debug|Win32.ActiveCfg = Debug|Win32
|
||||
{D893D6B3-805D-9848-8EA4-CDA1B79151F6}.Release|Win32.ActiveCfg = Release|Win32
|
||||
{D893D6B3-805D-9848-8EA4-CDA1B79151F6}.Release|Win32.Build.0 = Release|Win32
|
||||
{F70CFB27-8A2F-E447-B452-4E1C590EDA6D}.Debug|Win32.ActiveCfg = Debug|Win32
|
||||
{F70CFB27-8A2F-E447-B452-4E1C590EDA6D}.Release|Win32.ActiveCfg = Release|Win32
|
||||
{F70CFB27-8A2F-E447-B452-4E1C590EDA6D}.Release|Win32.Build.0 = Release|Win32
|
||||
|
@ -47,6 +47,10 @@ Global
|
|||
{4A17DAEE-2261-4E2C-96F6-BA4132A09551}.Debug|Win32.ActiveCfg = Debug|Win32
|
||||
{4A17DAEE-2261-4E2C-96F6-BA4132A09551}.Debug|Win32.Build.0 = Debug|Win32
|
||||
{4A17DAEE-2261-4E2C-96F6-BA4132A09551}.Release|Win32.ActiveCfg = Release|Win32
|
||||
{228B8781-480F-4608-BA14-1EB82581EADE}.Debug|Win32.ActiveCfg = Debug|Win32
|
||||
{228B8781-480F-4608-BA14-1EB82581EADE}.Debug|Win32.Build.0 = Debug|Win32
|
||||
{228B8781-480F-4608-BA14-1EB82581EADE}.Release|Win32.ActiveCfg = Release|Win32
|
||||
{228B8781-480F-4608-BA14-1EB82581EADE}.Release|Win32.Build.0 = Release|Win32
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
|
|
|
@ -0,0 +1,7 @@
|
|||
#include "common/settings.h"
|
||||
#include <platform.h>
|
||||
#include "stdafx.h"
|
||||
|
||||
#include <util/listbox.c>
|
||||
#include <gmtool.c>
|
||||
#include <eressea.c>
|
|
@ -187,7 +187,6 @@ extern keyword_t findkeyword(const char *s, const struct locale * lang);
|
|||
extern param_t findparam(const char *s, const struct locale * lang);
|
||||
extern param_t getparam(const struct locale * lang);
|
||||
|
||||
extern int atoi36(const char * s);
|
||||
extern int getid(void);
|
||||
#define unitid(x) itoa36((x)->no)
|
||||
|
||||
|
|
|
@ -0,0 +1,20 @@
|
|||
#include <external/cutest/CuTest.h>
|
||||
#include <stdio.h>
|
||||
|
||||
CuSuite* get_base36_suite();
|
||||
|
||||
void RunAllTests(void) {
|
||||
CuString *output = CuStringNew();
|
||||
CuSuite* suite = CuSuiteNew();
|
||||
|
||||
CuSuiteAddSuite(suite, get_base36_suite());
|
||||
|
||||
CuSuiteRun(suite);
|
||||
CuSuiteSummary(suite, output);
|
||||
CuSuiteDetails(suite, output);
|
||||
printf("%s\n", output->buffer);
|
||||
}
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
RunAllTests();
|
||||
}
|
|
@ -0,0 +1,29 @@
|
|||
#include <external/cutest/CuTest.h>
|
||||
|
||||
#include "base36.c"
|
||||
|
||||
static void test_atoi36(CuTest * tc) {
|
||||
CuAssertIntEquals(tc, 0, atoi36("0"));
|
||||
CuAssertIntEquals(tc, 666, atoi36("ii"));
|
||||
CuAssertIntEquals(tc, -10, atoi36("-a"));
|
||||
CuAssertIntEquals(tc, -1, atoi36("-1"));
|
||||
CuAssertIntEquals(tc, -10, atoi("-10"));
|
||||
CuAssertIntEquals(tc, -10, atoi("-10"));
|
||||
}
|
||||
|
||||
static void test_itoa36(CuTest * tc) {
|
||||
CuAssertStrEquals(tc, itoa36(0), "0");
|
||||
CuAssertStrEquals(tc, itoa10(INT_MAX), "2147483647");
|
||||
CuAssertStrEquals(tc, itoab(-1, 5), "-1");
|
||||
CuAssertStrEquals(tc, itoa36(-1), "-1");
|
||||
CuAssertStrEquals(tc, itoa36(-10), "-a");
|
||||
CuAssertStrEquals(tc, itoa36(666), "ii");
|
||||
}
|
||||
|
||||
CuSuite* get_base36_suite()
|
||||
{
|
||||
CuSuite* suite = CuSuiteNew();
|
||||
SUITE_ADD_TEST(suite, test_itoa36);
|
||||
SUITE_ADD_TEST(suite, test_atoi36);
|
||||
return suite;
|
||||
}
|
|
@ -206,6 +206,26 @@
|
|||
/>
|
||||
</FileConfiguration>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\src\bindings.c"
|
||||
>
|
||||
<FileConfiguration
|
||||
Name="Debug|Win32"
|
||||
ExcludedFromBuild="true"
|
||||
>
|
||||
<Tool
|
||||
Name="VCCLCompilerTool"
|
||||
/>
|
||||
</FileConfiguration>
|
||||
<FileConfiguration
|
||||
Name="Release|Win32"
|
||||
ExcludedFromBuild="true"
|
||||
>
|
||||
<Tool
|
||||
Name="VCCLCompilerTool"
|
||||
/>
|
||||
</FileConfiguration>
|
||||
</File>
|
||||
<File
|
||||
RelativePath=".\src\spells\buildingcurse.c"
|
||||
>
|
||||
|
|
|
@ -183,7 +183,6 @@
|
|||
<function name="wage" value="minimum_wage"/>
|
||||
</rules>
|
||||
<xi:include href="eressea/strings.xml"/>
|
||||
<xi:include href="eressea/races.xml"/>
|
||||
<xi:include href="eressea/items.xml"/>
|
||||
<xi:include href="eressea/artrewards.xml"/>
|
||||
<strings>
|
||||
|
|
|
@ -22,14 +22,13 @@
|
|||
<xi:include href="eressea:///core/common/buildings.xml"/>
|
||||
<xi:include href="eressea:///core/equipment.xml"/>
|
||||
<xi:include href="eressea:///core/terrains.xml"/>
|
||||
<xi:include href="eressea:///core/directions.xml"/>
|
||||
<xi:include href="directions.xml"/>
|
||||
<xi:include href="artrewards.xml"/>
|
||||
<xi:include href="buildings.xml"/>
|
||||
<xi:include href="eressea:///core/calendar.xml"/>
|
||||
<calendar name="secondage" newyear="month_1" start="184"/>
|
||||
<xi:include href="equipment.xml"/>
|
||||
<xi:include href="items.xml"/>
|
||||
<xi:include href="races.xml"/>
|
||||
<xi:include href="spells.xml"/>
|
||||
<xi:include href="strings.xml"/>
|
||||
<equipment>
|
||||
|
@ -44,11 +43,11 @@
|
|||
</set>
|
||||
</equipment>
|
||||
|
||||
<xi:include href="eressea:///core/names-undead.xml"/>
|
||||
<xi:include href="eressea:///core/names-skeletons.xml"/>
|
||||
<xi:include href="eressea:///core/names-zombies.xml"/>
|
||||
<xi:include href="eressea:///core/names-ghouls.xml"/>
|
||||
<xi:include href="eressea:///core/names-dragons.xml"/>
|
||||
<xi:include href="names-undead.xml"/>
|
||||
<xi:include href="names-skeletons.xml"/>
|
||||
<xi:include href="names-zombies.xml"/>
|
||||
<xi:include href="names-ghouls.xml"/>
|
||||
<xi:include href="names-dragons.xml"/>
|
||||
|
||||
<game name="Eressea">
|
||||
<!-- Game specific settings -->
|
||||
|
|
|
@ -0,0 +1,32 @@
|
|||
#include <platform.h>
|
||||
|
||||
#include "spells/shipcurse.h"
|
||||
|
||||
#include <kernel/ship.h>
|
||||
#include <kernel/unit.h>
|
||||
|
||||
#include <tolua.h>
|
||||
|
||||
static int
|
||||
tolua_levitate_ship(lua_State * L)
|
||||
{
|
||||
ship * sh = (ship *)tolua_tousertype(L, 1, 0);
|
||||
unit * mage = (unit *)tolua_tousertype(L, 2, 0);
|
||||
double power = (double)tolua_tonumber(L, 3, 0);
|
||||
int duration = (int)tolua_tonumber(L, 4, 0);
|
||||
int cno = levitate_ship(sh, mage, power, duration);
|
||||
tolua_pushnumber(L, (lua_Number)cno);
|
||||
return 1;
|
||||
}
|
||||
|
||||
void
|
||||
bind_eressea(struct lua_State * L)
|
||||
{
|
||||
tolua_module(L, NULL, 0);
|
||||
tolua_beginmodule(L, NULL);
|
||||
{
|
||||
tolua_function(L, TOLUA_CAST "levitate_ship", tolua_levitate_ship);
|
||||
}
|
||||
tolua_endmodule(L);
|
||||
}
|
||||
|
|
@ -216,6 +216,8 @@ void locale_init(void)
|
|||
assert(towlower(0xC4)==0xE4); /* Ä => ä */
|
||||
}
|
||||
|
||||
extern void bind_eressea(struct lua_State * L);
|
||||
|
||||
int main(int argc, char ** argv)
|
||||
{
|
||||
static int write_csv = 0;
|
||||
|
@ -243,6 +245,7 @@ int main(int argc, char ** argv)
|
|||
}
|
||||
register_curses();
|
||||
register_spells();
|
||||
bind_eressea((struct lua_State *)global.vm_state);
|
||||
|
||||
if (write_csv) {
|
||||
write_skills();
|
||||
|
|
|
@ -11,3 +11,4 @@
|
|||
#include "spells/unitcurse.c"
|
||||
|
||||
#include "main.c"
|
||||
#include "bindings.c"
|
||||
|
|
|
@ -97,9 +97,9 @@ static int flyingship_age(curse * c) {
|
|||
ship * sh = (ship *)c->data.v;
|
||||
if (sh && c->duration==1) {
|
||||
freset(sh, SF_FLYING);
|
||||
return AT_AGE_REMOVE;
|
||||
return 1;
|
||||
}
|
||||
return AT_AGE_KEEP;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct curse_type ct_flyingship = { "flyingship",
|
||||
|
@ -127,6 +127,7 @@ shipcurse_flyingship(ship* sh, unit * mage, double power, int duration)
|
|||
} else {
|
||||
/* mit C_SHIP_NODRIFT haben wir kein Problem */
|
||||
curse * c = create_curse(mage, &sh->attribs, ct_flyingship, power, duration, 0.0, 0);
|
||||
c->data.v = sh;
|
||||
if (c && c->duration>0) {
|
||||
sh->flags |= SF_FLYING;
|
||||
}
|
||||
|
|
|
@ -24,7 +24,6 @@ extern "C" {
|
|||
|
||||
extern void register_spells(void);
|
||||
|
||||
int levitate_ship(struct ship * sh, struct unit * mage, double power, int duration);
|
||||
void set_spelldata(struct spell * sp);
|
||||
|
||||
#ifdef __cplusplus
|
||||
|
|
|
@ -0,0 +1,25 @@
|
|||
#include <stdio.h>
|
||||
|
||||
#include "CuTest.h"
|
||||
|
||||
CuSuite* CuGetSuite();
|
||||
CuSuite* CuStringGetSuite();
|
||||
|
||||
void RunAllTests(void)
|
||||
{
|
||||
CuString *output = CuStringNew();
|
||||
CuSuite* suite = CuSuiteNew();
|
||||
|
||||
CuSuiteAddSuite(suite, CuGetSuite());
|
||||
CuSuiteAddSuite(suite, CuStringGetSuite());
|
||||
|
||||
CuSuiteRun(suite);
|
||||
CuSuiteSummary(suite, output);
|
||||
CuSuiteDetails(suite, output);
|
||||
printf("%s\n", output->buffer);
|
||||
}
|
||||
|
||||
int main(void)
|
||||
{
|
||||
RunAllTests();
|
||||
}
|
|
@ -0,0 +1,309 @@
|
|||
#include <assert.h>
|
||||
#include <setjmp.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <math.h>
|
||||
|
||||
#include "CuTest.h"
|
||||
|
||||
/*-------------------------------------------------------------------------*
|
||||
* CuStr
|
||||
*-------------------------------------------------------------------------*/
|
||||
|
||||
char* CuStrAlloc(int size)
|
||||
{
|
||||
char* newStr = (char*) malloc( sizeof(char) * (size) );
|
||||
return newStr;
|
||||
}
|
||||
|
||||
char* CuStrCopy(const char* old)
|
||||
{
|
||||
int len = (int)strlen(old);
|
||||
char* newStr = CuStrAlloc(len + 1);
|
||||
strcpy(newStr, old);
|
||||
return newStr;
|
||||
}
|
||||
|
||||
/*-------------------------------------------------------------------------*
|
||||
* CuString
|
||||
*-------------------------------------------------------------------------*/
|
||||
|
||||
void CuStringInit(CuString* str)
|
||||
{
|
||||
str->length = 0;
|
||||
str->size = STRING_MAX;
|
||||
str->buffer = (char*) malloc(sizeof(char) * str->size);
|
||||
str->buffer[0] = '\0';
|
||||
}
|
||||
|
||||
CuString* CuStringNew(void)
|
||||
{
|
||||
CuString* str = (CuString*) malloc(sizeof(CuString));
|
||||
str->length = 0;
|
||||
str->size = STRING_MAX;
|
||||
str->buffer = (char*) malloc(sizeof(char) * str->size);
|
||||
str->buffer[0] = '\0';
|
||||
return str;
|
||||
}
|
||||
|
||||
void CuStringResize(CuString* str, int newSize)
|
||||
{
|
||||
str->buffer = (char*) realloc(str->buffer, sizeof(char) * newSize);
|
||||
str->size = newSize;
|
||||
}
|
||||
|
||||
void CuStringAppend(CuString* str, const char* text)
|
||||
{
|
||||
int length;
|
||||
|
||||
if (text == NULL) {
|
||||
text = "NULL";
|
||||
}
|
||||
|
||||
length = (int)strlen(text);
|
||||
if (str->length + length + 1 >= str->size)
|
||||
CuStringResize(str, str->length + length + 1 + STRING_INC);
|
||||
str->length += length;
|
||||
strcat(str->buffer, text);
|
||||
}
|
||||
|
||||
void CuStringAppendChar(CuString* str, char ch)
|
||||
{
|
||||
char text[2];
|
||||
text[0] = ch;
|
||||
text[1] = '\0';
|
||||
CuStringAppend(str, text);
|
||||
}
|
||||
|
||||
void CuStringAppendFormat(CuString* str, const char* format, ...)
|
||||
{
|
||||
va_list argp;
|
||||
char buf[HUGE_STRING_LEN];
|
||||
va_start(argp, format);
|
||||
vsprintf(buf, format, argp);
|
||||
va_end(argp);
|
||||
CuStringAppend(str, buf);
|
||||
}
|
||||
|
||||
void CuStringInsert(CuString* str, const char* text, int pos)
|
||||
{
|
||||
int length = (int)strlen(text);
|
||||
if (pos > str->length)
|
||||
pos = str->length;
|
||||
if (str->length + length + 1 >= str->size)
|
||||
CuStringResize(str, str->length + length + 1 + STRING_INC);
|
||||
memmove(str->buffer + pos + length, str->buffer + pos, (str->length - pos) + 1);
|
||||
str->length += length;
|
||||
memcpy(str->buffer + pos, text, length);
|
||||
}
|
||||
|
||||
/*-------------------------------------------------------------------------*
|
||||
* CuTest
|
||||
*-------------------------------------------------------------------------*/
|
||||
|
||||
void CuTestInit(CuTest* t, const char* name, TestFunction function)
|
||||
{
|
||||
t->name = CuStrCopy(name);
|
||||
t->failed = 0;
|
||||
t->ran = 0;
|
||||
t->message = NULL;
|
||||
t->function = function;
|
||||
t->jumpBuf = NULL;
|
||||
}
|
||||
|
||||
CuTest* CuTestNew(const char* name, TestFunction function)
|
||||
{
|
||||
CuTest* tc = CU_ALLOC(CuTest);
|
||||
CuTestInit(tc, name, function);
|
||||
return tc;
|
||||
}
|
||||
|
||||
void CuTestRun(CuTest* tc)
|
||||
{
|
||||
jmp_buf buf;
|
||||
tc->jumpBuf = &buf;
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
tc->ran = 1;
|
||||
(tc->function)(tc);
|
||||
}
|
||||
tc->jumpBuf = 0;
|
||||
}
|
||||
|
||||
static void CuFailInternal(CuTest* tc, const char* file, int line, CuString* string)
|
||||
{
|
||||
char buf[HUGE_STRING_LEN];
|
||||
|
||||
sprintf(buf, "%s:%d: ", file, line);
|
||||
CuStringInsert(string, buf, 0);
|
||||
|
||||
tc->failed = 1;
|
||||
tc->message = string->buffer;
|
||||
if (tc->jumpBuf != 0) longjmp(*(tc->jumpBuf), 0);
|
||||
}
|
||||
|
||||
void CuFail_Line(CuTest* tc, const char* file, int line, const char* message2, const char* message)
|
||||
{
|
||||
CuString string;
|
||||
|
||||
CuStringInit(&string);
|
||||
if (message2 != NULL)
|
||||
{
|
||||
CuStringAppend(&string, message2);
|
||||
CuStringAppend(&string, ": ");
|
||||
}
|
||||
CuStringAppend(&string, message);
|
||||
CuFailInternal(tc, file, line, &string);
|
||||
}
|
||||
|
||||
void CuAssert_Line(CuTest* tc, const char* file, int line, const char* message, int condition)
|
||||
{
|
||||
if (condition) return;
|
||||
CuFail_Line(tc, file, line, NULL, message);
|
||||
}
|
||||
|
||||
void CuAssertStrEquals_LineMsg(CuTest* tc, const char* file, int line, const char* message,
|
||||
const char* expected, const char* actual)
|
||||
{
|
||||
CuString string;
|
||||
if ((expected == NULL && actual == NULL) ||
|
||||
(expected != NULL && actual != NULL &&
|
||||
strcmp(expected, actual) == 0))
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
CuStringInit(&string);
|
||||
if (message != NULL)
|
||||
{
|
||||
CuStringAppend(&string, message);
|
||||
CuStringAppend(&string, ": ");
|
||||
}
|
||||
CuStringAppend(&string, "expected <");
|
||||
CuStringAppend(&string, expected);
|
||||
CuStringAppend(&string, "> but was <");
|
||||
CuStringAppend(&string, actual);
|
||||
CuStringAppend(&string, ">");
|
||||
CuFailInternal(tc, file, line, &string);
|
||||
}
|
||||
|
||||
void CuAssertIntEquals_LineMsg(CuTest* tc, const char* file, int line, const char* message,
|
||||
int expected, int actual)
|
||||
{
|
||||
char buf[STRING_MAX];
|
||||
if (expected == actual) return;
|
||||
sprintf(buf, "expected <%d> but was <%d>", expected, actual);
|
||||
CuFail_Line(tc, file, line, message, buf);
|
||||
}
|
||||
|
||||
void CuAssertDblEquals_LineMsg(CuTest* tc, const char* file, int line, const char* message,
|
||||
double expected, double actual, double delta)
|
||||
{
|
||||
char buf[STRING_MAX];
|
||||
if (fabs(expected - actual) <= delta) return;
|
||||
sprintf(buf, "expected <%lf> but was <%lf>", expected, actual);
|
||||
CuFail_Line(tc, file, line, message, buf);
|
||||
}
|
||||
|
||||
void CuAssertPtrEquals_LineMsg(CuTest* tc, const char* file, int line, const char* message,
|
||||
void* expected, void* actual)
|
||||
{
|
||||
char buf[STRING_MAX];
|
||||
if (expected == actual) return;
|
||||
sprintf(buf, "expected pointer <0x%p> but was <0x%p>", expected, actual);
|
||||
CuFail_Line(tc, file, line, message, buf);
|
||||
}
|
||||
|
||||
|
||||
/*-------------------------------------------------------------------------*
|
||||
* CuSuite
|
||||
*-------------------------------------------------------------------------*/
|
||||
|
||||
void CuSuiteInit(CuSuite* testSuite)
|
||||
{
|
||||
testSuite->count = 0;
|
||||
testSuite->failCount = 0;
|
||||
}
|
||||
|
||||
CuSuite* CuSuiteNew(void)
|
||||
{
|
||||
CuSuite* testSuite = CU_ALLOC(CuSuite);
|
||||
CuSuiteInit(testSuite);
|
||||
return testSuite;
|
||||
}
|
||||
|
||||
void CuSuiteAdd(CuSuite* testSuite, CuTest *testCase)
|
||||
{
|
||||
assert(testSuite->count < MAX_TEST_CASES);
|
||||
testSuite->list[testSuite->count] = testCase;
|
||||
testSuite->count++;
|
||||
}
|
||||
|
||||
void CuSuiteAddSuite(CuSuite* testSuite, CuSuite* testSuite2)
|
||||
{
|
||||
int i;
|
||||
for (i = 0 ; i < testSuite2->count ; ++i)
|
||||
{
|
||||
CuTest* testCase = testSuite2->list[i];
|
||||
CuSuiteAdd(testSuite, testCase);
|
||||
}
|
||||
}
|
||||
|
||||
void CuSuiteRun(CuSuite* testSuite)
|
||||
{
|
||||
int i;
|
||||
for (i = 0 ; i < testSuite->count ; ++i)
|
||||
{
|
||||
CuTest* testCase = testSuite->list[i];
|
||||
CuTestRun(testCase);
|
||||
if (testCase->failed) { testSuite->failCount += 1; }
|
||||
}
|
||||
}
|
||||
|
||||
void CuSuiteSummary(CuSuite* testSuite, CuString* summary)
|
||||
{
|
||||
int i;
|
||||
for (i = 0 ; i < testSuite->count ; ++i)
|
||||
{
|
||||
CuTest* testCase = testSuite->list[i];
|
||||
CuStringAppend(summary, testCase->failed ? "F" : ".");
|
||||
}
|
||||
CuStringAppend(summary, "\n\n");
|
||||
}
|
||||
|
||||
void CuSuiteDetails(CuSuite* testSuite, CuString* details)
|
||||
{
|
||||
int i;
|
||||
int failCount = 0;
|
||||
|
||||
if (testSuite->failCount == 0)
|
||||
{
|
||||
int passCount = testSuite->count - testSuite->failCount;
|
||||
const char* testWord = passCount == 1 ? "test" : "tests";
|
||||
CuStringAppendFormat(details, "OK (%d %s)\n", passCount, testWord);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (testSuite->failCount == 1)
|
||||
CuStringAppend(details, "There was 1 failure:\n");
|
||||
else
|
||||
CuStringAppendFormat(details, "There were %d failures:\n", testSuite->failCount);
|
||||
|
||||
for (i = 0 ; i < testSuite->count ; ++i)
|
||||
{
|
||||
CuTest* testCase = testSuite->list[i];
|
||||
if (testCase->failed)
|
||||
{
|
||||
failCount++;
|
||||
CuStringAppendFormat(details, "%d) %s: %s\n",
|
||||
failCount, testCase->name, testCase->message);
|
||||
}
|
||||
}
|
||||
CuStringAppend(details, "\n!!!FAILURES!!!\n");
|
||||
|
||||
CuStringAppendFormat(details, "Runs: %d ", testSuite->count);
|
||||
CuStringAppendFormat(details, "Passes: %d ", testSuite->count - testSuite->failCount);
|
||||
CuStringAppendFormat(details, "Fails: %d\n", testSuite->failCount);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,111 @@
|
|||
#ifndef CU_TEST_H
|
||||
#define CU_TEST_H
|
||||
|
||||
#include <setjmp.h>
|
||||
#include <stdarg.h>
|
||||
|
||||
/* CuString */
|
||||
|
||||
char* CuStrAlloc(int size);
|
||||
char* CuStrCopy(const char* old);
|
||||
|
||||
#define CU_ALLOC(TYPE) ((TYPE*) malloc(sizeof(TYPE)))
|
||||
|
||||
#define HUGE_STRING_LEN 8192
|
||||
#define STRING_MAX 256
|
||||
#define STRING_INC 256
|
||||
|
||||
typedef struct
|
||||
{
|
||||
int length;
|
||||
int size;
|
||||
char* buffer;
|
||||
} CuString;
|
||||
|
||||
void CuStringInit(CuString* str);
|
||||
CuString* CuStringNew(void);
|
||||
void CuStringRead(CuString* str, const char* path);
|
||||
void CuStringAppend(CuString* str, const char* text);
|
||||
void CuStringAppendChar(CuString* str, char ch);
|
||||
void CuStringAppendFormat(CuString* str, const char* format, ...);
|
||||
void CuStringInsert(CuString* str, const char* text, int pos);
|
||||
void CuStringResize(CuString* str, int newSize);
|
||||
|
||||
/* CuTest */
|
||||
|
||||
typedef struct CuTest CuTest;
|
||||
|
||||
typedef void (*TestFunction)(CuTest *);
|
||||
|
||||
struct CuTest
|
||||
{
|
||||
const char* name;
|
||||
TestFunction function;
|
||||
int failed;
|
||||
int ran;
|
||||
const char* message;
|
||||
jmp_buf *jumpBuf;
|
||||
};
|
||||
|
||||
void CuTestInit(CuTest* t, const char* name, TestFunction function);
|
||||
CuTest* CuTestNew(const char* name, TestFunction function);
|
||||
void CuTestRun(CuTest* tc);
|
||||
|
||||
/* Internal versions of assert functions -- use the public versions */
|
||||
void CuFail_Line(CuTest* tc, const char* file, int line, const char* message2, const char* message);
|
||||
void CuAssert_Line(CuTest* tc, const char* file, int line, const char* message, int condition);
|
||||
void CuAssertStrEquals_LineMsg(CuTest* tc,
|
||||
const char* file, int line, const char* message,
|
||||
const char* expected, const char* actual);
|
||||
void CuAssertIntEquals_LineMsg(CuTest* tc,
|
||||
const char* file, int line, const char* message,
|
||||
int expected, int actual);
|
||||
void CuAssertDblEquals_LineMsg(CuTest* tc,
|
||||
const char* file, int line, const char* message,
|
||||
double expected, double actual, double delta);
|
||||
void CuAssertPtrEquals_LineMsg(CuTest* tc,
|
||||
const char* file, int line, const char* message,
|
||||
void* expected, void* actual);
|
||||
|
||||
/* public assert functions */
|
||||
|
||||
#define CuFail(tc, ms) CuFail_Line( (tc), __FILE__, __LINE__, NULL, (ms))
|
||||
#define CuAssert(tc, ms, cond) CuAssert_Line((tc), __FILE__, __LINE__, (ms), (cond))
|
||||
#define CuAssertTrue(tc, cond) CuAssert_Line((tc), __FILE__, __LINE__, "assert failed", (cond))
|
||||
|
||||
#define CuAssertStrEquals(tc,ex,ac) CuAssertStrEquals_LineMsg((tc),__FILE__,__LINE__,NULL,(ex),(ac))
|
||||
#define CuAssertStrEquals_Msg(tc,ms,ex,ac) CuAssertStrEquals_LineMsg((tc),__FILE__,__LINE__,(ms),(ex),(ac))
|
||||
#define CuAssertIntEquals(tc,ex,ac) CuAssertIntEquals_LineMsg((tc),__FILE__,__LINE__,NULL,(ex),(ac))
|
||||
#define CuAssertIntEquals_Msg(tc,ms,ex,ac) CuAssertIntEquals_LineMsg((tc),__FILE__,__LINE__,(ms),(ex),(ac))
|
||||
#define CuAssertDblEquals(tc,ex,ac,dl) CuAssertDblEquals_LineMsg((tc),__FILE__,__LINE__,NULL,(ex),(ac),(dl))
|
||||
#define CuAssertDblEquals_Msg(tc,ms,ex,ac,dl) CuAssertDblEquals_LineMsg((tc),__FILE__,__LINE__,(ms),(ex),(ac),(dl))
|
||||
#define CuAssertPtrEquals(tc,ex,ac) CuAssertPtrEquals_LineMsg((tc),__FILE__,__LINE__,NULL,(ex),(ac))
|
||||
#define CuAssertPtrEquals_Msg(tc,ms,ex,ac) CuAssertPtrEquals_LineMsg((tc),__FILE__,__LINE__,(ms),(ex),(ac))
|
||||
|
||||
#define CuAssertPtrNotNull(tc,p) CuAssert_Line((tc),__FILE__,__LINE__,"null pointer unexpected",(p != NULL))
|
||||
#define CuAssertPtrNotNullMsg(tc,msg,p) CuAssert_Line((tc),__FILE__,__LINE__,(msg),(p != NULL))
|
||||
|
||||
/* CuSuite */
|
||||
|
||||
#define MAX_TEST_CASES 1024
|
||||
|
||||
#define SUITE_ADD_TEST(SUITE,TEST) CuSuiteAdd(SUITE, CuTestNew(#TEST, TEST))
|
||||
|
||||
typedef struct
|
||||
{
|
||||
int count;
|
||||
CuTest* list[MAX_TEST_CASES];
|
||||
int failCount;
|
||||
|
||||
} CuSuite;
|
||||
|
||||
|
||||
void CuSuiteInit(CuSuite* testSuite);
|
||||
CuSuite* CuSuiteNew(void);
|
||||
void CuSuiteAdd(CuSuite* testSuite, CuTest *testCase);
|
||||
void CuSuiteAddSuite(CuSuite* testSuite, CuSuite* testSuite2);
|
||||
void CuSuiteRun(CuSuite* testSuite);
|
||||
void CuSuiteSummary(CuSuite* testSuite, CuString* summary);
|
||||
void CuSuiteDetails(CuSuite* testSuite, CuString* details);
|
||||
|
||||
#endif /* CU_TEST_H */
|
|
@ -0,0 +1,709 @@
|
|||
#include <assert.h>
|
||||
#include <setjmp.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "CuTest.h"
|
||||
|
||||
/*-------------------------------------------------------------------------*
|
||||
* Helper functions
|
||||
*-------------------------------------------------------------------------*/
|
||||
|
||||
#define CompareAsserts(tc, message, expected, actual) X_CompareAsserts((tc), __FILE__, __LINE__, (message), (expected), (actual))
|
||||
|
||||
static void X_CompareAsserts(CuTest* tc, const char *file, int line, const char* message, const char* expected, const char* actual)
|
||||
{
|
||||
int mismatch;
|
||||
if (expected == NULL || actual == NULL) {
|
||||
mismatch = (expected != NULL || actual != NULL);
|
||||
} else {
|
||||
const char *front = __FILE__ ":";
|
||||
const size_t frontLen = strlen(front);
|
||||
const size_t expectedLen = strlen(expected);
|
||||
|
||||
const char *matchStr = actual;
|
||||
|
||||
mismatch = (strncmp(matchStr, front, frontLen) != 0);
|
||||
if (!mismatch) {
|
||||
matchStr = strchr(matchStr + frontLen, ':');
|
||||
mismatch |= (matchStr == NULL || strncmp(matchStr, ": ", 2));
|
||||
if (!mismatch) {
|
||||
matchStr += 2;
|
||||
mismatch |= (strncmp(matchStr, expected, expectedLen) != 0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
CuAssert_Line(tc, file, line, message, !mismatch);
|
||||
}
|
||||
|
||||
/*-------------------------------------------------------------------------*
|
||||
* CuString Test
|
||||
*-------------------------------------------------------------------------*/
|
||||
|
||||
void TestCuStringNew(CuTest* tc)
|
||||
{
|
||||
CuString* str = CuStringNew();
|
||||
CuAssertTrue(tc, 0 == str->length);
|
||||
CuAssertTrue(tc, 0 != str->size);
|
||||
CuAssertStrEquals(tc, "", str->buffer);
|
||||
}
|
||||
|
||||
|
||||
void TestCuStringAppend(CuTest* tc)
|
||||
{
|
||||
CuString* str = CuStringNew();
|
||||
CuStringAppend(str, "hello");
|
||||
CuAssertIntEquals(tc, 5, str->length);
|
||||
CuAssertStrEquals(tc, "hello", str->buffer);
|
||||
CuStringAppend(str, " world");
|
||||
CuAssertIntEquals(tc, 11, str->length);
|
||||
CuAssertStrEquals(tc, "hello world", str->buffer);
|
||||
}
|
||||
|
||||
|
||||
void TestCuStringAppendNULL(CuTest* tc)
|
||||
{
|
||||
CuString* str = CuStringNew();
|
||||
CuStringAppend(str, NULL);
|
||||
CuAssertIntEquals(tc, 4, str->length);
|
||||
CuAssertStrEquals(tc, "NULL", str->buffer);
|
||||
}
|
||||
|
||||
|
||||
void TestCuStringAppendChar(CuTest* tc)
|
||||
{
|
||||
CuString* str = CuStringNew();
|
||||
CuStringAppendChar(str, 'a');
|
||||
CuStringAppendChar(str, 'b');
|
||||
CuStringAppendChar(str, 'c');
|
||||
CuStringAppendChar(str, 'd');
|
||||
CuAssertIntEquals(tc, 4, str->length);
|
||||
CuAssertStrEquals(tc, "abcd", str->buffer);
|
||||
}
|
||||
|
||||
|
||||
void TestCuStringInserts(CuTest* tc)
|
||||
{
|
||||
CuString* str = CuStringNew();
|
||||
CuStringAppend(str, "world");
|
||||
CuAssertIntEquals(tc, 5, str->length);
|
||||
CuAssertStrEquals(tc, "world", str->buffer);
|
||||
CuStringInsert(str, "hell", 0);
|
||||
CuAssertIntEquals(tc, 9, str->length);
|
||||
CuAssertStrEquals(tc, "hellworld", str->buffer);
|
||||
CuStringInsert(str, "o ", 4);
|
||||
CuAssertIntEquals(tc, 11, str->length);
|
||||
CuAssertStrEquals(tc, "hello world", str->buffer);
|
||||
CuStringInsert(str, "!", 11);
|
||||
CuAssertIntEquals(tc, 12, str->length);
|
||||
CuAssertStrEquals(tc, "hello world!", str->buffer);
|
||||
}
|
||||
|
||||
|
||||
void TestCuStringResizes(CuTest* tc)
|
||||
{
|
||||
CuString* str = CuStringNew();
|
||||
int i;
|
||||
for(i = 0 ; i < STRING_MAX ; ++i)
|
||||
{
|
||||
CuStringAppend(str, "aa");
|
||||
}
|
||||
CuAssertTrue(tc, STRING_MAX * 2 == str->length);
|
||||
CuAssertTrue(tc, STRING_MAX * 2 <= str->size);
|
||||
}
|
||||
|
||||
CuSuite* CuStringGetSuite(void)
|
||||
{
|
||||
CuSuite* suite = CuSuiteNew();
|
||||
|
||||
SUITE_ADD_TEST(suite, TestCuStringNew);
|
||||
SUITE_ADD_TEST(suite, TestCuStringAppend);
|
||||
SUITE_ADD_TEST(suite, TestCuStringAppendNULL);
|
||||
SUITE_ADD_TEST(suite, TestCuStringAppendChar);
|
||||
SUITE_ADD_TEST(suite, TestCuStringInserts);
|
||||
SUITE_ADD_TEST(suite, TestCuStringResizes);
|
||||
|
||||
return suite;
|
||||
}
|
||||
|
||||
/*-------------------------------------------------------------------------*
|
||||
* CuTest Test
|
||||
*-------------------------------------------------------------------------*/
|
||||
|
||||
void TestPasses(CuTest* tc)
|
||||
{
|
||||
CuAssert(tc, "test should pass", 1 == 0 + 1);
|
||||
}
|
||||
|
||||
void zTestFails(CuTest* tc)
|
||||
{
|
||||
CuAssert(tc, "test should fail", 1 == 1 + 1);
|
||||
}
|
||||
|
||||
|
||||
void TestCuTestNew(CuTest* tc)
|
||||
{
|
||||
CuTest* tc2 = CuTestNew("MyTest", TestPasses);
|
||||
CuAssertStrEquals(tc, "MyTest", tc2->name);
|
||||
CuAssertTrue(tc, !tc2->failed);
|
||||
CuAssertTrue(tc, tc2->message == NULL);
|
||||
CuAssertTrue(tc, tc2->function == TestPasses);
|
||||
CuAssertTrue(tc, tc2->ran == 0);
|
||||
CuAssertTrue(tc, tc2->jumpBuf == NULL);
|
||||
}
|
||||
|
||||
|
||||
void TestCuTestInit(CuTest *tc)
|
||||
{
|
||||
CuTest tc2;
|
||||
CuTestInit(&tc2, "MyTest", TestPasses);
|
||||
CuAssertStrEquals(tc, "MyTest", tc2.name);
|
||||
CuAssertTrue(tc, !tc2.failed);
|
||||
CuAssertTrue(tc, tc2.message == NULL);
|
||||
CuAssertTrue(tc, tc2.function == TestPasses);
|
||||
CuAssertTrue(tc, tc2.ran == 0);
|
||||
CuAssertTrue(tc, tc2.jumpBuf == NULL);
|
||||
}
|
||||
|
||||
void TestCuAssert(CuTest* tc)
|
||||
{
|
||||
CuTest tc2;
|
||||
CuTestInit(&tc2, "MyTest", TestPasses);
|
||||
|
||||
CuAssert(&tc2, "test 1", 5 == 4 + 1);
|
||||
CuAssertTrue(tc, !tc2.failed);
|
||||
CuAssertTrue(tc, tc2.message == NULL);
|
||||
|
||||
CuAssert(&tc2, "test 2", 0);
|
||||
CuAssertTrue(tc, tc2.failed);
|
||||
CompareAsserts(tc, "CuAssert didn't fail", "test 2", tc2.message);
|
||||
|
||||
CuAssert(&tc2, "test 3", 1);
|
||||
CuAssertTrue(tc, tc2.failed);
|
||||
CompareAsserts(tc, "CuAssert didn't fail", "test 2", tc2.message);
|
||||
|
||||
CuAssert(&tc2, "test 4", 0);
|
||||
CuAssertTrue(tc, tc2.failed);
|
||||
CompareAsserts(tc, "CuAssert didn't fail", "test 4", tc2.message);
|
||||
|
||||
}
|
||||
|
||||
void TestCuAssertPtrEquals_Success(CuTest* tc)
|
||||
{
|
||||
CuTest tc2;
|
||||
int x;
|
||||
|
||||
CuTestInit(&tc2, "MyTest", TestPasses);
|
||||
|
||||
/* test success case */
|
||||
CuAssertPtrEquals(&tc2, &x, &x);
|
||||
CuAssertTrue(tc, ! tc2.failed);
|
||||
CuAssertTrue(tc, NULL == tc2.message);
|
||||
}
|
||||
|
||||
void TestCuAssertPtrEquals_Failure(CuTest* tc)
|
||||
{
|
||||
CuTest tc2;
|
||||
int x;
|
||||
int* nullPtr = NULL;
|
||||
char expected_message[STRING_MAX];
|
||||
|
||||
CuTestInit(&tc2, "MyTest", TestPasses);
|
||||
|
||||
/* test failing case */
|
||||
sprintf(expected_message, "expected pointer <0x%p> but was <0x%p>", nullPtr, &x);
|
||||
CuAssertPtrEquals(&tc2, NULL, &x);
|
||||
CuAssertTrue(tc, tc2.failed);
|
||||
CompareAsserts(tc, "CuAssertPtrEquals failed", expected_message, tc2.message);
|
||||
}
|
||||
|
||||
void TestCuAssertPtrNotNull_Success(CuTest* tc)
|
||||
{
|
||||
CuTest tc2;
|
||||
int x;
|
||||
|
||||
CuTestInit(&tc2, "MyTest", TestPasses);
|
||||
|
||||
/* test success case */
|
||||
CuAssertPtrNotNull(&tc2, &x);
|
||||
CuAssertTrue(tc, ! tc2.failed);
|
||||
CuAssertTrue(tc, NULL == tc2.message);
|
||||
}
|
||||
|
||||
void TestCuAssertPtrNotNull_Failure(CuTest* tc)
|
||||
{
|
||||
CuTest tc2;
|
||||
|
||||
CuTestInit(&tc2, "MyTest", TestPasses);
|
||||
|
||||
/* test failing case */
|
||||
CuAssertPtrNotNull(&tc2, NULL);
|
||||
CuAssertTrue(tc, tc2.failed);
|
||||
CompareAsserts(tc, "CuAssertPtrNotNull failed", "null pointer unexpected", tc2.message);
|
||||
}
|
||||
|
||||
void TestCuTestRun(CuTest* tc)
|
||||
{
|
||||
CuTest tc2;
|
||||
CuTestInit(&tc2, "MyTest", zTestFails);
|
||||
CuTestRun(&tc2);
|
||||
|
||||
CuAssertStrEquals(tc, "MyTest", tc2.name);
|
||||
CuAssertTrue(tc, tc2.failed);
|
||||
CuAssertTrue(tc, tc2.ran);
|
||||
CompareAsserts(tc, "TestRun failed", "test should fail", tc2.message);
|
||||
}
|
||||
|
||||
/*-------------------------------------------------------------------------*
|
||||
* CuSuite Test
|
||||
*-------------------------------------------------------------------------*/
|
||||
|
||||
void TestCuSuiteInit(CuTest* tc)
|
||||
{
|
||||
CuSuite ts;
|
||||
CuSuiteInit(&ts);
|
||||
CuAssertTrue(tc, ts.count == 0);
|
||||
CuAssertTrue(tc, ts.failCount == 0);
|
||||
}
|
||||
|
||||
void TestCuSuiteNew(CuTest* tc)
|
||||
{
|
||||
CuSuite* ts = CuSuiteNew();
|
||||
CuAssertTrue(tc, ts->count == 0);
|
||||
CuAssertTrue(tc, ts->failCount == 0);
|
||||
}
|
||||
|
||||
void TestCuSuiteAddTest(CuTest* tc)
|
||||
{
|
||||
CuSuite ts;
|
||||
CuTest tc2;
|
||||
|
||||
CuSuiteInit(&ts);
|
||||
CuTestInit(&tc2, "MyTest", zTestFails);
|
||||
|
||||
CuSuiteAdd(&ts, &tc2);
|
||||
CuAssertTrue(tc, ts.count == 1);
|
||||
|
||||
CuAssertStrEquals(tc, "MyTest", ts.list[0]->name);
|
||||
}
|
||||
|
||||
void TestCuSuiteAddSuite(CuTest* tc)
|
||||
{
|
||||
CuSuite* ts1 = CuSuiteNew();
|
||||
CuSuite* ts2 = CuSuiteNew();
|
||||
|
||||
CuSuiteAdd(ts1, CuTestNew("TestFails1", zTestFails));
|
||||
CuSuiteAdd(ts1, CuTestNew("TestFails2", zTestFails));
|
||||
|
||||
CuSuiteAdd(ts2, CuTestNew("TestFails3", zTestFails));
|
||||
CuSuiteAdd(ts2, CuTestNew("TestFails4", zTestFails));
|
||||
|
||||
CuSuiteAddSuite(ts1, ts2);
|
||||
CuAssertIntEquals(tc, 4, ts1->count);
|
||||
|
||||
CuAssertStrEquals(tc, "TestFails1", ts1->list[0]->name);
|
||||
CuAssertStrEquals(tc, "TestFails2", ts1->list[1]->name);
|
||||
CuAssertStrEquals(tc, "TestFails3", ts1->list[2]->name);
|
||||
CuAssertStrEquals(tc, "TestFails4", ts1->list[3]->name);
|
||||
}
|
||||
|
||||
void TestCuSuiteRun(CuTest* tc)
|
||||
{
|
||||
CuSuite ts;
|
||||
CuTest tc1, tc2, tc3, tc4;
|
||||
|
||||
CuSuiteInit(&ts);
|
||||
CuTestInit(&tc1, "TestPasses", TestPasses);
|
||||
CuTestInit(&tc2, "TestPasses", TestPasses);
|
||||
CuTestInit(&tc3, "TestFails", zTestFails);
|
||||
CuTestInit(&tc4, "TestFails", zTestFails);
|
||||
|
||||
CuSuiteAdd(&ts, &tc1);
|
||||
CuSuiteAdd(&ts, &tc2);
|
||||
CuSuiteAdd(&ts, &tc3);
|
||||
CuSuiteAdd(&ts, &tc4);
|
||||
CuAssertTrue(tc, ts.count == 4);
|
||||
|
||||
CuSuiteRun(&ts);
|
||||
CuAssertTrue(tc, ts.count - ts.failCount == 2);
|
||||
CuAssertTrue(tc, ts.failCount == 2);
|
||||
}
|
||||
|
||||
void TestCuSuiteSummary(CuTest* tc)
|
||||
{
|
||||
CuSuite ts;
|
||||
CuTest tc1, tc2;
|
||||
CuString summary;
|
||||
|
||||
CuSuiteInit(&ts);
|
||||
CuTestInit(&tc1, "TestPasses", TestPasses);
|
||||
CuTestInit(&tc2, "TestFails", zTestFails);
|
||||
CuStringInit(&summary);
|
||||
|
||||
CuSuiteAdd(&ts, &tc1);
|
||||
CuSuiteAdd(&ts, &tc2);
|
||||
CuSuiteRun(&ts);
|
||||
|
||||
CuSuiteSummary(&ts, &summary);
|
||||
|
||||
CuAssertTrue(tc, ts.count == 2);
|
||||
CuAssertTrue(tc, ts.failCount == 1);
|
||||
CuAssertStrEquals(tc, ".F\n\n", summary.buffer);
|
||||
}
|
||||
|
||||
|
||||
void TestCuSuiteDetails_SingleFail(CuTest* tc)
|
||||
{
|
||||
CuSuite ts;
|
||||
CuTest tc1, tc2;
|
||||
CuString details;
|
||||
const char* front;
|
||||
const char* back;
|
||||
|
||||
CuSuiteInit(&ts);
|
||||
CuTestInit(&tc1, "TestPasses", TestPasses);
|
||||
CuTestInit(&tc2, "TestFails", zTestFails);
|
||||
CuStringInit(&details);
|
||||
|
||||
CuSuiteAdd(&ts, &tc1);
|
||||
CuSuiteAdd(&ts, &tc2);
|
||||
CuSuiteRun(&ts);
|
||||
|
||||
CuSuiteDetails(&ts, &details);
|
||||
|
||||
CuAssertTrue(tc, ts.count == 2);
|
||||
CuAssertTrue(tc, ts.failCount == 1);
|
||||
|
||||
front = "There was 1 failure:\n"
|
||||
"1) TestFails: ";
|
||||
back = "test should fail\n"
|
||||
"\n!!!FAILURES!!!\n"
|
||||
"Runs: 2 Passes: 1 Fails: 1\n";
|
||||
|
||||
CuAssertStrEquals(tc, back, details.buffer + strlen(details.buffer) - strlen(back));
|
||||
details.buffer[strlen(front)] = 0;
|
||||
CuAssertStrEquals(tc, front, details.buffer);
|
||||
}
|
||||
|
||||
|
||||
void TestCuSuiteDetails_SinglePass(CuTest* tc)
|
||||
{
|
||||
CuSuite ts;
|
||||
CuTest tc1;
|
||||
CuString details;
|
||||
const char* expected;
|
||||
|
||||
CuSuiteInit(&ts);
|
||||
CuTestInit(&tc1, "TestPasses", TestPasses);
|
||||
CuStringInit(&details);
|
||||
|
||||
CuSuiteAdd(&ts, &tc1);
|
||||
CuSuiteRun(&ts);
|
||||
|
||||
CuSuiteDetails(&ts, &details);
|
||||
|
||||
CuAssertTrue(tc, ts.count == 1);
|
||||
CuAssertTrue(tc, ts.failCount == 0);
|
||||
|
||||
expected =
|
||||
"OK (1 test)\n";
|
||||
|
||||
CuAssertStrEquals(tc, expected, details.buffer);
|
||||
}
|
||||
|
||||
void TestCuSuiteDetails_MultiplePasses(CuTest* tc)
|
||||
{
|
||||
CuSuite ts;
|
||||
CuTest tc1, tc2;
|
||||
CuString details;
|
||||
const char* expected;
|
||||
|
||||
CuSuiteInit(&ts);
|
||||
CuTestInit(&tc1, "TestPasses", TestPasses);
|
||||
CuTestInit(&tc2, "TestPasses", TestPasses);
|
||||
CuStringInit(&details);
|
||||
|
||||
CuSuiteAdd(&ts, &tc1);
|
||||
CuSuiteAdd(&ts, &tc2);
|
||||
CuSuiteRun(&ts);
|
||||
|
||||
CuSuiteDetails(&ts, &details);
|
||||
|
||||
CuAssertTrue(tc, ts.count == 2);
|
||||
CuAssertTrue(tc, ts.failCount == 0);
|
||||
|
||||
expected =
|
||||
"OK (2 tests)\n";
|
||||
|
||||
CuAssertStrEquals(tc, expected, details.buffer);
|
||||
}
|
||||
|
||||
void TestCuSuiteDetails_MultipleFails(CuTest* tc)
|
||||
{
|
||||
CuSuite ts;
|
||||
CuTest tc1, tc2;
|
||||
CuString details;
|
||||
const char* front;
|
||||
const char* mid;
|
||||
const char* back;
|
||||
|
||||
CuSuiteInit(&ts);
|
||||
CuTestInit(&tc1, "TestFails1", zTestFails);
|
||||
CuTestInit(&tc2, "TestFails2", zTestFails);
|
||||
CuStringInit(&details);
|
||||
|
||||
CuSuiteAdd(&ts, &tc1);
|
||||
CuSuiteAdd(&ts, &tc2);
|
||||
CuSuiteRun(&ts);
|
||||
|
||||
CuSuiteDetails(&ts, &details);
|
||||
|
||||
CuAssertTrue(tc, ts.count == 2);
|
||||
CuAssertTrue(tc, ts.failCount == 2);
|
||||
|
||||
front =
|
||||
"There were 2 failures:\n"
|
||||
"1) TestFails1: ";
|
||||
mid = "test should fail\n"
|
||||
"2) TestFails2: ";
|
||||
back = "test should fail\n"
|
||||
"\n!!!FAILURES!!!\n"
|
||||
"Runs: 2 Passes: 0 Fails: 2\n";
|
||||
|
||||
CuAssertStrEquals(tc, back, details.buffer + strlen(details.buffer) - strlen(back));
|
||||
CuAssert(tc, "Couldn't find middle", strstr(details.buffer, mid) != NULL);
|
||||
details.buffer[strlen(front)] = 0;
|
||||
CuAssertStrEquals(tc, front, details.buffer);
|
||||
}
|
||||
|
||||
|
||||
/*-------------------------------------------------------------------------*
|
||||
* Misc Test
|
||||
*-------------------------------------------------------------------------*/
|
||||
|
||||
void TestCuStrCopy(CuTest* tc)
|
||||
{
|
||||
const char* old = "hello world";
|
||||
const char* newStr = CuStrCopy(old);
|
||||
CuAssert(tc, "old is new", strcmp(old, newStr) == 0);
|
||||
}
|
||||
|
||||
|
||||
void TestCuStringAppendFormat(CuTest* tc)
|
||||
{
|
||||
int i;
|
||||
char* text = CuStrAlloc(301); /* long string */
|
||||
CuString* str = CuStringNew();
|
||||
for (i = 0 ; i < 300 ; ++i)
|
||||
text[i] = 'a';
|
||||
text[300] = '\0';
|
||||
CuStringAppendFormat(str, "%s", text);
|
||||
|
||||
/* buffer limit raised to HUGE_STRING_LEN so no overflow */
|
||||
|
||||
CuAssert(tc, "length of str->buffer is 300", 300 == strlen(str->buffer));
|
||||
}
|
||||
|
||||
void TestFail(CuTest* tc)
|
||||
{
|
||||
jmp_buf buf;
|
||||
int pointReached = 0;
|
||||
CuTest* tc2 = CuTestNew("TestFails", zTestFails);
|
||||
tc2->jumpBuf = &buf;
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuFail(tc2, "hello world");
|
||||
pointReached = 1;
|
||||
}
|
||||
CuAssert(tc, "point was not reached", pointReached == 0);
|
||||
}
|
||||
|
||||
void TestAssertStrEquals(CuTest* tc)
|
||||
{
|
||||
jmp_buf buf;
|
||||
CuTest *tc2 = CuTestNew("TestAssertStrEquals", zTestFails);
|
||||
|
||||
const char* expected = "expected <hello> but was <world>";
|
||||
const char *expectedMsg = "some text: expected <hello> but was <world>";
|
||||
|
||||
tc2->jumpBuf = &buf;
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertStrEquals(tc2, "hello", "world");
|
||||
}
|
||||
CuAssertTrue(tc, tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertStrEquals failed", expected, tc2->message);
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertStrEquals_Msg(tc2, "some text", "hello", "world");
|
||||
}
|
||||
CuAssertTrue(tc, tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertStrEquals failed", expectedMsg, tc2->message);
|
||||
}
|
||||
|
||||
void TestAssertStrEquals_NULL(CuTest* tc)
|
||||
{
|
||||
jmp_buf buf;
|
||||
CuTest *tc2 = CuTestNew("TestAssertStrEquals_NULL", zTestFails);
|
||||
|
||||
tc2->jumpBuf = &buf;
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertStrEquals(tc2, NULL, NULL);
|
||||
}
|
||||
CuAssertTrue(tc, !tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertStrEquals_NULL failed", NULL, tc2->message);
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertStrEquals_Msg(tc2, "some text", NULL, NULL);
|
||||
}
|
||||
CuAssertTrue(tc, !tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertStrEquals_NULL failed", NULL, tc2->message);
|
||||
}
|
||||
|
||||
void TestAssertStrEquals_FailNULLStr(CuTest* tc)
|
||||
{
|
||||
jmp_buf buf;
|
||||
CuTest *tc2 = CuTestNew("TestAssertStrEquals_FailNULLStr", zTestFails);
|
||||
|
||||
const char* expected = "expected <hello> but was <NULL>";
|
||||
const char *expectedMsg = "some text: expected <hello> but was <NULL>";
|
||||
|
||||
tc2->jumpBuf = &buf;
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertStrEquals(tc2, "hello", NULL);
|
||||
}
|
||||
CuAssertTrue(tc, tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertStrEquals_FailNULLStr failed", expected, tc2->message);
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertStrEquals_Msg(tc2, "some text", "hello", NULL);
|
||||
}
|
||||
CuAssertTrue(tc, tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertStrEquals_FailNULLStr failed", expectedMsg, tc2->message);
|
||||
}
|
||||
|
||||
void TestAssertStrEquals_FailStrNULL(CuTest* tc)
|
||||
{
|
||||
jmp_buf buf;
|
||||
CuTest *tc2 = CuTestNew("TestAssertStrEquals_FailStrNULL", zTestFails);
|
||||
|
||||
const char* expected = "expected <NULL> but was <hello>";
|
||||
const char *expectedMsg = "some text: expected <NULL> but was <hello>";
|
||||
|
||||
tc2->jumpBuf = &buf;
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertStrEquals(tc2, NULL, "hello");
|
||||
}
|
||||
CuAssertTrue(tc, tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertStrEquals_FailStrNULL failed", expected, tc2->message);
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertStrEquals_Msg(tc2, "some text", NULL, "hello");
|
||||
}
|
||||
CuAssertTrue(tc, tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertStrEquals_FailStrNULL failed", expectedMsg, tc2->message);
|
||||
}
|
||||
|
||||
void TestAssertIntEquals(CuTest* tc)
|
||||
{
|
||||
jmp_buf buf;
|
||||
CuTest *tc2 = CuTestNew("TestAssertIntEquals", zTestFails);
|
||||
const char* expected = "expected <42> but was <32>";
|
||||
const char* expectedMsg = "some text: expected <42> but was <32>";
|
||||
tc2->jumpBuf = &buf;
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertIntEquals(tc2, 42, 32);
|
||||
}
|
||||
CuAssertTrue(tc, tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertIntEquals failed", expected, tc2->message);
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertIntEquals_Msg(tc2, "some text", 42, 32);
|
||||
}
|
||||
CuAssertTrue(tc, tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertStrEquals failed", expectedMsg, tc2->message);
|
||||
}
|
||||
|
||||
void TestAssertDblEquals(CuTest* tc)
|
||||
{
|
||||
jmp_buf buf;
|
||||
double x = 3.33;
|
||||
double y = 10.0 / 3.0;
|
||||
CuTest *tc2 = CuTestNew("TestAssertDblEquals", zTestFails);
|
||||
char expected[STRING_MAX];
|
||||
char expectedMsg[STRING_MAX];
|
||||
sprintf(expected, "expected <%lf> but was <%lf>", x, y);
|
||||
sprintf(expectedMsg, "some text: expected <%lf> but was <%lf>", x, y);
|
||||
|
||||
CuTestInit(tc2, "TestAssertDblEquals", TestPasses);
|
||||
|
||||
CuAssertDblEquals(tc2, x, x, 0.0);
|
||||
CuAssertTrue(tc, ! tc2->failed);
|
||||
CuAssertTrue(tc, NULL == tc2->message);
|
||||
|
||||
CuAssertDblEquals(tc2, x, y, 0.01);
|
||||
CuAssertTrue(tc, ! tc2->failed);
|
||||
CuAssertTrue(tc, NULL == tc2->message);
|
||||
|
||||
tc2->jumpBuf = &buf;
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertDblEquals(tc2, x, y, 0.001);
|
||||
}
|
||||
CuAssertTrue(tc, tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertDblEquals failed", expected, tc2->message);
|
||||
tc2->jumpBuf = &buf;
|
||||
if (setjmp(buf) == 0)
|
||||
{
|
||||
CuAssertDblEquals_Msg(tc2, "some text", x, y, 0.001);
|
||||
}
|
||||
CuAssertTrue(tc, tc2->failed);
|
||||
CompareAsserts(tc, "CuAssertDblEquals failed", expectedMsg, tc2->message);
|
||||
}
|
||||
|
||||
/*-------------------------------------------------------------------------*
|
||||
* main
|
||||
*-------------------------------------------------------------------------*/
|
||||
|
||||
CuSuite* CuGetSuite(void)
|
||||
{
|
||||
CuSuite* suite = CuSuiteNew();
|
||||
|
||||
SUITE_ADD_TEST(suite, TestCuStringAppendFormat);
|
||||
SUITE_ADD_TEST(suite, TestCuStrCopy);
|
||||
SUITE_ADD_TEST(suite, TestFail);
|
||||
SUITE_ADD_TEST(suite, TestAssertStrEquals);
|
||||
SUITE_ADD_TEST(suite, TestAssertStrEquals_NULL);
|
||||
SUITE_ADD_TEST(suite, TestAssertStrEquals_FailStrNULL);
|
||||
SUITE_ADD_TEST(suite, TestAssertStrEquals_FailNULLStr);
|
||||
SUITE_ADD_TEST(suite, TestAssertIntEquals);
|
||||
SUITE_ADD_TEST(suite, TestAssertDblEquals);
|
||||
|
||||
SUITE_ADD_TEST(suite, TestCuTestNew);
|
||||
SUITE_ADD_TEST(suite, TestCuTestInit);
|
||||
SUITE_ADD_TEST(suite, TestCuAssert);
|
||||
SUITE_ADD_TEST(suite, TestCuAssertPtrEquals_Success);
|
||||
SUITE_ADD_TEST(suite, TestCuAssertPtrEquals_Failure);
|
||||
SUITE_ADD_TEST(suite, TestCuAssertPtrNotNull_Success);
|
||||
SUITE_ADD_TEST(suite, TestCuAssertPtrNotNull_Failure);
|
||||
SUITE_ADD_TEST(suite, TestCuTestRun);
|
||||
|
||||
SUITE_ADD_TEST(suite, TestCuSuiteInit);
|
||||
SUITE_ADD_TEST(suite, TestCuSuiteNew);
|
||||
SUITE_ADD_TEST(suite, TestCuSuiteAddTest);
|
||||
SUITE_ADD_TEST(suite, TestCuSuiteAddSuite);
|
||||
SUITE_ADD_TEST(suite, TestCuSuiteRun);
|
||||
SUITE_ADD_TEST(suite, TestCuSuiteSummary);
|
||||
SUITE_ADD_TEST(suite, TestCuSuiteDetails_SingleFail);
|
||||
SUITE_ADD_TEST(suite, TestCuSuiteDetails_SinglePass);
|
||||
SUITE_ADD_TEST(suite, TestCuSuiteDetails_MultiplePasses);
|
||||
SUITE_ADD_TEST(suite, TestCuSuiteDetails_MultipleFails);
|
||||
|
||||
return suite;
|
||||
}
|
|
@ -0,0 +1,209 @@
|
|||
HOW TO USE
|
||||
|
||||
You can use CuTest to create unit tests to drive your development
|
||||
in the style of Extreme Programming. You can also add unit tests to
|
||||
existing code to ensure that it works as you suspect.
|
||||
|
||||
Your unit tests are an investment. They let you to change your
|
||||
code and add new features confidently without worrying about
|
||||
accidentally breaking earlier features.
|
||||
|
||||
|
||||
LICENSING
|
||||
|
||||
For details on licensing see license.txt.
|
||||
|
||||
|
||||
GETTING STARTED
|
||||
|
||||
To add unit testing to your C code the only files you need are
|
||||
CuTest.c and CuTest.h.
|
||||
|
||||
CuTestTest.c and AllTests.c have been included to provide an
|
||||
example of how to write unit tests and then how to aggregate them
|
||||
into suites and into a single AllTests.c file. Suites allow you
|
||||
to put group tests into logical sets. AllTests.c combines all the
|
||||
suites and runs them.
|
||||
|
||||
You should not have to look inside CuTest.c. Looking in
|
||||
CuTestTest.c and AllTests.c (for example usage) should be
|
||||
sufficient.
|
||||
|
||||
After downloading the sources, run your compiler to create an
|
||||
executable called AllTests.exe. For example, if you are using
|
||||
Windows with the cl.exe compiler you would type:
|
||||
|
||||
cl.exe AllTests.c CuTest.c CuTestTest.c
|
||||
AllTests.exe
|
||||
|
||||
This will run all the unit tests associated with CuTest and print
|
||||
the output on the console. You can replace cl.exe with gcc or
|
||||
your favorite compiler in the command above.
|
||||
|
||||
|
||||
DETAILED EXAMPLE
|
||||
|
||||
Here is a more detailed example. We will work through a simple
|
||||
test first exercise. The goal is to create a library of string
|
||||
utilities. First, lets write a function that converts a
|
||||
null-terminated string to all upper case.
|
||||
|
||||
Ensure that CuTest.c and CuTest.h are accessible from your C
|
||||
project. Next, create a file called StrUtil.c with these
|
||||
contents:
|
||||
|
||||
#include "CuTest.h"
|
||||
|
||||
char* StrToUpper(char* str) {
|
||||
return str;
|
||||
}
|
||||
|
||||
void TestStrToUpper(CuTest *tc) {
|
||||
char* input = strdup("hello world");
|
||||
char* actual = StrToUpper(input);
|
||||
char* expected = "HELLO WORLD";
|
||||
CuAssertStrEquals(tc, expected, actual);
|
||||
}
|
||||
|
||||
CuSuite* StrUtilGetSuite() {
|
||||
CuSuite* suite = CuSuiteNew();
|
||||
SUITE_ADD_TEST(suite, TestStrToUpper);
|
||||
return suite;
|
||||
}
|
||||
|
||||
Create another file called AllTests.c with these contents:
|
||||
|
||||
#include "CuTest.h"
|
||||
|
||||
CuSuite* StrUtilGetSuite();
|
||||
|
||||
void RunAllTests(void) {
|
||||
CuString *output = CuStringNew();
|
||||
CuSuite* suite = CuSuiteNew();
|
||||
|
||||
CuSuiteAddSuite(suite, StrUtilGetSuite());
|
||||
|
||||
CuSuiteRun(suite);
|
||||
CuSuiteSummary(suite, output);
|
||||
CuSuiteDetails(suite, output);
|
||||
printf("%s\n", output->buffer);
|
||||
}
|
||||
|
||||
int main(void) {
|
||||
RunAllTests();
|
||||
}
|
||||
|
||||
Then type this on the command line:
|
||||
|
||||
gcc AllTests.c CuTest.c StrUtil.c
|
||||
|
||||
to compile. You can replace gcc with your favorite compiler.
|
||||
CuTest should be portable enough to handle all Windows and Unix
|
||||
compilers. Then to run the tests type:
|
||||
|
||||
a.out
|
||||
|
||||
This will print an error because we haven't implemented the
|
||||
StrToUpper function correctly. We are just returning the string
|
||||
without changing it to upper case.
|
||||
|
||||
char* StrToUpper(char* str) {
|
||||
return str;
|
||||
}
|
||||
|
||||
Rewrite this as follows:
|
||||
|
||||
char* StrToUpper(char* str) {
|
||||
char* p;
|
||||
for (p = str ; *p ; ++p) *p = toupper(*p);
|
||||
return str;
|
||||
}
|
||||
|
||||
Recompile and run the tests again. The test should pass this
|
||||
time.
|
||||
|
||||
|
||||
WHAT TO DO NEXT
|
||||
|
||||
At this point you might want to write more tests for the
|
||||
StrToUpper function. Here are some ideas:
|
||||
|
||||
TestStrToUpper_EmptyString : pass in ""
|
||||
TestStrToUpper_UpperCase : pass in "HELLO WORLD"
|
||||
TestStrToUpper_MixedCase : pass in "HELLO world"
|
||||
TestStrToUpper_Numbers : pass in "1234 hello"
|
||||
|
||||
As you write each one of these tests add it to StrUtilGetSuite
|
||||
function. If you don't the tests won't be run. Later as you write
|
||||
other functions and write tests for them be sure to include those
|
||||
in StrUtilGetSuite also. The StrUtilGetSuite function should
|
||||
include all the tests in StrUtil.c
|
||||
|
||||
Over time you will create another file called FunkyStuff.c
|
||||
containing other functions unrelated to StrUtil. Follow the same
|
||||
pattern. Create a FunkyStuffGetSuite function in FunkyStuff.c.
|
||||
And add FunkyStuffGetSuite to AllTests.c.
|
||||
|
||||
The framework is designed in the way it is so that it is easy to
|
||||
organize a lot of tests.
|
||||
|
||||
THE BIG PICTURE
|
||||
|
||||
Each individual test corresponds to a CuTest. These are grouped
|
||||
to form a CuSuite. CuSuites can hold CuTests or other CuSuites.
|
||||
AllTests.c collects all the CuSuites in the program into a single
|
||||
CuSuite which it then runs as a single CuSuite.
|
||||
|
||||
The project is open source so feel free to take a peek under the
|
||||
hood at the CuTest.c file to see how it works. CuTestTest.c
|
||||
contains tests for CuTest.c. So CuTest tests itself.
|
||||
|
||||
Since AllTests.c has a main() you will need to exclude this when
|
||||
you are building your product. Here is a nicer way to do this if
|
||||
you want to avoid messing with multiple builds. Remove the main()
|
||||
in AllTests.c. Note that it just calls RunAllTests(). Instead
|
||||
we'll call this directly from the main program.
|
||||
|
||||
Now in the main() of the actual program check to see if the
|
||||
command line option "--test" was passed. If it was then I call
|
||||
RunAllTests() from AllTests.c. Otherwise run the real program.
|
||||
|
||||
Shipping the tests with the code can be useful. If you customers
|
||||
complain about a problem you can ask them to run the unit tests
|
||||
and send you the output. This can help you to quickly isolate the
|
||||
piece of your system that is malfunctioning in the customer's
|
||||
environment.
|
||||
|
||||
CuTest offers a rich set of CuAssert functions. Here is a list:
|
||||
|
||||
void CuAssert(CuTest* tc, char* message, int condition);
|
||||
void CuAssertTrue(CuTest* tc, int condition);
|
||||
void CuAssertStrEquals(CuTest* tc, char* expected, char* actual);
|
||||
void CuAssertIntEquals(CuTest* tc, int expected, int actual);
|
||||
void CuAssertPtrEquals(CuTest* tc, void* expected, void* actual);
|
||||
void CuAssertPtrNotNull(CuTest* tc, void* pointer);
|
||||
|
||||
The project is open source and so you can add other more powerful
|
||||
asserts to make your tests easier to write and more concise.
|
||||
Please feel free to send me changes you make so that I can
|
||||
incorporate them into future releases.
|
||||
|
||||
If you see any errors in this document please contact me at
|
||||
asimjalis@peakprogramming.com.
|
||||
|
||||
|
||||
AUTOMATING TEST SUITE GENERATION
|
||||
|
||||
make-tests.sh will grep through all the .c files in the current
|
||||
directory and generate the code to run all the tests contained in
|
||||
them. Using this script you don't have to worry about writing
|
||||
AllTests.c or dealing with any of the other suite code.
|
||||
|
||||
|
||||
CREDITS
|
||||
|
||||
[02.23.2003] Dave Glowacki <dglo@hyde.ssec.wisc.edu> has added
|
||||
(1) file name and line numbers to the error messages, (2)
|
||||
AssertDblEquals for doubles, (3) Assert<X>Equals_Msg version of
|
||||
all the Assert<X>Equals to pass in optional message which is
|
||||
printed out on assert failure.
|
|
@ -0,0 +1,38 @@
|
|||
NOTE
|
||||
|
||||
The license is based on the zlib/libpng license. For more details see
|
||||
http://www.opensource.org/licenses/zlib-license.html. The intent of the
|
||||
license is to:
|
||||
|
||||
- keep the license as simple as possible
|
||||
- encourage the use of CuTest in both free and commercial applications
|
||||
and libraries
|
||||
- keep the source code together
|
||||
- give credit to the CuTest contributors for their work
|
||||
|
||||
If you ship CuTest in source form with your source distribution, the
|
||||
following license document must be included with it in unaltered form.
|
||||
If you find CuTest useful we would like to hear about it.
|
||||
|
||||
LICENSE
|
||||
|
||||
Copyright (c) 2003 Asim Jalis
|
||||
|
||||
This software is provided 'as-is', without any express or implied
|
||||
warranty. In no event will the authors be held liable for any damages
|
||||
arising from the use of this software.
|
||||
|
||||
Permission is granted to anyone to use this software for any purpose,
|
||||
including commercial applications, and to alter it and redistribute it
|
||||
freely, subject to the following restrictions:
|
||||
|
||||
1. The origin of this software must not be misrepresented; you must not
|
||||
claim that you wrote the original software. If you use this software in
|
||||
a product, an acknowledgment in the product documentation would be
|
||||
appreciated but is not required.
|
||||
|
||||
2. Altered source versions must be plainly marked as such, and must not
|
||||
be misrepresented as being the original software.
|
||||
|
||||
3. This notice may not be removed or altered from any source
|
||||
distribution.
|
|
@ -0,0 +1,51 @@
|
|||
#!/usr/local/bin/bash
|
||||
|
||||
# Auto generate single AllTests file for CuTest.
|
||||
# Searches through all *.c files in the current directory.
|
||||
# Prints to stdout.
|
||||
# Author: Asim Jalis
|
||||
# Date: 01/08/2003
|
||||
|
||||
if test $# -eq 0 ; then FILES=*.c ; else FILES=$* ; fi
|
||||
|
||||
echo '
|
||||
|
||||
/* This is auto-generated code. Edit at your own peril. */
|
||||
|
||||
#include "CuTest.h"
|
||||
|
||||
'
|
||||
|
||||
cat $FILES | grep '^void Test' |
|
||||
sed -e 's/(.*$//' \
|
||||
-e 's/$/(CuTest*);/' \
|
||||
-e 's/^/extern /'
|
||||
|
||||
echo \
|
||||
'
|
||||
|
||||
void RunAllTests(void)
|
||||
{
|
||||
CuString *output = CuStringNew();
|
||||
CuSuite* suite = CuSuiteNew();
|
||||
|
||||
'
|
||||
cat $FILES | grep '^void Test' |
|
||||
sed -e 's/^void //' \
|
||||
-e 's/(.*$//' \
|
||||
-e 's/^/ SUITE_ADD_TEST(suite, /' \
|
||||
-e 's/$/);/'
|
||||
|
||||
echo \
|
||||
'
|
||||
CuSuiteRun(suite);
|
||||
CuSuiteSummary(suite, output);
|
||||
CuSuiteDetails(suite, output);
|
||||
printf("%s\n", output->buffer);
|
||||
}
|
||||
|
||||
int main(void)
|
||||
{
|
||||
RunAllTests();
|
||||
}
|
||||
'
|
|
@ -25,7 +25,7 @@ end
|
|||
|
||||
module( "common", package.seeall, lunit.testcase )
|
||||
|
||||
function test_eventbus_fire()
|
||||
function DISABLE_test_eventbus_fire()
|
||||
local r = region.create(0, 0, "plain")
|
||||
local f = faction.create("noreply@eressea.de", "human", "de")
|
||||
local u = unit.create(f, r)
|
||||
|
|
|
@ -91,8 +91,11 @@ function test_levitate()
|
|||
local u = unit.create(f, r, 2)
|
||||
local s = ship.create(r, "boat")
|
||||
u.ship = s
|
||||
u.age = 20
|
||||
u:set_skill("sailing", 5)
|
||||
u:add_item("money", 100)
|
||||
u:clear_orders()
|
||||
u:add_order("ARBEITE")
|
||||
levitate_ship(u.ship, u, 2, 1)
|
||||
assert_equal(32, u.ship.flags)
|
||||
process_orders()
|
||||
|
|
Loading…
Reference in New Issue