180 {
181 std::regex enabled(".*"), disabled;
182 bool list_tests = false;
183 std::string output_xml;
184 for(int n = 1; n < argc; n++)
185 {
186
187 if(argv[n][0] == '-' && argv[n][1] == '-')
188 {
189 if(strstr(argv[n] + 2, "help"))
190 {
191 std::cout << "\nQuickCppLib minimal unit test framework\n\n"
192 << "Usage: " << argv[0] << " [options] [<regex for tests to run, defaults to .*>] [-<regex for tests to not run>]\n\n"
193 << " --help : Prints this help\n"
194 << " --list-tests : List matching tests\n"
195 << " --reporter <format> : Reporter to use, only format possible is junit\n"
196 << " --out <filename> : Write JUnit XML test report to filename\n"
197 << std::endl;
198 return 0;
199 }
200 else if(strstr(argv[n] + 2, "list-tests"))
201 {
202 list_tests = true;
203 }
204 else if(strstr(argv[n] + 2, "reporter"))
205 {
206 if(n + 1 >= argc || strcmp(argv[n + 1], "junit"))
207 {
208 std::cerr << "--reporter must be followed by 'junit'" << std::endl;
209 return 1;
210 }
211 n++;
212 }
213 else if(strstr(argv[n] + 2, "out"))
214 {
215 if(n + 1 >= argc)
216 {
217 std::cerr << "--out must be followed by the output filename" << std::endl;
218 return 1;
219 }
220 output_xml = argv[n + 1];
221 n++;
222 }
223 }
224 else
225 {
226
227 if(argv[n][0] == '-')
228 disabled.assign(argv[n] + 1);
229 else
230 enabled.assign(argv[n]);
231 }
232 }
233 if(list_tests)
234 {
235 size_t maxname = 0;
237 {
238 for(const auto &i : j.test_cases)
239 {
240 if(strlen(i.name) > maxname)
241 maxname = strlen(i.name);
242 }
243 }
245 {
246 std::cout << "\n" << j.name << ":\n";
247 for(const auto &i : j.test_cases)
248 {
249 if(std::regex_match(i.name, enabled) && !std::regex_match(i.name, disabled))
250 {
251 std::string padding(maxname - strlen(i.name), ' ');
252 std::cout << " " << i.name << padding << " (" << i.desc << ")\n";
253 }
254 }
255 }
256 std::cout << std::endl;
257 return 0;
258 }
260 {
261 for(auto &i : j.test_cases)
262 {
263 if(std::regex_match(i.name, enabled) && !std::regex_match(i.name, disabled))
264 {
266 std::cout << std::endl <<
bold <<
blue << i.name <<
white <<
" : " << i.desc <<
normal << std::endl;
267 std::chrono::steady_clock::time_point begin, end;
268#ifdef __cpp_exceptions
269 try
270 {
271#else
273 {
274 end = std::chrono::steady_clock::now();
275 i.requirement_failed = true;
276 }
277 else
278#endif
279 {
280 begin = std::chrono::steady_clock::now();
281 i.func();
282 end = std::chrono::steady_clock::now();
283 }
284#ifdef __cpp_exceptions
285 }
286 catch(const requirement_failed &)
287 {
288 end = std::chrono::steady_clock::now();
289 i.requirement_failed = true;
290 }
291 catch(const std::exception &e)
292 {
293 end = std::chrono::steady_clock::now();
294 ++i.fails;
295 std::cerr <<
red <<
"FAILURE: std::exception '" << e.what() <<
"' thrown out of test case" <<
normal << std::endl;
296 }
297 catch(...)
298 {
299 end = std::chrono::steady_clock::now();
300 ++i.fails;
301 std::cerr <<
red <<
"FAILURE: Exception thrown out of test case" <<
normal << std::endl;
302 }
303#endif
304 i.duration = end - begin;
305 if(i.passes)
306 std::cout <<
green << i.passes <<
" checks passed ";
307 if(i.fails)
308 std::cout <<
red << i.fails <<
" checks failed ";
309 std::cout <<
normal <<
"duration " << std::chrono::duration_cast<std::chrono::milliseconds>(i.duration).count() <<
" ms" << std::endl;
310 }
311 else
312 i.skipped = true;
313 }
314 }
316 std::ofstream oh;
317 if(!output_xml.empty())
318 {
319 oh.open(output_xml);
320 oh << R"(<?xml version="1.0" encoding="UTF-8"?>
321<testsuites>
322)";
323 }
324 size_t totalpassed = 0, totalfailed = 0, totalskipped = 0;
325 double totaltime = 0;
327 {
328 size_t passed = 0, failed = 0, skipped = 0;
329 double time = 0;
330 for(const auto &i : j.test_cases)
331 {
332 if(i.skipped)
333 ++skipped;
334 else if(i.fails)
335 ++failed;
336 else
337 ++passed;
338 time += std::chrono::duration_cast<std::chrono::duration<double>>(i.duration).count();
339 }
340 if(!output_xml.empty())
341 {
342 oh << " <testsuite name=\"" << j.name << "\" errors=\"" << 0 << "\" failures=\"" << failed << "\" skipped=\"" << skipped << "\" tests=\"" << j.test_cases.size() << "\" hostname=\"\" time=\"" << time << "\" timestamp=\"\">\n";
343 for(const auto &i : j.test_cases)
344 {
345 oh << " <testcase classname=\"\" name=\"" << i.name << "\" time=\"" << std::chrono::duration_cast<std::chrono::duration<double>>(i.duration).count() << "\">";
346 if(i.skipped)
347 oh << "<skipped/>";
348 else if(i.fails)
349 oh << "<failure/>";
350 oh << "</testcase>\n";
351 }
352 oh << " </testsuite>\n";
353 }
354 totalpassed += passed;
355 totalfailed += failed;
356 totalskipped += skipped;
357 totaltime += time;
358 }
359 if(!output_xml.empty())
360 {
361 oh << R"(</testsuites>
362)";
363 }
constexpr const char bold[]
Definition console_colours.hpp:135
constexpr const char normal[]
Definition console_colours.hpp:136
constexpr const char blue[]
Definition console_colours.hpp:130
constexpr const char red[]
Definition console_colours.hpp:128
constexpr const char green[]
Definition console_colours.hpp:129
constexpr const char white[]
Definition console_colours.hpp:134
std::vector< test_suite > & test_suites()
Definition unit_test.hpp:163
jmp_buf & test_case_failed()
Definition unit_test.hpp:112
test_case *& current_test_case()
Definition unit_test.hpp:173