cut.c 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. #include "cut.h"
  2. struct cut_runtime cut;
  3. static char suite_pattern[64];
  4. static char case_pattern[64];
  5. static void _filter(int argc, char **argv)
  6. {
  7. int i = 0;
  8. struct cut_case *c = NULL;
  9. if (argc == 2 && 0 == strcmp(argv[1], "all")) {
  10. return;
  11. }
  12. for (i = 0; i < cut.ccnt_total; i++) {
  13. c = cut.clist[i];
  14. if ((argc == 2 && (0 != strcmp(c->sname, argv[1]))) ||
  15. (argc == 3 && (0 != strcmp(c->sname, argv[1]) || 0 != strcmp(c->cname, argv[2])))) {
  16. if (!(argc == 2 && strlen(suite_pattern) && strstr(c->sname, suite_pattern))) {
  17. cut.clist[i]->skip = 1;
  18. cut.ccnt_skip++;
  19. }
  20. }
  21. }
  22. }
  23. static void _usage(const char *me)
  24. {
  25. cut_printf("Usage: %s [OPTION] S-FILTER [C-FILTER]\n\n" \
  26. "OPTION:\n" \
  27. " --verbose: verbose when exec cases\n" \
  28. " --list: list cases\n" \
  29. " --count: print case count\n" \
  30. "\n" \
  31. "S-FILTER: suite name filter, e.g. '%s all' means run all suites\n" \
  32. "C-FILTER: case name filter\n", me, me);
  33. }
  34. static int _verbose_opt = 0;
  35. static int _parse_arg(int argc, char **argv)
  36. {
  37. if (argc >= 2) {
  38. if (0 == strcmp(argv[1], "--list")) {
  39. int i = 0;
  40. int cnt = 0;
  41. for (i = 0; i < cut.ccnt_total; i++) {
  42. struct cut_case *c = cut.clist[i];
  43. if (argc == 2 ||
  44. (argc == 3 && 0 == strcmp(argv[2], "all")) ||
  45. (argc == 3 && 0 == strcmp(c->sname, argv[2])) ||
  46. (argc == 3 && strlen(suite_pattern) && strstr(c->sname, suite_pattern)) ||
  47. (argc == 4 && strlen(suite_pattern) && strlen(case_pattern) && strstr(c->sname, suite_pattern)
  48. && strstr(c->cname, case_pattern)) ||
  49. (argc == 4 && 0 == strcmp(c->sname, argv[2]) && 0 == strcmp(c->cname, argv[3]))) {
  50. cut_printf(" [%02d] %s.%s\n", ++cnt, c->sname, c->cname);
  51. }
  52. }
  53. cut_printf("\n");
  54. cut_printf("In total %d case(s), matched %d case(s)\n", cut.ccnt_total, cnt);
  55. cut_printf("\n");
  56. return 0;
  57. }
  58. if (0 == strcmp(argv[1], "--count")) {
  59. cut_printf("total %d case(s).\n", cut.ccnt_total);
  60. return 0;
  61. }
  62. if (0 == strcmp(argv[1], "--help")) {
  63. _usage(argv[0]);
  64. return 0;
  65. }
  66. }
  67. return 1;
  68. }
  69. int cut_main(int argc, char **argv)
  70. {
  71. int i = 0, j = 0, cnt = 0;
  72. char tmpbuf[128];
  73. char *pos;
  74. if (argc >= 2) {
  75. int idx = 1;
  76. char *q = NULL;
  77. if (!strcmp(argv[1], "--list") || !strncmp(argv[1], "--verbose", strlen("--verbose"))) {
  78. idx += 1;
  79. }
  80. if (idx < argc) {
  81. if ((q = strchr(argv[idx], '%')) != NULL) {
  82. strncpy(suite_pattern, argv[idx], q - argv[idx]);
  83. }
  84. idx += 1;
  85. if (idx < argc) {
  86. if ((q = strchr(argv[idx], '%')) != NULL) {
  87. strncpy(case_pattern, argv[idx], q - argv[idx]);
  88. }
  89. }
  90. }
  91. }
  92. if (0 == _parse_arg(argc, argv)) {
  93. return 0;
  94. }
  95. if (argc >= 2 && !strncmp(argv[1], "--verbose", strlen("--verbose"))) {
  96. _verbose_opt = 1;
  97. argc --;
  98. argv ++;
  99. }
  100. _filter(argc, argv);
  101. for (; i < cut.ccnt_total; i++) {
  102. pos = tmpbuf;
  103. cut.ccur = cut.clist[i];
  104. if (cut.ccur->skip) {
  105. continue;
  106. }
  107. memset(tmpbuf, 0, sizeof(tmpbuf));
  108. pos += cut_snprintf(pos,
  109. sizeof(tmpbuf),
  110. "TEST [%02d/%02d] %s.%s ",
  111. ++cnt,
  112. cut.ccnt_total - cut.ccnt_skip,
  113. cut.ccur->sname,
  114. cut.ccur->cname);
  115. for (j = 80 - strlen(tmpbuf); j >= 0; --j) {
  116. pos += sprintf(pos, "%s", ".");
  117. }
  118. if (_verbose_opt) {
  119. pos += sprintf(pos, " [%sEXEC%s]\n", COL_YEL, COL_DEF);
  120. cut_printf("%s", tmpbuf);
  121. pos -= 19;
  122. }
  123. TRY {
  124. if (cut.ccur->setup)
  125. {
  126. cut.ccur->setup(cut.ccur->data, cut.ccur);
  127. }
  128. cut.ccur->run(cut.ccur->data, (struct cut_case *)cut.ccur);
  129. if (cut.ccur->teardown)
  130. {
  131. cut.ccur->teardown(cut.ccur->data);
  132. }
  133. pos += sprintf(pos, " [%sSUCC%s]\n", COL_GRE, COL_DEF);
  134. cut_printf("%s", tmpbuf);
  135. cut.ccnt_pass++;
  136. continue;
  137. }
  138. EXCEPT {
  139. pos += sprintf(pos, " [%sFAIL%s]\n", COL_RED, COL_DEF);
  140. cut_printf("%s", tmpbuf);
  141. cut.ccnt_fail++;
  142. continue;
  143. }
  144. }
  145. cut_printf("\n");
  146. cut_printf("===========================================================================\n");
  147. if (cut.ccnt_fail > 0) {
  148. cut_printf("FAIL LIST:\n");
  149. for (i = 0; i < cut.ccnt_fail; i++) {
  150. cut_printf(" [%02d] %s\n", i + 1, cut.cerrmsg[i]);
  151. cut_free(cut.cerrmsg[i]);
  152. }
  153. cut_printf("---------------------------------------------------------------------------\n");
  154. }
  155. cut_printf("SUMMARY:\n" \
  156. " TOTAL: %d\n" \
  157. " SKIPPED: %d\n" \
  158. " MATCHED: %d\n" \
  159. " PASS: %d\n" \
  160. " FAILED: %d\n", cut.ccnt_total, cut.ccnt_skip,
  161. cut.ccnt_total - cut.ccnt_skip, cut.ccnt_pass, cut.ccnt_fail);
  162. cut_printf("===========================================================================\n");
  163. return cut.ccnt_fail;
  164. }