micro-test-plus 3.2.2
The µTest++ Testing Framework
Loading...
Searching...
No Matches
test-reporter.cpp
Go to the documentation of this file.
1/*
2 * This file is part of the µOS++ project (https://micro-os-plus.github.io/).
3 * Copyright (c) 2021 Liviu Ionescu. All rights reserved.
4 *
5 * Permission to use, copy, modify, and/or distribute this software
6 * for any purpose is hereby granted, under the terms of the MIT license.
7 *
8 * If a copy of the license was not distributed with this file, it can
9 * be obtained from https://opensource.org/licenses/mit.
10 *
11 * Major parts of the code are inspired from v1.1.8 of the Boost UT project,
12 * released under the terms of the Boost Version 1.0 Software License,
13 * which can be obtained from https://www.boost.org/LICENSE_1_0.txt.
14 */
15
16// ----------------------------------------------------------------------------
17
18#if defined(MICRO_OS_PLUS_INCLUDE_CONFIG_H)
19#include <micro-os-plus/config.h>
20#endif // MICRO_OS_PLUS_INCLUDE_CONFIG_H
21
23
24// ----------------------------------------------------------------------------
25
26#pragma GCC diagnostic ignored "-Waggregate-return"
27#if defined(__clang__)
28#pragma clang diagnostic ignored "-Wc++98-compat"
29#pragma clang diagnostic ignored "-Wc++98-compat-pedantic"
30#endif
31
33{
34 // --------------------------------------------------------------------------
35
38 {
39 reporter.endline ();
40 return stream;
41 }
42
43 // --------------------------------------------------------------------------
44
45 void
47 {
48 *this << colors_.pass;
50 {
51 *this << " ";
52 }
53 *this << " ✓ ";
54 *this << colors_.none;
55 if (!message.empty ())
56 {
57 *this << message.c_str ();
58 }
59 }
60
61 void
63 {
64 *this << endl;
65
66 flush ();
67 }
68
69 void
71 std::string& message, const reflection::source_location& location)
72 {
73 *this << colors_.fail;
75 {
76 *this << " ";
77 }
78 *this << " ✗ ";
79 *this << colors_.none;
80 if (!message.empty ())
81 {
82 *this << message.c_str ();
83 *this << " ";
84 }
85 *this << colors_.fail << "FAILED" << colors_.none;
86#pragma GCC diagnostic push
87#if defined(__clang__)
88#pragma clang diagnostic ignored "-Wsign-conversion"
89#elif defined(__GNUC__)
90#pragma GCC diagnostic ignored "-Wnarrowing"
91#pragma GCC diagnostic ignored "-Wsign-conversion"
92#endif
93 *this << " (" << reflection::short_name (location.file_name ()) << ":"
95 location.line ()
96 };
97#pragma GCC diagnostic pop
98 }
99
100 void
102 {
103 *this << ")";
104 if (abort)
105 {
106 *this << " aborted...";
107 }
108 *this << endl;
109
110 flush ();
111 }
112
115 {
116 // Call the endl function.
117 (*func) (*this);
118 return *this;
119 }
120
121 void
123 {
124 out_.append ("\n");
125 flush ();
126 }
127
128 void
130 {
131 fflush (stdout); // Sync STDOUT.
132 }
133
135 test_reporter::operator<< (std::string_view sv)
136 {
137 out_.append (sv);
138 return *this;
139 }
140
143 {
144 out_.append (1, c);
145 return *this;
146 }
147
149 test_reporter::operator<< (const char* s)
150 {
151 out_.append (s);
152 return *this;
153 }
154
157 {
158 out_.append (s);
159 return *this;
160 }
161
164 {
165 out_.append (v ? "true" : "false");
166 return *this;
167 }
168
170 test_reporter::operator<< (std::nullptr_t)
171 {
172 out_.append ("nullptr");
173 return *this;
174 }
175
177 test_reporter::operator<< (signed char c)
178 {
179 out_.append (std::to_string (c));
180 out_.append ("c");
181 return *this;
182 }
183
185 test_reporter::operator<< (unsigned char c)
186 {
187 out_.append (std::to_string (static_cast<int> (c)));
188 out_.append ("uc");
189 return *this;
190 }
191
193 test_reporter::operator<< (signed short v)
194 {
195 out_.append (std::to_string (v));
196 out_.append ("s");
197 return *this;
198 }
199
201 test_reporter::operator<< (unsigned short v)
202 {
203 out_.append (std::to_string (static_cast<long> (v)));
204 out_.append ("us");
205 return *this;
206 }
207
209 test_reporter::operator<< (signed int v)
210 {
211 out_.append (std::to_string (v));
212 return *this;
213 }
214
216 test_reporter::operator<< (unsigned int v)
217 {
218 out_.append (std::to_string (v));
219 out_.append ("u");
220 return *this;
221 }
222
224 test_reporter::operator<< (signed long v)
225 {
226 out_.append (std::to_string (v));
227 out_.append ("l");
228 return *this;
229 }
230
232 test_reporter::operator<< (unsigned long v)
233 {
234 out_.append (std::to_string (v));
235 out_.append ("ul");
236 return *this;
237 }
238
240 test_reporter::operator<< (signed long long v)
241 {
242 out_.append (std::to_string (v));
243 out_.append ("ll");
244 return *this;
245 }
246
248 test_reporter::operator<< (unsigned long long v)
249 {
250 out_.append (std::to_string (v));
251 out_.append ("ull");
252 return *this;
253 }
254
257 {
258 out_.append (std::to_string (v));
259 out_.append ("f");
260 return *this;
261 }
262
265 {
266 out_.append (std::to_string (v));
267 return *this;
268 }
269
271 test_reporter::operator<< (long double v)
272 {
273 out_.append (std::to_string (v));
274 out_.append ("l");
275 return *this;
276 }
277
278 void
279 test_reporter::begin_test_case ([[maybe_unused]] const char* name)
280 {
281 is_in_test_case_ = true;
282
283 if (!out_.empty () && (verbosity == verbosity::verbose))
284 {
285 if (add_empty_line)
286 {
287 printf ("\n");
288 }
289 output ();
290 add_empty_line = true;
291 }
292
293 out_.clear ();
294
295 flush ();
296 }
297
298 void
299 test_reporter::end_test_case ([[maybe_unused]] const char* name)
300 {
302 {
303 if (current_test_suite->current_test_case.failed_checks > 0)
304 {
305 if (true /* add_empty_line */)
306 {
307 printf ("\n");
308 }
309 printf (" • %s - test case started\n", name);
310 output ();
311 printf (
312 " %s✗%s %s - test case %sFAILED%s (%d %s passed, %d "
313 "failed)\n",
314 colors_.fail, colors_.none, name, colors_.fail, colors_.none,
315 current_test_suite->current_test_case.successful_checks,
316 current_test_suite->current_test_case.successful_checks == 1
317 ? "check"
318 : "checks",
319 current_test_suite->current_test_case.failed_checks);
320 add_empty_line = true;
321 }
322 else
323 {
324 if (add_empty_line)
325 {
326 printf ("\n");
327 }
329 {
330 printf (" • %s - test case started\n", name);
331 output ();
332 printf (
333 " %s✓%s %s - test case passed (%d %s)\n", colors_.pass,
334 colors_.none, name,
335 current_test_suite->current_test_case.successful_checks,
336 current_test_suite->current_test_case.successful_checks
337 == 1
338 ? "check"
339 : "checks");
340
341 add_empty_line = true;
342 }
343 else
344 {
345 printf (
346 " %s✓%s %s - test case passed (%d %s)\n", colors_.pass,
347 colors_.none, name,
348 current_test_suite->current_test_case.successful_checks,
349 current_test_suite->current_test_case.successful_checks
350 == 1
351 ? "check"
352 : "checks");
353
354 add_empty_line = false;
355 }
356 }
357 }
358
359 out_.clear ();
360 flush ();
361
362 is_in_test_case_ = false;
363 }
364
365 void
367 {
368 if (add_empty_line)
369 {
370 flush ();
371 printf ("\n");
372 }
373
375 {
376 add_empty_line = false;
377 return;
378 }
379
380 printf ("• %s - test suite started\n", name);
381
382 add_empty_line = true;
383 }
384
385 void
387 {
389 {
390 return;
391 }
392
393 if (suite.test_cases () > 0 && verbosity != verbosity::quiet)
394 {
395 printf ("\n");
396 add_empty_line = true;
397 }
398
399 // Also fail if none passed.
400 if (suite.failed_checks () == 0 && suite.successful_checks () != 0)
401 {
402 printf ("%s✓%s %s - test suite passed (%d %s in %d test %s)\n",
403 colors_.pass, colors_.none, suite.name (),
404 suite.successful_checks (),
405 suite.successful_checks () == 1 ? "check" : "checks",
406 suite.test_cases (),
407 suite.test_cases () == 1 ? "case" : "cases");
408 }
409 else
410 {
411 printf ("%s✗%s %s - test suite %sFAILED%s (%d %s passed, %d failed, "
412 "in %d test %s)\n",
413 colors_.fail, colors_.none, suite.name (), colors_.fail,
414 colors_.none, suite.successful_checks (),
415 suite.successful_checks () == 1 ? "check" : "checks",
416 suite.failed_checks (), suite.test_cases (),
417 suite.test_cases () == 1 ? "case" : "cases");
418 }
419 flush ();
420 }
421
422 void
424 {
425 printf ("%s", out_.c_str ()); // No `\n` here.
426 out_.clear ();
427 }
428
429 // --------------------------------------------------------------------------
430} // namespace micro_os_plus::micro_test_plus
431
432// ----------------------------------------------------------------------------
Local implementation of the std::source_location.
Definition reflection.h:57
Reporter to display the test results. For failed tests it prints the actual values of the operands,...
void flush(void)
Flush the current buffered content.
test_reporter & operator<<(std::string_view sv)
void output_fail_prefix_(std::string &message, const reflection::source_location &location)
Base class for all test suites.
Definition test-suite.h:50
constexpr const char * name()
Get the suite name.
Definition test-suite.h:106
constexpr int test_cases(void)
Get the number of test cases.
Definition test-suite.h:162
const char * short_name(const char *name)
test_reporter & endl(test_reporter &stream)