CuTest.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  1. /*
  2. * Copyright (c) 2003 Asim Jalis
  3. *
  4. * This software is provided 'as-is', without any express or implied
  5. * warranty. In no event will the authors be held liable for any
  6. * damages arising from the use of this software.
  7. *
  8. * Permission is granted to anyone to use this software for any
  9. * purpose, including commercial applications, and to alter it and
  10. * redistribute it freely, subject to the following restrictions:
  11. *
  12. * 1. The origin of this software must not be misrepresented; you
  13. * must not claim that you wrote the original software. If you use
  14. * this software in a product, an acknowledgment in the product
  15. * documentation would be appreciated but is not required.
  16. *
  17. * 2. Altered source versions must be plainly marked as such, and
  18. * must not be misrepresented as being the original software.
  19. *
  20. * 3. This notice may not be removed or altered from any source
  21. * distribution.
  22. *-------------------------------------------------------------------------*
  23. *
  24. * Originally obtained from "http://cutest.sourceforge.net/" version 1.4.
  25. *
  26. * See CuTest.h for a list of serf-specific modifications.
  27. */
  28. #include <assert.h>
  29. #include <setjmp.h>
  30. #include <stdlib.h>
  31. #include <stdio.h>
  32. #include <string.h>
  33. #include <math.h>
  34. #include "CuTest.h"
  35. /*-------------------------------------------------------------------------*
  36. * CuStr
  37. *-------------------------------------------------------------------------*/
  38. char* CuStrAlloc(int size)
  39. {
  40. char* newStr = (char*) malloc( sizeof(char) * (size) );
  41. return newStr;
  42. }
  43. char* CuStrCopy(const char* old)
  44. {
  45. int len = strlen(old);
  46. char* newStr = CuStrAlloc(len + 1);
  47. strcpy(newStr, old);
  48. return newStr;
  49. }
  50. /*-------------------------------------------------------------------------*
  51. * CuString
  52. *-------------------------------------------------------------------------*/
  53. void CuStringInit(CuString* str)
  54. {
  55. str->length = 0;
  56. str->size = STRING_MAX;
  57. str->buffer = (char*) malloc(sizeof(char) * str->size);
  58. str->buffer[0] = '\0';
  59. }
  60. CuString* CuStringNew(void)
  61. {
  62. CuString* str = (CuString*) malloc(sizeof(CuString));
  63. str->length = 0;
  64. str->size = STRING_MAX;
  65. str->buffer = (char*) malloc(sizeof(char) * str->size);
  66. str->buffer[0] = '\0';
  67. return str;
  68. }
  69. void CuStringFree(CuString *str)
  70. {
  71. free(str->buffer);
  72. free(str);
  73. }
  74. void CuStringResize(CuString* str, int newSize)
  75. {
  76. str->buffer = (char*) realloc(str->buffer, sizeof(char) * newSize);
  77. str->size = newSize;
  78. }
  79. void CuStringAppend(CuString* str, const char* text)
  80. {
  81. int length;
  82. if (text == NULL) {
  83. text = "NULL";
  84. }
  85. length = strlen(text);
  86. if (str->length + length + 1 >= str->size)
  87. CuStringResize(str, str->length + length + 1 + STRING_INC);
  88. str->length += length;
  89. strcat(str->buffer, text);
  90. }
  91. void CuStringAppendChar(CuString* str, char ch)
  92. {
  93. char text[2];
  94. text[0] = ch;
  95. text[1] = '\0';
  96. CuStringAppend(str, text);
  97. }
  98. void CuStringAppendFormat(CuString* str, const char* format, ...)
  99. {
  100. va_list argp;
  101. char buf[HUGE_STRING_LEN];
  102. va_start(argp, format);
  103. vsprintf(buf, format, argp);
  104. va_end(argp);
  105. CuStringAppend(str, buf);
  106. }
  107. void CuStringInsert(CuString* str, const char* text, int pos)
  108. {
  109. int length = strlen(text);
  110. if (pos > str->length)
  111. pos = str->length;
  112. if (str->length + length + 1 >= str->size)
  113. CuStringResize(str, str->length + length + 1 + STRING_INC);
  114. memmove(str->buffer + pos + length, str->buffer + pos, (str->length - pos) + 1);
  115. str->length += length;
  116. memcpy(str->buffer + pos, text, length);
  117. }
  118. /*-------------------------------------------------------------------------*
  119. * CuTest
  120. *-------------------------------------------------------------------------*/
  121. void CuTestInit(CuTest* t, const char* name, TestFunction function)
  122. {
  123. t->name = CuStrCopy(name);
  124. t->failed = 0;
  125. t->ran = 0;
  126. t->message = NULL;
  127. t->function = function;
  128. t->jumpBuf = NULL;
  129. t->setup = NULL;
  130. t->teardown = NULL;
  131. t->testBaton = NULL;
  132. }
  133. CuTest* CuTestNew(const char* name, TestFunction function)
  134. {
  135. CuTest* tc = CU_ALLOC(CuTest);
  136. CuTestInit(tc, name, function);
  137. return tc;
  138. }
  139. void CuTestFree(CuTest* tc)
  140. {
  141. free(tc->name);
  142. free(tc);
  143. }
  144. void CuTestRun(CuTest* tc)
  145. {
  146. jmp_buf buf;
  147. tc->jumpBuf = &buf;
  148. if (tc->setup)
  149. tc->testBaton = tc->setup(tc);
  150. if (setjmp(buf) == 0)
  151. {
  152. tc->ran = 1;
  153. (tc->function)(tc);
  154. }
  155. if (tc->teardown)
  156. tc->teardown(tc->testBaton);
  157. tc->jumpBuf = 0;
  158. }
  159. static void CuFailInternal(CuTest* tc, const char* file, int line, CuString* string)
  160. {
  161. char buf[HUGE_STRING_LEN];
  162. sprintf(buf, "%s:%d: ", file, line);
  163. CuStringInsert(string, buf, 0);
  164. tc->failed = 1;
  165. tc->message = string->buffer;
  166. if (tc->jumpBuf != 0) longjmp(*(tc->jumpBuf), 0);
  167. }
  168. void CuFail_Line(CuTest* tc, const char* file, int line, const char* message2, const char* message)
  169. {
  170. CuString string;
  171. CuStringInit(&string);
  172. if (message2 != NULL)
  173. {
  174. CuStringAppend(&string, message2);
  175. CuStringAppend(&string, ": ");
  176. }
  177. CuStringAppend(&string, message);
  178. CuFailInternal(tc, file, line, &string);
  179. }
  180. void CuAssert_Line(CuTest* tc, const char* file, int line, const char* message, int condition)
  181. {
  182. if (condition) return;
  183. CuFail_Line(tc, file, line, NULL, message);
  184. }
  185. void CuAssertStrnEquals_LineMsg(CuTest* tc, const char* file, int line, const char* message,
  186. const char* expected, size_t explen,
  187. const char* actual)
  188. {
  189. CuString string;
  190. if ((explen == 0) ||
  191. (expected == NULL && actual == NULL) ||
  192. (expected != NULL && actual != NULL &&
  193. strncmp(expected, actual, explen) == 0))
  194. {
  195. return;
  196. }
  197. CuStringInit(&string);
  198. if (message != NULL)
  199. {
  200. CuStringAppend(&string, message);
  201. CuStringAppend(&string, ": ");
  202. }
  203. CuStringAppend(&string, "expected <");
  204. CuStringAppend(&string, expected);
  205. CuStringAppend(&string, "> but was <");
  206. CuStringAppend(&string, actual);
  207. CuStringAppend(&string, ">");
  208. CuFailInternal(tc, file, line, &string);
  209. }
  210. void CuAssertStrEquals_LineMsg(CuTest* tc, const char* file, int line, const char* message,
  211. const char* expected, const char* actual)
  212. {
  213. CuString string;
  214. if ((expected == NULL && actual == NULL) ||
  215. (expected != NULL && actual != NULL &&
  216. strcmp(expected, actual) == 0))
  217. {
  218. return;
  219. }
  220. CuStringInit(&string);
  221. if (message != NULL)
  222. {
  223. CuStringAppend(&string, message);
  224. CuStringAppend(&string, ": ");
  225. }
  226. CuStringAppend(&string, "expected <");
  227. CuStringAppend(&string, expected);
  228. CuStringAppend(&string, "> but was <");
  229. CuStringAppend(&string, actual);
  230. CuStringAppend(&string, ">");
  231. CuFailInternal(tc, file, line, &string);}
  232. void CuAssertIntEquals_LineMsg(CuTest* tc, const char* file, int line, const char* message,
  233. int expected, int actual)
  234. {
  235. char buf[STRING_MAX];
  236. if (expected == actual) return;
  237. sprintf(buf, "expected <%d> but was <%d>", expected, actual);
  238. CuFail_Line(tc, file, line, message, buf);
  239. }
  240. void CuAssertDblEquals_LineMsg(CuTest* tc, const char* file, int line, const char* message,
  241. double expected, double actual, double delta)
  242. {
  243. char buf[STRING_MAX];
  244. if (fabs(expected - actual) <= delta) return;
  245. sprintf(buf, "expected <%lf> but was <%lf>", expected, actual);
  246. CuFail_Line(tc, file, line, message, buf);
  247. }
  248. void CuAssertPtrEquals_LineMsg(CuTest* tc, const char* file, int line, const char* message,
  249. void* expected, void* actual)
  250. {
  251. char buf[STRING_MAX];
  252. if (expected == actual) return;
  253. sprintf(buf, "expected pointer <0x%p> but was <0x%p>", expected, actual);
  254. CuFail_Line(tc, file, line, message, buf);
  255. }
  256. /*-------------------------------------------------------------------------*
  257. * CuSuite
  258. *-------------------------------------------------------------------------*/
  259. void CuSuiteInit(CuSuite* testSuite)
  260. {
  261. testSuite->count = 0;
  262. testSuite->failCount = 0;
  263. testSuite->setup = NULL;
  264. testSuite->teardown = NULL;
  265. }
  266. CuSuite* CuSuiteNew(void)
  267. {
  268. CuSuite* testSuite = CU_ALLOC(CuSuite);
  269. CuSuiteInit(testSuite);
  270. return testSuite;
  271. }
  272. void CuSuiteFree(CuSuite *testSuite)
  273. {
  274. free(testSuite);
  275. }
  276. void CuSuiteFreeDeep(CuSuite *testSuite)
  277. {
  278. int i;
  279. for (i = 0 ; i < testSuite->count ; ++i)
  280. {
  281. CuTest* testCase = testSuite->list[i];
  282. CuTestFree(testCase);
  283. }
  284. free(testSuite);
  285. }
  286. void CuSuiteAdd(CuSuite* testSuite, CuTest *testCase)
  287. {
  288. assert(testSuite->count < MAX_TEST_CASES);
  289. testSuite->list[testSuite->count] = testCase;
  290. testSuite->count++;
  291. /* CuSuiteAdd is called twice per test, don't reset the callbacks if
  292. already set. */
  293. if (!testCase->setup)
  294. testCase->setup = testSuite->setup;
  295. if (!testCase->teardown)
  296. testCase->teardown = testSuite->teardown;
  297. }
  298. void CuSuiteAddSuite(CuSuite* testSuite, CuSuite* testSuite2)
  299. {
  300. int i;
  301. for (i = 0 ; i < testSuite2->count ; ++i)
  302. {
  303. CuTest* testCase = testSuite2->list[i];
  304. CuSuiteAdd(testSuite, testCase);
  305. }
  306. }
  307. void CuSuiteRun(CuSuite* testSuite)
  308. {
  309. int i;
  310. for (i = 0 ; i < testSuite->count ; ++i)
  311. {
  312. CuTest* testCase = testSuite->list[i];
  313. CuTestRun(testCase);
  314. if (testCase->failed) { testSuite->failCount += 1; }
  315. }
  316. }
  317. void CuSuiteSummary(CuSuite* testSuite, CuString* summary)
  318. {
  319. int i;
  320. for (i = 0 ; i < testSuite->count ; ++i)
  321. {
  322. CuTest* testCase = testSuite->list[i];
  323. CuStringAppend(summary, testCase->failed ? "F" : ".");
  324. }
  325. CuStringAppend(summary, "\n\n");
  326. }
  327. void CuSuiteDetails(CuSuite* testSuite, CuString* details)
  328. {
  329. int i;
  330. int failCount = 0;
  331. if (testSuite->failCount == 0)
  332. {
  333. int passCount = testSuite->count - testSuite->failCount;
  334. const char* testWord = passCount == 1 ? "test" : "tests";
  335. CuStringAppendFormat(details, "OK (%d %s)\n", passCount, testWord);
  336. }
  337. else
  338. {
  339. if (testSuite->failCount == 1)
  340. CuStringAppend(details, "There was 1 failure:\n");
  341. else
  342. CuStringAppendFormat(details, "There were %d failures:\n", testSuite->failCount);
  343. for (i = 0 ; i < testSuite->count ; ++i)
  344. {
  345. CuTest* testCase = testSuite->list[i];
  346. if (testCase->failed)
  347. {
  348. failCount++;
  349. CuStringAppendFormat(details, "%d) %s: %s\n",
  350. failCount, testCase->name, testCase->message);
  351. }
  352. }
  353. CuStringAppend(details, "\n!!!FAILURES!!!\n");
  354. CuStringAppendFormat(details, "Runs: %d ", testSuite->count);
  355. CuStringAppendFormat(details, "Passes: %d ", testSuite->count - testSuite->failCount);
  356. CuStringAppendFormat(details, "Fails: %d\n", testSuite->failCount);
  357. }
  358. }
  359. void CuSuiteSetSetupTeardownCallbacks(CuSuite* testSuite, TestCallback setup,
  360. TestCallback teardown)
  361. {
  362. testSuite->setup = setup;
  363. testSuite->teardown = teardown;
  364. }