-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathRules.h
141 lines (129 loc) · 4.05 KB
/
Rules.h
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
#include<functional>
#include<iostream>
using namespace std;
/* Rules interface */
class IRule{
public:
virtual ~IRule(){};
virtual void process(int value){};
virtual bool isValid(){};
virtual string getName(){};
virtual operator bool() const{};
};
/* Concrete base class for all the rules */
class Rule: public IRule{
public:
~Rule(){};
Rule(const string &name):valid(valid), name(name){};
void process(int value){};
// Shouldn't have to override anything below here
string getName(){return name;};
bool isValid(){return valid;};
operator bool() const{return valid;};
friend ostream& operator<<(ostream &out, Rule &rule){
string result = rule.isValid() ? "passed":"failed";
out << "Rule " << rule.getName() << ' ' << result;
return out;
}
protected:
string name;
bool valid;
};
/* Applies a boolean operation to the 2 working values
* with an optional offset added to the incoming value
* before performing the boolean operation
*/
template<class T>
class BooleanRule : public Rule{
public:
~BooleanRule(){};
BooleanRule(string name, T op, int offset=0):
Rule(name), op(op), offset(offset){};
void process(int value){
if(valid) return;
if(started){
// Setup the initial value
val1 = value;
started = true;
return;
}
val2 = value;
//Alwas the new value first ( NEW < OLD, NEW > OLD, etc..)
if(op(val2+offset, val1)){
valid = true;
}
swap(val1, val2);
};
private:
int val1, val2, offset;
T op;
bool started;
};
/* Applies an arithmetic operation to the 2 working values
* and checks for equality of the operation result with the target value.
* If the optional boolean operator is given, it will use that operator
* instead of the equality operator
*/
template<class T1, class T2 >
class ArithmeticRule : public Rule{
public:
~ArithmeticRule(){};
ArithmeticRule(string name, T1 op, int target, T2 optional=NULL):
Rule(name), op(op), optional(optional), target(target){};
void process(int value){
if(valid) return;
if(!started){
// Setup the initial value
val1 = value;
started = true;
return;
}
val2 = value;
result = op(val2, val1);
//If we have the optional bool operator use that against target
//otherwise, just do an equality check
if(optional != NULL){
valid = optional(result, target);
}
else if(result == target){
valid = true;
}
swap(val1, val2);
};
private:
string name;
int val1, val2, target, result;
T1 op;
T2 optional;
bool started;
};
/* Computes the running average and compares for equality to the target value.
* If the optional boolean operator is given, it will use that operator
* instead of the equality operator
*/
template<class T>
class AvgRule : public Rule{
public:
~AvgRule(){};
AvgRule(string name, int target, T optional=NULL):
Rule(name), optional(optional), target(target), count(count),
total(total){};
void process(int value){
count++;
total += value;
avg = total/count;
//If we have the optional bool operator use that against target
//otherwise, just do an equality check
if(optional != NULL){
valid = optional(avg, target);
}
else{
valid = (avg == target);
}
};
bool isValid(){return valid;};
operator bool() const{return valid;};
private:
int target, avg, count, total;
T optional;
};