-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbadkan.hpp
175 lines (148 loc) · 5.45 KB
/
badkan.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
/**
* Defines the class TestCase, that represents a test-case
* in an automatic testing program.
*
* Use the macros CHECK_EQUAL and CHECK_OK for testing, e.g.:
badkan::TestCase ("title")
.CHECK_OK(do_something())
.CHECK_EQUAL(sum(1,2), 3)
.CHECK_EQUAL(factorial(5), 120)
.CHECK_OK(do_something_else())
.print();
*
* @author Erel Segal-Halevi
* @credit Nathan Oliver: https://stackoverflow.com/a/54752705/827927
* @since 2019-02
*
*/
#pragma once
#define CHECK_OK(actual) check_ok([&](){actual;}, __FILE__+string(":")+to_string(__LINE__))
#define CHECK_THROWS(actual) check_throws([&](){actual;}, __FILE__+string(":")+to_string(__LINE__))
#define CHECK_EQUAL(actual,expected) check_equal([&](){return actual;}, expected, __FILE__+string(":")+to_string(__LINE__))
#define CHECK_OUTPUT(actual,expected) check_equal([&](){ostringstream ostr; ostr<<"'"<<actual<<"'"; return ostr.str();}, "'"+string(expected)+"'", __FILE__+string(":")+to_string(__LINE__))
#define CHECK_INPUT(actual,expected) check_input(actual, expected)
#include <string>
#include <iostream>
#include <sstream>
#include <stdexcept>
#include <ctime>
#include <csignal>
#include <csetjmp>
using std::string, std::ostream, std::endl, std::cerr, std::exception, std::ostringstream, std::istringstream, std::to_string;
namespace badkan {
/* The following code is used for catching signals (e.g. for timeout) during the execution. */
jmp_buf longjmp_buffer; // A buffer to hold info on where to jump to.
void catch_signal(int signal_number) {
longjmp (longjmp_buffer,signal_number);
}
class TestCase {
string name; // Name of test-case - for showing in error messages
int passed, failed; // Number of tests passed/failed so far.
ostream& output; // Stream to which the error messages will be sent.
clock_t start_time; // time in which the test-case is constructed.
public:
TestCase(const string& name="", ostream& output=cerr):
name(name),
output(output),
passed(0), failed(0),
start_time(clock())
{
output << std::boolalpha;
std::signal(SIGTERM, catch_signal);
std::signal(SIGSEGV, catch_signal);
}
int right() const { return this->passed; }
int wrong() const { return this->failed; }
int total() const { return failed+passed; }
int grade() const { return (total()==0? 0: 100*passed/total()); }
ostream& print(ostream& out, bool show_grade=true) const {
double elapsed_seconds = double(clock() - start_time) / CLOCKS_PER_SEC;
out << endl << "*** CPU time: " << elapsed_seconds << " sec. Right: " << passed << ". Wrong: " << failed << ". ";
if (show_grade) out << "Grade: " << grade() << ". ";
out << "***" << endl;
return out;
}
TestCase& print() {
print(output, /*show_grade=*/true);
return *this;
}
TestCase& print_signal(int signal) {
if (signal==SIGTERM) {
output << "Your program timed out! (caught signal TERM)" << endl;
} else if (signal==SIGSEGV) {
output << "Your program made a segmentation fault! (caught signal SEGV)" << endl;
} else {
output << "Your program caught signal #" << signal << endl;
}
print(output, /*show_grade=*/false);
return *this;
}
TestCase& setname(const string& newname) {
name = newname;
return *this;
}
/** Checks that the given function is OK, i.e., does not throw an exception */
template<typename TFUNC> TestCase& check_ok(const TFUNC actual_func, const string& context) {
try {
get_actual_value<TFUNC,void>(actual_func, context);
passed++;
} catch(...) {}
return *this;
}
/** Checks that the given function throws an exception */
template<typename TFUNC> TestCase& check_throws(const TFUNC actual_func, const string& context) {
try {
actual_func();
failed++;
output << title(context) << "There should be an exception!" << endl;
} catch(...) {
passed++;
}
return *this;
}
template<typename TFUNC, typename TVAL> TestCase& check_equal(const TFUNC actual_func, const TVAL& expected_value, const string& context) {
try {
TVAL actual_value = get_actual_value<TFUNC,TVAL>(actual_func, context);
if (incorrect(actual_value==expected_value))
output << title(context) << "the result was " << actual_value << " but it should equal " << expected_value << "!" << endl;
} catch(...) {}
return *this;
}
private:
string title(const string& context) {
return name + /*" test #" + std::to_string(total()) + */ ", " + context + ": ";
}
template<typename TFUNC, typename TVAL> TVAL get_actual_value(TFUNC actual_func, const string& context) {
try {
if (std::is_same<TVAL,void>::value) {
// output << title(context) << " void " << endl;
actual_func();
} else {
// output << title(context) << " TVAL " << endl;
return actual_func();
}
} catch (const string& message) {
failed++;
output << title(context) << "There was an exception: " << message << endl;
throw;
} catch (const exception& ex) {
failed++;
output << title(context) << "There was an exception: " << ex.what() << endl;
throw;
} catch (...) {
failed++;
output << title(context) << "There was an exception." << endl;
throw;
}
}
bool incorrect(bool condition) {
if (condition) {
passed++;
return false;
} else {
failed++;
return true;
}
}
}; // end class TestCase
}; // end namespace badkan