QuickCppLib 0.10
Eliminate all the tedious hassle when making state-of-the-art C++ 14 - 23 libraries!
Loading...
Searching...
No Matches
quickcpplib::_xxx::unit_test Namespace Reference

Classes

struct  requirement_failed
 
struct  test_case
 
struct  test_case_registration
 
struct  test_suite
 
struct  test_suite_registration
 

Functions

jmp_buf & test_case_failed ()
 
std::vector< test_suite > & test_suites ()
 
test_suite *& current_test_suite ()
 
test_case *& current_test_case ()
 
int run (int argc, const char *const argv[])
 

Function Documentation

◆ test_case_failed()

jmp_buf & quickcpplib::_xxx::unit_test::test_case_failed ( )
externinline
113 {
114 static jmp_buf b;
115 return b;
116 }

◆ test_suites()

std::vector< test_suite > & quickcpplib::_xxx::unit_test::test_suites ( )
externinline
164 {
165 static std::vector<test_suite> v;
166 return v;
167 }

◆ current_test_suite()

test_suite *& quickcpplib::_xxx::unit_test::current_test_suite ( )
externinline
169 {
170 static test_suite *v;
171 return v;
172 }
Definition unit_test.hpp:155

◆ current_test_case()

test_case *& quickcpplib::_xxx::unit_test::current_test_case ( )
externinline
174 {
175 static test_case default_test_case("unnamed", "Default test case for unit test which don't declare test cases", nullptr);
176 static test_case *v = &default_test_case;
177 return v;
178 }
Definition unit_test.hpp:119

◆ run()

int quickcpplib::_xxx::unit_test::run ( int  argc,
const char *const  argv[] 
)
externinline
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 // Double -- means it's a flag
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 // -regex is disabled, otherwise it's an enabled
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;
236 for(const auto &j : test_suites())
237 {
238 for(const auto &i : j.test_cases)
239 {
240 if(strlen(i.name) > maxname)
241 maxname = strlen(i.name);
242 }
243 }
244 for(const auto &j : test_suites())
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 }
259 for(auto &j : test_suites())
260 {
261 for(auto &i : j.test_cases)
262 {
263 if(std::regex_match(i.name, enabled) && !std::regex_match(i.name, disabled))
264 {
265 current_test_case() = &i;
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
272 if(setjmp(test_case_failed()))
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 }
315 current_test_case() = nullptr;
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;
326 for(const auto &j : test_suites())
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